mirror of
https://github.com/misode/misode.github.io.git
synced 2026-04-25 08:06:51 +00:00
63 lines
2.4 KiB
TypeScript
63 lines
2.4 KiB
TypeScript
import { useMemo, useState } from 'preact/hooks'
|
|
import type { ConfigGenerator } from '../../Config.js'
|
|
import config from '../../Config.js'
|
|
import { useLocale, useVersion } from '../../contexts/index.js'
|
|
import { checkVersion } from '../../services/Versions.js'
|
|
import { GeneratorCard, TextInput, VersionSwitcher } from '../index.js'
|
|
|
|
interface Props {
|
|
path?: string,
|
|
predicate?: (gen: ConfigGenerator) => boolean | undefined,
|
|
}
|
|
export function GeneratorList({ predicate }: Props) {
|
|
const { locale } = useLocale()
|
|
const { version, changeVersion } = useVersion()
|
|
|
|
const [search, setSearch] = useState('')
|
|
|
|
const [versionFilter, setVersionFiler] = useState(true)
|
|
|
|
const versionedGenerators = useMemo(() => {
|
|
return config.generators.filter(gen => {
|
|
if (predicate === undefined || !predicate(gen)) return false
|
|
if (versionFilter === false) return true
|
|
return checkVersion(version, gen.minVersion, gen.maxVersion)
|
|
})
|
|
}, [version, versionFilter])
|
|
|
|
const filteredGenerators = useMemo(() => {
|
|
const results = versionedGenerators
|
|
.map(g => ({ ...g, name: locale(`generator.${g.id}`).toLowerCase() }))
|
|
return searchGenerators(results, search)
|
|
}, [versionedGenerators, search, locale])
|
|
|
|
return <div class="generator-list">
|
|
<div class="navigation">
|
|
<TextInput class="btn btn-input query-search" placeholder={locale('generators.search')} value={search} onChange={setSearch} autofocus />
|
|
<VersionSwitcher value={versionFilter ? version : undefined} onChange={v => {changeVersion(v); setVersionFiler(true)}} hasAny onAny={() => setVersionFiler(false)} />
|
|
</div>
|
|
{filteredGenerators.length === 0 ? <>
|
|
<span class="note">{locale('generators.no_results')}</span>
|
|
</> : <div class="card-grid">
|
|
{filteredGenerators.map(gen =>
|
|
<GeneratorCard id={gen.id} />
|
|
)}
|
|
</div>}
|
|
</div>
|
|
}
|
|
|
|
export function searchGenerators(generators: (ConfigGenerator & { name: string})[], search?: string) {
|
|
if (search) {
|
|
const parts = search.split(' ').map(q => q.trim().toLowerCase()).filter(q => q.length > 0)
|
|
generators = generators.filter(g => parts.some(p => g.name.includes(p))
|
|
|| parts.some(p => g.path?.includes(p) ?? false)
|
|
|| parts.some(p => g.tags?.some(t => t.includes(p)) ?? false)
|
|
|| parts.some(p => g.aliases?.some(a => a.includes(p)) ?? false))
|
|
}
|
|
generators.sort((a, b) => a.name.localeCompare(b.name))
|
|
if (search) {
|
|
generators.sort((a, b) => (b.name.startsWith(search) ? 1 : 0) - (a.name.startsWith(search) ? 1 : 0))
|
|
}
|
|
return generators
|
|
}
|