From 43ab6dfb6acc94b1cc3e85569e1ff24d66ecad79 Mon Sep 17 00:00:00 2001 From: Nikhil Tanwar <2002nikhiltanwar@gmail.com> Date: Fri, 11 Feb 2022 21:59:26 +0530 Subject: [PATCH] Add ability to filter by tags in kiwix serve This change introduces filtering by tags. To filter, the user can click on the tag name and it will filter it. A label is added (clickable) to show the tag filter, it can be clicked to remove the filter --- static/skin/index.css | 38 ++++++++++++++++++++++-- static/skin/index.js | 59 +++++++++++++++++++++++++++++++++---- static/templates/index.html | 1 + test/server.cpp | 4 +-- 4 files changed, 92 insertions(+), 10 deletions(-) diff --git a/static/skin/index.css b/static/skin/index.css index 800aca210..8443feacb 100644 --- a/static/skin/index.css +++ b/static/skin/index.css @@ -134,6 +134,27 @@ body { position: relative; } +.tagFilterLabel { + width: max-content; + padding: 10px; + font-family: roboto; + font-size: 12px; + margin: 0 0 0 17px; + text-align: center; + background-color: transparent; + color: #909090; + display: none; +} + +.tag__link { + cursor: pointer; + color: #909090; +} + +.tag__link:hover { + font-weight: bolder; +} + .book__list { position: relative; } @@ -144,6 +165,8 @@ body { .book__link { text-decoration: none; + grid-column: 1 / 3; + grid-row: 1 / 3; } .book__wrapper { @@ -165,6 +188,12 @@ body { transform: scale(1.07); } +.book__link__wrapper { + display: grid; + grid-template-columns: 65px 1fr; + grid-template-rows: 70px 120px 1fr 1fr; +} + .book__icon { background-image: url(''); background-repeat: no-repeat; @@ -249,7 +278,6 @@ body { text-align: end; font-size: 1.1rem; justify-content: flex-end; - color: #909090; font-family: roboto; margin-right: 10px; margin-top: 10px; @@ -424,7 +452,7 @@ body { @media screen and (max-width: 590px) { .kiwixNav { - height: 257px; + height: 285px; } .kiwixHomeBody { @@ -436,10 +464,14 @@ body { } .kiwixButton { - margin: 19px 0; + margin: 15px 0; width: 229px; } + .tagFilterLabel { + margin: 15px 0 0 0; + } + .kiwixNav__filters { grid-template-columns: 1fr; } diff --git a/static/skin/index.js b/static/skin/index.js index a6bcdd744..83ad0e8c8 100644 --- a/static/skin/index.js +++ b/static/skin/index.js @@ -77,6 +77,13 @@ return queryNode != null ? queryNode.innerHTML : ""; } + function generateTagLink(tagValue) { + tagValue = tagValue.toLowerCase(); + const humanFriendlyTagValue = humanFriendlyTitle(tagValue); + const tagMessage = `Filter by tag "${humanFriendlyTagValue}"`; + return `${humanFriendlyTagValue}` + } + function generateBookHtml(book, sort = false) { const link = book.querySelector('link[type="text/html"]').getAttribute('href'); let iconUrl; @@ -91,9 +98,9 @@ const langCode = getInnerHtml(book, 'language'); const language = languages[langCode]; const tags = getInnerHtml(book, 'tags'); - let tagHtml = tags.split(';').filter(tag => {return !(tag.split(':')[0].startsWith('_'))}) - .map((tag) => {return tag.charAt(0).toUpperCase() + tag.slice(1)}) - .join(' | ').replace(/_/g, ' '); + const tagList = tags.split(';').filter(tag => {return !(tag.startsWith('_'))}); + const tagFilterLinks = tagList.map((tagValue) => generateTagLink(tagValue)); + const tagHtml = tagFilterLinks.join(' | '); let downloadLink; let zimSize = 0; try { @@ -113,17 +120,21 @@ } const faviconAttr = iconUrl != undefined ? `style="background-image: url('${iconUrl}')"` : ''; const languageAttr = langCode != '' ? `title="${language}" aria-label="${language}"` : 'style="background-color: transparent"'; - divTag.innerHTML = ` + divTag.innerHTML = `
+ + +
${getLanguageCodeToDisplay(langCode)}
${tagHtml}
-
`; + `; return divTag; } @@ -333,6 +344,7 @@ insertModal(downloadButton); } }); + refreshTagLinks(); } async function resetAndFilter(filterType = '', filterValue = '') { @@ -376,10 +388,45 @@ } }); } + + function addTagElement(tagValue, resetFilter) { + const tagElement = document.getElementsByClassName('tagFilterLabel')[0]; + tagElement.style.display = 'inline-block'; + const humanFriendlyTagValue = humanFriendlyTitle(tagValue); + tagElement.innerHTML = `${humanFriendlyTagValue}`; + const tagMessage = `Stop filtering by tag "${humanFriendlyTagValue}"`; + tagElement.setAttribute('aria-label', tagMessage); + tagElement.setAttribute('title', tagMessage); + if (resetFilter) + resetAndFilter('tag', tagValue); + } + + function refreshTagLinks() { + const tagLinks = document.getElementsByClassName('tag__link'); + [...tagLinks].forEach(elem => { + if (!elem.getAttribute('click-listener')) { + elem.addEventListener('click', () => addTagElement(elem.dataset.tag, true)); + elem.setAttribute('click-listener', 'true'); + } + }); + } + + function removeTagElement(resetFilter) { + const tagElement = document.getElementsByClassName('tagFilterLabel')[0]; + tagElement.style.display = 'none'; + if (resetFilter) + resetAndFilter('tag', ''); + } function updateVisibleParams() { document.querySelectorAll('.filter').forEach(filter => {filter.value = params.get(filter.name) || ''}); updateFilterColors(); + const tagKey = params.get('tag'); + if (tagKey !== null && tagKey.trim() !== '') { + addTagElement(tagKey, false); + } else { + removeTagElement(false); + } } window.addEventListener('resize', (event) => { @@ -417,6 +464,8 @@ document.querySelectorAll('.filter').forEach(filter => { filter.addEventListener('change', () => {resetAndFilter(filter.name, filter.value)}); }); + const tagElement = document.getElementsByClassName('tagFilterLabel')[0]; + tagElement.addEventListener('click', () => removeTagElement(true)); if (filters) { const currentLink = window.location.search; const newLink = `?${params.toString()}`; diff --git a/static/templates/index.html b/static/templates/index.html index a62185fa8..0dabc45b7 100644 --- a/static/templates/index.html +++ b/static/templates/index.html @@ -58,6 +58,7 @@
+
diff --git a/test/server.cpp b/test/server.cpp index 145f59226..3e31d1f6d 100644 --- a/test/server.cpp +++ b/test/server.cpp @@ -179,12 +179,12 @@ R"EXPECTEDRESULT( src="/ROOT/skin/jquery-ui/external/jquery/jquery.js?cache src="/ROOT/skin/jquery-ui/jquery-ui.min.js?cacheid=d927c2ff" href="/ROOT/skin/jquery-ui/jquery-ui.min.css?cacheid=e1de77b3" href="/ROOT/skin/jquery-ui/jquery-ui.theme.min.css?cacheid=2a5841f9" - href="/ROOT/skin/index.css?cacheid=1aca980a" + href="/ROOT/skin/index.css?cacheid=a1acc52f" src: url("/ROOT/skin/fonts/Poppins.ttf?cacheid=af705837") format("truetype"); src: url("/ROOT/skin/fonts/Roboto.ttf?cacheid=84d10248") format("truetype"); - + )EXPECTEDRESULT" }, {