| 
									
										
										
										
											2023-11-29 14:29:11 +01:00
										 |  |  | import Script from "./Script" | 
					
						
							|  |  |  | import { LayerConfigJson } from "../src/Models/ThemeConfig/Json/LayerConfigJson" | 
					
						
							| 
									
										
										
										
											2023-12-05 00:04:29 +01:00
										 |  |  | import { existsSync, readFileSync, writeFileSync } from "fs" | 
					
						
							| 
									
										
										
										
											2023-11-29 14:29:11 +01:00
										 |  |  | import { AllSharedLayers } from "../src/Customizations/AllSharedLayers" | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  | import { AllKnownLayoutsLazy } from "../src/Customizations/AllKnownLayouts" | 
					
						
							|  |  |  | import { Utils } from "../src/Utils" | 
					
						
							|  |  |  | import { AddEditingElements } from "../src/Models/ThemeConfig/Conversion/PrepareLayer" | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |     MappingConfigJson, | 
					
						
							|  |  |  |     QuestionableTagRenderingConfigJson, | 
					
						
							|  |  |  | } from "../src/Models/ThemeConfig/Json/QuestionableTagRenderingConfigJson" | 
					
						
							|  |  |  | import { TagConfigJson } from "../src/Models/ThemeConfig/Json/TagConfigJson" | 
					
						
							|  |  |  | import { TagUtils } from "../src/Logic/Tags/TagUtils" | 
					
						
							|  |  |  | import { TagRenderingConfigJson } from "../src/Models/ThemeConfig/Json/TagRenderingConfigJson" | 
					
						
							|  |  |  | import { Translatable } from "../src/Models/ThemeConfig/Json/Translatable" | 
					
						
							| 
									
										
										
										
											2023-12-06 12:12:53 +01:00
										 |  |  | import Icon from "../src/UI/Map/Icon.svelte" | 
					
						
							| 
									
										
										
										
											2023-12-04 16:10:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  | export class GenerateFavouritesLayer extends Script { | 
					
						
							|  |  |  |     private readonly layers: LayerConfigJson[] = [] | 
					
						
							| 
									
										
										
										
											2023-11-29 14:29:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     constructor() { | 
					
						
							|  |  |  |         super("Prepares the 'favourites'-layer") | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |         const allThemes = new AllKnownLayoutsLazy(false).values() | 
					
						
							|  |  |  |         for (const theme of allThemes) { | 
					
						
							|  |  |  |             if (theme.hideFromOverview) { | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             for (const layer of theme.layers) { | 
					
						
							|  |  |  |                 if (!layer.source) { | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (layer.source.geojsonSource) { | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 const layerConfig = AllSharedLayers.getSharedLayersConfigs().get(layer.id) | 
					
						
							|  |  |  |                 if (!layerConfig) { | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 this.layers.push(layerConfig) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-04 16:10:05 +01:00
										 |  |  |     private sortMappings(mappings: MappingConfigJson[]): MappingConfigJson[] { | 
					
						
							|  |  |  |         const sortedMappings: MappingConfigJson[] = [...mappings] | 
					
						
							|  |  |  |         sortedMappings.sort((a, b) => { | 
					
						
							|  |  |  |             const aTag = TagUtils.Tag(a.if) | 
					
						
							|  |  |  |             const bTag = TagUtils.Tag(b.if) | 
					
						
							|  |  |  |             const aPop = TagUtils.GetPopularity(aTag) | 
					
						
							|  |  |  |             const bPop = TagUtils.GetPopularity(bTag) | 
					
						
							|  |  |  |             return aPop - bPop | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return sortedMappings | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |     private addTagRenderings(proto: LayerConfigJson) { | 
					
						
							|  |  |  |         const blacklistedIds = new Set([ | 
					
						
							|  |  |  |             "images", | 
					
						
							|  |  |  |             "questions", | 
					
						
							|  |  |  |             "mapillary", | 
					
						
							|  |  |  |             "leftover-questions", | 
					
						
							|  |  |  |             "last_edit", | 
					
						
							|  |  |  |             "minimap", | 
					
						
							|  |  |  |             "move-button", | 
					
						
							|  |  |  |             "delete-button", | 
					
						
							|  |  |  |             "all-tags", | 
					
						
							| 
									
										
										
										
											2023-12-02 03:18:53 +01:00
										 |  |  |             "all_tags", | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |             ...AddEditingElements.addedElements, | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-02 03:18:53 +01:00
										 |  |  |         const generatedTagRenderings: (string | QuestionableTagRenderingConfigJson)[] = [] | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |         const trPerId = new Map< | 
					
						
							|  |  |  |             string, | 
					
						
							|  |  |  |             { conditions: TagConfigJson[]; tr: QuestionableTagRenderingConfigJson } | 
					
						
							|  |  |  |         >() | 
					
						
							|  |  |  |         for (const layerConfig of this.layers) { | 
					
						
							|  |  |  |             if (!layerConfig.tagRenderings) { | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             for (const tagRendering of layerConfig.tagRenderings) { | 
					
						
							|  |  |  |                 if (typeof tagRendering === "string") { | 
					
						
							|  |  |  |                     if (blacklistedIds.has(tagRendering)) { | 
					
						
							|  |  |  |                         continue | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2023-12-02 03:18:53 +01:00
										 |  |  |                     generatedTagRenderings.push(tagRendering) | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |                     blacklistedIds.add(tagRendering) | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (tagRendering["builtin"]) { | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 const id = tagRendering.id | 
					
						
							|  |  |  |                 if (blacklistedIds.has(id)) { | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (trPerId.has(id)) { | 
					
						
							|  |  |  |                     const old = trPerId.get(id).tr | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     // We need to figure out if this was a 'recycled' tag rendering or just happens to have the same id
 | 
					
						
							|  |  |  |                     function isSame(fieldName: string) { | 
					
						
							|  |  |  |                         return old[fieldName]?.["en"] === tagRendering[fieldName]?.["en"] | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     const sameQuestion = isSame("question") && isSame("render") | 
					
						
							|  |  |  |                     if (!sameQuestion) { | 
					
						
							|  |  |  |                         const newTr = <QuestionableTagRenderingConfigJson>Utils.Clone(tagRendering) | 
					
						
							|  |  |  |                         newTr.id = layerConfig.id + "_" + newTr.id | 
					
						
							|  |  |  |                         if (blacklistedIds.has(newTr.id)) { | 
					
						
							|  |  |  |                             continue | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         newTr.condition = { | 
					
						
							| 
									
										
										
										
											2023-12-04 16:10:05 +01:00
										 |  |  |                             and: Utils.NoNull([newTr.condition, layerConfig.source["osmTags"]]), | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |                         } | 
					
						
							| 
									
										
										
										
											2023-12-02 03:18:53 +01:00
										 |  |  |                         generatedTagRenderings.push(newTr) | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |                         blacklistedIds.add(newTr.id) | 
					
						
							|  |  |  |                         continue | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (!trPerId.has(id)) { | 
					
						
							|  |  |  |                     const newTr = <QuestionableTagRenderingConfigJson>Utils.Clone(tagRendering) | 
					
						
							| 
									
										
										
										
											2023-12-02 03:18:53 +01:00
										 |  |  |                     generatedTagRenderings.push(newTr) | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |                     trPerId.set(newTr.id, { tr: newTr, conditions: [] }) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 const conditions = trPerId.get(id).conditions | 
					
						
							|  |  |  |                 if (tagRendering["condition"]) { | 
					
						
							|  |  |  |                     conditions.push({ | 
					
						
							|  |  |  |                         and: [tagRendering["condition"], layerConfig.source["osmTags"]], | 
					
						
							|  |  |  |                     }) | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     conditions.push(layerConfig.source["osmTags"]) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (const { tr, conditions } of Array.from(trPerId.values())) { | 
					
						
							|  |  |  |             const optimized = TagUtils.optimzeJson({ or: conditions }) | 
					
						
							|  |  |  |             if (optimized === true) { | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (optimized === false) { | 
					
						
							|  |  |  |                 throw "Optimized into 'false', this is weird..." | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             tr.condition = optimized | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const allTags: QuestionableTagRenderingConfigJson = { | 
					
						
							|  |  |  |             id: "all-tags", | 
					
						
							|  |  |  |             render: { "*": "{all_tags()}" }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             metacondition: { | 
					
						
							|  |  |  |                 or: [ | 
					
						
							|  |  |  |                     "__featureSwitchIsDebugging=true", | 
					
						
							|  |  |  |                     "mapcomplete-show_tags=full", | 
					
						
							|  |  |  |                     "mapcomplete-show_debug=yes", | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         proto.tagRenderings = [ | 
					
						
							| 
									
										
										
										
											2023-12-02 03:18:53 +01:00
										 |  |  |             "images", | 
					
						
							|  |  |  |             ...generatedTagRenderings, | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |             ...proto.tagRenderings, | 
					
						
							|  |  |  |             "questions", | 
					
						
							|  |  |  |             allTags, | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-02 03:18:53 +01:00
										 |  |  |     private addTitleIcons(proto: LayerConfigJson) { | 
					
						
							| 
									
										
										
										
											2023-12-06 12:12:53 +01:00
										 |  |  |         let iconsLibrary: Map<string, TagRenderingConfigJson[]> = new Map< | 
					
						
							|  |  |  |             string, | 
					
						
							|  |  |  |             TagRenderingConfigJson[] | 
					
						
							|  |  |  |         >() | 
					
						
							|  |  |  |         const path = "./src/assets/generated/layers/icons.json" | 
					
						
							|  |  |  |         if (existsSync(path)) { | 
					
						
							|  |  |  |             const config = <LayerConfigJson>JSON.parse(readFileSync(path, "utf8")) | 
					
						
							|  |  |  |             for (const tagRendering of config.tagRenderings) { | 
					
						
							|  |  |  |                 const qtr = <QuestionableTagRenderingConfigJson>tagRendering | 
					
						
							|  |  |  |                 const id = qtr.id | 
					
						
							|  |  |  |                 if (id) { | 
					
						
							|  |  |  |                     iconsLibrary.set(id, [qtr]) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 for (const label of tagRendering["labels"] ?? []) { | 
					
						
							|  |  |  |                     if (!iconsLibrary.has(label)) { | 
					
						
							|  |  |  |                         iconsLibrary.set(label, []) | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     iconsLibrary.get(label).push(qtr) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-12-02 03:18:53 +01:00
										 |  |  |         proto.titleIcons = [] | 
					
						
							|  |  |  |         const seenTitleIcons = new Set<string>() | 
					
						
							|  |  |  |         for (const layer of this.layers) { | 
					
						
							|  |  |  |             for (const titleIcon of layer.titleIcons) { | 
					
						
							|  |  |  |                 if (typeof titleIcon === "string") { | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (titleIcon["labels"]?.indexOf("defaults") >= 0) { | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (titleIcon.id === "rating") { | 
					
						
							|  |  |  |                     if (!seenTitleIcons.has("rating")) { | 
					
						
							| 
									
										
										
										
											2023-12-06 12:12:53 +01:00
										 |  |  |                         proto.titleIcons.unshift(...iconsLibrary.get("rating")) | 
					
						
							| 
									
										
										
										
											2023-12-02 03:18:53 +01:00
										 |  |  |                         seenTitleIcons.add("rating") | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (seenTitleIcons.has(titleIcon.id)) { | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 seenTitleIcons.add(titleIcon.id) | 
					
						
							|  |  |  |                 console.log("Adding ", titleIcon.id) | 
					
						
							|  |  |  |                 proto.titleIcons.push(titleIcon) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-12-06 12:12:53 +01:00
										 |  |  |         proto.titleIcons.push(...(iconsLibrary.get("defaults") ?? [])) | 
					
						
							| 
									
										
										
										
											2023-12-02 03:18:53 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |     private addTitle(proto: LayerConfigJson) { | 
					
						
							| 
									
										
										
										
											2023-12-04 16:10:05 +01:00
										 |  |  |         let mappings: MappingConfigJson[] = [] | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |         for (const layer of this.layers) { | 
					
						
							|  |  |  |             const t = layer.title | 
					
						
							|  |  |  |             const tags: TagConfigJson = layer.source["osmTags"] | 
					
						
							|  |  |  |             if (!t) { | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (typeof t === "string") { | 
					
						
							|  |  |  |                 mappings.push({ if: tags, then: t }) | 
					
						
							|  |  |  |             } else if (t["render"] !== undefined || t["mappings"] !== undefined) { | 
					
						
							|  |  |  |                 const tr = <TagRenderingConfigJson>t | 
					
						
							|  |  |  |                 for (let i = 0; i < (tr.mappings ?? []).length; i++) { | 
					
						
							|  |  |  |                     const mapping = tr.mappings[i] | 
					
						
							|  |  |  |                     const optimized = TagUtils.optimzeJson({ | 
					
						
							|  |  |  |                         and: [mapping.if, tags], | 
					
						
							|  |  |  |                     }) | 
					
						
							|  |  |  |                     if (optimized === false) { | 
					
						
							|  |  |  |                         console.warn( | 
					
						
							|  |  |  |                             "The following tags yielded 'false':", | 
					
						
							|  |  |  |                             JSON.stringify(mapping.if), | 
					
						
							|  |  |  |                             JSON.stringify(tags) | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                         continue | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     if (optimized === true) { | 
					
						
							|  |  |  |                         console.error( | 
					
						
							|  |  |  |                             "The following tags yielded 'false':", | 
					
						
							|  |  |  |                             JSON.stringify(mapping.if), | 
					
						
							|  |  |  |                             JSON.stringify(tags) | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                         throw "Tags for title optimized to true" | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if (!mapping.then) { | 
					
						
							|  |  |  |                         throw ( | 
					
						
							|  |  |  |                             "The title has a missing 'then' for mapping " + | 
					
						
							|  |  |  |                             i + | 
					
						
							|  |  |  |                             " in layer " + | 
					
						
							|  |  |  |                             layer.id | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     mappings.push({ | 
					
						
							|  |  |  |                         if: optimized, | 
					
						
							|  |  |  |                         then: mapping.then, | 
					
						
							|  |  |  |                     }) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (tr.render) { | 
					
						
							|  |  |  |                     mappings.push({ | 
					
						
							|  |  |  |                         if: tags, | 
					
						
							|  |  |  |                         then: <Translatable>tr.render, | 
					
						
							|  |  |  |                     }) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 mappings.push({ if: tags, then: <Record<string, string>>t }) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-04 16:10:05 +01:00
										 |  |  |         mappings = this.sortMappings(mappings) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |         if (proto.title["mappings"]) { | 
					
						
							|  |  |  |             mappings.unshift(...proto.title["mappings"]) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (proto.title["render"]) { | 
					
						
							|  |  |  |             mappings.push({ | 
					
						
							|  |  |  |                 if: "id~*", | 
					
						
							|  |  |  |                 then: proto.title["render"], | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-04 16:10:05 +01:00
										 |  |  |         for (const mapping of mappings) { | 
					
						
							|  |  |  |             const opt = TagUtils.optimzeJson(mapping.if) | 
					
						
							|  |  |  |             if (typeof opt === "boolean") { | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             mapping.if = opt | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |         proto.title = { | 
					
						
							|  |  |  |             mappings, | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-11-29 14:29:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     async main(args: string[]): Promise<void> { | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |         console.log("Generating the favourite layer: stealing _all_ tagRenderings") | 
					
						
							| 
									
										
										
										
											2023-11-29 14:29:11 +01:00
										 |  |  |         const proto = this.readLayer("favourite/favourite.proto.json") | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |         this.addTagRenderings(proto) | 
					
						
							|  |  |  |         this.addTitle(proto) | 
					
						
							| 
									
										
										
										
											2023-12-02 03:18:53 +01:00
										 |  |  |         this.addTitleIcons(proto) | 
					
						
							| 
									
										
										
										
											2023-12-05 00:04:29 +01:00
										 |  |  |         const targetContent = JSON.stringify(proto, null, "  ") | 
					
						
							|  |  |  |         const path = "./assets/layers/favourite/favourite.json" | 
					
						
							|  |  |  |         if (existsSync(path)) { | 
					
						
							|  |  |  |             if (readFileSync(path, "utf8") === targetContent) { | 
					
						
							|  |  |  |                 return // No need to actually write the file, it is identical
 | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         writeFileSync(path, targetContent) | 
					
						
							| 
									
										
										
										
											2023-11-29 14:29:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private readLayer(path: string): LayerConfigJson { | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  |         try { | 
					
						
							|  |  |  |             return JSON.parse(readFileSync("./assets/layers/" + path, "utf8")) | 
					
						
							|  |  |  |         } catch (e) { | 
					
						
							|  |  |  |             console.error("Could not read ./assets/layers/" + path) | 
					
						
							|  |  |  |             throw e | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-11-29 14:29:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-30 00:39:55 +01:00
										 |  |  | new GenerateFavouritesLayer().run() |