forked from MapComplete/MapComplete
		
	
		
			
				
	
	
		
			241 lines
		
	
	
	
		
			9.2 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			241 lines
		
	
	
	
		
			9.2 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
import {Utils} from "../../Utils";
 | 
						|
import {FixedInputElement} from "../Input/FixedInputElement";
 | 
						|
import {RadioButton} from "../Input/RadioButton";
 | 
						|
import {VariableUiElement} from "../Base/VariableUIElement";
 | 
						|
import Toggle from "../Input/Toggle";
 | 
						|
import Combine from "../Base/Combine";
 | 
						|
import Translations from "../i18n/Translations";
 | 
						|
import {Translation} from "../i18n/Translation";
 | 
						|
import Svg from "../../Svg";
 | 
						|
import {UIEventSource} from "../../Logic/UIEventSource";
 | 
						|
import BaseUIElement from "../BaseUIElement";
 | 
						|
import State from "../../State";
 | 
						|
import FilteredLayer from "../../Models/FilteredLayer";
 | 
						|
import BackgroundSelector from "./BackgroundSelector";
 | 
						|
import FilterConfig from "../../Models/ThemeConfig/FilterConfig";
 | 
						|
import TilesourceConfig from "../../Models/ThemeConfig/TilesourceConfig";
 | 
						|
 | 
						|
export default class FilterView extends VariableUiElement {
 | 
						|
    constructor(filteredLayer: UIEventSource<FilteredLayer[]>, tileLayers: { config: TilesourceConfig, isDisplayed: UIEventSource<boolean> }[]) {
 | 
						|
        const backgroundSelector = new Toggle(
 | 
						|
            new BackgroundSelector(),
 | 
						|
            undefined,
 | 
						|
            State.state.featureSwitchBackgroundSlection
 | 
						|
        )
 | 
						|
        super(
 | 
						|
            filteredLayer.map((filteredLayers) => {
 | 
						|
                    let elements = filteredLayers?.map(l => FilterView.createOneFilteredLayerElement(l))
 | 
						|
                    elements = elements.concat(tileLayers.map(tl => FilterView.createOverlayToggle(tl)))
 | 
						|
                    return elements.concat(backgroundSelector);
 | 
						|
                }
 | 
						|
            )
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    private static createOverlayToggle(config: { config: TilesourceConfig, isDisplayed: UIEventSource<boolean> }) {
 | 
						|
 | 
						|
        const iconStyle = "width:1.5rem;height:1.5rem;margin-left:1.25rem;flex-shrink: 0;";
 | 
						|
 | 
						|
        const icon = new Combine([Svg.checkbox_filled]).SetStyle(iconStyle);
 | 
						|
        const iconUnselected = new Combine([Svg.checkbox_empty]).SetStyle(
 | 
						|
            iconStyle
 | 
						|
        );
 | 
						|
        const name: Translation = config.config.name;
 | 
						|
 | 
						|
        const styledNameChecked = name.Clone().SetStyle("font-size:large").SetClass("ml-2");
 | 
						|
        const styledNameUnChecked = name.Clone().SetStyle("font-size:large").SetClass("ml-2");
 | 
						|
 | 
						|
        const zoomStatus =
 | 
						|
            new Toggle(
 | 
						|
                undefined,
 | 
						|
                Translations.t.general.layerSelection.zoomInToSeeThisLayer
 | 
						|
                    .SetClass("alert")
 | 
						|
                    .SetStyle("display: block ruby;width:min-content;"),
 | 
						|
                State.state.locationControl.map(location => location.zoom >= config.config.minzoom)
 | 
						|
            )
 | 
						|
 | 
						|
 | 
						|
        const style =
 | 
						|
            "display:flex;align-items:center;padding:0.5rem 0;";
 | 
						|
        const layerChecked = new Combine([icon, styledNameChecked, zoomStatus])
 | 
						|
            .SetStyle(style)
 | 
						|
            .onClick(() => config.isDisplayed.setData(false));
 | 
						|
 | 
						|
        const layerNotChecked = new Combine([iconUnselected, styledNameUnChecked])
 | 
						|
            .SetStyle(style)
 | 
						|
            .onClick(() => config.isDisplayed.setData(true));
 | 
						|
 | 
						|
 | 
						|
        return new Toggle(
 | 
						|
            layerChecked,
 | 
						|
            layerNotChecked,
 | 
						|
            config.isDisplayed
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    private static createOneFilteredLayerElement(filteredLayer: FilteredLayer) {
 | 
						|
        if (filteredLayer.layerDef.name === undefined) {
 | 
						|
            // Name is not defined: we hide this one
 | 
						|
            return undefined;
 | 
						|
        }
 | 
						|
        const iconStyle = "width:1.5rem;height:1.5rem;margin-left:1.25rem;flex-shrink: 0;";
 | 
						|
 | 
						|
        const icon = new Combine([Svg.checkbox_filled]).SetStyle(iconStyle);
 | 
						|
        const layer = filteredLayer.layerDef
 | 
						|
 | 
						|
        const iconUnselected = new Combine([Svg.checkbox_empty]).SetStyle(
 | 
						|
            iconStyle
 | 
						|
        );
 | 
						|
 | 
						|
        if (filteredLayer.layerDef.name === undefined) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
 | 
						|
        const name: Translation = Translations.WT(
 | 
						|
            filteredLayer.layerDef.name
 | 
						|
        );
 | 
						|
 | 
						|
        const styledNameChecked = name.Clone().SetStyle("font-size:large").SetClass("ml-3");
 | 
						|
 | 
						|
        const styledNameUnChecked = name.Clone().SetStyle("font-size:large").SetClass("ml-3");
 | 
						|
 | 
						|
        const zoomStatus =
 | 
						|
            new Toggle(
 | 
						|
                undefined,
 | 
						|
                Translations.t.general.layerSelection.zoomInToSeeThisLayer
 | 
						|
                    .SetClass("alert")
 | 
						|
                    .SetStyle("display: block ruby;width:min-content;"),
 | 
						|
                State.state.locationControl.map(location => location.zoom >= filteredLayer.layerDef.minzoom)
 | 
						|
            )
 | 
						|
 | 
						|
 | 
						|
        const style =
 | 
						|
            "display:flex;align-items:center;padding:0.5rem 0;";
 | 
						|
        const layerIcon = layer.defaultIcon()?.SetClass("w-8 h-8 ml-2")
 | 
						|
        const layerIconUnchecked = layer.defaultIcon()?.SetClass("opacity-50  w-8 h-8 ml-2")
 | 
						|
 | 
						|
        const layerChecked = new Combine([icon, layerIcon, styledNameChecked, zoomStatus])
 | 
						|
            .SetStyle(style)
 | 
						|
            .onClick(() => filteredLayer.isDisplayed.setData(false));
 | 
						|
 | 
						|
        const layerNotChecked = new Combine([iconUnselected, layerIconUnchecked, styledNameUnChecked])
 | 
						|
            .SetStyle(style)
 | 
						|
            .onClick(() => filteredLayer.isDisplayed.setData(true));
 | 
						|
 | 
						|
 | 
						|
        const filterPanel: BaseUIElement = FilterView.createFilterPanel(filteredLayer)
 | 
						|
 | 
						|
 | 
						|
        return new Toggle(
 | 
						|
            new Combine([layerChecked, filterPanel]),
 | 
						|
            layerNotChecked,
 | 
						|
            filteredLayer.isDisplayed
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    private static createFilterPanel(flayer: FilteredLayer): BaseUIElement {
 | 
						|
        const layer = flayer.layerDef
 | 
						|
        if (layer.filters.length === 0) {
 | 
						|
            return undefined;
 | 
						|
        }
 | 
						|
 | 
						|
        const filterIndexes = new Map<string, number>()
 | 
						|
        layer.filters.forEach((f, i) => filterIndexes.set(f.id, i))
 | 
						|
 | 
						|
        let listFilterElements: [BaseUIElement, UIEventSource<{ filter: FilterConfig, selected: number }>][] = layer.filters.map(
 | 
						|
            FilterView.createFilter
 | 
						|
        );
 | 
						|
 | 
						|
        listFilterElements.forEach((inputElement, i) =>
 | 
						|
            inputElement[1].addCallback((changed) => {
 | 
						|
                const oldValue = flayer.appliedFilters.data
 | 
						|
 | 
						|
                if (changed === undefined) {
 | 
						|
                    // Lets figure out which filter should be removed
 | 
						|
                    // We know this inputElement corresponds with layer.filters[i]
 | 
						|
                    // SO, if there is a value in 'oldValue' with this filter, we have to recalculated
 | 
						|
                    if (!oldValue.some(f => f.filter === layer.filters[i])) {
 | 
						|
                        // The filter to remove is already gone, we can stop
 | 
						|
                        return;
 | 
						|
                    }
 | 
						|
                } else if (oldValue.some(f => f.filter === changed.filter && f.selected === changed.selected)) {
 | 
						|
                    // The changed value is already there
 | 
						|
                    return;
 | 
						|
                }
 | 
						|
                const listTagsFilters = Utils.NoNull(
 | 
						|
                    listFilterElements.map((input) => input[1].data)
 | 
						|
                );
 | 
						|
 | 
						|
                flayer.appliedFilters.setData(listTagsFilters);
 | 
						|
            })
 | 
						|
        );
 | 
						|
 | 
						|
        flayer.appliedFilters.addCallbackAndRun(appliedFilters => {
 | 
						|
            for (let i = 0; i < layer.filters.length; i++) {
 | 
						|
                const filter = layer.filters[i];
 | 
						|
                let foundMatch = undefined
 | 
						|
                for (const appliedFilter of appliedFilters) {
 | 
						|
                    if (appliedFilter.filter === filter) {
 | 
						|
                        foundMatch = appliedFilter
 | 
						|
                        break;
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                listFilterElements[i][1].setData(foundMatch)
 | 
						|
            }
 | 
						|
 | 
						|
        })
 | 
						|
 | 
						|
        return new Combine(listFilterElements.map(input => input[0].SetClass("mt-3")))
 | 
						|
            .SetClass("flex flex-col ml-8 bg-gray-300 rounded-xl p-2")
 | 
						|
 | 
						|
    }
 | 
						|
 | 
						|
    private static createFilter(filterConfig: FilterConfig): [BaseUIElement, UIEventSource<{ filter: FilterConfig, selected: number }>] {
 | 
						|
        if (filterConfig.options.length === 1) {
 | 
						|
            let option = filterConfig.options[0];
 | 
						|
 | 
						|
            const icon = Svg.checkbox_filled_svg().SetClass("block mr-2");
 | 
						|
            const iconUnselected = Svg.checkbox_empty_svg().SetClass("block mr-2");
 | 
						|
 | 
						|
            const toggle = new Toggle(
 | 
						|
                new Combine([icon, option.question.Clone().SetClass("block")]).SetClass("flex"),
 | 
						|
                new Combine([iconUnselected, option.question.Clone().SetClass("block")]).SetClass("flex")
 | 
						|
            )
 | 
						|
                .ToggleOnClick()
 | 
						|
                .SetClass("block m-1")
 | 
						|
 | 
						|
            const selected = {
 | 
						|
                filter: filterConfig,
 | 
						|
                selected: 0
 | 
						|
            }
 | 
						|
            return [toggle, toggle.isEnabled.map(enabled => enabled ? selected : undefined, [],
 | 
						|
                f => f?.filter === filterConfig && f?.selected === 0)
 | 
						|
            ]
 | 
						|
        }
 | 
						|
 | 
						|
        let options = filterConfig.options;
 | 
						|
 | 
						|
        const values = options.map((f, i) => ({
 | 
						|
            filter: filterConfig, selected: i
 | 
						|
        }))
 | 
						|
        const radio = new RadioButton(
 | 
						|
            options.map(
 | 
						|
                (option, i) =>
 | 
						|
                    new FixedInputElement(option.question.Clone().SetClass("block"), i)
 | 
						|
            ),
 | 
						|
            {
 | 
						|
                dontStyle: true
 | 
						|
            }
 | 
						|
        );
 | 
						|
        return [radio,
 | 
						|
            radio.GetValue().map(
 | 
						|
                i => values[i],
 | 
						|
                [],
 | 
						|
                selected => {
 | 
						|
                    return selected?.selected
 | 
						|
                }
 | 
						|
            )]
 | 
						|
    }
 | 
						|
}
 |