MapComplete/src/UI/AllThemesGui.svelte

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

271 lines
9.3 KiB
Svelte
Raw Normal View History

<script lang="ts">
2023-12-01 15:23:28 +01:00
import { OsmConnectionFeatureSwitches } from "../Logic/State/FeatureSwitchState"
import { OsmConnection } from "../Logic/Osm/OsmConnection"
import { QueryParameters } from "../Logic/Web/QueryParameters"
2025-02-08 11:50:32 +01:00
import type UserRelatedState from "../Logic/State/UserRelatedState"
2023-12-01 15:23:28 +01:00
import LanguagePicker from "./InputElement/LanguagePicker.svelte"
import Translations from "./i18n/Translations"
import Logo from "../assets/svg/Logo.svelte"
import Tr from "./Base/Tr.svelte"
import LoginToggle from "./Base/LoginToggle.svelte"
import Pencil from "../assets/svg/Pencil.svelte"
import Constants from "../Models/Constants"
import { ImmutableStore, Store, Stores, UIEventSource } from "../Logic/UIEventSource"
import ThemesList from "./BigComponents/ThemesList.svelte"
import { MinimalThemeInformation } from "../Models/ThemeConfig/ThemeConfig"
import Eye from "../assets/svg/Eye.svelte"
import LoginButton from "./Base/LoginButton.svelte"
import Mastodon from "../assets/svg/Mastodon.svelte"
import Liberapay from "../assets/svg/Liberapay.svelte"
import Bug from "../assets/svg/Bug.svelte"
import Github from "../assets/svg/Github.svelte"
2024-06-21 02:46:54 +02:00
import { Utils } from "../Utils"
import { ArrowTrendingUp } from "@babeard/svelte-heroicons/solid/ArrowTrendingUp"
2024-09-02 03:47:54 +02:00
import Searchbar from "./Base/Searchbar.svelte"
import ThemeSearch from "../Logic/Search/ThemeSearch"
import SearchUtils from "../Logic/Search/SearchUtils"
2024-09-24 17:23:04 +02:00
import ChevronDoubleRight from "@babeard/svelte-heroicons/mini/ChevronDoubleRight"
import { AndroidPolyfill } from "../Logic/Web/AndroidPolyfill"
2025-02-08 01:27:49 +01:00
import Forgejo from "../assets/svg/Forgejo.svelte"
2025-01-23 14:43:35 +01:00
AndroidPolyfill.init().then(() => console.log("Android polyfill setup completed"))
2023-12-01 15:23:28 +01:00
const featureSwitches = new OsmConnectionFeatureSwitches()
const osmConnection = new OsmConnection({
fakeUser: featureSwitches.featureSwitchFakeUser.data,
oauth_token: QueryParameters.GetQueryParameter(
"oauth_token",
undefined,
2023-12-21 01:46:18 +01:00
"Used to complete the login"
2024-10-19 14:44:55 +02:00
),
2023-12-01 15:23:28 +01:00
})
const state = new UserRelatedState(osmConnection)
const t = Translations.t.index
const tu = Translations.t.general
const tr = Translations.t.general.morescreen
let userLanguages = osmConnection.userDetails.map((ud) => ud?.languages ?? [])
let search: UIEventSource<string | undefined> = new UIEventSource<string>("")
let searchStable = search.stabilized(100)
2024-10-19 14:44:55 +02:00
const officialThemes: MinimalThemeInformation[] = ThemeSearch.officialThemes.themes.filter(
(th) => th.hideFromOverview === false
)
const hiddenThemes: MinimalThemeInformation[] = ThemeSearch.officialThemes.themes.filter(
(th) => th.hideFromOverview === true
)
let visitedHiddenThemes: Store<undefined | MinimalThemeInformation[]> =
UserRelatedState.initDiscoveredHiddenThemes(state.osmConnection).mapD((knownIds) =>
2024-10-19 14:44:55 +02:00
hiddenThemes.filter(
(theme) =>
knownIds.indexOf(theme.id) >= 0 ||
2024-12-31 19:55:08 +01:00
state.osmConnection.userDetails.data?.name === "Pieter Vander Vennet"
2024-10-19 14:44:55 +02:00
)
)
2024-10-19 14:44:55 +02:00
const customThemes: Store<MinimalThemeInformation[]> = Stores.ListStabilized<string>(
state.installedUserThemes
).mapD((stableIds) => Utils.NoNullInplace(stableIds.map((id) => state.getUnofficialTheme(id))))
function filtered(themes: Store<MinimalThemeInformation[]>): Store<MinimalThemeInformation[]> {
2024-10-19 14:44:55 +02:00
return searchStable.map(
(search) => {
if (!themes.data) {
return []
}
2024-10-19 14:44:55 +02:00
if (!search) {
return themes.data
}
const start = new Date().getTime()
const scores = ThemeSearch.sortedByLowestScores(search, themes.data)
const end = new Date().getTime()
console.trace("Scores for", search, "are", scores, "searching took", end - start, "ms")
const strict = scores.filter((sc) => sc.lowest < 2)
if (strict.length > 0) {
return strict.map((sc) => sc.theme)
}
return scores
.filter((sc) => sc.lowest < 4)
.slice(0, 6)
.map((sc) => sc.theme)
},
[themes]
)
}
2024-10-19 14:44:55 +02:00
let officialSearched: Store<MinimalThemeInformation[]> = filtered(
2025-01-02 15:34:59 +01:00
osmConnection.isLoggedIn.map((loggedIn) =>
loggedIn ? officialThemes : officialThemes.filter((th) => th.id !== "personal")
)
2024-10-19 14:44:55 +02:00
)
let hiddenSearched: Store<MinimalThemeInformation[]> = filtered(visitedHiddenThemes)
let customSearched: Store<MinimalThemeInformation[]> = filtered(customThemes)
let searchIsFocussed = new UIEventSource(false)
2024-10-19 14:44:55 +02:00
document.addEventListener("keydown", function (event) {
if (event.ctrlKey && event.code === "KeyF") {
searchIsFocussed.set(true)
event.preventDefault()
}
})
AndroidPolyfill.onBackButton((() => {
if(searchIsFocussed.data){
searchIsFocussed.set(false)
return true
}
return false
}), {returnToIndex: new ImmutableStore(false)})
/**
* Opens the first search candidate
*/
function applySearch() {
const didRedirect = SearchUtils.applySpecialSearch(search.data)
if (didRedirect) {
return
}
const candidate = officialSearched.data[0] ?? hiddenSearched.data[0] ?? customSearched.data[0]
if (!candidate) {
return
}
window.location.href = ThemeSearch.createUrlFor(candidate, undefined)
}
</script>
<main>
<div class="m-4 flex flex-col">
<LanguagePicker
clss="self-end max-w-full"
assignTo={state.language}
availableLanguages={t.title.SupportedLanguages()}
preferredLanguages={userLanguages}
/>
<div class="mt-4 flex">
<div class="m-3 flex-none">
<Logo alt="MapComplete Logo" class="h-12 w-12 sm:h-24 sm:w-24" />
</div>
2024-06-20 04:21:29 +02:00
<div class="link-underline flex flex-col">
<h1 class="m-0 font-extrabold tracking-tight md:text-6xl">
<Tr t={t.title} />
</h1>
2024-06-20 04:21:29 +02:00
<Tr
cls="mr-4 text-base font-semibold sm:text-lg md:mt-5 md:text-xl lg:mx-0"
t={Translations.t.index.intro}
/>
<a href="#about">
<Tr t={Translations.t.index.learnMore} />
<ChevronDoubleRight class="inline h-4 w-4" />
</a>
</div>
</div>
2024-10-19 14:44:55 +02:00
<Searchbar
value={search}
placeholder={tr.searchForATheme}
on:search={() => applySearch()}
autofocus
isFocused={searchIsFocussed}
/>
<ThemesList {search} {state} themes={$officialSearched} />
<LoginToggle {state}>
2024-07-25 19:06:30 +02:00
<LoginButton clss="primary" {osmConnection} slot="not-logged-in">
2024-08-09 16:55:08 +02:00
<Tr t={t.logIn} />
2024-07-25 19:06:30 +02:00
</LoginButton>
<ThemesList
{search}
{state}
themes={$hiddenSearched}
hasSelection={$officialSearched.length === 0}
>
<svelte:fragment slot="title">
<h3>
<Tr t={tr.previouslyHiddenTitle} />
</h3>
<p>
<Tr
t={tr.hiddenExplanation.Subs({
2024-06-16 16:41:57 +02:00
hidden_discovered: $visitedHiddenThemes.length.toString(),
total_hidden: hiddenThemes.length.toString(),
})}
/>
</p>
</svelte:fragment>
</ThemesList>
{#if $customThemes.length > 0}
2024-10-19 14:44:55 +02:00
<ThemesList
{search}
{state}
themes={$customSearched}
hasSelection={$officialSearched.length === 0 && $hiddenSearched.length === 0}
>
<svelte:fragment slot="title">
<h3>
<Tr t={tu.customThemeTitle} />
</h3>
<Tr t={tu.customThemeIntro} />
</svelte:fragment>
</ThemesList>
{/if}
2024-06-16 19:00:43 +02:00
</LoginToggle>
2024-06-21 02:46:54 +02:00
<a
2024-06-24 13:11:35 +02:00
class="button flex"
2024-06-21 02:46:54 +02:00
href={window.location.protocol + "//" + window.location.host + "/studio.html"}
>
<Pencil class="mr-2 h-6 w-6" />
<Tr t={Translations.t.general.morescreen.createYourOwnTheme} />
</a>
2024-06-20 04:21:29 +02:00
<h3 id="about">
<Tr t={Translations.t.index.about} />
</h3>
<Tr cls="link-underline" t={Translations.t.general.aboutMapComplete.intro} />
<span class="link-underline flex flex-col gap-y-1">
2025-02-08 01:27:49 +01:00
<a class="flex" href="https://source.mapcomplete.org/MapComplete/MapComplete/" target="_blank">
<Forgejo class="mr-2 h-6 w-6" />
2024-06-20 04:21:29 +02:00
<Tr t={Translations.t.general.attribution.gotoSourceCode} />
</a>
2025-02-08 01:27:49 +01:00
<a class="flex" href="https://source.mapcomplete.org/MapComplete/MapComplete/issues" target="_blank">
2024-06-20 04:21:29 +02:00
<Bug class="mr-2 h-6 w-6" />
<Tr t={Translations.t.general.attribution.openIssueTracker} />
</a>
2024-06-21 02:46:54 +02:00
<a class="flex" href={Utils.OsmChaLinkFor(7)} target="_blank">
2024-06-24 13:11:35 +02:00
<ArrowTrendingUp class="mr-2 h-6 w-6" />
<Tr t={Translations.t.general.attribution.openOsmchaLastWeek} />
2024-06-21 02:46:54 +02:00
</a>
2024-06-20 04:21:29 +02:00
<a class="flex" href="https://en.osm.town/@MapComplete" target="_blank">
<Mastodon class="mr-2 h-6 w-6" />
<Tr t={Translations.t.general.attribution.followOnMastodon} />
</a>
<a class="flex" href="https://liberapay.com/pietervdvn/" target="_blank">
<Liberapay class="mr-2 h-6 w-6" />
<Tr t={Translations.t.general.attribution.donate} />
</a>
<a
class="flex"
href={window.location.protocol + "//" + window.location.host + "/privacy.html"}
>
<Eye class="mr-2 h-6 w-6" />
<Tr t={Translations.t.privacy.title} />
</a>
</span>
<Tr t={tr.streetcomplete} />
<div class="subtle mb-16 self-end">
v{Constants.vNumber}
</div>
</div>
</main>