MapComplete/UI/Input/SearchableMappingsSelector.ts

298 lines
11 KiB
TypeScript
Raw Normal View History

import {UIElement} from "../UIElement"
import {InputElement} from "./InputElement"
2022-09-08 21:40:48 +02:00
import BaseUIElement from "../BaseUIElement"
import {Store, UIEventSource} from "../../Logic/UIEventSource"
2022-09-08 21:40:48 +02:00
import Translations from "../i18n/Translations"
import Locale from "../i18n/Locale"
import Combine from "../Base/Combine"
import {TextField} from "./TextField"
2022-09-08 21:40:48 +02:00
import Svg from "../../Svg"
import {VariableUiElement} from "../Base/VariableUIElement"
2022-07-10 03:58:07 +02:00
/**
* A single 'pill' which can hide itself if the search criteria is not met
*/
class SelfHidingToggle extends UIElement implements InputElement<boolean> {
public readonly _selected: UIEventSource<boolean>
2022-09-08 21:40:48 +02:00
public readonly isShown: Store<boolean> = new UIEventSource<boolean>(true)
public readonly matchesSearchCriteria: Store<boolean>
2022-07-10 03:58:07 +02:00
public readonly forceSelected: UIEventSource<boolean>
private readonly _shown: BaseUIElement
2022-09-08 21:40:48 +02:00
private readonly _squared: boolean
2022-07-10 03:58:07 +02:00
public constructor(
shown: string | BaseUIElement,
mainTerm: Record<string, string>,
search: Store<string>,
options?: {
2022-09-08 21:40:48 +02:00
searchTerms?: Record<string, string[]>
selected?: UIEventSource<boolean>
forceSelected?: UIEventSource<boolean>
squared?: boolean,
/* Hide, if not selected*/
hide?: Store<boolean>
2022-07-10 03:58:07 +02:00
}
) {
2022-09-08 21:40:48 +02:00
super()
this._shown = Translations.W(shown)
this._squared = options?.squared ?? false
const searchTerms: Record<string, string[]> = {}
2022-07-10 03:58:07 +02:00
for (const lng in options?.searchTerms ?? []) {
if (lng === "_context") {
continue
}
searchTerms[lng] = options?.searchTerms[lng]?.map(SelfHidingToggle.clean)
}
for (const lng in mainTerm) {
if (lng === "_context") {
continue
}
2022-09-08 21:40:48 +02:00
const main = SelfHidingToggle.clean(mainTerm[lng])
2022-07-10 03:58:07 +02:00
searchTerms[lng] = [main].concat(searchTerms[lng] ?? [])
}
2022-09-08 21:40:48 +02:00
const selected = (this._selected = options?.selected ?? new UIEventSource<boolean>(false))
const forceSelected = (this.forceSelected =
options?.forceSelected ?? new UIEventSource<boolean>(false))
this.matchesSearchCriteria = search.map(s => {
if (s === undefined || s.length === 0) {
return true
}
s = s?.trim()?.toLowerCase()
if (searchTerms[Locale.language.data]?.some((t) => t.indexOf(s) >= 0)) {
return true
}
if (searchTerms["*"]?.some((t) => t.indexOf(s) >= 0)) {
return true
}
return false
})
this.isShown = this.matchesSearchCriteria.map(
(matchesSearch) => {
2022-09-08 21:40:48 +02:00
if (selected.data && !forceSelected.data) {
return true
}
if (options?.hide?.data) {
return false
2022-09-08 21:40:48 +02:00
}
return matchesSearch
2022-09-08 21:40:48 +02:00
},
[selected, Locale.language, options?.hide]
2022-09-08 21:40:48 +02:00
)
2022-07-10 03:58:07 +02:00
2022-09-08 21:40:48 +02:00
const self = this
this.isShown.addCallbackAndRun((shown) => {
2022-07-10 03:58:07 +02:00
if (shown) {
self.RemoveClass("hidden")
} else {
self.SetClass("hidden")
}
})
}
2022-09-08 21:40:48 +02:00
private static clean(s: string): string {
2022-07-10 03:58:07 +02:00
return s?.trim()?.toLowerCase()?.replace(/[-]/, "")
}
GetValue(): UIEventSource<boolean> {
return this._selected
}
IsValid(t: boolean): boolean {
2022-09-08 21:40:48 +02:00
return true
2022-07-10 03:58:07 +02:00
}
protected InnerRender(): string | BaseUIElement {
2022-09-08 21:40:48 +02:00
let el: BaseUIElement = this._shown
const selected = this._selected
2022-07-10 03:58:07 +02:00
2022-09-08 21:40:48 +02:00
selected.addCallbackAndRun((selected) => {
2022-07-10 03:58:07 +02:00
if (selected) {
el.SetClass("border-4")
el.RemoveClass("border")
el.SetStyle("margin: 0")
} else {
el.SetStyle("margin: 3px")
el.SetClass("border")
el.RemoveClass("border-4")
}
})
const forcedSelection = this.forceSelected
el.onClick(() => {
2022-09-08 21:40:48 +02:00
if (forcedSelection.data) {
selected.setData(true)
2022-09-08 21:40:48 +02:00
} else {
selected.setData(!selected.data)
}
})
2022-07-10 03:58:07 +02:00
2022-09-08 21:40:48 +02:00
if (!this._squared) {
el.SetClass("rounded-full")
}
return el.SetClass("border border-black p-1 px-4")
2022-07-10 03:58:07 +02:00
}
}
/**
* The searchable mappings selector is a selector which shows various pills from which one (or more) options can be chosen.
* A searchfield can be used to filter the values
*/
export class SearchablePillsSelector<T> extends Combine implements InputElement<T[]> {
2022-09-08 21:40:48 +02:00
public readonly someMatchFound: Store<boolean>
private readonly selectedElements: UIEventSource<T[]>
2022-07-10 03:58:07 +02:00
/**
2022-09-08 21:40:48 +02:00
*
* @param values: the values that can be selected
* @param options
*/
2022-07-10 03:58:07 +02:00
constructor(
2022-09-08 21:40:48 +02:00
values: {
show: BaseUIElement
value: T
mainTerm: Record<string, string>
searchTerms?: Record<string, string[]>,
/* If there are more then 200 elements, should this element still be shown? */
hasPriority?: Store<boolean>
2022-09-08 21:40:48 +02:00
}[],
2022-07-10 03:58:07 +02:00
options?: {
/*
* If one single value can be selected (like a radio button) or if many values can be selected (like checkboxes)
*/
2022-09-08 21:40:48 +02:00
mode?: "select-one" | "select-many"
/**
* The values of the selected elements.
* Use this to tie input elements together
*/
2022-09-08 21:40:48 +02:00
selectedElements?: UIEventSource<T[]>
/**
* The search bar. Use this to seed the search value or to tie to another value
*/
2022-09-08 21:40:48 +02:00
searchValue?: UIEventSource<string>
/**
* What is shown if the search yielded no results.
* By default: a translated "no search results"
*/
2022-09-08 21:40:48 +02:00
onNoMatches?: BaseUIElement
/**
* An element that is shown if no search is entered
* Default behaviour is to show all options
*/
2022-09-08 21:40:48 +02:00
onNoSearchMade?: BaseUIElement
/**
* Extra element to show if there are many (>200) possible mappings and when non-priority mappings are hidden
*
*/
2022-09-08 21:40:48 +02:00
onManyElements?: BaseUIElement
searchAreaClass?: string
hideSearchBar?: false | boolean
2022-09-08 21:40:48 +02:00
}
) {
const search = new TextField({value: options?.searchValue})
2022-09-08 21:40:48 +02:00
const searchBar = options?.hideSearchBar
? undefined
: new Combine([
Svg.search_svg().SetClass("w-8 normal-background"),
search.SetClass("w-full"),
]).SetClass("flex items-center border-2 border-black m-2")
2022-09-08 21:40:48 +02:00
const searchValue = search.GetValue().map((s) => s?.trim()?.toLowerCase())
const selectedElements = options?.selectedElements ?? new UIEventSource<T[]>([])
const mode = options?.mode ?? "select-one"
2022-07-10 03:58:07 +02:00
const onEmpty = options?.onNoMatches ?? Translations.t.general.noMatchingMapping
const forceHide = new UIEventSource(false)
2022-09-08 21:40:48 +02:00
const mappedValues: {
show: SelfHidingToggle
mainTerm: Record<string, string>
value: T
}[] = values.map((v) => {
const vIsSelected = new UIEventSource(false)
2022-07-10 03:58:07 +02:00
2022-09-08 21:40:48 +02:00
selectedElements.addCallbackAndRunD((selectedElements) => {
vIsSelected.setData(selectedElements.some((t) => t === v.value))
2022-07-10 03:58:07 +02:00
})
2022-09-08 21:40:48 +02:00
vIsSelected.addCallback((selected) => {
2022-07-10 03:58:07 +02:00
if (selected) {
if (mode === "select-one") {
selectedElements.setData([v.value])
2022-09-08 21:40:48 +02:00
} else if (!selectedElements.data.some((t) => t === v.value)) {
selectedElements.data.push(v.value)
2022-07-10 03:58:07 +02:00
selectedElements.ping()
}
} else {
for (let i = 0; i < selectedElements.data.length; i++) {
const t = selectedElements.data[i]
if (t == v.value) {
selectedElements.data.splice(i, 1)
selectedElements.ping()
2022-09-08 21:40:48 +02:00
break
2022-07-10 03:58:07 +02:00
}
}
}
})
const toggle = new SelfHidingToggle(v.show, v.mainTerm, searchValue, {
searchTerms: v.searchTerms,
selected: vIsSelected,
2022-09-08 21:40:48 +02:00
squared: mode === "select-many",
hide: v.hasPriority === undefined ? forceHide : forceHide.map(fh => fh && !v.hasPriority?.data, [v.hasPriority])
2022-07-10 03:58:07 +02:00
})
return {
...v,
2022-09-08 21:40:48 +02:00
show: toggle,
}
2022-07-10 03:58:07 +02:00
})
// The total number of elements that would be displayed based on the search criteria alone
let totalShown: Store<number>
totalShown = searchValue.map((_) => mappedValues.filter((mv) => mv.show.matchesSearchCriteria.data).length)
const tooMuchElementsCutoff = 40
totalShown.addCallbackAndRunD(shown => forceHide.setData(tooMuchElementsCutoff < shown))
2022-07-10 03:58:07 +02:00
super([
searchBar,
2022-09-08 21:40:48 +02:00
new VariableUiElement(
Locale.language.map(
(lng) => {
2022-09-08 21:40:48 +02:00
if (
options?.onNoSearchMade !== undefined &&
(searchValue.data === undefined || searchValue.data.length === 0)
) {
return options?.onNoSearchMade
}
if (totalShown.data == 0) {
return onEmpty
}
2022-07-10 03:58:07 +02:00
2022-09-08 21:40:48 +02:00
mappedValues.sort((a, b) => (a.mainTerm[lng] < b.mainTerm[lng] ? -1 : 1))
let pills = new Combine(mappedValues.map((e) => e.show))
2022-09-08 21:40:48 +02:00
.SetClass("flex flex-wrap w-full content-start")
.SetClass(options?.searchAreaClass ?? "")
if (totalShown.data >= tooMuchElementsCutoff) {
pills = new Combine([options?.onManyElements ?? Translations.t.general.useSearch, pills])
}
return pills
2022-09-08 21:40:48 +02:00
},
[totalShown, searchValue]
)
),
2022-07-10 03:58:07 +02:00
])
2022-09-08 21:40:48 +02:00
this.selectedElements = selectedElements
this.someMatchFound = totalShown.map((t) => t > 0)
2022-07-10 03:58:07 +02:00
}
public GetValue(): UIEventSource<T[]> {
2022-09-08 21:40:48 +02:00
return this.selectedElements
2022-07-10 03:58:07 +02:00
}
IsValid(t: T[]): boolean {
2022-09-08 21:40:48 +02:00
return true
2022-07-10 03:58:07 +02:00
}
}