| 
									
										
										
										
											2023-02-03 22:28:11 +01:00
										 |  |  | <script lang="ts"> | 
					
						
							|  |  |  |   import SubtleButton from "../Base/SubtleButton.svelte" | 
					
						
							|  |  |  |   import { Translation } from "../i18n/Translation" | 
					
						
							|  |  |  |   import * as personal from "../../assets/themes/personal/personal.json" | 
					
						
							|  |  |  |   import { ImmutableStore, Store, UIEventSource } from "../../Logic/UIEventSource" | 
					
						
							|  |  |  |   import UserDetails, { OsmConnection } from "../../Logic/Osm/OsmConnection" | 
					
						
							|  |  |  |   import Constants from "../../Models/Constants" | 
					
						
							|  |  |  |   import type Loc from "../../Models/Loc" | 
					
						
							| 
									
										
										
										
											2023-02-11 15:04:20 +01:00
										 |  |  |   import type { LayoutInformation } from "../../Models/ThemeConfig/LayoutConfig"; | 
					
						
							| 
									
										
										
										
											2023-03-29 17:21:20 +02:00
										 |  |  |   import Tr from "../Base/Tr.svelte"; | 
					
						
							| 
									
										
										
										
											2023-02-03 22:28:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-11 15:04:20 +01:00
										 |  |  |   export let theme: LayoutInformation | 
					
						
							| 
									
										
										
										
											2023-02-03 22:28:11 +01:00
										 |  |  |   export let isCustom: boolean = false | 
					
						
							|  |  |  |   export let userDetails: UIEventSource<UserDetails> | 
					
						
							|  |  |  |   export let state: { osmConnection: OsmConnection; locationControl?: UIEventSource<Loc> } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   $: title = new Translation( | 
					
						
							|  |  |  |     theme.title, | 
					
						
							|  |  |  |     !isCustom && !theme.mustHaveLanguage ? "themes:" + theme.id + ".title" : undefined | 
					
						
							| 
									
										
										
										
											2023-03-29 17:21:20 +02:00
										 |  |  |   ) | 
					
						
							|  |  |  |   $: description = new Translation(theme.shortDescription) | 
					
						
							| 
									
										
										
										
											2023-02-03 22:28:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // TODO: Improve this function | 
					
						
							|  |  |  |   function createUrl( | 
					
						
							|  |  |  |     layout: { id: string; definition?: string }, | 
					
						
							|  |  |  |     isCustom: boolean, | 
					
						
							|  |  |  |     state?: { locationControl?: UIEventSource<{ lat; lon; zoom }>; layoutToUse?: { id } } | 
					
						
							|  |  |  |   ): Store<string> { | 
					
						
							|  |  |  |     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 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const currentLocation = state?.locationControl | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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 (isCustom) { | 
					
						
							|  |  |  |       linkPrefix = `${path}/theme.html?userlayout=${layout.id}&` | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let hash = "" | 
					
						
							|  |  |  |     if (layout.definition !== undefined) { | 
					
						
							|  |  |  |       hash = "#" + btoa(JSON.stringify(layout.definition)) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ( | 
					
						
							|  |  |  |       currentLocation?.map((currentLocation) => { | 
					
						
							|  |  |  |         const params = [ | 
					
						
							|  |  |  |           ["z", currentLocation?.zoom], | 
					
						
							|  |  |  |           ["lat", currentLocation?.lat], | 
					
						
							|  |  |  |           ["lon", currentLocation?.lon], | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |           .filter((part) => part[1] !== undefined) | 
					
						
							|  |  |  |           .map((part) => part[0] + "=" + part[1]) | 
					
						
							|  |  |  |           .join("&") | 
					
						
							|  |  |  |         return `${linkPrefix}${params}${hash}` | 
					
						
							|  |  |  |       }) ?? new ImmutableStore<string>(`${linkPrefix}`) | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | </script> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | {#if theme.id !== personal.id || $userDetails.csCount > Constants.userJourney.personalLayoutUnlock} | 
					
						
							|  |  |  |   <div> | 
					
						
							|  |  |  |     <SubtleButton options={{ url: createUrl(theme, isCustom, state) }}> | 
					
						
							| 
									
										
										
										
											2023-02-11 15:04:20 +01:00
										 |  |  |       <img slot="image" src={theme.icon} class="block h-11 w-11 bg-red mx-4" alt="" /> | 
					
						
							| 
									
										
										
										
											2023-02-03 22:28:11 +01:00
										 |  |  |       <span slot="message" class="message"> | 
					
						
							|  |  |  |         <span> | 
					
						
							| 
									
										
										
										
											2023-03-29 17:21:20 +02:00
										 |  |  |           <Tr t={title}></Tr> | 
					
						
							|  |  |  |           <span class="subtle"> | 
					
						
							|  |  |  |             <Tr t={description}></Tr> | 
					
						
							|  |  |  |           </span> | 
					
						
							| 
									
										
										
										
											2023-02-03 22:28:11 +01:00
										 |  |  |         </span> | 
					
						
							|  |  |  |       </span> | 
					
						
							|  |  |  |     </SubtleButton> | 
					
						
							|  |  |  |   </div> | 
					
						
							|  |  |  | {/if} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <style lang="scss"> | 
					
						
							|  |  |  |   div { | 
					
						
							|  |  |  |     @apply h-32 min-h-[8rem] max-h-32 text-ellipsis overflow-hidden; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     span.message { | 
					
						
							|  |  |  |       @apply flex flex-col justify-center h-24; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       & > span { | 
					
						
							|  |  |  |         @apply flex flex-col overflow-hidden; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         span:nth-child(2) { | 
					
						
							|  |  |  |           @apply text-[#999]; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | </style> |