forked from MapComplete/MapComplete
		
	Formatting
This commit is contained in:
		
							parent
							
								
									6d822b42ca
								
							
						
					
					
						commit
						61aebc61eb
					
				
					 32 changed files with 664 additions and 511 deletions
				
			
		|  | @ -2,7 +2,7 @@ import FeatureSource, {FeatureSourceForLayer, Tiled} from "../FeatureSource" | |||
| import { ImmutableStore, Store } from "../../UIEventSource" | ||||
| import FilteredLayer from "../../../Models/FilteredLayer" | ||||
| import { BBox } from "../../BBox" | ||||
| import {Feature} from "geojson"; | ||||
| import { Feature } from "geojson" | ||||
| 
 | ||||
| /** | ||||
|  * A simple, read only feature store. | ||||
|  |  | |||
|  | @ -19,7 +19,11 @@ export default class CreateNewNodeAction extends OsmCreateAction { | |||
|     private readonly _lon: number | ||||
|     private readonly _snapOnto: OsmWay | ||||
|     private readonly _reusePointDistance: number | ||||
|     private readonly meta: { changeType: "create" | "import"; theme: string; specialMotivation?: string } | ||||
|     private readonly meta: { | ||||
|         changeType: "create" | "import" | ||||
|         theme: string | ||||
|         specialMotivation?: string | ||||
|     } | ||||
|     private readonly _reusePreviouslyCreatedPoint: boolean | ||||
| 
 | ||||
|     constructor( | ||||
|  |  | |||
|  | @ -6,7 +6,7 @@ import ChangeTagAction from "./ChangeTagAction" | |||
| import { TagsFilter } from "../../Tags/TagsFilter" | ||||
| import { And } from "../../Tags/And" | ||||
| import { Tag } from "../../Tags/Tag" | ||||
| import {OsmId} from "../../../Models/OsmFeature"; | ||||
| import { OsmId } from "../../../Models/OsmFeature" | ||||
| import { Utils } from "../../../Utils" | ||||
| 
 | ||||
| export default class DeleteAction extends OsmChangeAction { | ||||
|  | @ -19,7 +19,6 @@ export default class DeleteAction extends OsmChangeAction { | |||
|     private readonly _id: OsmId | ||||
|     private readonly _hardDelete: boolean | ||||
| 
 | ||||
| 
 | ||||
|     constructor( | ||||
|         id: OsmId, | ||||
|         softDeletionTags: TagsFilter | undefined, | ||||
|  | @ -42,8 +41,9 @@ export default class DeleteAction extends OsmChangeAction { | |||
|                     new Tag( | ||||
|                         "fixme", | ||||
|                         `A mapcomplete user marked this feature to be deleted (${meta.specialMotivation})` | ||||
|                     ), | ||||
|                 ]) | ||||
|             ) | ||||
|             ])) | ||||
|         } | ||||
|     } | ||||
|     /** | ||||
|  | @ -63,8 +63,11 @@ export default class DeleteAction extends OsmChangeAction { | |||
|      * const descr = await da.CreateChangeDescriptions(new Changes(), obj) | ||||
|      * descr[0] // => {doDelete: true, meta: {theme: "test", specialMotivation: "Testcase", changeType: "deletion"}, type: "node",id: 1 }
 | ||||
|      */ | ||||
|     public async CreateChangeDescriptions(changes: Changes, object?: OsmObject): Promise<ChangeDescription[]> { | ||||
|         const osmObject = object ?? await OsmObject.DownloadObjectAsync(this._id) | ||||
|     public async CreateChangeDescriptions( | ||||
|         changes: Changes, | ||||
|         object?: OsmObject | ||||
|     ): Promise<ChangeDescription[]> { | ||||
|         const osmObject = object ?? (await OsmObject.DownloadObjectAsync(this._id)) | ||||
| 
 | ||||
|         if (this._hardDelete) { | ||||
|             return [ | ||||
|  |  | |||
|  | @ -13,7 +13,7 @@ import { Utils } from "../../../Utils" | |||
| import { OsmConnection } from "../OsmConnection" | ||||
| import { Feature } from "@turf/turf" | ||||
| import FeaturePipeline from "../../FeatureSource/FeaturePipeline" | ||||
| import {Geometry, LineString, Point, Polygon} from "geojson"; | ||||
| import { Geometry, LineString, Point, Polygon } from "geojson" | ||||
| 
 | ||||
| export default class ReplaceGeometryAction extends OsmChangeAction { | ||||
|     /** | ||||
|  |  | |||
|  | @ -4,7 +4,7 @@ import {Store, UIEventSource} from "../UIEventSource" | |||
| import { BBox } from "../BBox" | ||||
| import * as OsmToGeoJson from "osmtogeojson" | ||||
| import { NodeId, OsmFeature, OsmId, OsmTags, RelationId, WayId } from "../../Models/OsmFeature" | ||||
| import {Feature, LineString, Polygon} from "geojson"; | ||||
| import { Feature, LineString, Polygon } from "geojson" | ||||
| 
 | ||||
| export abstract class OsmObject { | ||||
|     private static defaultBackend = "https://www.openstreetmap.org/" | ||||
|  | @ -41,9 +41,9 @@ export abstract class OsmObject { | |||
|         this.backendURL = url | ||||
|     } | ||||
| 
 | ||||
|     public static DownloadObject(id: NodeId, forceRefresh?: boolean): Store<OsmNode> ; | ||||
|     public static DownloadObject(id: RelationId, forceRefresh?: boolean): Store<OsmRelation> ; | ||||
|     public static DownloadObject(id: WayId, forceRefresh?: boolean): Store<OsmWay> ; | ||||
|     public static DownloadObject(id: NodeId, forceRefresh?: boolean): Store<OsmNode> | ||||
|     public static DownloadObject(id: RelationId, forceRefresh?: boolean): Store<OsmRelation> | ||||
|     public static DownloadObject(id: WayId, forceRefresh?: boolean): Store<OsmWay> | ||||
|     public static DownloadObject(id: string, forceRefresh: boolean = false): Store<OsmObject> { | ||||
|         let src: UIEventSource<OsmObject> | ||||
|         if (OsmObject.objectCache.has(id)) { | ||||
|  | @ -73,12 +73,30 @@ export abstract class OsmObject { | |||
|         return rawData.elements[0].tags | ||||
|     } | ||||
| 
 | ||||
|     static async DownloadObjectAsync(id: NodeId, maxCacheAgeInSecs?: number): Promise<OsmNode | undefined> | ||||
|     static async DownloadObjectAsync(id: WayId, maxCacheAgeInSecs?: number): Promise<OsmWay | undefined> | ||||
|     static async DownloadObjectAsync(id: RelationId, maxCacheAgeInSecs?: number): Promise<OsmRelation | undefined> | ||||
|     static async DownloadObjectAsync(id: OsmId, maxCacheAgeInSecs?: number): Promise<OsmObject | undefined> | ||||
|     static async DownloadObjectAsync(id: string, maxCacheAgeInSecs?: number): Promise<OsmObject | undefined> | ||||
|     static async DownloadObjectAsync(id: string, maxCacheAgeInSecs?: number): Promise<OsmObject | undefined> { | ||||
|     static async DownloadObjectAsync( | ||||
|         id: NodeId, | ||||
|         maxCacheAgeInSecs?: number | ||||
|     ): Promise<OsmNode | undefined> | ||||
|     static async DownloadObjectAsync( | ||||
|         id: WayId, | ||||
|         maxCacheAgeInSecs?: number | ||||
|     ): Promise<OsmWay | undefined> | ||||
|     static async DownloadObjectAsync( | ||||
|         id: RelationId, | ||||
|         maxCacheAgeInSecs?: number | ||||
|     ): Promise<OsmRelation | undefined> | ||||
|     static async DownloadObjectAsync( | ||||
|         id: OsmId, | ||||
|         maxCacheAgeInSecs?: number | ||||
|     ): Promise<OsmObject | undefined> | ||||
|     static async DownloadObjectAsync( | ||||
|         id: string, | ||||
|         maxCacheAgeInSecs?: number | ||||
|     ): Promise<OsmObject | undefined> | ||||
|     static async DownloadObjectAsync( | ||||
|         id: string, | ||||
|         maxCacheAgeInSecs?: number | ||||
|     ): Promise<OsmObject | undefined> { | ||||
|         const splitted = id.split("/") | ||||
|         const type = splitted[0] | ||||
|         const idN = Number(splitted[1]) | ||||
|  | @ -264,8 +282,10 @@ export abstract class OsmObject { | |||
|         return false | ||||
|     } | ||||
| 
 | ||||
|     private static constructPolygonFeatures(): Map<string, | ||||
|         { values: Set<string>; blacklist: boolean }> { | ||||
|     private static constructPolygonFeatures(): Map< | ||||
|         string, | ||||
|         { values: Set<string>; blacklist: boolean } | ||||
|     > { | ||||
|         const result = new Map<string, { values: Set<string>; blacklist: boolean }>() | ||||
|         for (const polygonFeature of polygon_features["default"] ?? polygon_features) { | ||||
|             const key = polygonFeature.key | ||||
|  |  | |||
|  | @ -557,7 +557,7 @@ export class TagUtils { | |||
| 
 | ||||
|         if (tag.indexOf("~~") >= 0) { | ||||
|             const split = Utils.SplitFirst(tag, "~~") | ||||
|             let keyRegex: RegExp; | ||||
|             let keyRegex: RegExp | ||||
|             if (split[0] === "*") { | ||||
|                 keyRegex = new RegExp(".+", "i") | ||||
|             } else { | ||||
|  | @ -569,10 +569,7 @@ export class TagUtils { | |||
|             } else { | ||||
|                 valueRegex = new RegExp("^(" + split[1] + ")$", "s") | ||||
|             } | ||||
|             return new RegexTag( | ||||
|                 keyRegex, | ||||
|                 valueRegex | ||||
|             ) | ||||
|             return new RegexTag(keyRegex, valueRegex) | ||||
|         } | ||||
|         const withRegex = TagUtils.parseRegexOperator(tag) | ||||
|         if (withRegex != null) { | ||||
|  | @ -627,7 +624,7 @@ export class TagUtils { | |||
|                 ) | ||||
|             } | ||||
|             if (split[1] === "") { | ||||
|                 return new RegexTag(split[0], /.+/si) | ||||
|                 return new RegexTag(split[0], /.+/is) | ||||
|             } | ||||
|             return new RegexTag(split[0], split[1], true) | ||||
|         } | ||||
|  |  | |||
|  | @ -1,4 +1,14 @@ | |||
| import {Concat, Conversion, DesugaringContext, DesugaringStep, Each, FirstOf, Fuse, On, SetDefault,} from "./Conversion" | ||||
| import { | ||||
|     Concat, | ||||
|     Conversion, | ||||
|     DesugaringContext, | ||||
|     DesugaringStep, | ||||
|     Each, | ||||
|     FirstOf, | ||||
|     Fuse, | ||||
|     On, | ||||
|     SetDefault, | ||||
| } from "./Conversion" | ||||
| import { LayerConfigJson } from "../Json/LayerConfigJson" | ||||
| import { TagRenderingConfigJson } from "../Json/TagRenderingConfigJson" | ||||
| import { Utils } from "../../../Utils" | ||||
|  | @ -8,27 +18,32 @@ import Translations from "../../../UI/i18n/Translations" | |||
| import { Translation } from "../../../UI/i18n/Translation" | ||||
| import * as tagrenderingconfigmeta from "../../../assets/tagrenderingconfigmeta.json" | ||||
| import { AddContextToTranslations } from "./AddContextToTranslations" | ||||
| import FilterConfigJson from "../Json/FilterConfigJson"; | ||||
| import FilterConfigJson from "../Json/FilterConfigJson" | ||||
| import * as predifined_filters from "../../../assets/layers/filters/filters.json" | ||||
| 
 | ||||
| class ExpandFilter extends DesugaringStep<LayerConfigJson> { | ||||
| 
 | ||||
| 
 | ||||
|     private static load_filters(): Map<string, FilterConfigJson> { | ||||
|         let filters = new Map<string, FilterConfigJson>(); | ||||
|         for (const filter of (<FilterConfigJson[]>predifined_filters.filter)) { | ||||
|         let filters = new Map<string, FilterConfigJson>() | ||||
|         for (const filter of <FilterConfigJson[]>predifined_filters.filter) { | ||||
|             filters.set(filter.id, filter) | ||||
|         } | ||||
|         return filters; | ||||
|         return filters | ||||
|     } | ||||
| 
 | ||||
|     private static readonly predefinedFilters = ExpandFilter.load_filters(); | ||||
|     private static readonly predefinedFilters = ExpandFilter.load_filters() | ||||
| 
 | ||||
|     constructor() { | ||||
|         super("Expands filters: replaces a shorthand by the value found in 'filters.json'", ["filter"], "ExpandFilter"); | ||||
|         super( | ||||
|             "Expands filters: replaces a shorthand by the value found in 'filters.json'", | ||||
|             ["filter"], | ||||
|             "ExpandFilter" | ||||
|         ) | ||||
|     } | ||||
| 
 | ||||
|     convert(json: LayerConfigJson, context: string): { result: LayerConfigJson; errors?: string[]; warnings?: string[]; information?: string[] } { | ||||
|     convert( | ||||
|         json: LayerConfigJson, | ||||
|         context: string | ||||
|     ): { result: LayerConfigJson; errors?: string[]; warnings?: string[]; information?: string[] } { | ||||
|         if (json.filter === undefined || json.filter === null) { | ||||
|             return { result: json } // Nothing to change here
 | ||||
|         } | ||||
|  | @ -39,7 +54,7 @@ class ExpandFilter extends DesugaringStep<LayerConfigJson>{ | |||
| 
 | ||||
|         const newFilters: FilterConfigJson[] = [] | ||||
|         const errors: string[] = [] | ||||
|         for (const filter of (<(FilterConfigJson|string)[]> json.filter)) { | ||||
|         for (const filter of <(FilterConfigJson | string)[]>json.filter) { | ||||
|             if (typeof filter !== "string") { | ||||
|                 newFilters.push(filter) | ||||
|                 continue | ||||
|  | @ -47,18 +62,29 @@ class ExpandFilter extends DesugaringStep<LayerConfigJson>{ | |||
|             // Search for the filter:
 | ||||
|             const found = ExpandFilter.predefinedFilters.get(filter) | ||||
|             if (found === undefined) { | ||||
|                 const suggestions = Utils.sortedByLevenshteinDistance(filter, Array.from(ExpandFilter.predefinedFilters.keys()), t => t) | ||||
|                 const err = context+".filter: while searching for predifined filter "+filter+": this filter is not found. Perhaps you meant one of: "+suggestions | ||||
|                 const suggestions = Utils.sortedByLevenshteinDistance( | ||||
|                     filter, | ||||
|                     Array.from(ExpandFilter.predefinedFilters.keys()), | ||||
|                     (t) => t | ||||
|                 ) | ||||
|                 const err = | ||||
|                     context + | ||||
|                     ".filter: while searching for predifined filter " + | ||||
|                     filter + | ||||
|                     ": this filter is not found. Perhaps you meant one of: " + | ||||
|                     suggestions | ||||
|                 errors.push(err) | ||||
|             } | ||||
|             newFilters.push(found) | ||||
|         } | ||||
|         return {result: { | ||||
|             ...json, filter: newFilters | ||||
|             }, errors}; | ||||
|         return { | ||||
|             result: { | ||||
|                 ...json, | ||||
|                 filter: newFilters, | ||||
|             }, | ||||
|             errors, | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| class ExpandTagRendering extends Conversion< | ||||
|  |  | |||
|  | @ -14,8 +14,8 @@ import {And} from "../../../Logic/Tags/And" | |||
| import Translations from "../../../UI/i18n/Translations" | ||||
| import Svg from "../../../Svg" | ||||
| import { QuestionableTagRenderingConfigJson } from "../Json/QuestionableTagRenderingConfigJson" | ||||
| import FilterConfigJson from "../Json/FilterConfigJson"; | ||||
| import DeleteConfig from "../DeleteConfig"; | ||||
| import FilterConfigJson from "../Json/FilterConfigJson" | ||||
| import DeleteConfig from "../DeleteConfig" | ||||
| 
 | ||||
| class ValidateLanguageCompleteness extends DesugaringStep<any> { | ||||
|     private readonly _languages: string[] | ||||
|  | @ -227,7 +227,6 @@ class ValidateTheme extends DesugaringStep<LayoutConfigJson> { | |||
| 
 | ||||
|         try { | ||||
|             if (this._isBuiltin) { | ||||
| 
 | ||||
|                 if (theme.id !== theme.id.toLowerCase()) { | ||||
|                     errors.push("Theme ids should be in lowercase, but it is " + theme.id) | ||||
|                 } | ||||
|  | @ -870,27 +869,40 @@ export class ValidateLayer extends DesugaringStep<LayerConfigJson> { | |||
|     } | ||||
| } | ||||
| 
 | ||||
| export class DetectDuplicateFilters extends DesugaringStep<{ layers: LayerConfigJson[], themes: LayoutConfigJson[]}> { | ||||
| 
 | ||||
| export class DetectDuplicateFilters extends DesugaringStep<{ | ||||
|     layers: LayerConfigJson[] | ||||
|     themes: LayoutConfigJson[] | ||||
| }> { | ||||
|     constructor() { | ||||
|         super("Tries to detect layers where a shared filter can be used (or where similar filters occur)", [], "DetectDuplicateFilters"); | ||||
|         super( | ||||
|             "Tries to detect layers where a shared filter can be used (or where similar filters occur)", | ||||
|             [], | ||||
|             "DetectDuplicateFilters" | ||||
|         ) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Add all filter options into 'perOsmTag' | ||||
|      */ | ||||
|     private addLayerFilters(layer: LayerConfigJson, perOsmTag: Map<string, { | ||||
|     private addLayerFilters( | ||||
|         layer: LayerConfigJson, | ||||
|         layout: LayoutConfigJson | undefined, | ||||
|         perOsmTag: Map< | ||||
|             string, | ||||
|             { | ||||
|                 layer: LayerConfigJson | ||||
|                 layout: LayoutConfigJson | undefined | ||||
|                 filter: FilterConfigJson | ||||
|     }[]>, layout?: LayoutConfigJson | undefined): void { | ||||
|             }[] | ||||
|         >, | ||||
|         layout?: LayoutConfigJson | undefined | ||||
|     ): void { | ||||
|         if (layer.filter === undefined || layer.filter === null) { | ||||
|             return; | ||||
|             return | ||||
|         } | ||||
|         if (layer.filter["sameAs"] !== undefined) { | ||||
|             return; | ||||
|             return | ||||
|         } | ||||
|         for (const filter of (<(string | FilterConfigJson) []>layer.filter)) { | ||||
|         for (const filter of <(string | FilterConfigJson)[]>layer.filter) { | ||||
|             if (typeof filter === "string") { | ||||
|                 continue | ||||
|             } | ||||
|  | @ -908,26 +920,36 @@ export class DetectDuplicateFilters extends DesugaringStep<{ layers: LayerConfig | |||
|                     perOsmTag.set(key, []) | ||||
|                 } | ||||
|                 perOsmTag.get(key).push({ | ||||
|                     layer, filter, layout | ||||
|                     layer, | ||||
|                     filter, | ||||
|                     layout, | ||||
|                 }) | ||||
|             } | ||||
| 
 | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     convert(json: { layers: LayerConfigJson[]; themes: LayoutConfigJson[] }, context: string): { result: { layers: LayerConfigJson[]; themes: LayoutConfigJson[] }; errors?: string[]; warnings?: string[]; information?: string[] } { | ||||
| 
 | ||||
|     convert( | ||||
|         json: { layers: LayerConfigJson[]; themes: LayoutConfigJson[] }, | ||||
|         context: string | ||||
|     ): { | ||||
|         result: { layers: LayerConfigJson[]; themes: LayoutConfigJson[] } | ||||
|         errors?: string[] | ||||
|         warnings?: string[] | ||||
|         information?: string[] | ||||
|     } { | ||||
|         const errors: string[] = [] | ||||
|         const warnings: string[] = [] | ||||
|         const information: string[] = [] | ||||
| 
 | ||||
|         const { layers, themes } = json | ||||
|         const perOsmTag = new Map<string, { | ||||
|             layer: LayerConfigJson, | ||||
|             layout: LayoutConfigJson | undefined, | ||||
|         const perOsmTag = new Map< | ||||
|             string, | ||||
|             { | ||||
|                 layer: LayerConfigJson | ||||
|                 layout: LayoutConfigJson | undefined | ||||
|                 filter: FilterConfigJson | ||||
|         }[]>() | ||||
|             }[] | ||||
|         >() | ||||
| 
 | ||||
|         for (const layer of layers) { | ||||
|             this.addLayerFilters(layer, perOsmTag) | ||||
|  | @ -948,12 +970,11 @@ export class DetectDuplicateFilters extends DesugaringStep<{ layers: LayerConfig | |||
|             } | ||||
|         } | ||||
| 
 | ||||
| 
 | ||||
|         // At this point, we have gathered all filters per tag - time to find duplicates
 | ||||
|         perOsmTag.forEach((value, key) => { | ||||
|             if (value.length <= 1) { | ||||
|                 // Seen this key just once, it is unique
 | ||||
|                 return; | ||||
|                 return | ||||
|             } | ||||
|             let msg = "Possible duplicate filter: " + key | ||||
|             for (const { filter, layer, layout } of value) { | ||||
|  | @ -970,8 +991,7 @@ export class DetectDuplicateFilters extends DesugaringStep<{ layers: LayerConfig | |||
|             result: json, | ||||
|             errors, | ||||
|             warnings, | ||||
|             information | ||||
|         }; | ||||
|             information, | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  |  | |||
|  | @ -267,7 +267,10 @@ export default class TagRenderingConfig { | |||
|                 if (this.freeform.key === "wikidata" && txt.indexOf("{wikipedia()") >= 0) { | ||||
|                     continue | ||||
|                 } | ||||
|                 if (this.freeform.type === "wikidata" && txt.indexOf(`{wikidata_label(${this.freeform.key})`) >= 0) { | ||||
|                 if ( | ||||
|                     this.freeform.type === "wikidata" && | ||||
|                     txt.indexOf(`{wikidata_label(${this.freeform.key})`) >= 0 | ||||
|                 ) { | ||||
|                     continue | ||||
|                 } | ||||
|                 throw `${context}: The rendering for language ${ln} does not contain the freeform key {${this.freeform.key}}. This is a bug, as this rendering should show exactly this freeform key!\nThe rendering is ${txt} ` | ||||
|  |  | |||
|  | @ -7,7 +7,6 @@ import Svg from "../../Svg" | |||
|  * The little 'translate'-icon next to every icon + some static helper functions | ||||
|  */ | ||||
| export default class LinkToWeblate extends VariableUiElement { | ||||
| 
 | ||||
|     constructor(context: string, availableTranslations: object) { | ||||
|         super( | ||||
|             Locale.language.map( | ||||
|  |  | |||
|  | @ -3,7 +3,7 @@ import Loc from "../../Models/Loc" | |||
| import BaseLayer from "../../Models/BaseLayer" | ||||
| import { UIEventSource } from "../../Logic/UIEventSource" | ||||
| import { BBox } from "../../Logic/BBox" | ||||
| import {deprecate} from "util"; | ||||
| import { deprecate } from "util" | ||||
| 
 | ||||
| export interface MinimapOptions { | ||||
|     background?: UIEventSource<BaseLayer> | ||||
|  |  | |||
|  | @ -114,15 +114,15 @@ export default class MinimapImplementation extends BaseUIElement implements Mini | |||
|      * @param format: image: give a base64 encoded png image; | ||||
|      * @constructor | ||||
|      */ | ||||
|     public async TakeScreenshot(): Promise<string> ; | ||||
|     public async TakeScreenshot(format: "image"): Promise<string> ; | ||||
|     public async TakeScreenshot(format: "blob"): Promise<Blob> ; | ||||
|     public async TakeScreenshot(format: "image" | "blob"): Promise<string | Blob> ; | ||||
|     public async TakeScreenshot(): Promise<string> | ||||
|     public async TakeScreenshot(format: "image"): Promise<string> | ||||
|     public async TakeScreenshot(format: "blob"): Promise<Blob> | ||||
|     public async TakeScreenshot(format: "image" | "blob"): Promise<string | Blob> | ||||
|     public async TakeScreenshot(format: "image" | "blob" = "image"): Promise<string | Blob> { | ||||
|         console.log("Taking a screenshot...") | ||||
|         const screenshotter = new SimpleMapScreenshoter() | ||||
|         screenshotter.addTo(this.leafletMap.data) | ||||
|         const result = <any> await screenshotter.takeScreen((<any> format) ?? "image") | ||||
|         const result = <any>await screenshotter.takeScreen(<any>format ?? "image") | ||||
|         if (format === "image" && typeof result === "string") { | ||||
|             return result | ||||
|         } | ||||
|  |  | |||
|  | @ -1,62 +1,73 @@ | |||
| import Combine from "../Base/Combine"; | ||||
| import {FlowPanelFactory, FlowStep} from "../ImportFlow/FlowStep"; | ||||
| import {ImmutableStore, Store, UIEventSource} from "../../Logic/UIEventSource"; | ||||
| import {InputElement} from "../Input/InputElement"; | ||||
| import {SvgToPdf, SvgToPdfOptions} from "../../Utils/svgToPdf"; | ||||
| import {FixedInputElement} from "../Input/FixedInputElement"; | ||||
| import {FixedUiElement} from "../Base/FixedUiElement"; | ||||
| import FileSelectorButton from "../Input/FileSelectorButton"; | ||||
| import InputElementMap from "../Input/InputElementMap"; | ||||
| import {RadioButton} from "../Input/RadioButton"; | ||||
| import {Utils} from "../../Utils"; | ||||
| import {VariableUiElement} from "../Base/VariableUIElement"; | ||||
| import Loading from "../Base/Loading"; | ||||
| import BaseUIElement from "../BaseUIElement"; | ||||
| import Img from "../Base/Img"; | ||||
| import Title from "../Base/Title"; | ||||
| import {CheckBox} from "../Input/Checkboxes"; | ||||
| import Minimap from "../Base/Minimap"; | ||||
| import SearchAndGo from "./SearchAndGo"; | ||||
| import Toggle from "../Input/Toggle"; | ||||
| import List from "../Base/List"; | ||||
| import LeftIndex from "../Base/LeftIndex"; | ||||
| import Constants from "../../Models/Constants"; | ||||
| import Toggleable from "../Base/Toggleable"; | ||||
| import Lazy from "../Base/Lazy"; | ||||
| import LinkToWeblate from "../Base/LinkToWeblate"; | ||||
| import Link from "../Base/Link"; | ||||
| import {SearchablePillsSelector} from "../Input/SearchableMappingsSelector"; | ||||
| import Combine from "../Base/Combine" | ||||
| import { FlowPanelFactory, FlowStep } from "../ImportFlow/FlowStep" | ||||
| import { ImmutableStore, Store, UIEventSource } from "../../Logic/UIEventSource" | ||||
| import { InputElement } from "../Input/InputElement" | ||||
| import { SvgToPdf, SvgToPdfOptions } from "../../Utils/svgToPdf" | ||||
| import { FixedInputElement } from "../Input/FixedInputElement" | ||||
| import { FixedUiElement } from "../Base/FixedUiElement" | ||||
| import FileSelectorButton from "../Input/FileSelectorButton" | ||||
| import InputElementMap from "../Input/InputElementMap" | ||||
| import { RadioButton } from "../Input/RadioButton" | ||||
| import { Utils } from "../../Utils" | ||||
| import { VariableUiElement } from "../Base/VariableUIElement" | ||||
| import Loading from "../Base/Loading" | ||||
| import BaseUIElement from "../BaseUIElement" | ||||
| import Img from "../Base/Img" | ||||
| import Title from "../Base/Title" | ||||
| import { CheckBox } from "../Input/Checkboxes" | ||||
| import Minimap from "../Base/Minimap" | ||||
| import SearchAndGo from "./SearchAndGo" | ||||
| import Toggle from "../Input/Toggle" | ||||
| import List from "../Base/List" | ||||
| import LeftIndex from "../Base/LeftIndex" | ||||
| import Constants from "../../Models/Constants" | ||||
| import Toggleable from "../Base/Toggleable" | ||||
| import Lazy from "../Base/Lazy" | ||||
| import LinkToWeblate from "../Base/LinkToWeblate" | ||||
| import Link from "../Base/Link" | ||||
| import { SearchablePillsSelector } from "../Input/SearchableMappingsSelector" | ||||
| import * as languages from "../../assets/language_translations.json" | ||||
| import {Translation} from "../i18n/Translation"; | ||||
| import { Translation } from "../i18n/Translation" | ||||
| 
 | ||||
| class SelectTemplate extends Combine implements FlowStep<{ title: string, pages: string[] }> { | ||||
|     readonly IsValid: Store<boolean>; | ||||
|     readonly Value: Store<{ title: string, pages: string[] }>; | ||||
| class SelectTemplate extends Combine implements FlowStep<{ title: string; pages: string[] }> { | ||||
|     readonly IsValid: Store<boolean> | ||||
|     readonly Value: Store<{ title: string; pages: string[] }> | ||||
| 
 | ||||
|     constructor() { | ||||
|         const elements: InputElement<{ templateName: string, pages: string[] }>[] = [] | ||||
|         const elements: InputElement<{ templateName: string; pages: string[] }>[] = [] | ||||
|         for (const templateName in SvgToPdf.templates) { | ||||
|             const template = SvgToPdf.templates[templateName] | ||||
|             elements.push(new FixedInputElement( | ||||
|                 new Combine([new FixedUiElement(templateName).SetClass("font-bold pr-2"), | ||||
|                     template.description | ||||
|                 ]) | ||||
|                 , new UIEventSource({templateName, pages: template.pages}))) | ||||
|             elements.push( | ||||
|                 new FixedInputElement( | ||||
|                     new Combine([ | ||||
|                         new FixedUiElement(templateName).SetClass("font-bold pr-2"), | ||||
|                         template.description, | ||||
|                     ]), | ||||
|                     new UIEventSource({ templateName, pages: template.pages }) | ||||
|                 ) | ||||
|             ) | ||||
|         } | ||||
| 
 | ||||
|         const file = new FileSelectorButton(new FixedUiElement("Select an svg image which acts as template"), { | ||||
|         const file = new FileSelectorButton( | ||||
|             new FixedUiElement("Select an svg image which acts as template"), | ||||
|             { | ||||
|                 acceptType: "image/svg+xml", | ||||
|             allowMultiple: true | ||||
|         }) | ||||
|         const fileMapped = new InputElementMap<FileList, { templateName: string, pages: string[], fromFile: true }>(file, (x0, x1) => x0 === x1, | ||||
|                 allowMultiple: true, | ||||
|             } | ||||
|         ) | ||||
|         const fileMapped = new InputElementMap< | ||||
|             FileList, | ||||
|             { templateName: string; pages: string[]; fromFile: true } | ||||
|         >( | ||||
|             file, | ||||
|             (x0, x1) => x0 === x1, | ||||
|             (filelist) => { | ||||
|                 if (filelist === undefined) { | ||||
|                     return undefined; | ||||
|                     return undefined | ||||
|                 } | ||||
|                 const pages = [] | ||||
|                 let templateName: string = undefined; | ||||
|                 let templateName: string = undefined | ||||
|                 for (const file of Array.from(filelist)) { | ||||
| 
 | ||||
|                     if (templateName == undefined) { | ||||
|                         templateName = file.name.substring(file.name.lastIndexOf("/") + 1) | ||||
|                         templateName = templateName.substring(0, templateName.lastIndexOf(".")) | ||||
|  | @ -67,40 +78,46 @@ class SelectTemplate extends Combine implements FlowStep<{ title: string, pages: | |||
|                 return { | ||||
|                     templateName, | ||||
|                     pages, | ||||
|                     fromFile: true | ||||
|                     fromFile: true, | ||||
|                 } | ||||
| 
 | ||||
|             }, | ||||
|             _ => undefined | ||||
|             (_) => undefined | ||||
|         ) | ||||
|         elements.push(fileMapped) | ||||
|         const radio = new RadioButton(elements, { selectFirstAsDefault: true }) | ||||
| 
 | ||||
|         const loaded: Store<{ success: { title: string, pages: string[] } } | { error: any }> = radio.GetValue().bind(template => { | ||||
|         const loaded: Store<{ success: { title: string; pages: string[] } } | { error: any }> = | ||||
|             radio.GetValue().bind((template) => { | ||||
|                 if (template === undefined) { | ||||
|                     return undefined | ||||
|                 } | ||||
|                 if (template["fromFile"]) { | ||||
|                 return UIEventSource.FromPromiseWithErr(Promise.all(template.pages).then(pages => ({ | ||||
|                     return UIEventSource.FromPromiseWithErr( | ||||
|                         Promise.all(template.pages).then((pages) => ({ | ||||
|                             title: template.templateName, | ||||
|                     pages | ||||
|                 }))) | ||||
|             } | ||||
|             const urls = template.pages.map(p => SelectTemplate.ToUrl(p)) | ||||
|             const dloadAll: Promise<{ title: string, pages: string[] }> = Promise.all(urls.map(url => Utils.download(url))).then(pages => ({ | ||||
|                             pages, | ||||
|                 title: template.templateName | ||||
|                         })) | ||||
|                     ) | ||||
|                 } | ||||
|                 const urls = template.pages.map((p) => SelectTemplate.ToUrl(p)) | ||||
|                 const dloadAll: Promise<{ title: string; pages: string[] }> = Promise.all( | ||||
|                     urls.map((url) => Utils.download(url)) | ||||
|                 ).then((pages) => ({ | ||||
|                     pages, | ||||
|                     title: template.templateName, | ||||
|                 })) | ||||
| 
 | ||||
|                 return UIEventSource.FromPromiseWithErr(dloadAll) | ||||
|             }) | ||||
|         const preview = new VariableUiElement( | ||||
|             loaded.map(pages => { | ||||
|             loaded.map((pages) => { | ||||
|                 if (pages === undefined) { | ||||
|                     return new Loading() | ||||
|                 } | ||||
|                 if (pages["error"] !== undefined) { | ||||
|                     return new FixedUiElement("Loading preview failed: " + pages["err"]).SetClass("alert") | ||||
|                     return new FixedUiElement("Loading preview failed: " + pages["err"]).SetClass( | ||||
|                         "alert" | ||||
|                     ) | ||||
|                 } | ||||
|                 const svgs = pages["success"].pages | ||||
|                 if (svgs.length === 0) { | ||||
|  | @ -108,22 +125,16 @@ class SelectTemplate extends Combine implements FlowStep<{ title: string, pages: | |||
|                 } | ||||
|                 const els: BaseUIElement[] = [] | ||||
|                 for (const pageSrc of svgs) { | ||||
|                     const el = new Img(pageSrc, true) | ||||
|                         .SetClass("w-96 m-2 border-black border-2") | ||||
|                     const el = new Img(pageSrc, true).SetClass("w-96 m-2 border-black border-2") | ||||
|                     els.push(el) | ||||
|                 } | ||||
|                 return new Combine(els).SetClass("flex border border-subtle rounded-xl"); | ||||
|                 return new Combine(els).SetClass("flex border border-subtle rounded-xl") | ||||
|             }) | ||||
|         ) | ||||
| 
 | ||||
|         super([ | ||||
|             new Title("Select template"), | ||||
|             radio, | ||||
|             new Title("Preview"), | ||||
|             preview | ||||
|         ]); | ||||
|         this.Value = loaded.map(l => l === undefined ? undefined : l["success"]) | ||||
|         this.IsValid = this.Value.map(v => v !== undefined) | ||||
|         super([new Title("Select template"), radio, new Title("Preview"), preview]) | ||||
|         this.Value = loaded.map((l) => (l === undefined ? undefined : l["success"])) | ||||
|         this.IsValid = this.Value.map((v) => v !== undefined) | ||||
|     } | ||||
| 
 | ||||
|     public static ToUrl(spec: string) { | ||||
|  | @ -134,63 +145,78 @@ class SelectTemplate extends Combine implements FlowStep<{ title: string, pages: | |||
|         path = path.substring(0, path.lastIndexOf("/")) | ||||
|         return window.location.protocol + "//" + window.location.host + path + "/" + spec | ||||
|     } | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| class SelectPdfOptions extends Combine implements FlowStep<{ title: string, pages: string[], options: SvgToPdfOptions }> { | ||||
|     readonly IsValid: Store<boolean>; | ||||
|     readonly Value: Store<{ title: string, pages: string[], options: SvgToPdfOptions }>; | ||||
| class SelectPdfOptions | ||||
|     extends Combine | ||||
|     implements FlowStep<{ title: string; pages: string[]; options: SvgToPdfOptions }> | ||||
| { | ||||
|     readonly IsValid: Store<boolean> | ||||
|     readonly Value: Store<{ title: string; pages: string[]; options: SvgToPdfOptions }> | ||||
| 
 | ||||
|     constructor(title: string, pages: string[], getFreeDiv: () => string) { | ||||
|         const dummy = new CheckBox("Don't add data to the map (to quickly preview the PDF)", false) | ||||
|         const overrideMapLocation = new CheckBox("Override map location: use a selected location instead of the location set in the template", false) | ||||
|         const overrideMapLocation = new CheckBox( | ||||
|             "Override map location: use a selected location instead of the location set in the template", | ||||
|             false | ||||
|         ) | ||||
|         const locationInput = Minimap.createMiniMap().SetClass("block w-full") | ||||
|         const searchField = new SearchAndGo({ leafletMap: locationInput.leafletMap }) | ||||
|         const selectLocation = | ||||
|             new Combine([ | ||||
|                 new Toggle(new Combine([new Title("Select override location"), searchField]).SetClass("flex"), undefined, overrideMapLocation.GetValue()), | ||||
|                 new Toggle(locationInput.SetStyle("height: 20rem"), undefined, overrideMapLocation.GetValue()).SetStyle("height: 20rem") | ||||
|             ]).SetClass("block").SetStyle("height: 25rem") | ||||
|         super([new Title("Select options"), | ||||
|             dummy, | ||||
|             overrideMapLocation, | ||||
|             selectLocation | ||||
|         ]); | ||||
|         this.Value = dummy.GetValue().map((disableMaps) => { | ||||
|         const selectLocation = new Combine([ | ||||
|             new Toggle( | ||||
|                 new Combine([new Title("Select override location"), searchField]).SetClass("flex"), | ||||
|                 undefined, | ||||
|                 overrideMapLocation.GetValue() | ||||
|             ), | ||||
|             new Toggle( | ||||
|                 locationInput.SetStyle("height: 20rem"), | ||||
|                 undefined, | ||||
|                 overrideMapLocation.GetValue() | ||||
|             ).SetStyle("height: 20rem"), | ||||
|         ]) | ||||
|             .SetClass("block") | ||||
|             .SetStyle("height: 25rem") | ||||
|         super([new Title("Select options"), dummy, overrideMapLocation, selectLocation]) | ||||
|         this.Value = dummy.GetValue().map( | ||||
|             (disableMaps) => { | ||||
|                 return { | ||||
|                     pages, | ||||
|                     title, | ||||
|                     options: <SvgToPdfOptions>{ | ||||
|                         disableMaps, | ||||
|                         getFreeDiv, | ||||
|                     overrideLocation: overrideMapLocation.GetValue().data ? locationInput.location.data : undefined | ||||
|                         overrideLocation: overrideMapLocation.GetValue().data | ||||
|                             ? locationInput.location.data | ||||
|                             : undefined, | ||||
|                     }, | ||||
|                 } | ||||
|             } | ||||
|         }, [overrideMapLocation.GetValue(), locationInput.location]) | ||||
|             }, | ||||
|             [overrideMapLocation.GetValue(), locationInput.location] | ||||
|         ) | ||||
|         this.IsValid = new ImmutableStore(true) | ||||
|     } | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| class PreparePdf extends Combine implements FlowStep<{ svgToPdf: SvgToPdf, languages: string[] }> { | ||||
|     readonly IsValid: Store<boolean>; | ||||
|     readonly Value: Store<{ svgToPdf: SvgToPdf, languages: string[] }>; | ||||
| class PreparePdf extends Combine implements FlowStep<{ svgToPdf: SvgToPdf; languages: string[] }> { | ||||
|     readonly IsValid: Store<boolean> | ||||
|     readonly Value: Store<{ svgToPdf: SvgToPdf; languages: string[] }> | ||||
| 
 | ||||
|     constructor(title: string, pages: string[], options: SvgToPdfOptions) { | ||||
|         const svgToPdf = new SvgToPdf(title, pages, options) | ||||
|         const languageOptions = [ | ||||
|             new FixedInputElement("Nederlands", "nl"), | ||||
|             new FixedInputElement("English", "en") | ||||
|             new FixedInputElement("English", "en"), | ||||
|         ] | ||||
|         const langs: string[] = Array.from(Object.keys(languages["default"] ?? languages)) | ||||
|         console.log("Available languages are:", langs) | ||||
|         const languageSelector = new SearchablePillsSelector( | ||||
|             langs.map(l => ({ | ||||
|             langs.map((l) => ({ | ||||
|                 show: new Translation(languages[l]), | ||||
|                 value: l, | ||||
|                 mainTerm: languages[l] | ||||
|             })), { | ||||
|                 mode: "select-many" | ||||
|                 mainTerm: languages[l], | ||||
|             })), | ||||
|             { | ||||
|                 mode: "select-many", | ||||
|             } | ||||
|         ) | ||||
| 
 | ||||
|  | @ -202,10 +228,11 @@ class PreparePdf extends Combine implements FlowStep<{ svgToPdf: SvgToPdf, langu | |||
|             new Toggle( | ||||
|                 new Loading("Preparing maps..."), | ||||
|                 undefined, | ||||
|                 isPrepared.map(p => p === undefined) | ||||
|             ) | ||||
|         ]); | ||||
|         this.Value = isPrepared.map(isPrepped => { | ||||
|                 isPrepared.map((p) => p === undefined) | ||||
|             ), | ||||
|         ]) | ||||
|         this.Value = isPrepared.map( | ||||
|             (isPrepped) => { | ||||
|                 if (isPrepped === undefined) { | ||||
|                     return undefined | ||||
|                 } | ||||
|  | @ -218,28 +245,36 @@ class PreparePdf extends Combine implements FlowStep<{ svgToPdf: SvgToPdf, langu | |||
|                     } | ||||
|                     return { svgToPdf, languages: langs } | ||||
|                 } | ||||
|             return undefined; | ||||
|         }, [languageSelector.GetValue()]) | ||||
|         this.IsValid = this.Value.map(v => v !== undefined) | ||||
|                 return undefined | ||||
|             }, | ||||
|             [languageSelector.GetValue()] | ||||
|         ) | ||||
|         this.IsValid = this.Value.map((v) => v !== undefined) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| class InspectStrings extends Toggle implements FlowStep<{ svgToPdf: SvgToPdf, languages: string[] }> { | ||||
|     readonly IsValid: Store<boolean>; | ||||
|     readonly Value: Store<{ svgToPdf: SvgToPdf; languages: string[] }>; | ||||
| class InspectStrings | ||||
|     extends Toggle | ||||
|     implements FlowStep<{ svgToPdf: SvgToPdf; languages: string[] }> | ||||
| { | ||||
|     readonly IsValid: Store<boolean> | ||||
|     readonly Value: Store<{ svgToPdf: SvgToPdf; languages: string[] }> | ||||
| 
 | ||||
|     constructor(svgToPdf: SvgToPdf, languages: string[]) { | ||||
|         const didLoadLanguages = UIEventSource.FromPromiseWithErr( | ||||
|             svgToPdf.PrepareLanguages(languages) | ||||
|         ).map((l) => l !== undefined && l["success"] !== undefined) | ||||
| 
 | ||||
|         const didLoadLanguages = UIEventSource.FromPromiseWithErr(svgToPdf.PrepareLanguages(languages)).map(l => l !== undefined && l["success"] !== undefined) | ||||
| 
 | ||||
|         super(new Combine([ | ||||
|         super( | ||||
|             new Combine([ | ||||
|                 new Title("Inspect translation strings"), | ||||
|                 ...languages.map(l => new Lazy(() => InspectStrings.createOverviewPanel(svgToPdf, l))) | ||||
|                 ...languages.map( | ||||
|                     (l) => new Lazy(() => InspectStrings.createOverviewPanel(svgToPdf, l)) | ||||
|                 ), | ||||
|             ]), | ||||
|             new Loading(), | ||||
|             didLoadLanguages | ||||
|         ); | ||||
|         ) | ||||
|         this.Value = new ImmutableStore({ svgToPdf, languages }) | ||||
|         this.IsValid = didLoadLanguages | ||||
|     } | ||||
|  | @ -259,65 +294,83 @@ class InspectStrings extends Toggle implements FlowStep<{ svgToPdf: SvgToPdf, la | |||
|             if (translated) { | ||||
|                 foundTranslations++ | ||||
|             } | ||||
|             const linkToWeblate = new Link(spec, LinkToWeblate.hrefToWeblate(language, spec), true).SetClass("font-bold link-underline") | ||||
|             elements.push(new Combine([ | ||||
|             const linkToWeblate = new Link( | ||||
|                 spec, | ||||
|                 LinkToWeblate.hrefToWeblate(language, spec), | ||||
|                 true | ||||
|             ).SetClass("font-bold link-underline") | ||||
|             elements.push( | ||||
|                 new Combine([ | ||||
|                     linkToWeblate, | ||||
|                     " ", | ||||
|                 translated ?? new FixedUiElement("No translation found!").SetClass("alert") | ||||
| 
 | ||||
|             ])) | ||||
|                     translated ?? new FixedUiElement("No translation found!").SetClass("alert"), | ||||
|                 ]) | ||||
|             ) | ||||
|         } | ||||
| 
 | ||||
|         return new Toggleable( | ||||
|             new Title("Translations for " + language), | ||||
|             new Combine([ | ||||
|                 `${foundTranslations}/${allKeys.length} of translations are found (${Math.floor(100 * foundTranslations / allKeys.length)}%)`, | ||||
|                 `${foundTranslations}/${allKeys.length} of translations are found (${Math.floor( | ||||
|                     (100 * foundTranslations) / allKeys.length | ||||
|                 )}%)`,
 | ||||
|                 "The following keys are used:", | ||||
|                 new List(elements) | ||||
|                 new List(elements), | ||||
|             ]), | ||||
|             {closeOnClick: false, height: "15rem"}) | ||||
|             { closeOnClick: false, height: "15rem" } | ||||
|         ) | ||||
|     } | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| class SavePdf extends Combine { | ||||
| 
 | ||||
|     constructor(svgToPdf: SvgToPdf, languages: string[]) { | ||||
| 
 | ||||
|         super([ | ||||
|             new Title("Generating your pdfs..."), | ||||
|             new List(languages.map(lng => new Toggle( | ||||
|             new List( | ||||
|                 languages.map( | ||||
|                     (lng) => | ||||
|                         new Toggle( | ||||
|                             lng + " is done!", | ||||
|                             new Loading("Creating pdf for " + lng), | ||||
|                 UIEventSource.FromPromiseWithErr(svgToPdf.ConvertSvg(lng).then(() => true)) | ||||
|                     .map(x => x !== undefined && x["success"] === true) | ||||
|             ))) | ||||
|         ]); | ||||
|                             UIEventSource.FromPromiseWithErr( | ||||
|                                 svgToPdf.ConvertSvg(lng).then(() => true) | ||||
|                             ).map((x) => x !== undefined && x["success"] === true) | ||||
|                         ) | ||||
|                 ) | ||||
|             ), | ||||
|         ]) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| export class PdfExportGui extends LeftIndex { | ||||
| 
 | ||||
| 
 | ||||
|     constructor(freeDivId: string) { | ||||
| 
 | ||||
|         let i = 0 | ||||
|         const createDiv = (): string => { | ||||
|             const div = document.createElement("div") | ||||
|             div.id = "freediv-" + (i++) | ||||
|             div.id = "freediv-" + i++ | ||||
|             document.getElementById(freeDivId).append(div) | ||||
|             return div.id | ||||
|         } | ||||
| 
 | ||||
|         Constants.defaultOverpassUrls.splice(0, 1) | ||||
|         const { flow, furthestStep, titles } = FlowPanelFactory.start( | ||||
|             new Title("Select template"), new SelectTemplate() | ||||
|         ).then(new Title("Select options"), ({title, pages}) => new SelectPdfOptions(title, pages, createDiv)) | ||||
|             .then("Generate maps...", ({title, pages, options}) => new PreparePdf(title, pages, options)) | ||||
|             .then("Inspect translations", (({svgToPdf, languages}) => new InspectStrings(svgToPdf, languages))) | ||||
|             new Title("Select template"), | ||||
|             new SelectTemplate() | ||||
|         ) | ||||
|             .then( | ||||
|                 new Title("Select options"), | ||||
|                 ({ title, pages }) => new SelectPdfOptions(title, pages, createDiv) | ||||
|             ) | ||||
|             .then( | ||||
|                 "Generate maps...", | ||||
|                 ({ title, pages, options }) => new PreparePdf(title, pages, options) | ||||
|             ) | ||||
|             .then( | ||||
|                 "Inspect translations", | ||||
|                 ({ svgToPdf, languages }) => new InspectStrings(svgToPdf, languages) | ||||
|             ) | ||||
|             .finish("Generating...", ({ svgToPdf, languages }) => new SavePdf(svgToPdf, languages)) | ||||
| 
 | ||||
| 
 | ||||
|         const toc = new List( | ||||
|             titles.map( | ||||
|                 (title, i) => | ||||
|  | @ -338,9 +391,7 @@ export class PdfExportGui extends LeftIndex { | |||
|             true | ||||
|         ) | ||||
| 
 | ||||
|         const leftContents: BaseUIElement[] = [ | ||||
|             toc | ||||
|         ].map((el) => el?.SetClass("pl-4")) | ||||
|         const leftContents: BaseUIElement[] = [toc].map((el) => el?.SetClass("pl-4")) | ||||
| 
 | ||||
|         super(leftContents, flow) | ||||
|     } | ||||
|  |  | |||
|  | @ -26,7 +26,7 @@ import BaseLayer from "../../Models/BaseLayer" | |||
| import Loading from "../Base/Loading" | ||||
| import Hash from "../../Logic/Web/Hash" | ||||
| import { GlobalFilter } from "../../Logic/State/MapState" | ||||
| import {WayId} from "../../Models/OsmFeature"; | ||||
| import { WayId } from "../../Models/OsmFeature" | ||||
| 
 | ||||
| /* | ||||
|  * The SimpleAddUI is a single panel, which can have multiple states: | ||||
|  |  | |||
|  | @ -3,10 +3,10 @@ import { UIEventSource } from "../../Logic/UIEventSource" | |||
| import { Utils } from "../../Utils" | ||||
| import BaseUIElement from "../BaseUIElement" | ||||
| import InputElementMap from "./InputElementMap" | ||||
| import Translations from "../i18n/Translations"; | ||||
| import Translations from "../i18n/Translations" | ||||
| 
 | ||||
| export class CheckBox extends InputElementMap<number[], boolean> { | ||||
|     constructor(el: (BaseUIElement | string), defaultValue?: boolean) { | ||||
|     constructor(el: BaseUIElement | string, defaultValue?: boolean) { | ||||
|         super( | ||||
|             new CheckBoxes([Translations.W(el)]), | ||||
|             (x0, x1) => x0 === x1, | ||||
|  |  | |||
|  | @ -15,24 +15,26 @@ import ShowDataLayer from "../ShowDataLayer/ShowDataLayer" | |||
| import BaseUIElement from "../BaseUIElement" | ||||
| import Toggle from "./Toggle" | ||||
| import * as matchpoint from "../../assets/layers/matchpoint/matchpoint.json" | ||||
| import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig"; | ||||
| import FilteredLayer from "../../Models/FilteredLayer"; | ||||
| import {ElementStorage} from "../../Logic/ElementStorage"; | ||||
| import AvailableBaseLayers from "../../Logic/Actors/AvailableBaseLayers"; | ||||
| import {RelationId, WayId} from "../../Models/OsmFeature"; | ||||
| import {Feature, LineString, Polygon} from "geojson"; | ||||
| import {OsmObject, OsmWay} from "../../Logic/Osm/OsmObject"; | ||||
| import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig" | ||||
| import FilteredLayer from "../../Models/FilteredLayer" | ||||
| import { ElementStorage } from "../../Logic/ElementStorage" | ||||
| import AvailableBaseLayers from "../../Logic/Actors/AvailableBaseLayers" | ||||
| import { RelationId, WayId } from "../../Models/OsmFeature" | ||||
| import { Feature, LineString, Polygon } from "geojson" | ||||
| import { OsmObject, OsmWay } from "../../Logic/Osm/OsmObject" | ||||
| 
 | ||||
| export default class LocationInput | ||||
|     extends BaseUIElement | ||||
|     implements ReadonlyInputElement<Loc>, MinimapObj { | ||||
|     implements ReadonlyInputElement<Loc>, MinimapObj | ||||
| { | ||||
|     private static readonly matchLayer = new LayerConfig( | ||||
|         matchpoint, | ||||
|         "LocationInput.matchpoint", | ||||
|         true | ||||
|     ) | ||||
| 
 | ||||
|     public readonly snappedOnto: UIEventSource<Feature & { properties : { id : WayId} }> = new UIEventSource(undefined) | ||||
|     public readonly snappedOnto: UIEventSource<Feature & { properties: { id: WayId } }> = | ||||
|         new UIEventSource(undefined) | ||||
|     public readonly _matching_layer: LayerConfig | ||||
|     public readonly leafletMap: UIEventSource<any> | ||||
|     public readonly bounds | ||||
|  | @ -43,7 +45,9 @@ export default class LocationInput | |||
|      * The features to which the input should be snapped | ||||
|      * @private | ||||
|      */ | ||||
|     private readonly _snapTo: Store< (Feature<LineString | Polygon> & {properties: {id : WayId}})[]> | ||||
|     private readonly _snapTo: Store< | ||||
|         (Feature<LineString | Polygon> & { properties: { id: WayId } })[] | ||||
|     > | ||||
|     /** | ||||
|      * The features to which the input should be snapped without cleanup of relations and memberships | ||||
|      * Used for rendering | ||||
|  | @ -59,10 +63,10 @@ export default class LocationInput | |||
|     private readonly clickLocation: UIEventSource<Loc> | ||||
|     private readonly _minZoom: number | ||||
|     private readonly _state: { | ||||
|         readonly filteredLayers: Store<FilteredLayer[]>; | ||||
|         readonly backgroundLayer: UIEventSource<BaseLayer>; | ||||
|         readonly layoutToUse: LayoutConfig; | ||||
|         readonly selectedElement: UIEventSource<any>; | ||||
|         readonly filteredLayers: Store<FilteredLayer[]> | ||||
|         readonly backgroundLayer: UIEventSource<BaseLayer> | ||||
|         readonly layoutToUse: LayoutConfig | ||||
|         readonly selectedElement: UIEventSource<any> | ||||
|         readonly allElements: ElementStorage | ||||
|     } | ||||
| 
 | ||||
|  | @ -74,8 +78,12 @@ export default class LocationInput | |||
|      * | ||||
|      * @private | ||||
|      */ | ||||
|     private static async prepareSnapOnto(features: Feature[]): Promise<(Feature<LineString | Polygon> & {properties : {id: WayId}})[]> { | ||||
|         const linesAndPolygon : Feature<LineString | Polygon>[] = <any> features.filter(f => f.geometry.type !== "Point") | ||||
|     private static async prepareSnapOnto( | ||||
|         features: Feature[] | ||||
|     ): Promise<(Feature<LineString | Polygon> & { properties: { id: WayId } })[]> { | ||||
|         const linesAndPolygon: Feature<LineString | Polygon>[] = <any>( | ||||
|             features.filter((f) => f.geometry.type !== "Point") | ||||
|         ) | ||||
|         // Clean the features: multipolygons are split into their it's members
 | ||||
|         const linestrings: (Feature<LineString | Polygon> & { properties: { id: WayId } })[] = [] | ||||
|         for (const feature of linesAndPolygon) { | ||||
|  | @ -87,14 +95,18 @@ export default class LocationInput | |||
| 
 | ||||
|             // We have a multipolygon, thus: a relation
 | ||||
|             // Download the members
 | ||||
|             const relation = await OsmObject.DownloadObjectAsync(<RelationId> feature.properties.id, 60 * 60) | ||||
|             const members: OsmWay[] = await Promise.all(relation.members | ||||
|                 .filter(m => m.type === "way") | ||||
|                 .map(m => OsmObject.DownloadObjectAsync(<WayId> ("way/"+m.ref), 60 * 60))) | ||||
|             linestrings.push(...members.map(m => m.asGeoJson())) | ||||
|             const relation = await OsmObject.DownloadObjectAsync( | ||||
|                 <RelationId>feature.properties.id, | ||||
|                 60 * 60 | ||||
|             ) | ||||
|             const members: OsmWay[] = await Promise.all( | ||||
|                 relation.members | ||||
|                     .filter((m) => m.type === "way") | ||||
|                     .map((m) => OsmObject.DownloadObjectAsync(<WayId>("way/" + m.ref), 60 * 60)) | ||||
|             ) | ||||
|             linestrings.push(...members.map((m) => m.asGeoJson())) | ||||
|         } | ||||
|         return linestrings | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     constructor(options?: { | ||||
|  | @ -107,19 +119,31 @@ export default class LocationInput | |||
|         centerLocation?: UIEventSource<Loc> | ||||
|         bounds?: UIEventSource<BBox> | ||||
|         state?: { | ||||
|             readonly filteredLayers: Store<FilteredLayer[]>; | ||||
|             readonly backgroundLayer: UIEventSource<BaseLayer>; | ||||
|             readonly layoutToUse: LayoutConfig; | ||||
|             readonly selectedElement: UIEventSource<any>; | ||||
|             readonly filteredLayers: Store<FilteredLayer[]> | ||||
|             readonly backgroundLayer: UIEventSource<BaseLayer> | ||||
|             readonly layoutToUse: LayoutConfig | ||||
|             readonly selectedElement: UIEventSource<any> | ||||
|             readonly allElements: ElementStorage | ||||
|         } | ||||
|     }) { | ||||
|         super() | ||||
|         this._snapToRaw = options?.snapTo?.map(feats => feats.filter(f => f.feature.geometry.type !== "Point")) | ||||
|         this._snapTo = options?.snapTo?.bind((features) => UIEventSource.FromPromise(LocationInput.prepareSnapOnto(features.map(f => f.feature))))?.map(f => f ?? []) | ||||
|         this._snapToRaw = options?.snapTo?.map((feats) => | ||||
|             feats.filter((f) => f.feature.geometry.type !== "Point") | ||||
|         ) | ||||
|         this._snapTo = options?.snapTo | ||||
|             ?.bind((features) => | ||||
|                 UIEventSource.FromPromise( | ||||
|                     LocationInput.prepareSnapOnto(features.map((f) => f.feature)) | ||||
|                 ) | ||||
|             ) | ||||
|             ?.map((f) => f ?? []) | ||||
|         this._maxSnapDistance = options?.maxSnapDistance | ||||
|         this._centerLocation = options?.centerLocation ?? new UIEventSource<Loc>({ | ||||
|             lat: 0, lon: 0, zoom: 0 | ||||
|         this._centerLocation = | ||||
|             options?.centerLocation ?? | ||||
|             new UIEventSource<Loc>({ | ||||
|                 lat: 0, | ||||
|                 lon: 0, | ||||
|                 zoom: 0, | ||||
|             }) | ||||
|         this._snappedPointTags = options?.snappedPointTags | ||||
|         this._bounds = options?.bounds | ||||
|  | @ -152,11 +176,11 @@ export default class LocationInput | |||
|                         return undefined | ||||
|                     } | ||||
| 
 | ||||
| 
 | ||||
|                     // We reproject the location onto every 'snap-to-feature' and select the closest
 | ||||
| 
 | ||||
|                     let min = undefined | ||||
|                     let matchedWay: Feature<LineString | Polygon> & {properties : {id : WayId}} = undefined | ||||
|                     let matchedWay: Feature<LineString | Polygon> & { properties: { id: WayId } } = | ||||
|                         undefined | ||||
|                     for (const feature of self._snapTo.data ?? []) { | ||||
|                         try { | ||||
|                             const nearestPointOnLine = GeoOperations.nearestPoint(feature, [ | ||||
|  | @ -199,8 +223,6 @@ export default class LocationInput | |||
|                     if (matchedWay.properties.id.startsWith("relation/")) { | ||||
|                         // We matched a relation instead of a way
 | ||||
|                         console.log("Snapping onto a relation. The relation is", matchedWay) | ||||
| 
 | ||||
| 
 | ||||
|                     } | ||||
|                     self.snappedOnto.setData(<any>matchedWay) | ||||
|                     return min | ||||
|  | @ -217,7 +239,10 @@ export default class LocationInput | |||
|                 } | ||||
|             }) | ||||
|         } | ||||
|         this.mapBackground = options?.mapBackground ?? this._state?.backgroundLayer ?? new UIEventSource<BaseLayer>(AvailableBaseLayers.osmCarto) | ||||
|         this.mapBackground = | ||||
|             options?.mapBackground ?? | ||||
|             this._state?.backgroundLayer ?? | ||||
|             new UIEventSource<BaseLayer>(AvailableBaseLayers.osmCarto) | ||||
|         this.SetClass("block h-full") | ||||
| 
 | ||||
|         this.clickLocation = new UIEventSource<Loc>(undefined) | ||||
|  | @ -335,9 +360,9 @@ export default class LocationInput | |||
|         } | ||||
|     } | ||||
| 
 | ||||
|     TakeScreenshot(format: "image"): Promise<string>; | ||||
|     TakeScreenshot(format: "blob"): Promise<Blob>; | ||||
|     TakeScreenshot(format: "image" | "blob"): Promise<string | Blob>; | ||||
|     TakeScreenshot(format: "image"): Promise<string> | ||||
|     TakeScreenshot(format: "blob"): Promise<Blob> | ||||
|     TakeScreenshot(format: "image" | "blob"): Promise<string | Blob> | ||||
|     TakeScreenshot(format: "image" | "blob"): Promise<string | Blob> { | ||||
|         return this.map.TakeScreenshot(format) | ||||
|     } | ||||
|  |  | |||
|  | @ -18,7 +18,7 @@ import Title from "../Base/Title" | |||
| import { GlobalFilter } from "../../Logic/State/MapState" | ||||
| import { VariableUiElement } from "../Base/VariableUIElement" | ||||
| import { Tag } from "../../Logic/Tags/Tag" | ||||
| import {WayId} from "../../Models/OsmFeature"; | ||||
| import { WayId } from "../../Models/OsmFeature" | ||||
| 
 | ||||
| export default class ConfirmLocationOfPoint extends Combine { | ||||
|     constructor( | ||||
|  | @ -76,7 +76,7 @@ export default class ConfirmLocationOfPoint extends Combine { | |||
|                 snappedPointTags: tags, | ||||
|                 maxSnapDistance: preset.preciseInput.maxSnapDistance, | ||||
|                 bounds: mapBounds, | ||||
|                 state: <any> state | ||||
|                 state: <any>state, | ||||
|             }) | ||||
|             preciseInput.installBounds(preset.boundsFactor ?? 0.25, true) | ||||
|             preciseInput | ||||
|  |  | |||
|  | @ -22,7 +22,7 @@ import Title from "../Base/Title" | |||
| import { SubstitutedTranslation } from "../SubstitutedTranslation" | ||||
| import FeaturePipelineState from "../../Logic/State/FeaturePipelineState" | ||||
| import TagRenderingQuestion from "./TagRenderingQuestion" | ||||
| import {OsmId} from "../../Models/OsmFeature"; | ||||
| import { OsmId } from "../../Models/OsmFeature" | ||||
| 
 | ||||
| export default class DeleteWizard extends Toggle { | ||||
|     /** | ||||
|  |  | |||
|  | @ -248,8 +248,7 @@ export default class FeatureInfoBox extends ScrollableFullScreen { | |||
|         ) | ||||
| 
 | ||||
|         editElements.push( | ||||
|             Toggle.If(state.featureSwitchIsDebugging, | ||||
|                 () => { | ||||
|             Toggle.If(state.featureSwitchIsDebugging, () => { | ||||
|                 const config_all_tags: TagRenderingConfig = new TagRenderingConfig( | ||||
|                     { render: "{all_tags()}" }, | ||||
|                     "" | ||||
|  | @ -269,8 +268,7 @@ export default class FeatureInfoBox extends ScrollableFullScreen { | |||
|                     new TagRenderingAnswer(tags, config_id, state), | ||||
|                     "This is layer " + layerConfig.id, | ||||
|                 ]) | ||||
|                 } | ||||
|                 ) | ||||
|             }) | ||||
|         ) | ||||
| 
 | ||||
|         return new Combine(editElements).SetClass("flex flex-col") | ||||
|  |  | |||
|  | @ -145,7 +145,7 @@ export default class MoveWizard extends Toggle { | |||
|                 minZoom: reason.minZoom, | ||||
|                 centerLocation: loc, | ||||
|                 mapBackground: new UIEventSource<BaseLayer>(preferredBackground), // We detach the layer
 | ||||
|                 state: <any> state | ||||
|                 state: <any>state, | ||||
|             }) | ||||
| 
 | ||||
|             if (reason.lockBounds) { | ||||
|  |  | |||
|  | @ -304,7 +304,7 @@ export default class TagRenderingQuestion extends Combine { | |||
|             const patchedMapping = <Mapping>{ | ||||
|                 ...mapping, | ||||
|                 iconClass: mapping.iconClass ?? `small-height`, | ||||
|                 icon: mapping.icon ?? (addIcons ? "./assets/svg/none.svg" : undefined) | ||||
|                 icon: mapping.icon ?? (addIcons ? "./assets/svg/none.svg" : undefined), | ||||
|             } | ||||
|             const fancy = TagRenderingQuestion.GenerateMappingContent( | ||||
|                 patchedMapping, | ||||
|  |  | |||
|  | @ -10,12 +10,15 @@ import * as clusterstyle from "../../assets/layers/cluster_style/cluster_style.j | |||
| export default class ShowTileInfo { | ||||
|     public static readonly styling = new LayerConfig(clusterstyle, "ShowTileInfo", true) | ||||
| 
 | ||||
|     constructor(options: { | ||||
|     constructor( | ||||
|         options: { | ||||
|             source: FeatureSource & Tiled | ||||
|             leafletMap: UIEventSource<any> | ||||
|             layer?: LayerConfig | ||||
|             doShowLayer?: UIEventSource<boolean> | ||||
|     }, state) { | ||||
|         }, | ||||
|         state | ||||
|     ) { | ||||
|         const source = options.source | ||||
|         const metaFeature: Store<{ feature; freshness: Date }[]> = source.features.map( | ||||
|             (features) => { | ||||
|  | @ -55,7 +58,7 @@ export default class ShowTileInfo { | |||
|             features: new StaticFeatureSource(metaFeature), | ||||
|             leafletMap: options.leafletMap, | ||||
|             doShowLayer: options.doShowLayer, | ||||
|             state | ||||
|             state, | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -41,7 +41,10 @@ export default class Translations { | |||
|      * translation.textFor("nl") // => "Nederlands"
 | ||||
|      * | ||||
|      */ | ||||
|     static T(t: string | undefined | null | Translation | TypedTranslation<object>, context = undefined): TypedTranslation<object> { | ||||
|     static T( | ||||
|         t: string | undefined | null | Translation | TypedTranslation<object>, | ||||
|         context = undefined | ||||
|     ): TypedTranslation<object> { | ||||
|         if (t === undefined || t === null) { | ||||
|             return undefined | ||||
|         } | ||||
|  |  | |||
|  | @ -1,11 +1,11 @@ | |||
| import MinimapImplementation from "./UI/Base/MinimapImplementation"; | ||||
| import MinimapImplementation from "./UI/Base/MinimapImplementation" | ||||
| 
 | ||||
| import { Utils } from "./Utils" | ||||
| import AllThemesGui from "./UI/AllThemesGui" | ||||
| import { QueryParameters } from "./Logic/Web/QueryParameters" | ||||
| import StatisticsGUI from "./UI/StatisticsGUI" | ||||
| import { FixedUiElement } from "./UI/Base/FixedUiElement" | ||||
| import {PdfExportGui} from "./UI/BigComponents/PdfExportGui"; | ||||
| import { PdfExportGui } from "./UI/BigComponents/PdfExportGui" | ||||
| 
 | ||||
| const layout = QueryParameters.GetQueryParameter("layout", undefined).data ?? "" | ||||
| const customLayout = QueryParameters.GetQueryParameter("userlayout", undefined).data ?? "" | ||||
|  |  | |||
|  | @ -108,10 +108,11 @@ function main() { | |||
|             { | ||||
|                 id: "tactile_writing-braille", | ||||
|                 // @ts-ignore
 | ||||
|                 description: "Enables to pick *multiple* 'tactile_writing:braille=<lng>' within the mappings", | ||||
|                 description: | ||||
|                     "Enables to pick *multiple* 'tactile_writing:braille=<lng>' within the mappings", | ||||
|                 multiAnswer: true, | ||||
|                 mappings: brailemappings, | ||||
|             } | ||||
|             }, | ||||
|         ], | ||||
|     } | ||||
|     const dir = "./assets/layers/wikidata/" | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue