forked from MapComplete/MapComplete
		
	
		
			
	
	
		
			111 lines
		
	
	
	
		
			3.3 KiB
		
	
	
	
		
			Svelte
		
	
	
	
	
	
		
		
			
		
	
	
			111 lines
		
	
	
	
		
			3.3 KiB
		
	
	
	
		
			Svelte
		
	
	
	
	
	
|  | <script lang="ts"> | ||
|  |   import SubtleButton from "../Base/SubtleButton.svelte" | ||
|  |   import { Translation } from "../i18n/Translation" | ||
|  |   import type { Theme } from "./ThemesList.svelte" | ||
|  |   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" | ||
|  | 
 | ||
|  |   export let theme: Theme | ||
|  |   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 | ||
|  |   ).toString() | ||
|  |   $: description = new Translation(theme.shortDescription).toString() | ||
|  | 
 | ||
|  |   // 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) }}> | ||
|  |       <img slot="image" src={theme.icon} alt="" /> | ||
|  |       <span slot="message" class="message"> | ||
|  |         <span> | ||
|  |           <span>{title}</span> | ||
|  |           <span>{description}</span> | ||
|  |         </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> |