MapComplete/src/UI/StudioGUI.svelte

298 lines
9.9 KiB
Svelte
Raw Normal View History

2023-06-16 02:36:11 +02:00
<script lang="ts">
2023-11-09 16:30:26 +01:00
import NextButton from "./Base/NextButton.svelte"
import { Store, UIEventSource } from "../Logic/UIEventSource"
import EditLayerState, { EditThemeState } from "./Studio/EditLayerState"
import EditLayer from "./Studio/EditLayer.svelte"
import Loading from "../assets/svg/Loading.svelte"
import StudioServer from "./Studio/StudioServer"
import LoginToggle from "./Base/LoginToggle.svelte"
import { OsmConnection } from "../Logic/Osm/OsmConnection"
import { QueryParameters } from "../Logic/Web/QueryParameters"
2023-10-07 03:07:32 +02:00
2023-11-09 16:30:26 +01:00
import layerSchemaRaw from "../../src/assets/schemas/layerconfigmeta.json"
import layoutSchemaRaw from "../../src/assets/schemas/layoutconfigmeta.json"
2023-10-07 03:07:32 +02:00
2023-11-09 16:30:26 +01:00
import If from "./Base/If.svelte"
import BackButton from "./Base/BackButton.svelte"
import ChooseLayerToEdit from "./Studio/ChooseLayerToEdit.svelte"
import { LocalStorageSource } from "../Logic/Web/LocalStorageSource"
import FloatOver from "./Base/FloatOver.svelte"
import Walkthrough from "./Walkthrough/Walkthrough.svelte"
import * as intro from "../assets/studio_introduction.json"
import * as intro_tagrenderings from "../assets/studio_tagrenderings_intro.json"
import { QuestionMarkCircleIcon } from "@babeard/svelte-heroicons/mini"
import type { ConfigMeta } from "./Studio/configMeta"
import EditTheme from "./Studio/EditTheme.svelte"
import * as meta from "../../package.json"
import Checkbox from "./Base/Checkbox.svelte"
export let studioUrl =
window.location.hostname === "127.0.0.2"
? "http://127.0.0.1:1235"
: "https://studio.mapcomplete.org"
let osmConnection = new OsmConnection(
new OsmConnection({
oauth_token: QueryParameters.GetQueryParameter(
"oauth_token",
undefined,
"Used to complete the login"
),
})
)
const expertMode = UIEventSource.asBoolean(
osmConnection.GetPreference("studio-expert-mode", "false", {
documentation: "Indicates if more options are shown in mapcomplete studio",
})
)
expertMode.addCallbackAndRunD((expert) => console.log("Expert mode is", expert))
const createdBy = osmConnection.userDetails.data.name
const uid = osmConnection.userDetails.map((ud) => ud?.uid)
const studio = new StudioServer(studioUrl, uid)
let layersWithErr = UIEventSource.FromPromiseWithErr(studio.fetchOverview())
let layers: Store<{ owner: number }[]> = layersWithErr.mapD((l) =>
l.success?.filter((l) => l.category === "layers")
)
let selfLayers = layers.mapD((ls) => ls.filter((l) => l.owner === uid.data), [uid])
let otherLayers = layers.mapD(
(ls) => ls.filter((l) => l.owner !== undefined && l.owner !== uid.data),
[uid]
)
let officialLayers = layers.mapD((ls) => ls.filter((l) => l.owner === undefined), [uid])
let themes: Store<{ owner: number }[]> = layersWithErr.mapD((l) =>
l.success?.filter((l) => l.category === "themes")
)
let selfThemes = themes.mapD((ls) => ls.filter((l) => l.owner === uid.data), [uid])
let otherThemes = themes.mapD(
(ls) => ls.filter((l) => l.owner !== undefined && l.owner !== uid.data),
[uid]
)
let officialThemes = themes.mapD((ls) => ls.filter((l) => l.owner === undefined), [uid])
let state:
| undefined
| "edit_layer"
| "edit_theme"
| "editing_layer"
| "editing_theme"
| "loading" = undefined
const layerSchema: ConfigMeta[] = <any>layerSchemaRaw
let editLayerState = new EditLayerState(layerSchema, studio, osmConnection, { expertMode })
2023-11-07 18:51:50 +01:00
let showIntro = editLayerState.showIntro
2023-10-30 13:45:44 +01:00
2023-11-09 16:30:26 +01:00
const layoutSchema: ConfigMeta[] = <any>layoutSchemaRaw
let editThemeState = new EditThemeState(layoutSchema, studio, { expertMode })
2023-10-30 13:45:44 +01:00
2023-11-09 16:30:26 +01:00
let layerId = editLayerState.configuration.map((layerConfig) => layerConfig.id)
2023-10-07 03:07:32 +02:00
2023-11-09 16:30:26 +01:00
const version = meta.version
async function editLayer(event: Event) {
2023-11-09 16:30:26 +01:00
const layerId: { owner: number; id: string } = event.detail
state = "loading"
editLayerState.startSavingUpdates(false)
editLayerState.configuration.setData(await studio.fetch(layerId.id, "layers", layerId.owner))
editLayerState.startSavingUpdates()
state = "editing_layer"
}
2023-10-30 13:45:44 +01:00
async function editTheme(event: Event) {
2023-11-09 16:30:26 +01:00
const id: { id: string; owner: number } = event.detail
state = "loading"
editThemeState.startSavingUpdates(false)
editThemeState.configuration.setData(await studio.fetch(id.id, "themes", id.owner))
editThemeState.startSavingUpdates()
state = "editing_theme"
2023-10-30 13:45:44 +01:00
}
async function createNewLayer() {
2023-11-09 16:30:26 +01:00
state = "loading"
2023-10-30 13:45:44 +01:00
const initialLayerConfig = {
credits: createdBy,
minzoom: 15,
pointRendering: [
{
location: ["point", "centroid"],
2023-11-09 16:30:26 +01:00
marker: [
{
icon: "circle",
color: "white",
},
],
},
],
2023-11-05 12:05:00 +01:00
tagRenderings: ["images"],
2023-11-09 16:30:26 +01:00
lineRendering: [
{
width: 1,
color: "blue",
},
],
}
editLayerState.configuration.setData(initialLayerConfig)
editLayerState.startSavingUpdates()
state = "editing_layer"
}
2023-06-16 02:36:11 +02:00
</script>
2023-11-09 16:30:26 +01:00
<If condition={layersWithErr.map((d) => d?.error !== undefined)}>
2023-10-16 15:06:50 +02:00
<div>
<div class="alert">
Something went wrong while contacting the MapComplete Studio Server: {$layersWithErr["error"]}
</div>
The server might be offline. Please:
<ul>
2023-11-09 16:30:26 +01:00
<li>Try again in a few minutes</li>
2023-10-16 15:06:50 +02:00
<li>
2023-11-09 16:30:26 +01:00
Contact <a href="https://app.element.io/#/room/#MapComplete:matrix.org">
the MapComplete community via the chat.
</a>
Someone might be able to help you
2023-10-16 15:06:50 +02:00
</li>
<li>
File <a href="https://github.com/pietervdvn/MapComplete/issues">an issue</a>
</li>
<li>
Contact the devs via <a href="mailto:info@posteo.net">email</a>
</li>
</ul>
2023-10-07 03:07:32 +02:00
</div>
2023-11-09 16:30:26 +01:00
<LoginToggle ignoreLoading={true} slot="else" state={{ osmConnection }}>
2023-10-16 15:06:50 +02:00
<div slot="not-logged-in">
<NextButton clss="primary" on:click={() => osmConnection.AttemptLogin()}>
2023-10-16 15:06:50 +02:00
Please log in to use MapComplete Studio
</NextButton>
</div>
2023-10-16 15:06:50 +02:00
{#if state === undefined}
2023-11-09 16:30:26 +01:00
<div class="flex h-full flex-col justify-between p-4">
<div class="flex w-full flex-col">
<h1>MapComplete Studio</h1>
2023-11-09 16:30:26 +01:00
<NextButton on:click={() => (state = "edit_layer")}>Edit an existing layer</NextButton>
<NextButton on:click={() => createNewLayer()}>Create a new layer</NextButton>
<NextButton on:click={() => (state = "edit_theme")}>Edit a theme</NextButton>
<NextButton
on:click={() => {
editThemeState.configuration.setData({})
state = "editing_theme"
}}
>
Create a new theme
2023-10-16 15:06:50 +02:00
</NextButton>
2023-11-09 16:30:26 +01:00
<NextButton
clss="small"
on:click={() => {
showIntro.setData("intro")
}}
>
<QuestionMarkCircleIcon class="h-6 w-6" />
Show the introduction again
</NextButton>
</div>
<div>
2023-11-09 16:30:26 +01:00
<Checkbox selected={expertMode}>Enable more options (expert mode)</Checkbox>
<span class="subtle">MapComplete version {version}</span>
</div>
</div>
{:else if state === "edit_layer"}
2023-11-09 16:30:26 +01:00
<div class="m-4 flex flex-col">
<BackButton
clss="small p-1"
imageClass="w-8 h-8"
on:click={() => {
state = undefined
}}
>
MapComplete Studio
</BackButton>
<h2>Choose a layer to edit</h2>
2023-11-02 04:35:32 +01:00
<ChooseLayerToEdit {osmConnection} layerIds={$selfLayers} on:layerSelected={editLayer}>
<h3 slot="title">Your layers</h3>
</ChooseLayerToEdit>
2023-10-30 13:45:44 +01:00
<h3>Layers by other contributors</h3>
2023-11-02 04:35:32 +01:00
<ChooseLayerToEdit {osmConnection} layerIds={$otherLayers} on:layerSelected={editLayer} />
2023-10-30 13:45:44 +01:00
<h3>Official layers by MapComplete</h3>
2023-11-09 16:30:26 +01:00
<ChooseLayerToEdit
{osmConnection}
layerIds={$officialLayers}
on:layerSelected={editLayer}
/>
2023-10-30 13:45:44 +01:00
</div>
{:else if state === "edit_theme"}
2023-11-09 16:30:26 +01:00
<div class="m-4 flex flex-col">
<BackButton
clss="small p-1"
imageClass="w-8 h-8"
on:click={() => {
state = undefined
}}
>
MapComplete Studio
2023-10-30 13:45:44 +01:00
</BackButton>
<h2>Choose a theme to edit</h2>
2023-11-02 04:35:32 +01:00
<ChooseLayerToEdit {osmConnection} layerIds={$selfThemes} on:layerSelected={editTheme}>
2023-10-30 13:45:44 +01:00
<h3 slot="title">Your themes</h3>
</ChooseLayerToEdit>
<h3>Themes by other contributors</h3>
2023-11-02 04:35:32 +01:00
<ChooseLayerToEdit {osmConnection} layerIds={$otherThemes} on:layerSelected={editTheme} />
2023-10-30 13:45:44 +01:00
<h3>Official themes by MapComplete</h3>
2023-11-09 16:30:26 +01:00
<ChooseLayerToEdit
{osmConnection}
layerIds={$officialThemes}
on:layerSelected={editTheme}
/>
2023-10-16 15:06:50 +02:00
</div>
{:else if state === "loading"}
2023-11-09 16:30:26 +01:00
<div class="h-8 w-8">
2023-10-16 15:06:50 +02:00
<Loading />
</div>
{:else if state === "editing_layer"}
2023-10-30 13:45:44 +01:00
<EditLayer state={editLayerState}>
2023-11-09 16:30:26 +01:00
<BackButton
clss="small p-1"
imageClass="w-8 h-8"
on:click={() => {
state = undefined
}}
>
MapComplete Studio
</BackButton>
</EditLayer>
2023-10-30 13:45:44 +01:00
{:else if state === "editing_theme"}
<EditTheme state={editThemeState}>
2023-11-09 16:30:26 +01:00
<BackButton
clss="small p-1"
imageClass="w-8 h-8"
on:click={() => {
state = undefined
}}
>
MapComplete Studio
2023-10-30 13:45:44 +01:00
</BackButton>
</EditTheme>
2023-10-16 15:06:50 +02:00
{/if}
</LoginToggle>
</If>
2023-11-09 16:30:26 +01:00
{#if { intro, tagrenderings: intro_tagrenderings }[$showIntro]?.sections}
<FloatOver
on:close={() => {
showIntro.setData("no")
}}
>
<div class="flex h-full p-4 pr-12">
<Walkthrough
pages={{ intro, tagrenderings: intro_tagrenderings }[$showIntro]?.sections}
on:done={() => {
showIntro.setData("no")
}}
/>
</div>
</FloatOver>
{/if}