diff --git a/manifest.json b/manifest.json index 5941632..e15aa3b 100644 --- a/manifest.json +++ b/manifest.json @@ -3,7 +3,7 @@ "name": "Skater", "description": "Spotlight-like extension for Chrome bookmarks", - "version": "1.1.4", + "version": "1.1.5", "icons": { "16": "images/icon16.png", "24": "images/icon24.png", diff --git a/static/bundle.js b/static/bundle.js index 1c2d732..e6c5e1c 100644 --- a/static/bundle.js +++ b/static/bundle.js @@ -10,9 +10,18 @@ const { const { createOverlay, createListItem, - resetListElementCSS + resetListElementCSS, + animateFocusedSearchResult } = require('./src/htmlUtils.js'); +const { + getFocusedListElement, + getSearchInputElement, + getSearchResultsElement, + getOverlayDiv, + getSearchResultsElementChildren +} = require('./src/selectors'); + chrome.runtime.onMessage.addListener( async function(a, b, sendResponse) { // check if our overlay exists already @@ -31,6 +40,7 @@ chrome.runtime.onMessage.addListener( async function setUpOverlayEventListener() { const overlayDiv = getOverlayDiv(); + document.addEventListener('keydown', handleEscapeKey); overlayDiv.addEventListener('keydown', async function(documentEvent) { if (getSearchInputElement()) { focusInput(); @@ -62,14 +72,20 @@ async function setUpOverlayEventListener() { destroyOverlay(); await goTo(selectedResult.href); } - case "Escape": - destroyOverlay(); } } return true }); } +function handleEscapeKey(globalEvent) { + switch(globalEvent.key) { + case "Escape": + destroyOverlay(); + document.removeEventListener('keydown', handleEscapeKey); + } +} + function setUpInputEventListener() { const searchInput = getSearchInputElement(); searchInput.addEventListener('keyup', async function(inputEvent) { @@ -121,14 +137,6 @@ function moveDownOneResult() { focusInput(); } -function animateFocusedSearchResult(index) { - const focusedElement = document.querySelector(`.skater-result-${index}`); - if (focusedElement) { - focusedElement.parentElement.style['background-position-y'] = '100%'; - focusedElement.style.color = 'black'; - focusedElement.style.outline = "none"; - } -} function updateSearchResultsCSS(index) { const searchElements = getSearchResultsElementChildren(); @@ -146,30 +154,6 @@ function resetListElementClass(skaterLinkElement) { skaterLinkElement.setAttribute('class', elementRootClass); } -function getFocusedListElement() { - return document.querySelector('.skater-focused'); -} - -function getSearchInputElement() { - return document.getElementById("searchInput"); -} - -function getSearchResultsElement() { - return document.getElementById("searchResults"); -} - -function getSearchWrapperElement() { - return document.getElementById("searchWrapperDiv"); -} - -function getOverlayDiv() { - return document.getElementById("skater-overlay"); -} - -function getSearchResultsElementChildren() { - return document.querySelectorAll(".skater-link"); -} - function destroyOverlay() { const overlayDiv = getOverlayDiv(); if (overlayDiv) { @@ -219,7 +203,7 @@ function sendBackgroundMessage(query_object) { } } -},{"./src/htmlUtils.js":2,"./src/utils.js":3}],2:[function(require,module,exports){ +},{"./src/htmlUtils.js":2,"./src/selectors":3,"./src/utils.js":4}],2:[function(require,module,exports){ function createOverlay() { const searchIcon = createSearchIcon(); const overlayDiv = createOverlayDiv(); @@ -252,8 +236,7 @@ function createSearchIcon() { function createSearchInput() { const searchInput = document.createElement('input'); - searchInput.id = "searchInput"; - searchInput.setAttribute('class', ""); + searchInput.id = "search-input"; searchInput.style = "height: 60px; width: 80%; padding: 2px 23px 2px 35px; background-color: #f5f5f5; font-size:19px; font-family: Helvetica Neue,Helvetica,Arial,sans-serif;" searchInput.style.border = "0px"; searchInput.style.outline = "none"; @@ -265,7 +248,7 @@ function createSearchInput() { function createSearchWrapperDiv() { const searchWrapperDiv = document.createElement('div'); - searchWrapperDiv.id = "searchWrapperDiv"; + searchWrapperDiv.id = "search-wrapper-div"; searchWrapperDiv.style = "width:400px; margin:auto; position: absolute; top: 50%; left: 40%; margin-right: -50%; transform: translate(-50%, -%50);" searchWrapperDiv.style.padding = "5px"; searchWrapperDiv.style.border = "1px solid grey"; @@ -277,7 +260,7 @@ function createSearchWrapperDiv() { function createSearchResultsList() { const resultsDiv = document.createElement('div'); - resultsDiv.id = "searchResults"; + resultsDiv.id = "search-results"; resultsDiv.style = "padding 20px; background-color: #f5f5f5; font-size:17px; font-family: Helvetica Neue,Helvetica,Arial,sans-serif;" resultsDiv.style.visibility = "hidden"; resultsDiv.style.padding = "0px"; @@ -302,8 +285,7 @@ function createListItem(result, index) { listURL.style.color = "black"; listURL.style['text-decoration'] = 'none'; - listElement.setAttribute('class', `searchResultItem`); - resetListElementCSS(listElement, index); + resetListElementCSS(listElement); if (result.title.length > 27) { listURL.innerHTML = result.title.substring(0, 27) + '...'; @@ -317,8 +299,8 @@ function createListItem(result, index) { return listElement } -function resetListElementCSS(listElement, index) { - listElement.setAttribute('class', 'unselected'); +function resetListElementCSS(listElement) { + listElement.setAttribute('class', 'searchResultItem'); listElement.style['background-position-y'] = "-0%"; listElement.style['background-image'] = 'linear-gradient(#f5f5f5 50%, #c6f6d5 50%)'; listElement.style['transition'] = 'background 200ms ease'; @@ -327,16 +309,60 @@ function resetListElementCSS(listElement, index) { listElement.style.padding = '4px 0px 4px 0px'; } +function animateFocusedSearchResult(index) { + const focusedElement = document.querySelector(`.skater-result-${index}`); + if (focusedElement) { + focusedElement.parentElement.style['background-position-y'] = '100%'; + focusedElement.style.color = 'black'; + focusedElement.style.outline = 'none'; + } +} + module.exports = { createOverlay, createOverlayDiv, createSearchIcon, createSearchInput, createListItem, - resetListElementCSS + resetListElementCSS, + animateFocusedSearchResult } },{}],3:[function(require,module,exports){ +function getFocusedListElement() { + return document.querySelector('.skater-focused'); +} + +function getSearchInputElement() { + return document.getElementById("search-input"); +} + +function getSearchResultsElement() { + return document.getElementById("search-results"); +} + +function getSearchWrapperElement() { + return document.getElementById("search-wrapper-div"); +} + +function getOverlayDiv() { + return document.getElementById("skater-overlay"); +} + +function getSearchResultsElementChildren() { + return document.querySelectorAll(".skater-link"); +} + +module.exports = { + getFocusedListElement, + getSearchInputElement, + getSearchResultsElement, + getSearchWrapperElement, + getOverlayDiv, + getSearchResultsElementChildren +} + +},{}],4:[function(require,module,exports){ function isValidInputEvent(key) { const isAlphabetical = (key >= "a" && key <= "z") || (key >= "A" && key <= "Z") && key.length === 1; const isNumeric = (key >= "0" && key <= "9"); @@ -385,4 +411,5 @@ module.exports = { refineResults } -},{}]},{},[1]); +},{}]},{},[1]) +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../home/ec/.nvm/versions/node/v15.1.0/lib/node_modules/watchify/node_modules/browser-pack/_prelude.js","content_script.js","src/htmlUtils.js","src/selectors.js","src/utils.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3HA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c=\"function\"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error(\"Cannot find module '\"+i+\"'\");throw a.code=\"MODULE_NOT_FOUND\",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u=\"function\"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()","const {\n    isValidInputEvent,\n    stripIndexFromClass,\n    stripFocusFromClass,\n    giveElementFocusedClass,\n    refineResults\n} = require('./src/utils.js');\n\nconst {\n    createOverlay,\n    createListItem,\n    resetListElementCSS,\n    animateFocusedSearchResult\n} = require('./src/htmlUtils.js');\n\nconst {\n    getFocusedListElement,\n    getSearchInputElement,\n    getSearchResultsElement,\n    getOverlayDiv,\n    getSearchResultsElementChildren\n} = require('./src/selectors');\n\nchrome.runtime.onMessage.addListener(\n    async function(a, b, sendResponse) {\n        // check if our overlay exists already\n        if (!getOverlayDiv()) {\n            createOverlay();\n            setUpInputEventListener();\n            setTimeout(() => focusInput(), 100);\n            await setUpOverlayEventListener();\n        } else {\n            focusInput();\n        }\n        sendResponse([]);\n        return true\n    }\n);\n\nasync function setUpOverlayEventListener() {\n    const overlayDiv = getOverlayDiv();\n    document.addEventListener('keydown', handleEscapeKey);\n    overlayDiv.addEventListener('keydown', async function(documentEvent) {\n        if (getSearchInputElement()) {\n            focusInput();\n            const focusedElement = getFocusedListElement();\n            switch(documentEvent.key) {\n                case \"Up\":\n                case \"ArrowUp\":\n                    if (focusedElement.getAttribute('class') === 'skater-link skater-result-0 skater-focused') {\n                        focusInput();\n                    } else if (focusedElement.isSameNode(getSearchInputElement())) {\n                        // do nothing\n                    } else {\n                        // move to previous result\n                        moveUpOneResult();\n                        documentEvent.preventDefault();\n                    }\n                    return true\n                case \"Down\":\n                case \"ArrowDown\":\n                    // move to next search result\n                    moveDownOneResult();\n                    return true\n                case \"Enter\":\n                    // go to first inputEvent in the list\n                    const selectedResult = getFocusedListElement();\n                    if (documentEvent.ctrlKey){\n                        // open in same window\n                    } else {\n                        destroyOverlay();                   \n                        await goTo(selectedResult.href);\n                    }\n            }\n        }\n        return true\n    });\n}\n\nfunction handleEscapeKey(globalEvent) {\n    switch(globalEvent.key) {\n        case \"Escape\":\n            destroyOverlay();\n            document.removeEventListener('keydown', handleEscapeKey);\n    }\n}\n\nfunction setUpInputEventListener() {\n    const searchInput = getSearchInputElement();\n    searchInput.addEventListener('keyup', async function(inputEvent) {\n        const query = searchInput.value;\n        const bookmarkSearchResults = await sendBackgroundMessage({userSearch: query});\n        // refine results\n        if (Array.isArray(bookmarkSearchResults) && isValidInputEvent(inputEvent.key)) {\n            const refinedResults = refineResults(bookmarkSearchResults, query);\n            refreshActiveSearchResults(refinedResults);\n            if (refinedResults.length) {\n                if (!document.querySelector('.skater-focused')) {\n                    setTimeout(() => animateFocusedSearchResult(0), 100);\n                    giveElementFocusedClass(0);\n                }\n            }\n        }\n        return true\n    });\n}\n\nasync function goTo(url) {\n    // checks if a tab with this url already exists\n    // if so, go to it, else open a new window\n    await sendBackgroundMessage({url: url});\n}\n\n\nfunction moveUpOneResult() {\n    if (getFocusedListElement().getAttribute('class') === 'skater-link skater-result-0 skater-focused') {\n        focusInput();\n    } else if (getFocusedListElement().isSameNode(getSearchInputElement())) {\n        // do nothing\n    } else {\n        const indexOfLastFocus = stripIndexFromClass(getFocusedListElement());\n        const index = indexOfLastFocus - 1;\n        updateSearchResultsCSS(index);\n        focusInput();\n    }\n}\n\nfunction moveDownOneResult() {\n    let index;\n    const indexOfLastFocus = stripIndexFromClass(getFocusedListElement());\n    index = indexOfLastFocus + 1;\n    // handle if you are already focused on the last list item\n    if (document.querySelector(`.skater-result-${index}`)) {\n        updateSearchResultsCSS(index);\n    }\n    focusInput();\n}\n\n\nfunction updateSearchResultsCSS(index) {\n    const searchElements = getSearchResultsElementChildren();\n    searchElements.forEach(e => {\n        resetListElementCSS(e.parentElement, index);\n        resetListElementClass(e);\n    });\n    // color focused element\n    giveElementFocusedClass(index);\n    animateFocusedSearchResult(index);\n}\n\nfunction resetListElementClass(skaterLinkElement) {\n    const elementRootClass = stripFocusFromClass(skaterLinkElement.getAttribute('class')); \n    skaterLinkElement.setAttribute('class', elementRootClass);\n}\n\nfunction destroyOverlay() {\n    const overlayDiv = getOverlayDiv();\n    if (overlayDiv) {\n        overlayDiv.parentNode.removeChild(overlayDiv);\n    }\n}\n\nfunction ensureResultsListIsVisible() {\n    const searchResultsElement = getSearchResultsElement();\n    searchResultsElement.style.visibility = \"visible\";\n    searchResultsElement.style.padding = \"10px 10px 10px 10px\";\n}\n\nfunction ensureResultsListIsHidden() {\n    const searchResultsElement = getSearchResultsElement();\n    searchResultsElement.style.visibility = \"hidden\";\n    searchResultsElement.style.padding = \"0px\";\n}\n\nfunction refreshActiveSearchResults(results) {\n    const resultsDiv = getSearchResultsElement();\n    // wipes the unordered list\n    resultsDiv.innerHTML = '';\n    if (results.length) {\n        ensureResultsListIsVisible();\n        let index = 0;\n        results.forEach(result => {\n            resultsDiv.appendChild(createListItem(result, index));\n            index += 1;\n        });\n    } else {\n        ensureResultsListIsHidden();\n    };\n}\n\nfunction focusInput() {\n    getSearchInputElement().focus();\n}\n\nfunction sendBackgroundMessage(query_object) {\n    if ((query_object.url && query_object.url.length > 0) || (query_object.userSearch && query_object.userSearch.length)) {\n        return new Promise((resolve, _reject) => {\n            chrome.runtime.sendMessage(query_object, resolve);\n        });\n    } else {\n        return Promise.resolve([]);\n    }\n}\n","function createOverlay() {\n    const searchIcon = createSearchIcon();\n    const overlayDiv = createOverlayDiv();\n    const searchInput = createSearchInput();\n    const resultsDiv = createSearchResultsList();\n    const searchWrapperDiv= createSearchWrapperDiv();\n\n    searchWrapperDiv.appendChild(searchIcon);\n    searchWrapperDiv.appendChild(searchInput);\n    searchWrapperDiv.appendChild(resultsDiv);\n    overlayDiv.appendChild(searchWrapperDiv);\n    // append all to document\n    document.body.appendChild(overlayDiv);\n}\n\nfunction createOverlayDiv () {\n    const overlayDiv = document.createElement('div');\n    overlayDiv.style = \"position: fixed; left: 0; top: 0; width: 100%; height: 100%; background: rgba(204, 204, 204, 0.3); z-index:2147483647;\";\n    overlayDiv.id = \"skater-overlay\"\n    return overlayDiv\n}\n\nfunction createSearchIcon() {\n    const searchIcon = document.createElement('img');\n    searchIcon.setAttribute('class', \"search-icon\");\n    searchIcon.src = \"data:image/svg+xml;utf8;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSI/Pgo8IS0tIEdlbmVyYXRvcjogQWRvYmUgSWxsdXN0cmF0b3IgMTkuMC4wLCBTVkcgRXhwb3J0IFBsdWctSW4gLiBTVkcgVmVyc2lvbjogNi4wMCBCdWlsZCAwKSAgLS0+CjxzdmcgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB4bWxuczp4bGluaz0iaHR0cDovL3d3dy53My5vcmcvMTk5OS94bGluayIgdmVyc2lvbj0iMS4xIiBpZD0iQ2FwYV8xIiB4PSIwcHgiIHk9IjBweCIgdmlld0JveD0iMCAwIDU2Ljk2NiA1Ni45NjYiIHN0eWxlPSJlbmFibGUtYmFja2dyb3VuZDpuZXcgMCAwIDU2Ljk2NiA1Ni45NjY7IiB4bWw6c3BhY2U9InByZXNlcnZlIiB3aWR0aD0iMTZweCIgaGVpZ2h0PSIxNnB4Ij4KPHBhdGggZD0iTTU1LjE0Niw1MS44ODdMNDEuNTg4LDM3Ljc4NmMzLjQ4Ni00LjE0NCw1LjM5Ni05LjM1OCw1LjM5Ni0xNC43ODZjMC0xMi42ODItMTAuMzE4LTIzLTIzLTIzcy0yMywxMC4zMTgtMjMsMjMgIHMxMC4zMTgsMjMsMjMsMjNjNC43NjEsMCw5LjI5OC0xLjQzNiwxMy4xNzctNC4xNjJsMTMuNjYxLDE0LjIwOGMwLjU3MSwwLjU5MywxLjMzOSwwLjkyLDIuMTYyLDAuOTIgIGMwLjc3OSwwLDEuNTE4LTAuMjk3LDIuMDc5LTAuODM3QzU2LjI1NSw1NC45ODIsNTYuMjkzLDUzLjA4LDU1LjE0Niw1MS44ODd6IE0yMy45ODQsNmM5LjM3NCwwLDE3LDcuNjI2LDE3LDE3cy03LjYyNiwxNy0xNywxNyAgcy0xNy03LjYyNi0xNy0xN1MxNC42MSw2LDIzLjk4NCw2eiIgZmlsbD0iIzAwMDAwMCIvPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8L3N2Zz4K\"\n    searchIcon.style = \"position: absolute; margin-top: 23px; left: 10px; width: 20px; opacity: 50%\"\n    return searchIcon\n}\n\nfunction createSearchInput() {\n    const searchInput = document.createElement('input');\n    searchInput.id = \"search-input\";\n    searchInput.style = \"height: 60px; width: 80%; padding: 2px 23px 2px 35px; background-color: #f5f5f5; font-size:19px; font-family: Helvetica Neue,Helvetica,Arial,sans-serif;\"\n    searchInput.style.border = \"0px\";\n    searchInput.style.outline = \"none\";\n    searchInput.style['border-width'] = \"0px\";\n    searchInput.autocomplete = \"off\";\n    searchInput.placeholder=\"Skate to...\";\n    return searchInput\n}\n\nfunction createSearchWrapperDiv() {\n    const searchWrapperDiv = document.createElement('div');\n    searchWrapperDiv.id = \"search-wrapper-div\";\n    searchWrapperDiv.style = \"width:400px; margin:auto; position: absolute; top: 50%; left: 40%; margin-right: -50%; transform: translate(-50%, -%50);\"\n    searchWrapperDiv.style.padding = \"5px\";\n    searchWrapperDiv.style.border = \"1px solid grey\";\n    searchWrapperDiv.style['border-radius'] = \"8px\";\n    searchWrapperDiv.style['background-color'] = \"#f5f5f5\";\n    searchWrapperDiv.style['box-shadow'] = \"-13px 13px 16px 1px rgba(0, 0, 0, 0.2)\"\n    return searchWrapperDiv\n}\n\nfunction createSearchResultsList() {\n    const resultsDiv = document.createElement('div');\n    resultsDiv.id = \"search-results\";\n    resultsDiv.style = \"padding 20px; background-color: #f5f5f5; font-size:17px; font-family: Helvetica Neue,Helvetica,Arial,sans-serif;\"\n    resultsDiv.style.visibility = \"hidden\";\n    resultsDiv.style.padding = \"0px\";\n    return resultsDiv\n}\n\nfunction createListItem(result, index) {\n    const listElement = document.createElement('div');\n    const listURL = document.createElement('a');\n    const listIMG = document.createElement('img');\n\n    const matches = result.url.match(/^https?\\:\\/\\/(?:www\\.)?([^\\/?#]+)(?:[\\/?#]|$)/i);\n    const domain = matches && matches[1] // domain is null if no matches found\n\n    listIMG.src = `https://www.google.com/s2/favicons?domain_url=${domain}`;\n    listIMG.style.padding = '0% 5% 0% 5%';\n    listIMG.style['margin-bottom'] = '-1%';\n    listIMG.style.display = 'inline';\n    listIMG.setAttribute('class', 'domain-icon');\n\n    listURL.setAttribute('class', `skater-link skater-result-${index}`);\n    listURL.style.color = \"black\";\n    listURL.style['text-decoration'] = 'none';\n\n    resetListElementCSS(listElement);\n\n    if (result.title.length > 27) {\n        listURL.innerHTML = result.title.substring(0, 27) + '...';\n    }\n    else {\n        listURL.innerHTML = result.title;\n    }\n    listURL.href = result.url;\n    listElement.appendChild(listIMG);\n    listElement.appendChild(listURL);\n    return listElement\n}\n\nfunction resetListElementCSS(listElement) {\n    listElement.setAttribute('class', 'searchResultItem');\n    listElement.style['background-position-y'] = \"-0%\";\n    listElement.style['background-image'] = 'linear-gradient(#f5f5f5 50%, #c6f6d5 50%)';\n    listElement.style['transition'] = 'background 200ms ease';\n    listElement.style['background-size'] = 'auto 200%';\n    listElement.style['border-radius'] = '10px';\n    listElement.style.padding = '4px 0px 4px 0px';\n}\n\nfunction animateFocusedSearchResult(index) {\n    const focusedElement = document.querySelector(`.skater-result-${index}`);\n    if (focusedElement) {\n        focusedElement.parentElement.style['background-position-y'] = '100%';\n        focusedElement.style.color = 'black';\n        focusedElement.style.outline = 'none';\n    }\n}\n\nmodule.exports = {\n    createOverlay,\n    createOverlayDiv,\n    createSearchIcon,\n    createSearchInput,\n    createListItem,\n    resetListElementCSS,\n    animateFocusedSearchResult\n}\n","function getFocusedListElement() {\n    return document.querySelector('.skater-focused');\n}\n\nfunction getSearchInputElement() {\n    return document.getElementById(\"search-input\");\n}\n\nfunction getSearchResultsElement() {\n    return document.getElementById(\"search-results\");\n}\n\nfunction getSearchWrapperElement() {\n    return document.getElementById(\"search-wrapper-div\");\n}\n\nfunction getOverlayDiv() {\n    return document.getElementById(\"skater-overlay\");\n}\n\nfunction getSearchResultsElementChildren() {\n    return document.querySelectorAll(\".skater-link\");\n}\n\nmodule.exports = {\n    getFocusedListElement,\n    getSearchInputElement,\n    getSearchResultsElement,\n    getSearchWrapperElement,\n    getOverlayDiv,\n    getSearchResultsElementChildren\n}\n","function isValidInputEvent(key) {\n    const isAlphabetical = (key >= \"a\" && key <= \"z\") || (key >= \"A\" && key <= \"Z\") && key.length === 1;\n    const isNumeric = (key >= \"0\" && key <= \"9\");\n    const isBackspace = (key === \"Backspace\");\n    const isEnter = (key === \"Enter\");\n    const isShift = (key === \"Shift\");\n    const isArrowKey = (['ArrowUp', 'Up', 'ArrowDown', 'Down', 'ArrowLeft', 'Left', 'ArrowRight', 'Right'].includes(key))\n    return (isAlphabetical || isNumeric || isBackspace || isEnter || isShift) && !isArrowKey\n}\n\nfunction stripIndexFromClass(element) {\n    const classString = element.getAttribute('class').split(' ')[1].split('-'); \n    return parseInt(classString[classString.length - 1]);\n}\n\nfunction stripFocusFromClass(classString) {\n    const rootClassList = classString.split(' ');\n    if (classString.length > 2) {\n        return rootClassList.slice(0, 2).join(' ');\n    } else {\n        return classString\n    }\n}\n\n\nfunction giveElementFocusedClass(index) {\n    const focusedElement = document.querySelector(`.skater-result-${index}`);\n    focusedElement.setAttribute('class', `skater-link skater-result-${index} skater-focused`);\n}\n\n\nfunction refineResults(searchResults, query) {\n    return searchResults.filter(result => {\n      const queryLower = query.toLowerCase();\n      const bookmarkTitle = result.title.toLowerCase();\n      return (bookmarkTitle.includes(queryLower) || result.url.includes(queryLower)) && result.hasOwnProperty('url');\n    });\n}\n\n\nmodule.exports = {\n    isValidInputEvent,\n    stripIndexFromClass,\n    stripFocusFromClass,\n    giveElementFocusedClass,\n    refineResults\n}\n"]}