diff --git a/_redirects b/_redirects
index bceeb5f8..10d8dc22 100644
--- a/_redirects
+++ b/_redirects
@@ -60,4 +60,5 @@
/posts/2023/locally-stored-music-and-storage-as-a-meaningful-constraint/ /posts/2023/doppler-locally-stored-music-and-storage-as-a-beneficial-constraint/ 301
/blog/digital-privacy-tools /posts/2021/digital-privacy-tools/ 301
/posts/2023/now-page-update-matter-favorites/ /posts/2023/now-page-update-favorite-articles-from-matter/ 301
-/posts/2023/now-playing-eleventy-netlify-edge-functions-emoji/ /posts/2023/displaying-now-playing-data-with-matching-emoji-using-netlify-edge-functions-and-eleventy/ 301
\ No newline at end of file
+/posts/2023/now-playing-eleventy-netlify-edge-functions-emoji/ /posts/2023/displaying-now-playing-data-with-matching-emoji-using-netlify-edge-functions-and-eleventy/ 301
+/posts/2014/sublime-text-ctrl-tab-key-bindings/ /posts/2014/sublime-text-3-ctrl-tab-key-bindings/ 301
\ No newline at end of file
diff --git a/src/assets/scripts/index.js b/src/assets/scripts/index.js
index cb3570df..bff8a111 100644
--- a/src/assets/scripts/index.js
+++ b/src/assets/scripts/index.js
@@ -87,36 +87,40 @@ window.addEventListener('load', () => {
// search logic
;(() => {
- if (!MiniSearch) return
-
- const miniSearch = new MiniSearch({ fields: ['title', 'description', 'tags', 'type'] })
+ if (!MiniSearch || !document.querySelector('.search__form')) return
+
+ const miniSearch = new MiniSearch({
+ fields: ['title', 'description', 'tags', 'type'],
+ idField: 'id',
+ storeFields: ['id', 'title', 'url', 'description', 'type', 'tags', 'total_plays'],
+ searchOptions: {
+ boost: { title: 2, tags: 1.5 },
+ prefix: true,
+ fuzzy: 0.3,
+ },
+ })
+
const $form = document.querySelector('.search__form')
const $input = document.querySelector('.search__form--input')
- const $fallback = document.querySelector('.search__form--fallback')
const $typeCheckboxes = document.querySelectorAll('.search__form--type input[type="checkbox"]')
const $results = document.querySelector('.search__results')
const $loadMoreButton = document.querySelector('.search__load-more')
-
- $form.removeAttribute('action')
- $form.removeAttribute('method')
- $fallback.remove()
-
+
const PAGE_SIZE = 10
let currentPage = 1
let currentResults = []
+ let totalResults = 0
let resultsById = {}
let debounceTimeout
-
- const parseMarkdown = markdown => {
- if (!markdown) return ''
- return markdown
+
+ const parseMarkdown = (markdown = '') =>
+ markdown
.replace(/\*\*(.*?)\*\*/g, '$1')
.replace(/\*(.*?)\*/g, '$1')
.replace(/\[(.*?)\]\((.*?)\)/g, '$1')
.replace(/\n/g, '
')
.replace(/[#*_~`]/g, '')
- }
-
+
const truncateDescription = (markdown, maxLength = 150) => {
const htmlDescription = parseMarkdown(markdown)
const tempDiv = document.createElement('div')
@@ -124,41 +128,22 @@ window.addEventListener('load', () => {
const plainText = tempDiv.textContent || tempDiv.innerText || ''
return plainText.length > maxLength ? `${plainText.substring(0, maxLength)}...` : plainText
}
-
+
const formatArtistTitle = (title, totalPlays) =>
totalPlays > 0 ? `${title} ${totalPlays} plays` : title
-
- const renderSearchResults = results => {
- if (results.length > 0) {
- $results.innerHTML = results.map(({ title, url, description, type, total_plays }) => {
- const truncatedDesc = truncateDescription(description)
- const formattedTitle = type === 'artist' && total_plays !== undefined
- ? formatArtistTitle(title, total_plays)
- : title
-
- return `
-
-
- ${formattedTitle}
-
- ${truncatedDesc}
-
- `
- }).join('')
- $results.style.display = 'block'
- } else {
- $results.innerHTML = 'No results found.'
- $results.style.display = 'block'
- }
+
+ const updateLoadMoreButton = () => {
+ const moreResultsToShow = currentPage * PAGE_SIZE < totalResults
+ $loadMoreButton.style.display = moreResultsToShow ? 'block' : 'none'
}
-
- const appendSearchResults = results => {
- const newResults = results.map(({ title, url, description, type, total_plays }) => {
+
+ const renderSearchResults = results => {
+ const resultHTML = results.map(({ title, url, description, type, total_plays }) => {
const truncatedDesc = truncateDescription(description)
const formattedTitle = type === 'artist' && total_plays !== undefined
? formatArtistTitle(title, total_plays)
: title
-
+
return `
@@ -168,85 +153,108 @@ window.addEventListener('load', () => {
`
}).join('')
- $results.insertAdjacentHTML('beforeend', newResults)
+
+ $results.innerHTML = resultHTML || 'No results found.'
+ $results.style.display = 'block'
}
-
- const loadSearchIndex = async (query = '', types = []) => {
+
+ const appendSearchResults = results => {
+ const resultHTML = results.map(({ title, url, description, type, total_plays }) => {
+ const truncatedDesc = truncateDescription(description)
+ const formattedTitle = type === 'artist' && total_plays !== undefined
+ ? formatArtistTitle(title, total_plays)
+ : title
+
+ return `
+
+ ${formattedTitle}
+ ${truncatedDesc}
+
+ `
+ }).join('')
+
+ $results.insertAdjacentHTML('beforeend', resultHTML)
+ }
+
+ const loadSearchIndex = async (query = '', types = [], page = 1) => {
const typeQuery = types.join(',')
-
try {
- const response = await fetch(`https://coryd.dev/api/search?q=${query}&type=${typeQuery}`)
+ const response = await fetch(`http://localhost:8787/api/search?q=${query}&type=${typeQuery}&page=${page}&pageSize=${PAGE_SIZE}`)
const index = await response.json()
- const results = index.results
-
- if (!Array.isArray(results)) return {}
-
+
+ const results = index.results || []
+ totalResults = index.total || 0
+
+ if (!Array.isArray(results)) throw new Error('Expected results to be an array')
+
resultsById = results.reduce((acc, item) => {
acc[item.id] = item
return acc
}, {})
-
- miniSearch.removeAll()
- miniSearch.addAll(index)
- return resultsById
+
+ if (page === 1) {
+ miniSearch.removeAll()
+ miniSearch.addAll(results)
+ }
+
+ return results
} catch (error) {
console.error('Error fetching search data:', error)
- return {}
+ return []
}
}
-
+
+ const getSearchResults = query =>
+ miniSearch.search(query).map(({ id }) => resultsById[id])
+ .filter(result => getSelectedTypes().includes(result.type))
+
const getSelectedTypes = () =>
Array.from($typeCheckboxes)
.filter(checkbox => checkbox.checked)
.map(checkbox => checkbox.value)
-
- $input.addEventListener('input', () => {
+
+ const handleInputEvent = async () => {
const query = $input.value.trim()
clearTimeout(debounceTimeout)
-
+
if (!query) {
renderSearchResults([])
$loadMoreButton.style.display = 'none'
return
}
-
+
debounceTimeout = setTimeout(async () => {
- resultsById = await loadSearchIndex(query, getSelectedTypes())
- const results = getSearchResults(query)
+ const results = await loadSearchIndex(query, getSelectedTypes(), 1)
currentResults = results
currentPage = 1
-
+
renderSearchResults(getResultsForPage(currentPage))
- $loadMoreButton.style.display = results.length > PAGE_SIZE ? 'block' : 'none'
+ updateLoadMoreButton()
}, 300)
- })
-
- $typeCheckboxes.forEach(checkbox => {
- checkbox.addEventListener('change', async () => {
- resultsById = await loadSearchIndex($input.value.trim(), getSelectedTypes())
- const results = getSearchResults($input.value.trim())
- currentResults = results
- currentPage = 1
-
- renderSearchResults(getResultsForPage(currentPage))
- $loadMoreButton.style.display = results.length > PAGE_SIZE ? 'block' : 'none'
- })
- })
-
- $loadMoreButton.addEventListener('click', () => {
+ }
+
+ const handleCheckboxChange = async () => {
+ const results = await loadSearchIndex($input.value.trim(), getSelectedTypes(), 1)
+ currentResults = results
+ currentPage = 1
+
+ renderSearchResults(getResultsForPage(currentPage))
+ updateLoadMoreButton()
+ }
+
+ const handleLoadMoreClick = async () => {
currentPage++
- const nextResults = getResultsForPage(currentPage)
+ const nextResults = await loadSearchIndex($input.value.trim(), getSelectedTypes(), currentPage)
appendSearchResults(nextResults)
-
- if (currentPage * PAGE_SIZE >= currentResults.length) $loadMoreButton.style.display = 'none'
- })
-
- const getSearchResults = query =>
- miniSearch.search(query, { prefix: true, fuzzy: 0.2, boost: { title: 2 } })
- .map(({ id }) => resultsById[id])
- .filter(result => getSelectedTypes().includes(result.type))
-
+
+ updateLoadMoreButton()
+ }
+
const getResultsForPage = page =>
currentResults.slice((page - 1) * PAGE_SIZE, page * PAGE_SIZE)
+
+ $input.addEventListener('input', handleInputEvent)
+ $typeCheckboxes.forEach(checkbox => checkbox.addEventListener('change', handleCheckboxChange))
+ $loadMoreButton.addEventListener('click', handleLoadMoreClick)
})()
})
\ No newline at end of file