2019-01-11 10:46:02 +00:00
|
|
|
import withSitePaths from 'Components/utils/withSitePaths'
|
2018-03-29 08:55:42 +00:00
|
|
|
import { encodeRuleName } from 'Engine/rules.js'
|
2018-01-25 16:06:55 +00:00
|
|
|
import Fuse from 'fuse.js'
|
2019-01-11 10:46:02 +00:00
|
|
|
import { compose, pick, sortBy } from 'ramda'
|
2019-09-15 20:51:13 +00:00
|
|
|
import React, { useMemo, useRef, useState } from 'react'
|
2018-07-12 08:09:41 +00:00
|
|
|
import Highlighter from 'react-highlight-words'
|
2019-09-11 08:06:51 +00:00
|
|
|
import { useTranslation } from 'react-i18next'
|
2018-07-12 08:09:41 +00:00
|
|
|
import { Link, Redirect } from 'react-router-dom'
|
|
|
|
import Select from 'react-select'
|
|
|
|
import 'react-select/dist/react-select.css'
|
|
|
|
import { capitalise0 } from '../utils'
|
2018-01-25 16:06:55 +00:00
|
|
|
|
2019-09-11 08:06:26 +00:00
|
|
|
function SearchBar({
|
|
|
|
rules,
|
|
|
|
showDefaultList,
|
|
|
|
finally: finallyCallback,
|
|
|
|
sitePaths
|
|
|
|
}) {
|
|
|
|
const [inputValue, setInputValue] = useState(null)
|
|
|
|
const [selectedOption, setSelectedOption] = useState(null)
|
|
|
|
const inputElementRef = useRef()
|
2019-09-15 20:51:13 +00:00
|
|
|
// This operation is expensive, we don't want to do it everytime we re-render, so we cache its result
|
|
|
|
const fuse = useMemo(() => {
|
|
|
|
const list = rules.map(
|
|
|
|
pick(['title', 'espace', 'description', 'name', 'dottedName'])
|
2018-02-13 18:13:00 +00:00
|
|
|
)
|
2019-09-15 20:51:13 +00:00
|
|
|
const options = {
|
|
|
|
keys: [
|
|
|
|
{ name: 'name', weight: 0.3 },
|
|
|
|
{ name: 'title', weight: 0.3 },
|
|
|
|
{ name: 'espace', weight: 0.2 },
|
|
|
|
{ name: 'description', weight: 0.2 }
|
|
|
|
]
|
|
|
|
}
|
|
|
|
return new Fuse(list, options)
|
|
|
|
}, [rules])
|
|
|
|
const { i18n } = useTranslation()
|
2019-09-11 08:06:26 +00:00
|
|
|
|
|
|
|
const renderOption = ({ title, dottedName }) => (
|
2018-02-21 14:53:45 +00:00
|
|
|
<span>
|
2019-09-11 08:06:26 +00:00
|
|
|
<Highlighter searchWords={[inputValue]} textToHighlight={title} />
|
2018-02-22 17:22:07 +00:00
|
|
|
<span style={{ opacity: 0.6, fontSize: '75%', marginLeft: '.6em' }}>
|
2019-09-11 08:06:26 +00:00
|
|
|
<Highlighter searchWords={[inputValue]} textToHighlight={dottedName} />
|
2018-02-21 14:53:45 +00:00
|
|
|
</span>
|
|
|
|
</span>
|
2018-01-25 16:24:53 +00:00
|
|
|
)
|
2019-09-15 20:51:13 +00:00
|
|
|
const filterOptions = (options, filter) => fuse.search(filter)
|
2018-01-25 16:06:55 +00:00
|
|
|
|
2019-09-11 08:06:26 +00:00
|
|
|
if (selectedOption != null) {
|
|
|
|
finallyCallback && finallyCallback()
|
2018-01-25 16:06:55 +00:00
|
|
|
return (
|
2019-09-11 08:06:26 +00:00
|
|
|
<Redirect
|
|
|
|
to={
|
|
|
|
sitePaths.documentation.index +
|
|
|
|
'/' +
|
|
|
|
encodeRuleName(selectedOption.dottedName)
|
|
|
|
}
|
|
|
|
/>
|
2018-01-25 16:06:55 +00:00
|
|
|
)
|
|
|
|
}
|
2019-09-11 08:06:26 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<Select
|
|
|
|
value={selectedOption && selectedOption.dottedName}
|
2019-09-15 20:51:13 +00:00
|
|
|
onChange={setSelectedOption}
|
|
|
|
onInputChange={setInputValue}
|
2019-09-11 08:06:26 +00:00
|
|
|
valueKey="dottedName"
|
|
|
|
labelKey="title"
|
|
|
|
options={rules}
|
|
|
|
filterOptions={filterOptions}
|
|
|
|
optionRenderer={renderOption}
|
|
|
|
placeholder={i18n.t('Entrez des mots clefs ici')}
|
|
|
|
noResultsText={i18n.t('noresults', {
|
|
|
|
defaultValue: "Nous n'avons rien trouvé…"
|
|
|
|
})}
|
|
|
|
ref={inputElementRef}
|
|
|
|
/>
|
|
|
|
{showDefaultList && !inputValue && (
|
|
|
|
<ul>
|
|
|
|
{sortBy(rule => rule.title, rules).map(rule => (
|
|
|
|
<li key={rule.dottedName}>
|
|
|
|
<Link
|
|
|
|
to={
|
|
|
|
sitePaths.documentation.index +
|
|
|
|
'/' +
|
|
|
|
encodeRuleName(rule.name)
|
|
|
|
}>
|
|
|
|
{rule.title || capitalise0(rule.name)}
|
|
|
|
</Link>
|
|
|
|
</li>
|
|
|
|
))}
|
|
|
|
</ul>
|
|
|
|
)}
|
|
|
|
</>
|
|
|
|
)
|
2018-01-25 16:06:55 +00:00
|
|
|
}
|
2018-11-14 15:51:37 +00:00
|
|
|
|
2019-09-11 08:06:51 +00:00
|
|
|
export default compose(withSitePaths)(SearchBar)
|