2023-06-16 02:36:11 +02:00
|
|
|
<script lang="ts">
|
2023-12-01 15:23:28 +01:00
|
|
|
import NextButton from "./Base/NextButton.svelte"
|
|
|
|
import { Store, UIEventSource } from "../Logic/UIEventSource"
|
2024-04-28 00:23:20 +02:00
|
|
|
import EditLayerState, { EditJsonState, EditThemeState } from "./Studio/EditLayerState"
|
2023-12-01 15:23:28 +01:00
|
|
|
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"
|
|
|
|
|
|
|
|
import layerSchemaRaw from "../../src/assets/schemas/layerconfigmeta.json"
|
|
|
|
import layoutSchemaRaw from "../../src/assets/schemas/layoutconfigmeta.json"
|
|
|
|
|
|
|
|
import If from "./Base/If.svelte"
|
|
|
|
import BackButton from "./Base/BackButton.svelte"
|
|
|
|
import ChooseLayerToEdit from "./Studio/ChooseLayerToEdit.svelte"
|
|
|
|
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"
|
|
|
|
import { Utils } from "../Utils"
|
|
|
|
import Translations from "./i18n/Translations"
|
|
|
|
import Tr from "./Base/Tr.svelte"
|
|
|
|
import Add from "../assets/svg/Add.svelte"
|
2024-04-21 01:19:48 +02:00
|
|
|
import { SearchIcon } from "@rgossiaux/svelte-heroicons/solid"
|
2024-04-28 00:23:20 +02:00
|
|
|
import Hash from "../Logic/Web/Hash"
|
2024-10-19 14:44:55 +02:00
|
|
|
const directEntry = QueryParameters.GetBooleanQueryParameter(
|
|
|
|
"direct",
|
|
|
|
false,
|
|
|
|
"If set, write directly into the theme files"
|
|
|
|
)
|
2023-12-01 15:23:28 +01:00
|
|
|
|
|
|
|
export let studioUrl =
|
2024-09-25 11:40:45 +02:00
|
|
|
window.location.hostname === "127.0.0.2" || directEntry.data
|
2023-11-09 16:30:26 +01:00
|
|
|
? "http://127.0.0.1:1235"
|
|
|
|
: "https://studio.mapcomplete.org"
|
|
|
|
|
2024-09-25 11:40:45 +02:00
|
|
|
console.log("Using studio URL", studioUrl, "direct?", directEntry.data)
|
|
|
|
|
2024-01-19 17:31:35 +01:00
|
|
|
const oauth_token = QueryParameters.GetQueryParameter(
|
|
|
|
"oauth_token",
|
|
|
|
undefined,
|
2024-02-20 13:33:38 +01:00
|
|
|
"Used to complete the login"
|
2023-11-09 16:30:26 +01:00
|
|
|
)
|
2024-08-14 13:53:56 +02:00
|
|
|
const fakeUser = UIEventSource.asBoolean(
|
|
|
|
QueryParameters.GetQueryParameter("fake-user", "Test switch for fake login")
|
|
|
|
)
|
2024-01-19 17:31:35 +01:00
|
|
|
let osmConnection = new OsmConnection({
|
|
|
|
oauth_token,
|
2024-06-16 16:06:26 +02:00
|
|
|
checkOnlineRegularly: true,
|
2024-08-14 13:53:56 +02:00
|
|
|
fakeUser: fakeUser.data,
|
2024-01-19 17:31:35 +01:00
|
|
|
})
|
2023-11-09 16:30:26 +01:00
|
|
|
const expertMode = UIEventSource.asBoolean(
|
2025-02-28 13:14:14 +01:00
|
|
|
osmConnection.GetPreference("studio-expert-mode", "false")
|
2023-11-09 16:30:26 +01:00
|
|
|
)
|
|
|
|
expertMode.addCallbackAndRunD((expert) => console.log("Expert mode is", expert))
|
2025-02-08 23:21:34 +01:00
|
|
|
const createdBy = osmConnection.userDetails.data?.name
|
|
|
|
const uid = osmConnection.userDetails.mapD(
|
2024-10-19 14:44:55 +02:00
|
|
|
(ud) => (directEntry.data ? null : ud?.uid),
|
|
|
|
[directEntry]
|
|
|
|
)
|
2025-02-08 23:21:34 +01:00
|
|
|
const studio = new StudioServer(studioUrl, uid)
|
2023-11-09 16:30:26 +01:00
|
|
|
|
2024-04-23 21:31:58 +02:00
|
|
|
let layersWithErr = studio.fetchOverview()
|
2025-02-28 13:14:14 +01:00
|
|
|
let layerFilterTerm: UIEventSource<string> = new UIEventSource<string>("")
|
2024-02-20 13:33:38 +01:00
|
|
|
let layers: Store<{ owner: number; id: string }[]> = layersWithErr.mapD((l) =>
|
|
|
|
l["success"]?.filter((l) => l.category === "layers")
|
2023-11-09 16:30:26 +01:00
|
|
|
)
|
2025-02-28 13:14:14 +01:00
|
|
|
let selfLayers = layers.mapD(
|
2024-04-28 00:23:20 +02:00
|
|
|
(ls) =>
|
|
|
|
ls.filter(
|
2025-03-06 16:21:55 +01:00
|
|
|
(l) =>
|
|
|
|
l.owner === uid.data && l.id.toLowerCase().includes(layerFilterTerm.data.toLowerCase())
|
2024-04-28 00:23:20 +02:00
|
|
|
),
|
2025-02-28 13:14:14 +01:00
|
|
|
[uid, layerFilterTerm]
|
2024-04-28 00:23:20 +02:00
|
|
|
)
|
2025-02-28 13:14:14 +01:00
|
|
|
let otherLayers = layers.mapD(
|
2024-04-21 01:19:48 +02:00
|
|
|
(ls) =>
|
|
|
|
ls.filter(
|
|
|
|
(l) =>
|
|
|
|
l.owner !== undefined &&
|
|
|
|
l.owner !== uid.data &&
|
2025-02-28 13:14:14 +01:00
|
|
|
l.id.toLowerCase().includes(layerFilterTerm.data.toLowerCase())
|
2024-04-21 01:19:48 +02:00
|
|
|
),
|
2025-02-28 13:14:14 +01:00
|
|
|
[uid, layerFilterTerm]
|
2024-04-21 01:19:48 +02:00
|
|
|
)
|
2025-02-28 13:14:14 +01:00
|
|
|
let officialLayers = layers.mapD(
|
2024-04-21 01:19:48 +02:00
|
|
|
(ls) =>
|
|
|
|
ls.filter(
|
2025-03-06 16:21:55 +01:00
|
|
|
(l) =>
|
|
|
|
l.owner === undefined && l.id.toLowerCase().includes(layerFilterTerm.data.toLowerCase())
|
2024-04-21 01:19:48 +02:00
|
|
|
),
|
2025-02-28 13:14:14 +01:00
|
|
|
[uid, layerFilterTerm]
|
2023-11-09 16:30:26 +01:00
|
|
|
)
|
2024-04-21 01:19:48 +02:00
|
|
|
|
2025-02-28 13:14:14 +01:00
|
|
|
let themeFilterTerm: UIEventSource<string> = new UIEventSource("")
|
2023-11-09 16:30:26 +01:00
|
|
|
|
2024-02-20 13:33:38 +01:00
|
|
|
let themes: Store<{ owner: number; id: string }[]> = layersWithErr.mapD((l) =>
|
|
|
|
l["success"]?.filter((l) => l.category === "themes")
|
2023-11-09 16:30:26 +01:00
|
|
|
)
|
2025-02-28 13:14:14 +01:00
|
|
|
let selfThemes = themes.mapD(
|
2024-04-21 01:19:48 +02:00
|
|
|
(ls) =>
|
|
|
|
ls.filter(
|
2025-03-06 16:21:55 +01:00
|
|
|
(l) =>
|
|
|
|
l.owner === uid.data && l.id.toLowerCase().includes(themeFilterTerm.data.toLowerCase())
|
2024-04-21 01:19:48 +02:00
|
|
|
),
|
2025-02-28 13:14:14 +01:00
|
|
|
[uid, themeFilterTerm]
|
2024-04-21 01:19:48 +02:00
|
|
|
)
|
2025-02-28 13:14:14 +01:00
|
|
|
let otherThemes = themes.mapD(
|
2024-04-21 01:19:48 +02:00
|
|
|
(ls) =>
|
|
|
|
ls.filter(
|
|
|
|
(l) =>
|
|
|
|
l.owner !== undefined &&
|
|
|
|
l.owner !== uid.data &&
|
2025-02-28 13:14:14 +01:00
|
|
|
l.id.toLowerCase().includes(themeFilterTerm.data.toLowerCase())
|
2024-04-21 01:19:48 +02:00
|
|
|
),
|
2025-02-28 13:14:14 +01:00
|
|
|
[uid, themeFilterTerm]
|
2024-04-21 01:19:48 +02:00
|
|
|
)
|
2025-02-28 13:14:14 +01:00
|
|
|
let officialThemes = themes.mapD(
|
2024-04-21 01:19:48 +02:00
|
|
|
(ls) =>
|
|
|
|
ls.filter(
|
2025-03-06 16:21:55 +01:00
|
|
|
(l) =>
|
|
|
|
l.owner === undefined && l.id.toLowerCase().includes(themeFilterTerm.data.toLowerCase())
|
2024-04-21 01:19:48 +02:00
|
|
|
),
|
2025-02-28 13:14:14 +01:00
|
|
|
[uid, themeFilterTerm]
|
2023-11-09 16:30:26 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
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
|
2024-08-02 19:06:14 +02:00
|
|
|
let editThemeState = new EditThemeState(layoutSchema, studio, osmConnection, { expertMode })
|
2023-10-30 13:45:44 +01:00
|
|
|
|
2023-11-09 16:30:26 +01:00
|
|
|
const version = meta.version
|
2023-11-07 02:13:16 +01:00
|
|
|
|
2024-04-28 00:23:20 +02:00
|
|
|
async function editLayer(event: { detail }): Promise<EditLayerState> {
|
2024-01-03 14:57:26 +01:00
|
|
|
const layerId: { owner: number; id: string } = event["detail"]
|
2023-11-09 16:30:26 +01:00
|
|
|
state = "loading"
|
|
|
|
editLayerState.startSavingUpdates(false)
|
|
|
|
editLayerState.configuration.setData(await studio.fetch(layerId.id, "layers", layerId.owner))
|
|
|
|
editLayerState.startSavingUpdates()
|
|
|
|
state = "editing_layer"
|
2024-04-28 00:23:20 +02:00
|
|
|
return editLayerState
|
2023-10-13 18:46:56 +02:00
|
|
|
}
|
|
|
|
|
2024-04-28 00:23:20 +02:00
|
|
|
async function editTheme(event: { detail }): Promise<EditThemeState> {
|
2024-01-03 14:57:26 +01:00
|
|
|
const id: { id: string; owner: number } = event["detail"]
|
2023-11-09 16:30:26 +01:00
|
|
|
state = "loading"
|
|
|
|
editThemeState.startSavingUpdates(false)
|
2024-01-19 17:31:35 +01:00
|
|
|
const layout = await studio.fetch(id.id, "themes", id.owner)
|
|
|
|
editThemeState.configuration.setData(layout)
|
2023-11-09 16:30:26 +01:00
|
|
|
editThemeState.startSavingUpdates()
|
|
|
|
state = "editing_theme"
|
2024-04-28 00:23:20 +02:00
|
|
|
return editThemeState
|
2023-10-30 13:45:44 +01:00
|
|
|
}
|
|
|
|
|
2023-10-13 18:46:56 +02:00
|
|
|
async function createNewLayer() {
|
2023-11-09 16:30:26 +01:00
|
|
|
state = "loading"
|
2023-10-30 13:45:44 +01:00
|
|
|
const initialLayerConfig = {
|
2023-10-24 22:01:10 +02:00
|
|
|
credits: createdBy,
|
|
|
|
minzoom: 15,
|
|
|
|
pointRendering: [
|
|
|
|
{
|
|
|
|
location: ["point", "centroid"],
|
2023-11-09 16:30:26 +01:00
|
|
|
marker: [
|
|
|
|
{
|
|
|
|
icon: "circle",
|
2024-06-16 16:06:26 +02:00
|
|
|
color: "white",
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
2023-10-24 22:01:10 +02:00
|
|
|
],
|
2023-11-05 12:05:00 +01:00
|
|
|
tagRenderings: ["images"],
|
2023-11-09 16:30:26 +01:00
|
|
|
lineRendering: [
|
|
|
|
{
|
|
|
|
width: 1,
|
2024-06-16 16:06:26 +02:00
|
|
|
color: "blue",
|
|
|
|
},
|
|
|
|
],
|
2023-11-09 16:30:26 +01:00
|
|
|
}
|
|
|
|
editLayerState.configuration.setData(initialLayerConfig)
|
|
|
|
editLayerState.startSavingUpdates()
|
|
|
|
state = "editing_layer"
|
2023-10-11 04:16:52 +02:00
|
|
|
}
|
2024-04-28 00:23:20 +02:00
|
|
|
|
2024-06-26 11:57:18 +02:00
|
|
|
async function selectStateBasedOnHash(uid: number) {
|
2024-04-28 00:23:20 +02:00
|
|
|
const hash = Hash.hash.data
|
|
|
|
if (!hash) {
|
|
|
|
return
|
|
|
|
}
|
2024-06-26 11:57:18 +02:00
|
|
|
console.log("Selecting state based on ", hash, uid)
|
2024-04-28 00:23:20 +02:00
|
|
|
const [mode, id, tab] = hash.split("/")
|
|
|
|
// Not really an event, we just set the 'detail'
|
|
|
|
const event = {
|
|
|
|
detail: {
|
|
|
|
id,
|
2024-06-26 11:57:18 +02:00
|
|
|
owner: uid,
|
2024-06-16 16:06:26 +02:00
|
|
|
},
|
2024-04-28 00:23:20 +02:00
|
|
|
}
|
2025-07-08 12:19:25 +02:00
|
|
|
if(mode === undefined || id === undefined){
|
|
|
|
return
|
|
|
|
}
|
2024-06-16 16:06:26 +02:00
|
|
|
const statePromise: Promise<EditJsonState<any>> =
|
|
|
|
mode === "layer" ? editLayer(event) : editTheme(event)
|
2024-04-28 00:23:20 +02:00
|
|
|
const state = await statePromise
|
|
|
|
state.selectedTab.setData(Number(tab))
|
|
|
|
}
|
|
|
|
|
2024-07-09 13:42:08 +02:00
|
|
|
uid.AsPromise().then((uid) => selectStateBasedOnHash(uid))
|
2024-04-28 00:23:20 +02:00
|
|
|
|
|
|
|
function backToStudio() {
|
|
|
|
console.log("Back to studio")
|
|
|
|
state = undefined
|
|
|
|
Hash.hash.setData(undefined)
|
|
|
|
}
|
2023-06-16 02:36:11 +02:00
|
|
|
</script>
|
2023-10-11 04:16:52 +02:00
|
|
|
|
2024-02-20 13:33:38 +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">
|
2024-06-16 16:06:26 +02:00
|
|
|
the MapComplete community via the chat.
|
|
|
|
</a>
|
2023-11-23 15:47:16 +01:00
|
|
|
Someone might be able to help you
|
2023-10-16 15:06:50 +02:00
|
|
|
</li>
|
|
|
|
<li>
|
2025-04-07 19:10:11 +02:00
|
|
|
File <a href="https://source.mapcomplete.org/MapComplete/MapComplete/issues">an issue</a>
|
2023-10-16 15:06:50 +02:00
|
|
|
</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">
|
2023-10-17 17:03:13 +02:00
|
|
|
<NextButton clss="primary" on:click={() => osmConnection.AttemptLogin()}>
|
2023-10-16 15:06:50 +02:00
|
|
|
Please log in to use MapComplete Studio
|
2023-10-13 18:46:56 +02:00
|
|
|
</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">
|
2023-11-07 02:13:16 +01:00
|
|
|
<h1>MapComplete Studio</h1>
|
2023-10-24 22:01:10 +02:00
|
|
|
|
2024-01-21 23:50:23 +01:00
|
|
|
<NextButton on:click={() => (state = "edit_layer")}>
|
|
|
|
<div class="flex flex-col items-start">
|
2024-02-20 13:33:38 +01:00
|
|
|
<div>Edit an existing layer</div>
|
|
|
|
<div class="font-normal">
|
|
|
|
Edit layers you created, others created or from the official MapComplete
|
|
|
|
</div>
|
2024-01-21 23:50:23 +01:00
|
|
|
</div>
|
|
|
|
</NextButton>
|
2023-11-09 16:30:26 +01:00
|
|
|
<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({})
|
2023-11-10 16:34:35 +01:00
|
|
|
editThemeState.startSavingUpdates()
|
2023-11-09 16:30:26 +01:00
|
|
|
state = "editing_theme"
|
|
|
|
}}
|
|
|
|
>
|
2023-10-24 22:01:10 +02:00
|
|
|
Create a new theme
|
2023-10-16 15:06:50 +02:00
|
|
|
</NextButton>
|
2023-11-16 18:46:22 +01:00
|
|
|
<button
|
|
|
|
class="small"
|
2023-11-09 16:30:26 +01:00
|
|
|
on:click={() => {
|
|
|
|
showIntro.setData("intro")
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<QuestionMarkCircleIcon class="h-6 w-6" />
|
2023-10-24 22:01:10 +02:00
|
|
|
Show the introduction again
|
2023-11-16 18:46:22 +01:00
|
|
|
</button>
|
2023-12-01 15:23:28 +01:00
|
|
|
<a class="button flex" href={Utils.HomepageLink()}>
|
2023-11-19 04:38:34 +01:00
|
|
|
<Add class="h-6 w-6" />
|
2023-11-16 18:46:22 +01:00
|
|
|
<Tr t={Translations.t.general.backToIndex} />
|
|
|
|
</a>
|
2023-10-24 22:01:10 +02:00
|
|
|
</div>
|
2024-07-27 02:18:58 +02:00
|
|
|
<div class="flex justify-between">
|
2023-11-09 16:30:26 +01:00
|
|
|
<Checkbox selected={expertMode}>Enable more options (expert mode)</Checkbox>
|
|
|
|
<span class="subtle">MapComplete version {version}</span>
|
2024-10-19 14:44:55 +02:00
|
|
|
<div>
|
|
|
|
{$uid}
|
|
|
|
{studioUrl}
|
2024-09-25 11:40:45 +02:00
|
|
|
{#if $directEntry}
|
2024-10-19 14:44:55 +02:00
|
|
|
<b>direct</b>
|
2024-09-25 11:40:45 +02:00
|
|
|
{/if}
|
|
|
|
</div>
|
2023-11-07 02:13:16 +01:00
|
|
|
</div>
|
2023-10-24 22:01:10 +02:00
|
|
|
</div>
|
|
|
|
{:else if state === "edit_layer"}
|
2023-11-09 16:30:26 +01:00
|
|
|
<div class="m-4 flex flex-col">
|
2024-06-16 16:06:26 +02:00
|
|
|
<BackButton clss="small p-1" imageClass="w-8 h-8" on:click={() => backToStudio()}>
|
2023-11-09 16:30:26 +01:00
|
|
|
MapComplete Studio
|
2023-10-24 22:01:10 +02:00
|
|
|
</BackButton>
|
|
|
|
<h2>Choose a layer to edit</h2>
|
2024-04-21 01:19:48 +02:00
|
|
|
|
|
|
|
<form class="flex justify-center">
|
|
|
|
<label
|
|
|
|
class="neutral-label my-2 flex w-full items-center rounded-full border-2 border-black sm:w-1/2"
|
|
|
|
>
|
|
|
|
<SearchIcon aria-hidden="true" class="h-8 w-8" />
|
|
|
|
<input
|
|
|
|
class="mr-4 w-full outline-none"
|
|
|
|
id="layer-search"
|
|
|
|
type="search"
|
|
|
|
placeholder="Filter layers by name"
|
2025-02-28 13:14:14 +01:00
|
|
|
bind:value={$layerFilterTerm}
|
2024-04-21 01:19:48 +02:00
|
|
|
/>
|
|
|
|
</label>
|
|
|
|
</form>
|
|
|
|
|
2025-02-28 13:14:14 +01:00
|
|
|
<ChooseLayerToEdit {osmConnection} layerIds={selfLayers} on:layerSelected={editLayer}>
|
2023-10-24 22:01:10 +02:00
|
|
|
<h3 slot="title">Your layers</h3>
|
2024-07-27 02:18:58 +02:00
|
|
|
<div class="subtle">Your id is {$uid}</div>
|
2023-10-24 22:01:10 +02:00
|
|
|
</ChooseLayerToEdit>
|
2023-10-30 13:45:44 +01:00
|
|
|
<h3>Layers by other contributors</h3>
|
2024-02-20 13:33:38 +01:00
|
|
|
<div>
|
|
|
|
Selecting a layer will create a copy in your account that you edit. You will not change
|
|
|
|
the version of the other contributor
|
|
|
|
</div>
|
2025-02-28 13:14:14 +01:00
|
|
|
<ChooseLayerToEdit {osmConnection} layerIds={otherLayers} on:layerSelected={editLayer} />
|
2023-10-30 13:45:44 +01:00
|
|
|
|
|
|
|
<h3>Official layers by MapComplete</h3>
|
2024-02-20 13:33:38 +01:00
|
|
|
<div>
|
|
|
|
Selecting a layer will create a copy in your account. You will not change the version that
|
|
|
|
is in MapComplete
|
|
|
|
</div>
|
2025-03-06 16:21:55 +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">
|
2024-06-16 16:06:26 +02:00
|
|
|
<BackButton clss="small p-1" imageClass="w-8 h-8" on:click={() => backToStudio()}>
|
2023-11-09 16:30:26 +01:00
|
|
|
MapComplete Studio
|
2023-10-30 13:45:44 +01:00
|
|
|
</BackButton>
|
|
|
|
<h2>Choose a theme to edit</h2>
|
2024-04-21 01:19:48 +02:00
|
|
|
|
|
|
|
<form class="flex justify-center">
|
|
|
|
<label
|
|
|
|
class="neutral-label my-2 flex w-full items-center rounded-full border-2 border-black sm:w-1/2"
|
|
|
|
>
|
|
|
|
<SearchIcon aria-hidden="true" class="h-8 w-8" />
|
|
|
|
<input
|
|
|
|
class="mr-4 w-full outline-none"
|
|
|
|
id="theme-search"
|
|
|
|
type="search"
|
|
|
|
placeholder="Filter themes by name"
|
2025-03-28 15:00:50 +01:00
|
|
|
bind:value={$themeFilterTerm}
|
2024-04-21 01:19:48 +02:00
|
|
|
/>
|
|
|
|
</label>
|
|
|
|
</form>
|
|
|
|
|
2025-02-28 13:14:14 +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>
|
2025-02-28 13:14:14 +01:00
|
|
|
<ChooseLayerToEdit {osmConnection} layerIds={otherThemes} on:layerSelected={editTheme} />
|
2023-10-30 13:45:44 +01:00
|
|
|
<h3>Official themes by MapComplete</h3>
|
2025-03-06 16:21:55 +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">
|
2024-10-19 14:44:55 +02:00
|
|
|
<Loading>Fetching information from {studioUrl}</Loading>
|
2023-10-16 15:06:50 +02:00
|
|
|
</div>
|
|
|
|
{:else if state === "editing_layer"}
|
2024-06-16 16:06:26 +02:00
|
|
|
<EditLayer state={editLayerState} {backToStudio}>
|
|
|
|
<BackButton clss="small p-1" imageClass="w-8 h-8" on:click={() => backToStudio()}>
|
2024-08-09 16:55:08 +02:00
|
|
|
Studio
|
2023-10-24 22:01:10 +02:00
|
|
|
</BackButton>
|
2023-10-21 09:35:54 +02:00
|
|
|
</EditLayer>
|
2023-10-30 13:45:44 +01:00
|
|
|
{:else if state === "editing_theme"}
|
2024-06-16 16:06:26 +02:00
|
|
|
<EditTheme
|
|
|
|
state={editThemeState}
|
|
|
|
selfLayers={$selfLayers}
|
|
|
|
otherLayers={$otherLayers}
|
|
|
|
{osmConnection}
|
|
|
|
{backToStudio}
|
|
|
|
>
|
|
|
|
<BackButton clss="small p-1" imageClass="w-8 h-8" on:click={() => backToStudio()}>
|
2023-11-09 16:30:26 +01:00
|
|
|
MapComplete Studio
|
2023-10-30 13:45:44 +01:00
|
|
|
</BackButton>
|
|
|
|
</EditTheme>
|
2023-10-16 15:06:50 +02:00
|
|
|
{/if}
|
2023-10-24 22:01:10 +02:00
|
|
|
|
2023-12-02 00:24:55 +01:00
|
|
|
{#if { intro, tagrenderings: intro_tagrenderings }[$showIntro]?.sections}
|
|
|
|
<FloatOver
|
|
|
|
on:close={() => {
|
2023-12-19 22:08:00 +01:00
|
|
|
showIntro.setData("no")
|
|
|
|
}}
|
2023-12-02 00:24:55 +01:00
|
|
|
>
|
|
|
|
<div class="flex h-full p-4 pr-12">
|
|
|
|
<Walkthrough
|
|
|
|
pages={{ intro, tagrenderings: intro_tagrenderings }[$showIntro]?.sections}
|
|
|
|
on:done={() => {
|
2023-12-19 22:08:00 +01:00
|
|
|
showIntro.setData("no")
|
|
|
|
}}
|
2023-12-02 00:24:55 +01:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</FloatOver>
|
|
|
|
{/if}
|
|
|
|
</LoginToggle>
|
|
|
|
</If>
|