From a5dd3fbeb976d3a6b8d9736ddd56f437912b1be2 Mon Sep 17 00:00:00 2001 From: "H. Kamran" Date: Fri, 1 Nov 2024 18:39:22 -0700 Subject: [PATCH 1/5] Lower minimum query length Resolves #67 --- src/components/search.jsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/components/search.jsx b/src/components/search.jsx index f9609eb..ff287ee 100644 --- a/src/components/search.jsx +++ b/src/components/search.jsx @@ -37,7 +37,7 @@ function hitToAPI(hit) { * @param {Object} apiCategories - The regional categories from the API */ function sendSearch(query, apiCategories) { - if (query === undefined || query === "" || (query.length < 3 && !query.match('^[x|X]$')) || query.match('http(s)?:\/\/.*') || query.match('^2fa(:)?$')) { + if (query === undefined || query === "" || (query.length < 2 && !query.match('^[x|X]$')) || query.match('http(s)?:\/\/.*') || query.match('^2fa(:)?$')) { document.getElementById("categories").style.display = "grid"; render(null, document.getElementById("search-categories")); } else { From 2c0a8e1c7428513a45c2467532cea9aab7397faf Mon Sep 17 00:00:00 2001 From: "H. Kamran" Date: Fri, 1 Nov 2024 18:47:13 -0700 Subject: [PATCH 2/5] Remove categories from search results --- src/components/search.jsx | 33 ++++----------------------------- 1 file changed, 4 insertions(+), 29 deletions(-) diff --git a/src/components/search.jsx b/src/components/search.jsx index ff287ee..9e0c3e7 100644 --- a/src/components/search.jsx +++ b/src/components/search.jsx @@ -1,6 +1,4 @@ import { render } from "preact"; -import { useEffect, useState } from "preact/hooks"; -import { API_URL } from "../constants"; import algoliasearch from "algoliasearch"; import Table from "./table"; @@ -34,9 +32,8 @@ function hitToAPI(hit) { * Send a search query to Algolia * * @param {string} query - The query - * @param {Object} apiCategories - The regional categories from the API */ -function sendSearch(query, apiCategories) { +function sendSearch(query) { if (query === undefined || query === "" || (query.length < 2 && !query.match('^[x|X]$')) || query.match('http(s)?:\/\/.*') || query.match('^2fa(:)?$')) { document.getElementById("categories").style.display = "grid"; render(null, document.getElementById("search-categories")); @@ -66,25 +63,13 @@ function sendSearch(query, apiCategories) { .then(({ hits }) => { const entries = hits.map((hit) => hitToAPI(hit)) .filter(([, entry]) => !entry.regions || entry.regions.includes(region)); - const categories = {}; - - entries.forEach((entry) => { - for (const category of entry[1].categories) { - if (!(category in categories)) categories[category] = []; - categories[category].push(entry); - } - }); if (entries.length !== 0) { - const tables = <> - {Object.keys(categories).sort().map((category, index) => - - )} - ; + const table =
document.getElementById("categories").style.display = "none"; render(null, document.getElementById("search-categories")); - render(tables, document.getElementById("search-categories")); + render(table, document.getElementById("search-categories")); } else { render(

No results found.

, document.getElementById("search-categories")) } @@ -94,18 +79,8 @@ function sendSearch(query, apiCategories) { } function Search() { - const [categories, setCategories] = useState({}); - let timeout = null; - // Fetch categories from the API - useEffect(() => { - fetch(`${API_URL}/${region || 'int'}/categories.json`). - then(res => res.json()). - then(data => setCategories(data || {})). - catch(err => console.error('Error fetching categories:', err)); // Add error handling - }, []); - return
{ if (timeout) clearTimeout(timeout); - timeout = setTimeout(() => sendSearch(event.target.value, categories), 1000); + timeout = setTimeout(() => sendSearch(event.target.value), 1000); }} /> From f551b9ff16c3fb95b582c17b4e669a1606abab5e Mon Sep 17 00:00:00 2001 From: "H. Kamran" Date: Fri, 1 Nov 2024 18:53:01 -0700 Subject: [PATCH 3/5] Sort results by name --- src/components/search.jsx | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/components/search.jsx b/src/components/search.jsx index 9e0c3e7..c3f60a3 100644 --- a/src/components/search.jsx +++ b/src/components/search.jsx @@ -62,7 +62,8 @@ function sendSearch(query) { index.search(query, options) .then(({ hits }) => { const entries = hits.map((hit) => hitToAPI(hit)) - .filter(([, entry]) => !entry.regions || entry.regions.includes(region)); + .filter(([, entry]) => !entry.regions || entry.regions.includes(region)) + .sort(([a,], [b,]) => a.localeCompare(b)); if (entries.length !== 0) { const table =
From 82a2c96928c6c578098d8723a268d63014684e89 Mon Sep 17 00:00:00 2001 From: "H. Kamran" Date: Fri, 1 Nov 2024 19:00:44 -0700 Subject: [PATCH 4/5] Switch to HTM rendering --- index.html | 2 +- src/components/search.js | 137 ++++++++++++++++++++++++++++++++++++++ src/components/search.jsx | 104 ----------------------------- 3 files changed, 138 insertions(+), 105 deletions(-) create mode 100644 src/components/search.js delete mode 100644 src/components/search.jsx diff --git a/index.html b/index.html index 826f439..f2250de 100644 --- a/index.html +++ b/index.html @@ -59,6 +59,6 @@ - + diff --git a/src/components/search.js b/src/components/search.js new file mode 100644 index 0000000..f521430 --- /dev/null +++ b/src/components/search.js @@ -0,0 +1,137 @@ +import { html } from "htm/preact"; +import { render } from "preact"; +import algoliasearch from "algoliasearch"; +import Table from "./table"; + +const client = algoliasearch( + import.meta.env.VITE_ALGOLIA_APP_ID, + import.meta.env.VITE_ALGOLIA_API_KEY, +); +const index = client.initIndex(import.meta.env.VITE_ALGOLIA_INDEX_NAME); + +const region = window.location.pathname.replace(/\//g, ""); + +const searchOptions = { + hitsPerPage: 500, + attributesToRetrieve: [ + "name", + "category", + "2fa", + "regions", + "contact", + "documentation", + "recovery", + "notes", + "img", + "custom-software", + "custom-hardware", + ], +}; + +function hitToAPI(hit) { + const attributes = { domain: hit.objectID, categories: hit.category }; + + if (hit["2fa"]) attributes.methods = hit["2fa"]; + if (hit["custom-software"]) + attributes["custom-software"] = hit["custom-software"]; + if (hit["custom-hardware"]) + attributes["custom-hardware"] = hit["custom-hardware"]; + if (hit.documentation) attributes.documentation = hit.documentation; + if (hit.recovery) attributes.recovery = hit.recovery; + if (hit.notes) attributes.notes = hit.notes; + if (hit.img) attributes.img = hit.img; + if (hit.contact) attributes.contact = hit.contact; + if (hit.regions) attributes.regions = hit.regions; + + return [hit.name, attributes]; +} + +/** + * Send a search query to Algolia + * + * @param {string} query - The query + */ +function sendSearch(query) { + if ( + query === undefined || + query === "" || + (query.length < 2 && !query.match("^[x|X]$")) || + query.match("http(s)?://.*") || + query.match("^2fa(:)?$") + ) { + document.getElementById("categories").style.display = "grid"; + render(null, document.getElementById("search-categories")); + } else { + if (query.match("^[x|X]$")) query = '"X"'; + + let filter = []; + let _query = []; + query.split(" ").map((item) => { + // Add word to filter & remove word from _query + if (item.match(/\w:\w/g)) { + filter.push(item); + } else { + _query.push(item); + } + }); + + // Convert back _query to query + query = _query.join(" "); + + // Add fetched filters to search options array + const options = searchOptions; + if (filter) options["facetFilters"] = filter; + + // Execute search + index.search(query, options).then(({ hits }) => { + const entries = hits + .map((hit) => hitToAPI(hit)) + .filter(([, entry]) => !entry.regions || entry.regions.includes(region)) + .sort(([a], [b]) => a.localeCompare(b)); + + if (entries.length !== 0) { + const table = html`<${Table} + Category="search" + Title="Search Results" + search=${entries} + grid="" + />`; + + document.getElementById("categories").style.display = "none"; + render(null, document.getElementById("search-categories")); + render(table, document.getElementById("search-categories")); + } else { + render( +

No results found.

, + document.getElementById("search-categories"), + ); + } + }); + } +} + +function Search() { + let timeout = null; + + return html` +
+ { + if (timeout) clearTimeout(timeout); + timeout = setTimeout(() => sendSearch(event.target.value), 1000); + }} + /> + + +
+ `; +} + +render(html`<${Search} />`, document.getElementById("search")); diff --git a/src/components/search.jsx b/src/components/search.jsx deleted file mode 100644 index c3f60a3..0000000 --- a/src/components/search.jsx +++ /dev/null @@ -1,104 +0,0 @@ -import { render } from "preact"; -import algoliasearch from "algoliasearch"; -import Table from "./table"; - -const client = algoliasearch(import.meta.env.VITE_ALGOLIA_APP_ID, import.meta.env.VITE_ALGOLIA_API_KEY); -const index = client.initIndex(import.meta.env.VITE_ALGOLIA_INDEX_NAME); - -const region = window.location.pathname.replace(/\//g, ""); - -const searchOptions = { - hitsPerPage: 500, - attributesToRetrieve: ["name", "category", "2fa", "regions", "contact", "documentation", "recovery", "notes", "img", "custom-software", "custom-hardware"] -}; - -function hitToAPI(hit) { - const attributes = { domain: hit.objectID, categories: hit.category }; - - if (hit["2fa"]) attributes.methods = hit["2fa"]; - if (hit["custom-software"]) attributes["custom-software"] = hit["custom-software"]; - if (hit["custom-hardware"]) attributes["custom-hardware"] = hit["custom-hardware"]; - if (hit.documentation) attributes.documentation = hit.documentation; - if (hit.recovery) attributes.recovery = hit.recovery; - if (hit.notes) attributes.notes = hit.notes; - if (hit.img) attributes.img = hit.img; - if (hit.contact) attributes.contact = hit.contact; - if (hit.regions) attributes.regions = hit.regions; - - return [hit.name, attributes]; -} - -/** - * Send a search query to Algolia - * - * @param {string} query - The query - */ -function sendSearch(query) { - if (query === undefined || query === "" || (query.length < 2 && !query.match('^[x|X]$')) || query.match('http(s)?:\/\/.*') || query.match('^2fa(:)?$')) { - document.getElementById("categories").style.display = "grid"; - render(null, document.getElementById("search-categories")); - } else { - if (query.match('^[x|X]$')) query = '"X"'; - - let filter = [] - let _query = [] - query.split(' ').map(item => { - // Add word to filter & remove word from _query - if (item.match(/\w:\w/g)) { - filter.push(item); - } else { - _query.push(item) - } - }); - - // Convert back _query to query - query = _query.join(' '); - - // Add fetched filters to search options array - const options = searchOptions; - if (filter) options['facetFilters'] = filter; - - // Execute search - index.search(query, options) - .then(({ hits }) => { - const entries = hits.map((hit) => hitToAPI(hit)) - .filter(([, entry]) => !entry.regions || entry.regions.includes(region)) - .sort(([a,], [b,]) => a.localeCompare(b)); - - if (entries.length !== 0) { - const table =
- - document.getElementById("categories").style.display = "none"; - render(null, document.getElementById("search-categories")); - render(table, document.getElementById("search-categories")); - } else { - render(

No results found.

, document.getElementById("search-categories")) - } - }) - - } -} - -function Search() { - let timeout = null; - - return
- { - if (timeout) clearTimeout(timeout); - timeout = setTimeout(() => sendSearch(event.target.value), 1000); - }} - /> - - -
-} - -render(, document.getElementById('search')); From e4fc1ba84c1d33a088793c0bef109c4e10dbe5f7 Mon Sep 17 00:00:00 2001 From: "H. Kamran" Date: Fri, 1 Nov 2024 19:01:53 -0700 Subject: [PATCH 5/5] Remove `grid` parameter --- src/components/search.js | 1 - 1 file changed, 1 deletion(-) diff --git a/src/components/search.js b/src/components/search.js index f521430..069e5d5 100644 --- a/src/components/search.js +++ b/src/components/search.js @@ -94,7 +94,6 @@ function sendSearch(query) { Category="search" Title="Search Results" search=${entries} - grid="" />`; document.getElementById("categories").style.display = "none";