import { OsmConnection } from "../../Logic/Osm/OsmConnection" import { ConfigMeta } from "./configMeta" import { Store, UIEventSource } from "../../Logic/UIEventSource" import { LayerConfigJson } from "../../Models/ThemeConfig/Json/LayerConfigJson" import { QueryParameters } from "../../Logic/Web/QueryParameters" import { ConversionContext, ConversionMessage, DesugaringContext, Pipe, } from "../../Models/ThemeConfig/Conversion/Conversion" import { PrepareLayer } from "../../Models/ThemeConfig/Conversion/PrepareLayer" import { ValidateLayer } from "../../Models/ThemeConfig/Conversion/Validation" import { AllSharedLayers } from "../../Customizations/AllSharedLayers" import { QuestionableTagRenderingConfigJson } from "../../Models/ThemeConfig/Json/QuestionableTagRenderingConfigJson" import { TagUtils } from "../../Logic/Tags/TagUtils" import StudioServer from "./StudioServer" /** * Sends changes back to the server */ export class LayerStateSender { constructor(layerState: EditLayerState) { layerState.configuration.addCallback(async (config) => { const id = config.id if (id === undefined) { console.warn("No id found in layer, not updating") return } await layerState.server.updateLayer(config) }) } } export default class EditLayerState { public readonly osmConnection: OsmConnection public readonly schema: ConfigMeta[] public readonly featureSwitches: { featureSwitchIsDebugging: UIEventSource } public readonly configuration: UIEventSource> = new UIEventSource< Partial >({}) public readonly messages: Store public readonly server: StudioServer constructor(schema: ConfigMeta[], server: StudioServer) { this.schema = schema this.server = server this.osmConnection = new OsmConnection({ oauth_token: QueryParameters.GetQueryParameter( "oauth_token", undefined, "Used to complete the login" ), }) this.featureSwitches = { featureSwitchIsDebugging: new UIEventSource(true), } let state: DesugaringContext { const layers = AllSharedLayers.getSharedLayersConfigs() const questions = layers.get("questions") const sharedQuestions = new Map() for (const question of questions.tagRenderings) { sharedQuestions.set(question["id"], question) } state = { tagRenderings: sharedQuestions, sharedLayers: layers, } } this.messages = this.configuration.mapD((config) => { const context = ConversionContext.construct([], ["prepare"]) for (let i = 0; i < (config.tagRenderings ?? []).length; i++) { const tr = config.tagRenderings[i] if (typeof tr === "string") { continue } if (!tr["id"] && !tr["override"]) { const qtr = tr let id = "" + i if (qtr?.freeform?.key) { id = qtr?.freeform?.key } else if (qtr.mappings?.[0]?.if) { id = qtr.freeform?.key ?? TagUtils.Tag(qtr.mappings[0].if).usedKeys()?.[0] ?? "" + i } qtr["id"] = id } } const prepare = new Pipe( new PrepareLayer(state), new ValidateLayer("dynamic", false, undefined) ) prepare.convert(config, context) return context.messages }) } public getCurrentValueFor(path: ReadonlyArray): any | undefined { // Walk the path down to see if we find something let entry = this.configuration.data for (let i = 0; i < path.length; i++) { if (entry === undefined) { // We reached a dead end - no old vlaue return undefined } const breadcrumb = path[i] entry = entry[breadcrumb] } return entry } public getStoreFor(path: ReadonlyArray): UIEventSource { const store = new UIEventSource(this.getCurrentValueFor(path)) store.addCallback((v) => { this.setValueAt(path, v) }) return store } public register( path: ReadonlyArray, value: Store, noInitialSync: boolean = false ): () => void { const unsync = value.addCallback((v) => this.setValueAt(path, v)) if (!noInitialSync) { this.setValueAt(path, value.data) } return unsync } public getSchemaStartingWith(path: string[]) { return this.schema.filter( (sch) => !path.some((part, i) => !(sch.path.length > path.length && sch.path[i] === part)) ) } public getTranslationAt(path: string[]): ConfigMeta { const origConfig = this.getSchema(path)[0] return { path, type: "translation", hints: { typehint: "translation", }, required: origConfig.required ?? false, description: origConfig.description ?? "A translatable object", } } public getSchema(path: string[]): ConfigMeta[] { const schemas = this.schema.filter( (sch) => sch !== undefined && !path.some((part, i) => !(sch.path.length == path.length && sch.path[i] === part)) ) if (schemas.length == 0) { console.warn("No schemas found for path", path.join(".")) } return schemas } public setValueAt(path: ReadonlyArray, v: any) { let entry = this.configuration.data for (let i = 0; i < path.length - 1; i++) { const breadcrumb = path[i] if (entry[breadcrumb] === undefined) { entry[breadcrumb] = typeof path[i + 1] === "number" ? [] : {} } entry = entry[breadcrumb] } if ( v !== undefined && v !== null && v !== "" && !(typeof v === "object" && Object.keys({}).length === 0) ) { entry[path.at(-1)] = v } else if (entry) { delete entry[path.at(-1)] } this.configuration.ping() } }