Skip to content

Commit

Permalink
Bring in the aloglia netlify package
Browse files Browse the repository at this point in the history
  • Loading branch information
joebailey26 committed Jun 14, 2023
1 parent 2229c1b commit 019f19b
Show file tree
Hide file tree
Showing 16 changed files with 1,247 additions and 36 deletions.
2 changes: 1 addition & 1 deletion .eslintrc
Expand Up @@ -11,7 +11,7 @@
],
"parserOptions": {
"requireConfigFile": false,
"parser": "@babel/eslint-parser",
"parser": "@typescript-eslint/parser",
"ecmaVersion": 8,
"sourceType": "module"
},
Expand Down
171 changes: 171 additions & 0 deletions assets/scss/_algoliasearchNetlify.scss
@@ -0,0 +1,171 @@
@import '~@algolia/autocomplete-theme-classic';

$color-bg: white;
$color-muted: #969FAF;
$color-light: #797979;
$color-text: #23263B;
$color-mark: rgb(84 104 255);
$color-bg-selected: #F5F5FA;
$color-input-icon: #777777;
$color-source-icon: rgb(80 80 80 / 32%);
$font-size-xs: 12px;
$font-size-s: 14px;
$font-size-m: 16px;
$size-xs: 2px;
$size-s: 4px;
$size-m: 8px;
$size-l: 16px;
$size-xl: 32px;
$height: $size-xl;
$height-icon: $size-l;
$font-size: $font-size-m;
.aa-Autocomplete, .aa-Panel, .aa-DetachedContainer {
--color-mark: #{$color-mark};
--color-background: #{$color-bg};
--color-selected: #{$color-bg-selected};
--color-text: #{$color-text};
--color-input-icon: #{$color-input-icon};
--color-source-icon: #{$color-source-icon};
--height: #{$height};
--height-icon: #{$height-icon};
--font-size: #{$font-size}
}

//// ---- Overridden definitions of classic theme
.aa-Panel {
z-index: 1100;
min-width: 350px;
margin-top: $size-xs;
.aa-PanelLayout {
padding-top: 0;
padding-bottom: 0;
background-color: var(--color-background);
.aa-PanelLayoutPreview {
border-left: solid 1px var(--color-selected)
}
}
.aa-GradientBottom {
background-image: none
}
}
.aa-Autocomplete, .aa-DetachedFormContainer {
.aa-Form {
height: var(--height);
padding: 0;
font-size: var(--font-size);
background-color: var(--color-background);
&:focus-within {
box-shadow: none
}
.aa-InputWrapperPrefix {
padding: 0;
.aa-Label {
padding: 0;
svg {
left: 0;
width: var(--height-icon);
color: var(--color-input-icon);
vertical-align: middle
}
}
}
.aa-InputWrapper {
.aa-Input {
height: var(--height);
color: var(--color-text)
}
}
.aa-InputWrapperSuffix {
height: var(--height);
.aa-ClearButton {
padding: 0;
&:hover,
&:focus {
color: var(--color-text)
}
}
}
}
}
.aa-Item {
padding: $size-xs 0;
color: var(--color-text);
a {
color: inherit;
text-decoration: none
}
&[aria-selected='true'] {
background-color: var(--color-selected)
}
.aa-ItemContent {
display: flex;
color: var(--color-text);
mark {
color: var(--color-mark);
background-color: transparent
}
}
.aa-ItemIcon {
align-items: baseline;
margin: 0 var(--aa-spacing-half) 0 2px;
color: var(--color-source-icon);
background: none;
box-shadow: none
}
.aa-ItemTitle {
font-weight: bold;
font-size: $font-size-s;
line-height: 18px
}
.aa-ItemHierarchy {
padding: 1px 0;
font-size: $font-size-xs;
font-style: italic;
line-height: 18px;
opacity: .8
}
.aa-ItemDescription {
color: $color-light;
font-size: $font-size-xs;
line-height: 16px
}
}
.aa-DetachedContainer {
background: var(--color-background);
.aa-DetachedFormContainer {
.aa-DetachedCancelButton {
color: var(--color-text)
}
}
}
.aa-DetachedOverlay {
z-index: 10
}
.aa-DetachedSearchButton {
background-color: var(--color-background);
.aa-DetachedSearchButtonIcon {
color: var(--color-input-icon)
}
}

/* Search by */
.aa-powered-by-link {
display: inline-block;
width: 64px;
height: 18px;
margin-left: $size-s;
overflow: hidden;
white-space: nowrap;
text-indent: 101%;
vertical-align: middle;
background-image: url('');
background-repeat: no-repeat;
background-size: contain
}
.aa-powered-by {
padding: $size-m $size-m $size-s 0;
color: $color-muted;
font-weight: normal;
font-size: $font-size-xs;
text-align: right
}
49 changes: 49 additions & 0 deletions components/algolia-search-netlify/AlgoliasearchNetlify.ts
@@ -0,0 +1,49 @@
import { AutocompleteWrapper } from './AutocompleteWrapper'
import type { Options } from './types'

const defaultOptions: Omit<
Options,
'apiKey' | 'appId' | 'branch' | 'selector' | 'siteId'
> = {
analytics: true,
hitsPerPage: 5,
debug: false,
poweredBy: true,
placeholder: 'Search...',
openOnFocus: false
}

const mandatory: Array<keyof Options> = [
'appId',
'apiKey',
'selector',
'siteId',
'branch'
]

const instances: AutocompleteWrapper[] = []

export default function algoliasearchNetlify (_options: Options): void {
const options = {
...defaultOptions,
..._options
}
for (const key of mandatory) {
if (options[key]) { continue }

throw new Error(`[algoliasearch-netlify] Missing mandatory key: ${key}`)
}

const autocomplete = new AutocompleteWrapper(options)
instances.push(autocomplete)

// Wait for DOM initialization, then render
const render = (): void => {
autocomplete.render()
}
if (['complete', 'interactive'].includes(document.readyState)) {
render()
} else {
document.addEventListener('DOMContentLoaded', render)
}
}
164 changes: 164 additions & 0 deletions components/algolia-search-netlify/AutocompleteWrapper.ts
@@ -0,0 +1,164 @@
import { autocomplete, getAlgoliaResults } from '@algolia/autocomplete-js'
import type {
// eslint-disable-next-line no-unused-vars
AutocompleteApi,
AutocompleteSource,
SourceTemplates
} from '@algolia/autocomplete-js'
import type { HighlightedHit } from '@algolia/autocomplete-preset-algolia'
import algoliasearch from 'algoliasearch/lite'
import type { SearchClient } from 'algoliasearch/lite'

import { templates } from './templates'
import type { Options, AlgoliaRecord } from './types'

class AutocompleteWrapper {
private options;
private indexName;
private client;
private $themeNode: HTMLStyleElement | null = null;
private autocomplete: AutocompleteApi<AlgoliaRecord> | undefined;

constructor (options: Options) {
this.options = options
this.client = this.createClient()
this.indexName = this.computeIndexName()
}

render (): void {
const $input = document.querySelector(this.options.selector) as HTMLElement
if (!$input) {
// eslint-disable-next-line no-console
console.error(
`[algoliasearch-netlify] no element ${this.options.selector} found`
)
return
}

let detachedMediaQuery
if (this.options.detached !== undefined) {
if (this.options.detached === true) {
detachedMediaQuery = ''
} else if (this.options.detached === false) {
detachedMediaQuery = 'none'
} else {
detachedMediaQuery = this.options.detached.mediaQuery
}
}
const instance = autocomplete<AlgoliaRecord>({
container: $input,
autoFocus: false,
placeholder: this.options.placeholder,
debug: this.options.debug,
openOnFocus: this.options.openOnFocus,
panelPlacement: 'input-wrapper-width',
detachedMediaQuery,
getSources: () => {
return [this.getSources()]
}
})
this.applyTheme($input.firstElementChild as HTMLElement)

this.autocomplete = instance
}

private computeIndexName (): string {
const { siteId, branch } = this.options

// Keep in sync with crawler code in /netlify/crawl
const cleanBranch = branch
.trim()
.replace(/[^\p{L}\p{N}_.-]+/gu, '-')
.replace(/-{2,}/g, '-')
.toLocaleLowerCase()
return `netlify_${siteId}_${cleanBranch}_all`
}

private createClient (): SearchClient {
const client = algoliasearch(this.options.appId, this.options.apiKey)
client.addAlgoliaAgent('Netlify integration 1.0.15')
return client
}

private getSources (): AutocompleteSource<HighlightedHit<AlgoliaRecord>> {
const poweredBy = this.options.poweredBy
const tpls: SourceTemplates<HighlightedHit<AlgoliaRecord>> = {
header () {
return ''
},
item ({ item, components }) {
return templates.item(item, components)
},
footer () {
if (poweredBy) {
return templates.poweredBy({
hostname: window.location.host
})
}
return ''
}
}
const res: AutocompleteSource<HighlightedHit<AlgoliaRecord>> = {
sourceId: 'algoliaHits',
getItems: ({ query }) => {
return getAlgoliaResults({
searchClient: this.client,
queries: [
{
indexName: this.indexName,
query,
params: {
analytics: this.options.analytics,
hitsPerPage: this.options.hitsPerPage
}
}
]
})
},
getItemUrl ({ item }) {
return item.url
},
templates: tpls
}
return res
}

private applyTheme (el: HTMLElement | null): void {
if (!el || !this.options.theme) {
return
}

const theme = this.options.theme
this.$themeNode = addCss(
`.aa-Autocomplete, .aa-Panel, .aa-DetachedContainer {
${theme.mark && `--color-mark: ${theme.mark};`}
${theme.mark && `--color-background: ${theme.background};`}
${theme.mark && `--color-selected: ${theme.selected};`}
${theme.mark && `--color-text: ${theme.text};`}
${theme.mark && `--color-source-icon: ${theme.colorSourceIcon};`}
}`,
this.$themeNode
)
}
}

function addCss (
css: string,
$mainStyle: HTMLElement | null = null
): HTMLStyleElement {
const $usedSibling =
$mainStyle ??
document.querySelector(
'link[rel=stylesheet][href*="algoliasearchNetlify"]'
) ??
document.getElementsByTagName('head')[0].lastChild!
const $styleTag = document.createElement('style')
$styleTag.setAttribute('type', 'text/css')
$styleTag.appendChild(document.createTextNode(css))
return $usedSibling.parentNode!.insertBefore(
$styleTag,
$usedSibling.nextSibling
)
}

export { AutocompleteWrapper }

0 comments on commit 019f19b

Please sign in to comment.