| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  | import ThemeConfig, { MinimalThemeInformation } from "../../Models/ThemeConfig/ThemeConfig" | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  | import { Store } from "../UIEventSource" | 
					
						
							| 
									
										
										
										
											2024-09-10 02:19:55 +02:00
										 |  |  | import UserRelatedState from "../State/UserRelatedState" | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  | import { Utils } from "../../Utils" | 
					
						
							|  |  |  | import Locale from "../../UI/i18n/Locale" | 
					
						
							|  |  |  | import themeOverview from "../../assets/generated/theme_overview.json" | 
					
						
							|  |  |  | import LayerSearch from "./LayerSearch" | 
					
						
							|  |  |  | import SearchUtils from "./SearchUtils" | 
					
						
							| 
									
										
										
										
											2024-09-12 16:14:57 +02:00
										 |  |  | import { OsmConnection } from "../Osm/OsmConnection" | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | type ThemeSearchScore = { | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |     theme: MinimalThemeInformation | 
					
						
							|  |  |  |     lowest: number | 
					
						
							|  |  |  |     perLayer?: Record<string, number> | 
					
						
							|  |  |  |     other: number | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export default class ThemeSearch { | 
					
						
							|  |  |  |     public static readonly officialThemes: { | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |         themes: MinimalThemeInformation[] | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  |         layers: Record<string, Record<string, string[]>> | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |     } = <any>themeOverview | 
					
						
							|  |  |  |     public static readonly officialThemesById: Map<string, MinimalThemeInformation> = new Map< | 
					
						
							|  |  |  |         string, | 
					
						
							|  |  |  |         MinimalThemeInformation | 
					
						
							|  |  |  |     >() | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  |     static { | 
					
						
							|  |  |  |         for (const th of ThemeSearch.officialThemes.themes ?? []) { | 
					
						
							|  |  |  |             ThemeSearch.officialThemesById.set(th.id, th) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-08-22 02:54:46 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     private readonly _knownHiddenThemes: Store<Set<string>> | 
					
						
							| 
									
										
										
										
											2024-09-05 02:25:03 +02:00
										 |  |  |     private readonly _layersToIgnore: string[] | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |     private readonly _otherThemes: MinimalThemeInformation[] | 
					
						
							| 
									
										
										
										
											2024-08-22 02:54:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |     constructor(state: { osmConnection: OsmConnection; theme: ThemeConfig }) { | 
					
						
							|  |  |  |         this._layersToIgnore = state.theme.layers.filter((l) => l.isNormal()).map((l) => l.id) | 
					
						
							|  |  |  |         this._knownHiddenThemes = UserRelatedState.initDiscoveredHiddenThemes( | 
					
						
							|  |  |  |             state.osmConnection | 
					
						
							|  |  |  |         ).map((list) => new Set(list)) | 
					
						
							|  |  |  |         this._otherThemes = ThemeSearch.officialThemes.themes.filter( | 
					
						
							|  |  |  |             (th) => th.id !== state.theme.id | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2024-08-22 02:54:46 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |     public search(query: string, limit: number, threshold: number = 3): MinimalThemeInformation[] { | 
					
						
							| 
									
										
										
										
											2024-08-30 02:18:29 +02:00
										 |  |  |         if (query.length < 1) { | 
					
						
							| 
									
										
										
										
											2024-08-22 02:54:46 +02:00
										 |  |  |             return [] | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |         const sorted = ThemeSearch.sortedByLowestScores( | 
					
						
							|  |  |  |             query, | 
					
						
							|  |  |  |             this._otherThemes, | 
					
						
							|  |  |  |             this._layersToIgnore | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2024-09-05 02:25:03 +02:00
										 |  |  |         return sorted | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |             .filter((sorted) => sorted.lowest < threshold) | 
					
						
							|  |  |  |             .map((th) => th.theme) | 
					
						
							|  |  |  |             .filter((th) => !th.hideFromOverview || this._knownHiddenThemes.data.has(th.id)) | 
					
						
							| 
									
										
										
										
											2024-08-27 23:56:54 +02:00
										 |  |  |             .slice(0, limit) | 
					
						
							| 
									
										
										
										
											2024-08-22 02:54:46 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |     public static createUrlFor(layout: { id: string }, state?: { layoutToUse?: { id } }): string { | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  |         if (layout === undefined) { | 
					
						
							|  |  |  |             return undefined | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (layout.id === undefined) { | 
					
						
							|  |  |  |             console.error("ID is undefined for layout", layout) | 
					
						
							|  |  |  |             return undefined | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (layout.id === state?.layoutToUse?.id) { | 
					
						
							|  |  |  |             return undefined | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let path = window.location.pathname | 
					
						
							|  |  |  |         // Path starts with a '/' and contains everything, e.g. '/dir/dir/page.html'
 | 
					
						
							|  |  |  |         path = path.substr(0, path.lastIndexOf("/")) | 
					
						
							|  |  |  |         // Path will now contain '/dir/dir', or empty string in case of nothing
 | 
					
						
							|  |  |  |         if (path === "") { | 
					
						
							|  |  |  |             path = "." | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let linkPrefix = `${path}/${layout.id.toLowerCase()}.html?` | 
					
						
							|  |  |  |         if (location.hostname === "localhost" || location.hostname === "127.0.0.1") { | 
					
						
							|  |  |  |             linkPrefix = `${path}/theme.html?layout=${layout.id}&` | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (layout.id.startsWith("http://") || layout.id.startsWith("https://")) { | 
					
						
							|  |  |  |             linkPrefix = `${path}/theme.html?userlayout=${layout.id}&` | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return `${linkPrefix}` | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-24 17:23:44 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Returns a score based on textual search | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * Note that, if `query.length < 3`, layers are _not_ searched because this takes too much time | 
					
						
							|  |  |  |      * @param query | 
					
						
							|  |  |  |      * @param themes | 
					
						
							|  |  |  |      * @param ignoreLayers | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |     private static scoreThemes( | 
					
						
							|  |  |  |         query: string, | 
					
						
							|  |  |  |         themes: MinimalThemeInformation[], | 
					
						
							|  |  |  |         ignoreLayers: string[] = undefined | 
					
						
							|  |  |  |     ): Record<string, ThemeSearchScore> { | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  |         if (query?.length < 1) { | 
					
						
							|  |  |  |             return undefined | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         themes = Utils.NoNullInplace(themes) | 
					
						
							| 
									
										
										
										
											2024-09-24 17:23:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |         let options: { blacklist: Set<string> } = undefined | 
					
						
							|  |  |  |         if (ignoreLayers?.length > 0) { | 
					
						
							| 
									
										
										
										
											2024-09-24 17:53:31 +02:00
										 |  |  |             options = { blacklist: new Set(ignoreLayers) } | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |         const layerScores = query.length < 3 ? {} : LayerSearch.scoreLayers(query, options) | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  |         const results: Record<string, ThemeSearchScore> = {} | 
					
						
							|  |  |  |         for (const layoutInfo of themes) { | 
					
						
							|  |  |  |             const theme = layoutInfo.id | 
					
						
							|  |  |  |             if (theme === "personal") { | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (Utils.simplifyStringForSearch(theme) === query) { | 
					
						
							|  |  |  |                 results[theme] = { | 
					
						
							|  |  |  |                     theme: layoutInfo, | 
					
						
							|  |  |  |                     lowest: -1, | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |                     other: 0, | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |             const perLayer = Utils.asRecord(layoutInfo.layers ?? [], (layer) => layerScores[layer]) | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  |             const language = Locale.language.data | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |             const keywords = Utils.NoNullInplace([ | 
					
						
							|  |  |  |                 layoutInfo.shortDescription, | 
					
						
							|  |  |  |                 layoutInfo.title, | 
					
						
							|  |  |  |             ]).map((item) => (typeof item === "string" ? item : item[language] ?? item["*"])) | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |             const other = Math.min( | 
					
						
							|  |  |  |                 SearchUtils.scoreKeywords(query, keywords), | 
					
						
							|  |  |  |                 SearchUtils.scoreKeywords(query, layoutInfo.keywords) | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  |             const lowest = Math.min(other, ...Object.values(perLayer)) | 
					
						
							|  |  |  |             results[theme] = { | 
					
						
							|  |  |  |                 theme: layoutInfo, | 
					
						
							|  |  |  |                 perLayer, | 
					
						
							|  |  |  |                 other, | 
					
						
							|  |  |  |                 lowest, | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return results | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |     public static sortedByLowestScores( | 
					
						
							|  |  |  |         search: string, | 
					
						
							|  |  |  |         themes: MinimalThemeInformation[], | 
					
						
							|  |  |  |         ignoreLayers: string[] = [] | 
					
						
							|  |  |  |     ): ThemeSearchScore[] { | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  |         const scored = Object.values(this.scoreThemes(search, themes, ignoreLayers)) | 
					
						
							|  |  |  |         scored.sort((a, b) => a.lowest - b.lowest) | 
					
						
							|  |  |  |         return scored | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |     public static sortedByLowest( | 
					
						
							|  |  |  |         search: string, | 
					
						
							|  |  |  |         themes: MinimalThemeInformation[], | 
					
						
							|  |  |  |         ignoreLayers: string[] = [] | 
					
						
							|  |  |  |     ): MinimalThemeInformation[] { | 
					
						
							|  |  |  |         return this.sortedByLowestScores(search, themes, ignoreLayers).map((th) => th.theme) | 
					
						
							| 
									
										
										
										
											2024-09-11 17:31:38 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-08-22 02:54:46 +02:00
										 |  |  | } |