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>
|