Studio: UX-improvements after user testing
This commit is contained in:
parent
2041a9245d
commit
44c1548e89
19 changed files with 100 additions and 35 deletions
|
@ -210,7 +210,7 @@
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"tags": {
|
"tags": {
|
||||||
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag",
|
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag\ntypeHelper: uploadableOnly",
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
"type": "string"
|
"type": "string"
|
||||||
|
|
|
@ -210,7 +210,7 @@ export default {
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"tags": {
|
"tags": {
|
||||||
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag",
|
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag\ntypeHelper: uploadableOnly",
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
"type": "string"
|
"type": "string"
|
||||||
|
|
|
@ -1992,7 +1992,7 @@
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"tags": {
|
"tags": {
|
||||||
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag",
|
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag\ntypeHelper: uploadableOnly",
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
"type": "string"
|
"type": "string"
|
||||||
|
@ -2392,7 +2392,7 @@
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"tags": {
|
"tags": {
|
||||||
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag",
|
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag\ntypeHelper: uploadableOnly",
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
"type": "string"
|
"type": "string"
|
||||||
|
|
|
@ -1970,7 +1970,7 @@ export default {
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"tags": {
|
"tags": {
|
||||||
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag",
|
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag\ntypeHelper: uploadableOnly",
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
"type": "string"
|
"type": "string"
|
||||||
|
@ -2369,7 +2369,7 @@ export default {
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"tags": {
|
"tags": {
|
||||||
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag",
|
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag\ntypeHelper: uploadableOnly",
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
"type": "string"
|
"type": "string"
|
||||||
|
|
|
@ -4,6 +4,8 @@
|
||||||
|
|
||||||
Create a simple layer specification using MapComplete studio with 'images' and a question. The actual _topic_ of the layer can be chosen by the participant
|
Create a simple layer specification using MapComplete studio with 'images' and a question. The actual _topic_ of the layer can be chosen by the participant
|
||||||
|
|
||||||
|
This participant wanted to create a layer about solar panels.
|
||||||
|
|
||||||
## Background info
|
## Background info
|
||||||
|
|
||||||
Browser: Librewolf on a linux machine (actually: pietervdvn's dev machine)
|
Browser: Librewolf on a linux machine (actually: pietervdvn's dev machine)
|
||||||
|
@ -30,4 +32,4 @@ As such, many terms and the general structure of Studio were intuitively clear.
|
||||||
|
|
||||||
## Other misc issues
|
## Other misc issues
|
||||||
|
|
||||||
- The crosshair might be invisible if the aerial imagery is quite dark (fixed in 9dc222be433512d4d1ca530c1d09e28442d976ec)
|
- [x] The crosshair might be invisible if the aerial imagery is quite dark (fixed in 9dc222be433512d4d1ca530c1d09e28442d976ec)
|
||||||
|
|
22
Docs/UserTests/2023-10-17 User Test Studio bxl-forever.md
Normal file
22
Docs/UserTests/2023-10-17 User Test Studio bxl-forever.md
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
# User Test of the Studio
|
||||||
|
|
||||||
|
## Task
|
||||||
|
|
||||||
|
Create a simple layer specification using MapComplete studio with 'images' and a question. The actual _topic_ of the layer can be chosen by the participant
|
||||||
|
|
||||||
|
This participant wanted to create a layer to park escooters
|
||||||
|
|
||||||
|
## Background info
|
||||||
|
|
||||||
|
Browser: Participants machine, browser unknown (but no browser-specific bugs were encountered)
|
||||||
|
Testurl: hosted.Mapcomplete.org/studio.html
|
||||||
|
The participant has extensive OpenStreetMap-knowledge but only used MapComplete a few times, long ago.
|
||||||
|
|
||||||
|
## Surfaced issues
|
||||||
|
|
||||||
|
- [x] In presets, all options can be chosen (e.g. regex, '<', ...). However, these should be uploadable tags
|
||||||
|
- [x] The 'try it out'-button should be a 'next'-button
|
||||||
|
- [x] Entering an incorrect ID and pressing enter still takes you to the layer editor with an incorrect ID
|
||||||
|
- [x] A name and description are obligatory to use the layer as single-layer-theme; but those error messages are unclear.
|
||||||
|
- [ ]
|
||||||
|
- [ ]
|
|
@ -9,6 +9,7 @@ import { AllSharedLayers } from "../src/Customizations/AllSharedLayers"
|
||||||
|
|
||||||
const metainfo = {
|
const metainfo = {
|
||||||
type: "One of the inputValidator types",
|
type: "One of the inputValidator types",
|
||||||
|
typeHelper: "Helper arguments for the type input, comma-separated. Same as 'args'",
|
||||||
types: "Is multiple types are allowed for this field, then first show a mapping to pick the appropriate subtype. `Types` should be `;`-separated and contain precisely the same amount of subtypes",
|
types: "Is multiple types are allowed for this field, then first show a mapping to pick the appropriate subtype. `Types` should be `;`-separated and contain precisely the same amount of subtypes",
|
||||||
typesdefault: "Works in conjuction with `types`: this type will be selected by default",
|
typesdefault: "Works in conjuction with `types`: this type will be selected by default",
|
||||||
group: "A kind of label. Items with the same group name will be placed in the same region",
|
group: "A kind of label. Items with the same group name will be placed in the same region",
|
||||||
|
|
|
@ -97,8 +97,9 @@ export class TagUtils {
|
||||||
docs: "Both the `key` and `value` part of this specification are interpreted as regexes, both the key and value musth completely match their respective regexes",
|
docs: "Both the `key` and `value` part of this specification are interpreted as regexes, both the key and value musth completely match their respective regexes",
|
||||||
},
|
},
|
||||||
":=": {
|
":=": {
|
||||||
name: "Substitute `{some_key}` should match `key`",
|
name: "Substitute `... {some_key} ...` and match `key`",
|
||||||
overpassSupport: false,
|
overpassSupport: false,
|
||||||
|
uploadable: true,
|
||||||
docs:
|
docs:
|
||||||
"**This is an advanced feature - use with caution**\n" +
|
"**This is an advanced feature - use with caution**\n" +
|
||||||
"\n" +
|
"\n" +
|
||||||
|
|
|
@ -319,6 +319,7 @@ export interface LayerConfigJson {
|
||||||
*
|
*
|
||||||
* question: What tag should be added to the new object?
|
* question: What tag should be added to the new object?
|
||||||
* type: simple_tag
|
* type: simple_tag
|
||||||
|
* typeHelper: uploadableOnly
|
||||||
*/
|
*/
|
||||||
tags: string[]
|
tags: string[]
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -3,18 +3,29 @@
|
||||||
*/
|
*/
|
||||||
import { UIEventSource } from "../../../Logic/UIEventSource";
|
import { UIEventSource } from "../../../Logic/UIEventSource";
|
||||||
import BasicTagInput from "../../Studio/TagInput/BasicTagInput.svelte";
|
import BasicTagInput from "../../Studio/TagInput/BasicTagInput.svelte";
|
||||||
|
import { TagUtils } from "../../../Logic/Tags/TagUtils";
|
||||||
|
import * as nmd from "nano-markdown"
|
||||||
|
import FromHtml from "../../Base/FromHtml.svelte";
|
||||||
export let value: UIEventSource<undefined | string>;
|
export let value: UIEventSource<undefined | string>;
|
||||||
export let uploadableOnly: boolean;
|
export let args: string[] = [];
|
||||||
|
let uploadableOnly: boolean = args[0] === "uploadableOnly";
|
||||||
export let overpassSupportNeeded: boolean;
|
export let overpassSupportNeeded: boolean;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Only show the taginfo-statistics if they are suspicious (thus: less then 250 entries)
|
* Only show the taginfo-statistics if they are suspicious (thus: less then 250 entries)
|
||||||
*/
|
*/
|
||||||
export let silent: boolean = false;
|
export let silent: boolean = false;
|
||||||
|
let mode: string = "=";
|
||||||
|
let dropdownFocussed = new UIEventSource(false);
|
||||||
|
let documentation = TagUtils.modeDocumentation[mode];
|
||||||
|
$: documentation = TagUtils.modeDocumentation[mode];
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
||||||
<BasicTagInput {overpassSupportNeeded} {silent} tag={value} {uploadableOnly} />
|
<BasicTagInput bind:mode={mode} {dropdownFocussed} {overpassSupportNeeded} {silent} tag={value} {uploadableOnly} />
|
||||||
|
{#if $dropdownFocussed}
|
||||||
|
<div class="border border-dashed border-black p-2 m-2">
|
||||||
|
<b>{documentation.name}</b>
|
||||||
|
<FromHtml src={nmd(documentation.docs)}/>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
|
|
@ -46,7 +46,7 @@
|
||||||
{:else if type === "tag"}
|
{:else if type === "tag"}
|
||||||
<TagInput { value } />
|
<TagInput { value } />
|
||||||
{:else if type === "simple_tag"}
|
{:else if type === "simple_tag"}
|
||||||
<SimpleTagInput { value } />
|
<SimpleTagInput { value } {args} />
|
||||||
{:else if type === "opening_hours"}
|
{:else if type === "opening_hours"}
|
||||||
<OpeningHoursInput { value } />
|
<OpeningHoursInput { value } />
|
||||||
{:else if type === "wikidata"}
|
{:else if type === "wikidata"}
|
||||||
|
|
|
@ -17,7 +17,6 @@
|
||||||
export let getCountry: () => string | undefined
|
export let getCountry: () => string | undefined
|
||||||
export let placeholder: string | Translation | undefined
|
export let placeholder: string | Translation | undefined
|
||||||
export let unit: Unit = undefined
|
export let unit: Unit = undefined
|
||||||
|
|
||||||
export let value: UIEventSource<string>
|
export let value: UIEventSource<string>
|
||||||
/**
|
/**
|
||||||
* Internal state bound to the input element.
|
* Internal state bound to the input element.
|
||||||
|
|
|
@ -10,6 +10,7 @@
|
||||||
import type { LayerConfigJson } from "../../Models/ThemeConfig/Json/LayerConfigJson";
|
import type { LayerConfigJson } from "../../Models/ThemeConfig/Json/LayerConfigJson";
|
||||||
import type { ConversionMessage } from "../../Models/ThemeConfig/Conversion/Conversion";
|
import type { ConversionMessage } from "../../Models/ThemeConfig/Conversion/Conversion";
|
||||||
import ErrorIndicatorForRegion from "./ErrorIndicatorForRegion.svelte";
|
import ErrorIndicatorForRegion from "./ErrorIndicatorForRegion.svelte";
|
||||||
|
import { ChevronRightIcon } from "@rgossiaux/svelte-heroicons/solid";
|
||||||
|
|
||||||
const layerSchema: ConfigMeta[] = <any>layerSchemaRaw;
|
const layerSchema: ConfigMeta[] = <any>layerSchemaRaw;
|
||||||
|
|
||||||
|
@ -57,12 +58,15 @@
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="w-full flex justify-between">
|
<div class="w-full flex justify-between">
|
||||||
|
<slot />
|
||||||
<h3>Editing layer {$title}</h3>
|
<h3>Editing layer {$title}</h3>
|
||||||
{#if $hasErrors > 0}
|
{#if $hasErrors > 0}
|
||||||
<div class="alert">{$hasErrors} errors detected</div>
|
<div class="alert">{$hasErrors} errors detected</div>
|
||||||
{:else}
|
{:else}
|
||||||
<a class="primary button" href={baseUrl+state.server.layerUrl(title.data)} target="_blank" rel="noopener">Try it
|
<a class="primary button" href={baseUrl+state.server.layerUrl(title.data)} target="_blank" rel="noopener">
|
||||||
out</a>
|
Try it out
|
||||||
|
<ChevronRightIcon class= "h-6 w-6 shrink-0"/>
|
||||||
|
</a>
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
<div class="m4">
|
<div class="m4">
|
||||||
|
|
|
@ -21,16 +21,16 @@
|
||||||
const isTranslation = schema.hints.typehint === "translation" || schema.hints.typehint === "rendered" || ConfigMetaUtils.isTranslation(schema);
|
const isTranslation = schema.hints.typehint === "translation" || schema.hints.typehint === "rendered" || ConfigMetaUtils.isTranslation(schema);
|
||||||
let type = schema.hints.typehint ?? "string";
|
let type = schema.hints.typehint ?? "string";
|
||||||
|
|
||||||
let rendervalue = schema.type === "boolean" ? undefined : ((schema.hints.inline ?? schema.path.join(".")) + " <b>{translated(value)}</b>")
|
let rendervalue = schema.type === "boolean" ? undefined : ((schema.hints.inline ?? schema.path.join(".")) + " <b>{translated(value)}</b>");
|
||||||
let helperArgs = undefined
|
let helperArgs = schema.hints.typehelper?.split(",");
|
||||||
let inline = schema.hints.inline !== undefined
|
let inline = schema.hints.inline !== undefined;
|
||||||
if (isTranslation) {
|
if (isTranslation) {
|
||||||
type = "translation";
|
type = "translation";
|
||||||
if (schema.hints.inline) {
|
if (schema.hints.inline) {
|
||||||
const inlineValue = schema.hints.inline
|
const inlineValue = schema.hints.inline;
|
||||||
rendervalue = inlineValue
|
rendervalue = inlineValue;
|
||||||
inline = false
|
inline = false;
|
||||||
helperArgs = [inlineValue.substring(0, inlineValue.indexOf("{")), inlineValue.substring(inlineValue.indexOf("}") + 1)]
|
helperArgs = [inlineValue.substring(0, inlineValue.indexOf("{")), inlineValue.substring(inlineValue.indexOf("}") + 1)];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (type.endsWith("[]")) {
|
if (type.endsWith("[]")) {
|
||||||
|
@ -164,6 +164,8 @@
|
||||||
<div class="alert">{msg.message}</div>
|
<div class="alert">{msg.message}</div>
|
||||||
{/each}
|
{/each}
|
||||||
{/if}
|
{/if}
|
||||||
|
{#if window.location.hostname === "127.0.0.1"}
|
||||||
<span class="subtle">{schema.path.join(".")}</span>
|
<span class="subtle">{schema.path.join(".")}</span>
|
||||||
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
|
@ -12,8 +12,15 @@
|
||||||
export let uploadableOnly: boolean
|
export let uploadableOnly: boolean
|
||||||
export let overpassSupportNeeded: boolean
|
export let overpassSupportNeeded: boolean
|
||||||
|
|
||||||
|
export let dropdownFocussed = new UIEventSource(false)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If set, do not show tagInfo if there are many features matching
|
||||||
|
*/
|
||||||
export let silent : boolean = false
|
export let silent : boolean = false
|
||||||
|
|
||||||
|
export let selected: UIEventSource<boolean> = new UIEventSource<boolean>(false)
|
||||||
|
|
||||||
let feedbackGlobal = tag.map(tag => {
|
let feedbackGlobal = tag.map(tag => {
|
||||||
if (!tag) {
|
if (!tag) {
|
||||||
return undefined
|
return undefined
|
||||||
|
@ -38,7 +45,7 @@
|
||||||
let valueValue = new UIEventSource<string>(undefined)
|
let valueValue = new UIEventSource<string>(undefined)
|
||||||
|
|
||||||
|
|
||||||
let mode: string = "="
|
export let mode: string = "="
|
||||||
let modes: string[] = []
|
let modes: string[] = []
|
||||||
|
|
||||||
for (const k in TagUtils.modeDocumentation) {
|
for (const k in TagUtils.modeDocumentation) {
|
||||||
|
@ -105,7 +112,7 @@
|
||||||
|
|
||||||
<ValidatedInput feedback={feedbackKey} placeholder="The key of the tag" type="key"
|
<ValidatedInput feedback={feedbackKey} placeholder="The key of the tag" type="key"
|
||||||
value={keyValue}></ValidatedInput>
|
value={keyValue}></ValidatedInput>
|
||||||
<select bind:value={mode}>
|
<select bind:value={mode} on:focusin={() => dropdownFocussed.setData(true)} on:focusout={() => dropdownFocussed.setData(false)}>
|
||||||
{#each modes as option}
|
{#each modes as option}
|
||||||
<option value={option}>
|
<option value={option}>
|
||||||
{option}
|
{option}
|
||||||
|
|
|
@ -3,9 +3,13 @@ import { JsonSchema, JsonSchemaType } from "./jsonSchema"
|
||||||
export interface ConfigMeta {
|
export interface ConfigMeta {
|
||||||
path: string[]
|
path: string[]
|
||||||
type: JsonSchemaType | JsonSchema[]
|
type: JsonSchemaType | JsonSchema[]
|
||||||
|
/**
|
||||||
|
* All fields are lowercase, as they should be case-insensitive
|
||||||
|
*/
|
||||||
hints: {
|
hints: {
|
||||||
group?: string
|
group?: string
|
||||||
typehint?: string
|
typehint?: string
|
||||||
|
typehelper?: string
|
||||||
/**
|
/**
|
||||||
* If multiple subcategories can be chosen
|
* If multiple subcategories can be chosen
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -16,6 +16,7 @@
|
||||||
|
|
||||||
import layerSchemaRaw from "../../src/assets/schemas/layerconfigmeta.json";
|
import layerSchemaRaw from "../../src/assets/schemas/layerconfigmeta.json";
|
||||||
import If from "./Base/If.svelte";
|
import If from "./Base/If.svelte";
|
||||||
|
import BackButton from "./Base/BackButton.svelte";
|
||||||
|
|
||||||
export let studioUrl = window.location.hostname === "127.0.0.1" ? "http://127.0.0.1:1235" : "https://studio.mapcomplete.org";
|
export let studioUrl = window.location.hostname === "127.0.0.1" ? "http://127.0.0.1:1235" : "https://studio.mapcomplete.org";
|
||||||
const studio = new StudioServer(studioUrl);
|
const studio = new StudioServer(studioUrl);
|
||||||
|
@ -42,6 +43,7 @@
|
||||||
const layerSchema: ConfigMeta[] = <any>layerSchemaRaw;
|
const layerSchema: ConfigMeta[] = <any>layerSchemaRaw;
|
||||||
|
|
||||||
let editLayerState = new EditLayerState(layerSchema, studio);
|
let editLayerState = new EditLayerState(layerSchema, studio);
|
||||||
|
let layerId = editLayerState.configuration.map(layerConfig => layerConfig.id);
|
||||||
|
|
||||||
function fetchIconDescription(layerId): any {
|
function fetchIconDescription(layerId): any {
|
||||||
return AllSharedLayers.getSharedLayersConfigs().get(layerId)?._layerIcon;
|
return AllSharedLayers.getSharedLayersConfigs().get(layerId)?._layerIcon;
|
||||||
|
@ -49,8 +51,8 @@
|
||||||
|
|
||||||
async function createNewLayer() {
|
async function createNewLayer() {
|
||||||
if (layerIdFeedback.data !== undefined) {
|
if (layerIdFeedback.data !== undefined) {
|
||||||
console.warn("There is still some feedback - not starting to create a new layer")
|
console.warn("There is still some feedback - not starting to create a new layer");
|
||||||
return
|
return;
|
||||||
}
|
}
|
||||||
state = "loading";
|
state = "loading";
|
||||||
const id = newLayerId.data;
|
const id = newLayerId.data;
|
||||||
|
@ -142,6 +144,9 @@
|
||||||
-->
|
-->
|
||||||
</div>
|
</div>
|
||||||
{:else if state === "edit_layer"}
|
{:else if state === "edit_layer"}
|
||||||
|
|
||||||
|
<BackButton clss="small p-1" imageClass="w-8 h-8" on:click={() => {state =undefined}}>MapComplete Studio</BackButton>
|
||||||
|
<h3>Choose a layer to edit</h3>
|
||||||
<div class="flex flex-wrap">
|
<div class="flex flex-wrap">
|
||||||
{#each Array.from($layers) as layerId}
|
{#each Array.from($layers) as layerId}
|
||||||
<NextButton clss="small" on:click={async () => {
|
<NextButton clss="small" on:click={async () => {
|
||||||
|
@ -157,6 +162,7 @@
|
||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
{:else if state === "new_layer"}
|
{:else if state === "new_layer"}
|
||||||
|
|
||||||
<div class="interactive flex m-2 rounded-2xl flex-col p-2">
|
<div class="interactive flex m-2 rounded-2xl flex-col p-2">
|
||||||
<h3>Enter the ID for the new layer</h3>
|
<h3>Enter the ID for the new layer</h3>
|
||||||
A good ID is:
|
A good ID is:
|
||||||
|
@ -185,7 +191,9 @@
|
||||||
<Loading />
|
<Loading />
|
||||||
</div>
|
</div>
|
||||||
{:else if state === "editing_layer"}
|
{:else if state === "editing_layer"}
|
||||||
<EditLayer {initialLayerConfig} state={editLayerState} />
|
<EditLayer {initialLayerConfig} state={editLayerState} >
|
||||||
|
<BackButton clss="small p-1" imageClass="w-8 h-8" on:click={() => {state =undefined}}>MapComplete Studio</BackButton>
|
||||||
|
</EditLayer>
|
||||||
{/if}
|
{/if}
|
||||||
</LoginToggle>
|
</LoginToggle>
|
||||||
</If>
|
</If>
|
||||||
|
|
|
@ -9769,6 +9769,7 @@
|
||||||
"required": true,
|
"required": true,
|
||||||
"hints": {
|
"hints": {
|
||||||
"typehint": "simple_tag",
|
"typehint": "simple_tag",
|
||||||
|
"typehelper": "uploadableOnly",
|
||||||
"question": "What tag should be added to the new object?"
|
"question": "What tag should be added to the new object?"
|
||||||
},
|
},
|
||||||
"type": "array",
|
"type": "array",
|
||||||
|
|
|
@ -593,7 +593,7 @@
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"tags": {
|
"tags": {
|
||||||
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag",
|
"description": "A single tag (encoded as <code>key=value</code>) out of all the tags to add onto the newly created point.\nNote that the icon in the UI will be chosen automatically based on the tags provided here.\n\nquestion: What tag should be added to the new object?\ntype: simple_tag\ntypeHelper: uploadableOnly",
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
"type": "string"
|
"type": "string"
|
||||||
|
@ -11193,6 +11193,7 @@
|
||||||
"required": true,
|
"required": true,
|
||||||
"hints": {
|
"hints": {
|
||||||
"typehint": "simple_tag",
|
"typehint": "simple_tag",
|
||||||
|
"typehelper": "uploadableOnly",
|
||||||
"question": "What tag should be added to the new object?"
|
"question": "What tag should be added to the new object?"
|
||||||
},
|
},
|
||||||
"type": "array",
|
"type": "array",
|
||||||
|
@ -28077,6 +28078,7 @@
|
||||||
"required": true,
|
"required": true,
|
||||||
"hints": {
|
"hints": {
|
||||||
"typehint": "simple_tag",
|
"typehint": "simple_tag",
|
||||||
|
"typehelper": "uploadableOnly",
|
||||||
"question": "What tag should be added to the new object?"
|
"question": "What tag should be added to the new object?"
|
||||||
},
|
},
|
||||||
"type": "array",
|
"type": "array",
|
||||||
|
|
Loading…
Reference in a new issue