diff --git a/Customizations/Layers/Park.ts b/Customizations/Layers/Park.ts
index 373643500..e46dc2585 100644
--- a/Customizations/Layers/Park.ts
+++ b/Customizations/Layers/Park.ts
@@ -9,6 +9,18 @@ import {NameInline} from "../Questions/NameInline";
export class Park extends LayerDefinition {
+
+ private accessByDefault = new TagRenderingOptions({
+ question: "Is dit park publiek toegankelijk?",
+ mappings: [
+ {k: new Tag("access","yes"), txt: "Publiek toegankelijk"},
+ {k: new Tag("access",""), txt: "Publiek toegankelijk"},
+ {k: new Tag("access","no"), txt: "Niet-publiek toegankelijk park"},
+ {k: new Tag("access","guided"), txt: "Enkel toegankelijk met een gids of op een activiteit"}
+ ]
+ })
+
+
constructor() {
super();
this.name = "park";
@@ -22,7 +34,10 @@ export class Park extends LayerDefinition {
this.minzoom = 13;
this.style = this.generateStyleFunction();
this.title = new NameInline("park");
- this.elementsToShow = [new NameQuestion()];
+ this.elementsToShow = [new NameQuestion(),
+ this.accessByDefault
+
+ ];
}
diff --git a/Customizations/Layouts/BikePumps.ts b/Customizations/Layouts/BikePumps.ts
index a79730542..a3902fb0a 100644
--- a/Customizations/Layouts/BikePumps.ts
+++ b/Customizations/Layouts/BikePumps.ts
@@ -6,16 +6,16 @@ export class BikePumpsLayout extends Layout {
constructor() {
super(
"pomp",
- "Grb import fix tool",
+ "Cyclofix",
[new BikePumps()],
15,
51.2083,
3.2279,
- "
GRB Fix tool
\n" +
+ "Open CycloFix
\n" +
"\n" +
- "Expert use only"
+ "Something something bikes"
,
"", "");
diff --git a/Customizations/TagRendering.ts b/Customizations/TagRendering.ts
index 8c11961f3..f94b1124b 100644
--- a/Customizations/TagRendering.ts
+++ b/Customizations/TagRendering.ts
@@ -65,7 +65,6 @@ export class TagRenderingOptions {
mappings?: { k: TagsFilter, txt: string, priority?: number, substitute?: boolean }[]
}) {
this.options = options;
-
}
@@ -150,6 +149,7 @@ export class TagRendering extends UIElement {
// Prepare the choices for the Radio buttons
let i = 0;
const choices: UIElement[] = [];
+ const alreadyUsedTexts: string[] = [];
for (const choice of options.mappings ?? []) {
if (choice.k === null) {
@@ -159,16 +159,21 @@ export class TagRendering extends UIElement {
let choiceSubbed = choice;
if (choice.substitute) {
choiceSubbed = {
- k : choice.k.substituteValues(
+ k: choice.k.substituteValues(
options.tagsPreprocessor(this._source.data)),
- txt : this.ApplyTemplate(choice.txt),
+ txt: this.ApplyTemplate(choice.txt),
substitute: false,
priority: choice.priority
}
}
-
- choices.push(new FixedUiElement(choiceSubbed.txt));
+ const txt = choiceSubbed.txt;
+ if (alreadyUsedTexts.indexOf(txt) < 0) {
+ choices.push(new FixedUiElement(txt));
+ alreadyUsedTexts.push(txt);
+ }
+
+
this._mapping.push(choiceSubbed);
i++;
}
diff --git a/Logic/ImageSearcher.ts b/Logic/ImageSearcher.ts
index 7e7b5dd66..dc4815313 100644
--- a/Logic/ImageSearcher.ts
+++ b/Logic/ImageSearcher.ts
@@ -105,9 +105,8 @@ export class ImageSearcher extends UIEventSource {
if(key === undefined){
return;
}
- console.log("Deleting image...");
-
- // this._changes.addChange(this._tags.data.id, key, "");
+ console.log("Deleting image...", key, " --> ", url);
+ this._changes.addChange(this._tags.data.id, key, "");
}
@@ -133,16 +132,11 @@ export class ImageSearcher extends UIEventSource {
}
}
- const image0 = this._tags.data["image:0"];
- if (image0 !== undefined) {
- this.AddImage(image0);
- }
- let imageIndex = 1;
- let imagei = this._tags.data["image:" + imageIndex];
- while (imagei !== undefined) {
- this.AddImage(imagei);
- imageIndex++;
- imagei = this._tags.data["image:" + imageIndex];
+ for (const key in this._tags.data) {
+ // @ts-ignore
+ if(key.startsWith("image:")){
+ this.AddImage(this._tags.data[key]);
+ }
}
const wdItem = this._tags.data.wikidata;
diff --git a/README.md b/README.md
index 5cd383242..448373bcf 100644
--- a/README.md
+++ b/README.md
@@ -78,7 +78,8 @@ Camera Icon, Dave Gandy, CC-BY-SA 3.0
https://commons.wikimedia.org/wiki/File:OOjs_UI_indicator_search-rtl.svg
Search Icon, MIT
-
+https://commons.wikimedia.org/wiki/File:Trash_font_awesome.svg
+Trash icon by Dave Gandy, CC-BY-SA
https://commons.wikimedia.org/wiki/File:Home-icon.svg
Home icon by Timothy Miller, CC-BY-SA 3.0
diff --git a/UI/ConfirmDialog.ts b/UI/ConfirmDialog.ts
new file mode 100644
index 000000000..83954011a
--- /dev/null
+++ b/UI/ConfirmDialog.ts
@@ -0,0 +1,69 @@
+import {UIElement} from "./UIElement";
+import {UIEventSource} from "./UIEventSource";
+import {FixedUiElement} from "./Base/FixedUiElement";
+import {VariableUiElement} from "./Base/VariableUIElement";
+
+
+export class ConfirmDialog extends UIElement {
+ private _showOptions: UIEventSource = new UIEventSource(false);
+
+ private _question: UIElement;
+ private _optionA: UIElement;
+ private _optionB: UIElement;
+
+ constructor(
+ show: UIEventSource,
+ question: string,
+ optionA: string, optionB: string,
+ executeA: () => void,
+ executeB: () => void,
+ classA: string = "",
+ classB: string = "") {
+ super(show);
+ this.ListenTo(this._showOptions);
+ const self = this;
+ show.addCallback(() => {
+ self._showOptions.setData(false);
+ })
+ this._question = new FixedUiElement("" + question + "")
+ .onClick(() => {
+ self._showOptions.setData(!self._showOptions.data);
+ });
+ this._optionA = new VariableUiElement(
+ this._showOptions.map(
+ (show) => show ? "" + optionA + "
" : ""))
+ .onClick(() => {
+ self._showOptions.setData(false);
+ executeA();
+ }
+ );
+ this._optionB = new VariableUiElement(
+ this._showOptions.map((show) =>
+ show ? "" + optionB + "
" : "") )
+ .onClick(() => {
+ self._showOptions.setData(false);
+ executeB();
+ });
+
+
+
+ }
+
+ protected InnerRender(): string {
+ if (!this._source.data) {
+ return "";
+ }
+
+ return this._question.Render() +
+ this._optionA.Render() +
+ this._optionB.Render();
+ }
+
+ Update() {
+ super.Update();
+ this._question.Update();
+ this._optionA.Update();
+ this._optionB.Update();
+ }
+
+}
\ No newline at end of file
diff --git a/UI/Image/ImageCarousel.ts b/UI/Image/ImageCarousel.ts
index 05682c408..21d36f64d 100644
--- a/UI/Image/ImageCarousel.ts
+++ b/UI/Image/ImageCarousel.ts
@@ -6,6 +6,7 @@ import {FixedUiElement} from "../Base/FixedUiElement";
import {VerticalCombine} from "../Base/VerticalCombine";
import {Changes} from "../../Logic/Changes";
import {VariableUiElement} from "../Base/VariableUIElement";
+import {ConfirmDialog} from "../ConfirmDialog";
export class ImageCarousel extends UIElement {
/**
@@ -25,7 +26,6 @@ export class ImageCarousel extends UIElement {
private readonly _uiElements: UIEventSource;
- private readonly _deleteButtonText = new UIEventSource("");
private readonly _deleteButton: UIElement;
constructor(tags: UIEventSource, changes: Changes) {
@@ -48,24 +48,33 @@ export class ImageCarousel extends UIElement {
new FixedUiElement("")).HideOnEmpty(true);
- this._deleteButtonText = this.slideshow._currentSlide.map((i) => {
- if(self.searcher.IsDeletable(self.searcher.data[i])){
- return "DELETE";
- }else{
- return "";
- }
- });
+ const showDeleteButton = this.slideshow._currentSlide.map((i) => {
+ return self.searcher.IsDeletable(self.searcher.data[i]);
+ }, [this.searcher]);
+ this.slideshow._currentSlide.addCallback(() => {
+ showDeleteButton.ping(); // This pings the showDeleteButton, which indicates that it has to hide it's subbuttons
+ })
- this._deleteButton = new VariableUiElement(this._deleteButtonText)
- .HideOnEmpty(true)
- .onClick(() => {
- self.searcher.Delete(self.searcher.data[self.slideshow._currentSlide.data]);
- });
+
+ const deleteCurrent = () => self.searcher.Delete(self.searcher.data[self.slideshow._currentSlide.data]);
+
+ this._deleteButton = new ConfirmDialog(showDeleteButton,
+ "",
+ "Afbeelding verwijderen",
+ "Terug",
+ deleteCurrent,
+ () => {},
+ 'delete-image-confirm',
+ 'delete-image-cancel');
}
InnerRender(): string {
- return this.slideshow.Render() ;
- // + this._deleteButton.Render();
+ return "" +
+ "" +
+ this._deleteButton.Render() +
+ "
" +
+ this.slideshow.Render() +
+ "";
}
InnerUpdate(htmlElement: HTMLElement) {
diff --git a/UI/UIElement.ts b/UI/UIElement.ts
index 5a7204332..57f142753 100644
--- a/UI/UIElement.ts
+++ b/UI/UIElement.ts
@@ -58,6 +58,17 @@ export abstract class UIElement {
}
element.style.pointerEvents = "all";
element.style.cursor = "pointer";
+ /*
+ const childs = element.children;
+ for (let i = 0; i < childs.length; i++) {
+ const ch = childs[i];
+ console.log(ch);
+ ch.style.cursor = "pointer";
+ ch.onclick = () => {
+ self._onClick();
+ }
+ ch.style.pointerEvents = "all";
+ }*/
}
this.InnerUpdate(element);
diff --git a/UI/UIEventSource.ts b/UI/UIEventSource.ts
index f1a9909c8..eae21515c 100644
--- a/UI/UIEventSource.ts
+++ b/UI/UIEventSource.ts
@@ -27,15 +27,23 @@ export class UIEventSource{
}
}
- public map(f: ((T) => J)): UIEventSource {
+ public map(f: ((T) => J),
+ extraSources : UIEventSource[] = []): UIEventSource {
const self = this;
- this.addCallback(function () {
+
+ const update = function () {
newSource.setData(f(self.data));
newSource.ping();
- });
+ }
+
+ this.addCallback(update);
+ for (const extraSource of extraSources) {
+ extraSource.addCallback(update);
+ }
const newSource = new UIEventSource(
f(this.data)
);
+
return newSource;
diff --git a/assets/delete.svg b/assets/delete.svg
new file mode 100644
index 000000000..60bf193af
--- /dev/null
+++ b/assets/delete.svg
@@ -0,0 +1,55 @@
+
+
diff --git a/index.css b/index.css
index f95ddb02f..f61ecac65 100644
--- a/index.css
+++ b/index.css
@@ -682,6 +682,84 @@ body {
display: inline-block
}
+/******* THe remove image buttons ****/
+
+.image-carousel-container {
+ position: relative;
+}
+
+.image-delete-container {
+ position: absolute;
+ left: 6em;
+ top: 1.5em;
+ display: inline-block;
+ z-index: 7000;
+
+}
+
+.delete-image {
+ width: 1.5em;
+ height: 1.5em;
+ padding: 0.5em;
+ border-radius: 3em;
+ background-color: black;
+}
+
+.delete-image-confirm {
+ position: absolute;
+ display: inline-block;
+ left: 0;
+ top: 2.5em;
+
+ padding: 0.5em;
+ padding-left: 0.75em;
+
+ z-index: -1;
+ height: 3em;
+ width: 14em;
+ border-radius: 1em;
+ border-top-left-radius: 0;
+ border-top-right-radius: 0;
+ background-color: #ff8c8c;
+
+ color: white;
+ height: 1.5em; /* same as .delete-image */
+
+ z-index: 7000;
+}
+
+.delete-image-confirm span {
+ font-size: larger;
+ font-weight: bold;
+}
+
+
+.delete-image-cancel {
+ display: inline-block;
+ position: absolute;
+
+ left: 0em;
+ padding: 0.5em;
+ padding-left: 0.75em;
+
+ border-radius: 1em;
+ border-bottom-right-radius: 0;
+ border-bottom-left-radius: 0;
+
+ height: 1.5em; /* same as .delete-image */
+ width: 14em; /* Same as delete-image-confirm */
+
+
+ background-color: black;
+ color: white;
+ z-index: 7000;
+}
+
+.delete-image-cancel span {
+ font-size: larger;
+ font-weight: bold;
+
+}
/**** The save button *****/
diff --git a/test.html b/test.html
index b5f19fbc3..ad22b5de6 100644
--- a/test.html
+++ b/test.html
@@ -5,7 +5,9 @@
+
'maindiv' not attached
+
diff --git a/test.ts b/test.ts
index 90794353e..f00640079 100644
--- a/test.ts
+++ b/test.ts
@@ -4,18 +4,20 @@ import {OsmConnection} from "./Logic/OsmConnection";
import {ElementStorage} from "./Logic/ElementStorage";
import {WikipediaLink} from "./Customizations/Questions/WikipediaLink";
import {OsmLink} from "./Customizations/Questions/OsmLink";
-
-const tags = {name: "Test",
- wikipedia: "nl:Pieter",
- id: "node/-1"};
-const tagsES = new UIEventSource(tags);
-
-const login = new OsmConnection(true);
-
-const allElements = new ElementStorage();
-allElements.addElementById(tags.id, tagsES);
-
-const changes = new Changes("Test", login, allElements)
+import {ConfirmDialog} from "./UI/ConfirmDialog";
-new OsmLink(tagsES, changes).AttachTo("maindiv");
\ No newline at end of file
+new ConfirmDialog(new UIEventSource(true),
+ "",
+ "Deze afbeelding verwijderen",
+ "Terug",
+
+ () => {
+ console.log("Verwijderen");
+ },
+ () => {
+ console.log("terug")
+ },
+ 'delete-image-confirm',
+ 'delete-image-cancel')
+ .AttachTo("maindiv")
\ No newline at end of file