MapComplete/src/Models/FilteredLayer.ts

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

304 lines
10 KiB
TypeScript
Raw Normal View History

2023-07-28 01:02:31 +02:00
import { Store, UIEventSource } from "../Logic/UIEventSource"
import LayerConfig from "./ThemeConfig/LayerConfig"
2023-07-28 01:02:31 +02:00
import { OsmConnection } from "../Logic/Osm/OsmConnection"
import { LocalStorageSource } from "../Logic/Web/LocalStorageSource"
import { QueryParameters } from "../Logic/Web/QueryParameters"
import { FilterConfigOption } from "./ThemeConfig/FilterConfig"
import { TagsFilter } from "../Logic/Tags/TagsFilter"
import { Utils } from "../Utils"
import { TagUtils } from "../Logic/Tags/TagUtils"
import { And } from "../Logic/Tags/And"
import { GlobalFilter } from "./GlobalFilter"
2022-01-08 04:22:50 +01:00
2023-03-28 05:13:48 +02:00
export default class FilteredLayer {
/**
2024-04-10 15:30:12 +02:00
* Whether the specified layer is enabled by the user
2023-03-28 05:13:48 +02:00
*/
2021-07-27 19:39:57 +02:00
readonly isDisplayed: UIEventSource<boolean>
2023-03-28 05:13:48 +02:00
/**
* Maps the filter.option.id onto the actual used state.
* This state is either the chosen option (as number) or a representation of the fields
2023-03-28 05:13:48 +02:00
*/
readonly appliedFilters: ReadonlyMap<string, UIEventSource<undefined | number | string>>
2021-07-27 19:39:57 +02:00
readonly layerDef: LayerConfig
2023-03-28 05:13:48 +02:00
/**
* Indicates if some filter is set.
* If this is the case, adding a new element of this type might be a bad idea
*/
readonly hasFilter: Store<boolean>
/**
* Contains the current properties a feature should fulfill in order to match the filter
*/
readonly currentFilter: Store<TagsFilter | undefined>
2023-03-28 05:13:48 +02:00
constructor(
layer: LayerConfig,
2023-04-14 17:53:08 +02:00
appliedFilters?: ReadonlyMap<string, UIEventSource<undefined | number | string>>,
2024-11-07 11:19:15 +01:00
isDisplayed?: UIEventSource<boolean>
2023-03-28 05:13:48 +02:00
) {
this.layerDef = layer
this.isDisplayed = isDisplayed ?? new UIEventSource(true)
2023-04-24 03:22:43 +02:00
if (!appliedFilters) {
const appliedFiltersWritable = new Map<
string,
UIEventSource<number | string | undefined>
>()
for (const filter of this.layerDef.filters) {
appliedFiltersWritable.set(filter.id, new UIEventSource(undefined))
}
appliedFilters = appliedFiltersWritable
}
this.appliedFilters = appliedFilters
const currentTags = new UIEventSource<TagsFilter>(undefined)
this.appliedFilters.forEach((filterSrc) => {
2024-04-10 15:30:12 +02:00
filterSrc.addCallbackAndRun(() => {
currentTags.setData(this.calculateCurrentTags())
})
})
this.hasFilter = currentTags.map((ct) => ct !== undefined)
this.currentFilter = currentTags
}
public static fieldsToString(values: Record<string, string>): string {
for (const key in values) {
if (values[key] === "") {
delete values[key]
}
}
return JSON.stringify(values)
}
2023-07-16 02:08:43 +02:00
public static queryParameterKey(layer: LayerConfig) {
return "layer-" + layer.id
}
2023-03-28 05:13:48 +02:00
/**
* Creates a FilteredLayer which is tied into the QueryParameters and/or user preferences
*/
public static initLinkedState(
layer: LayerConfig,
context: string,
osmConnection?: OsmConnection,
2024-11-07 11:19:15 +01:00
enabledByDefault?: Store<boolean>
2023-03-28 05:13:48 +02:00
) {
let isDisplayed: UIEventSource<boolean>
if (layer.syncSelection === "local") {
isDisplayed = LocalStorageSource.getParsed(
2023-03-28 05:13:48 +02:00
context + "-layer-" + layer.id + "-enabled",
2024-11-07 11:19:15 +01:00
layer.shownByDefault
2023-03-28 05:13:48 +02:00
)
} else if (layer.syncSelection === "theme-only" && osmConnection) {
2023-03-28 05:13:48 +02:00
isDisplayed = FilteredLayer.getPref(
osmConnection,
context + "-layer-" + layer.id + "-enabled",
2024-11-07 11:19:15 +01:00
layer
2023-03-28 05:13:48 +02:00
)
} else if (layer.syncSelection === "global" && osmConnection) {
2023-03-28 05:13:48 +02:00
isDisplayed = FilteredLayer.getPref(
osmConnection,
"layer-" + layer.id + "-enabled",
2024-11-07 11:19:15 +01:00
layer
2023-03-28 05:13:48 +02:00
)
} else {
let isShown = layer.shownByDefault
2024-07-09 13:42:08 +02:00
if (enabledByDefault !== undefined && enabledByDefault.data === false) {
isShown = false
}
2023-03-28 05:13:48 +02:00
isDisplayed = QueryParameters.GetBooleanQueryParameter(
2023-07-16 02:08:43 +02:00
FilteredLayer.queryParameterKey(layer),
2024-07-09 13:42:08 +02:00
isShown,
2024-11-07 11:19:15 +01:00
"Whether or not layer " + layer.id + " is shown"
2023-03-28 05:13:48 +02:00
)
}
const appliedFilters = new Map<string, UIEventSource<undefined | number | string>>()
for (const subfilter of layer.filters) {
appliedFilters.set(subfilter.id, subfilter.initState(layer.id))
2023-03-28 05:13:48 +02:00
}
return new FilteredLayer(layer, appliedFilters, isDisplayed)
}
2023-04-24 03:22:43 +02:00
private static stringToFieldProperties(value: string): Record<string, string> {
const values = JSON.parse(value)
for (const key in values) {
if (values[key] === "") {
delete values[key]
}
}
return values
}
/**
* import Translations from "../UI/i18n/Translations"
* import { RegexTag } from "../Logic/Tags/RegexTag"
* import { ComparingTag } from "../Logic/Tags/ComparingTag"
*
* const option: FilterConfigOption = {question: Translations.T("question"), osmTags: undefined, originalTagsSpec: "key~.*{search}.*", fields: [{name: "search", type: "string"}] }
2024-09-17 03:17:21 +02:00
* FilteredLayer.fieldsToTags(option, {search: "value_regex"}) // => new RegexTag("key", /^(.*(value_regex).*)$/s)
*
* const option: FilterConfigOption = {question: Translations.T("question"), searchTerms: undefined, osmTags: undefined, originalTagsSpec: "edit_time>{search}", fields: [{name: "search", type: "date"}] }
* const comparingTag = FilteredLayer.fieldsToTags(option, {search: "2024-09-20"})
* comparingTag.asJson() // => "edit_time>1726790400000"
*/
private static fieldsToTags(
option: FilterConfigOption,
2024-11-07 11:19:15 +01:00
fieldstate: string | Record<string, string>
): TagsFilter | undefined {
let properties: Record<string, string>
if (typeof fieldstate === "string") {
properties = FilteredLayer.stringToFieldProperties(fieldstate)
} else {
properties = fieldstate
}
const missingKeys = option.fields
.map((f) => f.name)
.filter((key) => properties[key] === undefined)
if (missingKeys.length > 0) {
return undefined
}
const tagsSpec = Utils.WalkJson(option.originalTagsSpec, (v) => {
if (typeof v !== "string") {
return v
}
for (const key in properties) {
const needsParentheses = v.match(/[a-zA-Z0-9_:]+~/)
if (needsParentheses) {
v = (<string>v).replace("{" + key + "}", "(" + properties[key] + ")")
} else {
v = (<string>v).replace("{" + key + "}", properties[key])
}
}
return v
})
return TagUtils.Tag(tagsSpec)
}
2023-03-28 05:13:48 +02:00
private static getPref(
osmConnection: OsmConnection,
key: string,
2024-11-07 11:19:15 +01:00
layer: LayerConfig
2023-03-28 05:13:48 +02:00
): UIEventSource<boolean> {
return osmConnection.GetPreference(key, layer.shownByDefault + "").sync(
(v) => {
if (v === undefined) {
return undefined
}
return v === "true"
},
[],
(b) => {
if (b === undefined) {
return undefined
}
return "" + b
2024-11-07 11:19:15 +01:00
}
2023-03-28 05:13:48 +02:00
)
}
public disableAllFilters(): void {
this.appliedFilters.forEach((value) => value.setData(undefined))
}
2023-04-24 03:22:43 +02:00
/**
* Returns true if the given tags match
* - the current filters
* - the specified 'global filters'
* - the 'isShown'-filter set by the layer
2023-04-24 03:22:43 +02:00
*/
2024-11-28 12:00:23 +01:00
public isShown(
properties: Record<string, string>,
globalFilters?: GlobalFilter[],
zoomlevel?: number
): boolean {
2023-04-24 03:22:43 +02:00
if (properties._deleted === "yes") {
return false
}
2023-04-26 18:04:42 +02:00
for (const globalFilter of globalFilters ?? []) {
const neededTags = globalFilter.osmTags
if (neededTags !== undefined) {
const doesMatch = neededTags.matchesProperties(properties)
if (globalFilter.forceShowOnMatch) {
2024-11-28 12:00:23 +01:00
if (doesMatch) {
return true
}
} else if (!doesMatch) {
return false
}
2023-04-26 18:04:42 +02:00
}
}
{
2024-12-11 02:45:44 +01:00
if (!this.isDisplayed.data) {
return false
}
}
2023-04-24 03:22:43 +02:00
{
const isShown: TagsFilter = this.layerDef.isShown
if (isShown !== undefined && !isShown.matchesProperties(properties)) {
return false
}
}
{
2024-04-10 15:30:12 +02:00
const neededTags: TagsFilter = this.currentFilter.data
2023-04-24 03:22:43 +02:00
if (neededTags !== undefined && !neededTags.matchesProperties(properties)) {
return false
}
}
2024-11-28 12:00:23 +01:00
if (
zoomlevel !== undefined &&
(this.layerDef.minzoom > zoomlevel || this.layerDef.minzoomVisible < zoomlevel)
) {
return false
}
2023-04-24 03:22:43 +02:00
return true
}
private calculateCurrentTags(): TagsFilter {
let needed: TagsFilter[] = []
for (const filter of this.layerDef.filters) {
const state = this.appliedFilters.get(filter.id)
if (state.data === undefined) {
continue
}
if (filter.options[0].fields.length > 0) {
// This is a filter with fields
// We calculate the fields
const fieldProperties = FilteredLayer.stringToFieldProperties(<string>state.data)
const asTags = FilteredLayer.fieldsToTags(filter.options[0], fieldProperties)
if (asTags) {
needed.push(asTags)
}
continue
}
needed.push(filter.options[state.data].osmTags)
}
needed = Utils.NoNull(needed)
if (needed.length == 0) {
return undefined
}
let tags: TagsFilter
if (needed.length == 1) {
tags = needed[0]
} else {
tags = new And(needed)
}
2024-04-10 15:30:12 +02:00
const optimized = tags.optimize()
if (optimized === true) {
return undefined
}
if (optimized === false) {
return tags
}
return optimized
}
2021-07-27 19:39:57 +02:00
}