diff --git a/Logic/Actors/OverpassFeatureSource.ts b/Logic/Actors/OverpassFeatureSource.ts index e292f5b4e..e2ab5ea87 100644 --- a/Logic/Actors/OverpassFeatureSource.ts +++ b/Logic/Actors/OverpassFeatureSource.ts @@ -191,6 +191,9 @@ export default class OverpassFeatureSource implements FeatureSource { const self = this const overpassUrls = self.state.overpassUrl.data + if(overpassUrls === undefined || overpassUrls.length === 0){ + throw "Panic: overpassFeatureSource didn't receive any overpassUrls" + } let bounds: BBox do { try { diff --git a/Logic/Osm/Overpass.ts b/Logic/Osm/Overpass.ts index 7bb1dbf8f..f9ad43f7c 100644 --- a/Logic/Osm/Overpass.ts +++ b/Logic/Osm/Overpass.ts @@ -14,7 +14,7 @@ export class Overpass { private readonly _interpreterUrl: string private readonly _timeout: Store private readonly _extraScripts: string[] - private _includeMeta: boolean + private readonly _includeMeta: boolean private _relationTracker: RelationsTracker constructor( diff --git a/Logic/State/FeatureSwitchState.ts b/Logic/State/FeatureSwitchState.ts index c3d9a0bb9..e690307d0 100644 --- a/Logic/State/FeatureSwitchState.ts +++ b/Logic/State/FeatureSwitchState.ts @@ -166,9 +166,9 @@ export default class FeatureSwitchState { (layoutToUse?.overpassUrl ?? Constants.defaultOverpassUrls).join(","), "Point mapcomplete to a different overpass-instance. Example: https://overpass-api.de/api/interpreter" ).sync( - (param) => param.split(","), + (param) => param?.split(","), [], - (urls) => urls.join(",") + (urls) => urls?.join(",") ) this.overpassTimeout = UIEventSource.asFloat( diff --git a/Logic/Web/QueryParameters.ts b/Logic/Web/QueryParameters.ts index 398247281..501da39b4 100644 --- a/Logic/Web/QueryParameters.ts +++ b/Logic/Web/QueryParameters.ts @@ -6,11 +6,11 @@ import Hash from "./Hash" import { Utils } from "../../Utils" export class QueryParameters { - static defaults = {} + static defaults : Record = {} static documentation: Map = new Map() private static order: string[] = ["layout", "test", "z", "lat", "lon"] - private static _wasInitialized: Set = new Set() - private static knownSources = {} + protected static readonly _wasInitialized: Set = new Set() + protected static readonly knownSources: Record> = {} private static initialized = false public static GetQueryParameter( @@ -105,14 +105,19 @@ export class QueryParameters { } if (!Utils.runningFromConsole) { // Don't pollute the history every time a parameter changes - history.replaceState(null, "", "?" + parts.join("&") + Hash.Current()) + try{ + history.replaceState(null, "", "?" + parts.join("&") + Hash.Current()) + }catch(e){ + console.error(e) + } } } static ClearAll() { for (const name in QueryParameters.knownSources) { - QueryParameters.knownSources[name].setData("") + QueryParameters.knownSources[name].setData(undefined) } QueryParameters._wasInitialized.clear() + QueryParameters.order = [] } } diff --git a/UI/Base/MinimapImplementation.ts b/UI/Base/MinimapImplementation.ts index 8ec5c8522..3e507c223 100644 --- a/UI/Base/MinimapImplementation.ts +++ b/UI/Base/MinimapImplementation.ts @@ -119,6 +119,7 @@ export default class MinimapImplementation extends BaseUIElement implements Mini public async TakeScreenshot(format: "blob"): Promise ; public async TakeScreenshot(format: "image" | "blob"): Promise ; public async TakeScreenshot(format: "image" | "blob" = "image"): Promise { + console.log("Taking a screenshot...") const screenshotter = new SimpleMapScreenshoter() screenshotter.addTo(this.leafletMap.data) const result = await screenshotter.takeScreen(( format) ?? "image") diff --git a/UI/BigComponents/PdfExportGui.ts b/UI/BigComponents/PdfExportGui.ts new file mode 100644 index 000000000..b70dd3215 --- /dev/null +++ b/UI/BigComponents/PdfExportGui.ts @@ -0,0 +1,272 @@ +import Combine from "../Base/Combine"; +import {FlowPanelFactory, FlowStep} from "../ImportFlow/FlowStep"; +import {ImmutableStore, Store, UIEventSource} from "../../Logic/UIEventSource"; +import {InputElement} from "../Input/InputElement"; +import {SvgToPdf, SvgToPdfOptions} from "../../Utils/svgToPdf"; +import {FixedInputElement} from "../Input/FixedInputElement"; +import {FixedUiElement} from "../Base/FixedUiElement"; +import FileSelectorButton from "../Input/FileSelectorButton"; +import InputElementMap from "../Input/InputElementMap"; +import {RadioButton} from "../Input/RadioButton"; +import {Utils} from "../../Utils"; +import {VariableUiElement} from "../Base/VariableUIElement"; +import Loading from "../Base/Loading"; +import BaseUIElement from "../BaseUIElement"; +import Img from "../Base/Img"; +import Title from "../Base/Title"; +import CheckBoxes, {CheckBox} from "../Input/Checkboxes"; +import Minimap from "../Base/Minimap"; +import SearchAndGo from "./SearchAndGo"; +import Toggle from "../Input/Toggle"; +import List from "../Base/List"; +import LeftIndex from "../Base/LeftIndex"; +import Constants from "../../Models/Constants"; + +class SelectTemplate extends Combine implements FlowStep<{ title: string, pages: string[] }> { + readonly IsValid: Store; + readonly Value: Store<{ title: string, pages: string[] }>; + + constructor() { + const elements: InputElement<{ templateName: string, pages: string[] }>[] = [] + for (const templateName in SvgToPdf.templates) { + const template = SvgToPdf.templates[templateName] + elements.push(new FixedInputElement( + new Combine([new FixedUiElement(templateName).SetClass("font-bold pr-2"), + template.description + ]) + , new UIEventSource({templateName, pages: template.pages}))) + } + + const file = new FileSelectorButton(new FixedUiElement("Select an svg image which acts as template"), { + acceptType: "image/svg+xml", + allowMultiple: true + }) + const fileMapped = new InputElementMap(file, (x0, x1) => x0 === x1, + (filelist) => { + if (filelist === undefined) { + return undefined; + } + const pages = [] + let templateName: string = undefined; + for (const file of Array.from(filelist)) { + + if (templateName == undefined) { + templateName = file.name.substring(file.name.lastIndexOf("/") + 1) + templateName = templateName.substring(0, templateName.lastIndexOf(".")) + } + pages.push(file.text()) + } + + return { + templateName, + pages, + fromFile: true + } + + }, + fromX => undefined + ) + elements.push(fileMapped) + const radio = new RadioButton(elements, {selectFirstAsDefault: true}) + + const loaded: Store<{ success: { title: string, pages: string[] } } | { error: any }> = radio.GetValue().bind(template => { + if (template === undefined) { + return undefined + } + if (template["fromFile"]) { + return UIEventSource.FromPromiseWithErr(Promise.all(template.pages).then(pages => ({ + title: template.templateName, + pages + }))) + } + const urls = template.pages.map(p => SelectTemplate.ToUrl(p)) + const dloadAll: Promise<{ title: string, pages: string[] }> = Promise.all(urls.map(url => Utils.download(url))).then(pages => ({ + pages, + title: template.templateName + })) + + return UIEventSource.FromPromiseWithErr(dloadAll) + }) + const preview = new VariableUiElement( + loaded.map(pages => { + if (pages === undefined) { + return new Loading() + } + if (pages["err"] !== undefined) { + return new FixedUiElement("Loading preview failed: " + pages["err"]).SetClass("alert") + } + const svgs = pages["success"].pages + if (svgs.length === 0) { + return new FixedUiElement("No pages loaded...").SetClass("alert") + } + const els: BaseUIElement[] = [] + for (const pageSrc of svgs) { + const el = new Img(pageSrc, true) + .SetClass("w-96 m-2 border-black border-2") + els.push(el) + } + return new Combine(els).SetClass("flex border border-subtle rounded-xl"); + }) + ) + + super([ + new Title("Select template"), + radio, + new Title("Preview"), + preview + ]); + this.Value = loaded.map(l => l === undefined ? undefined : l["success"]) + this.IsValid = this.Value.map(v => v !== undefined) + } + + public static ToUrl(spec: string) { + if (spec.startsWith("http")) { + return spec + } + return window.location.protocol + "//" + window.location.host + "/" + spec + } + +} + +class SelectPdfOptions extends Combine implements FlowStep<{ title: string, pages: string[], options: SvgToPdfOptions }> { + readonly IsValid: Store; + readonly Value: Store<{ title: string, pages: string[], options: SvgToPdfOptions }>; + + constructor(title: string, pages: string[], getFreeDiv: () => string) { + const dummy = new CheckBox("Don't add data to the map (to quickly preview the PDF)", false) + const overrideMapLocation = new CheckBox("Override map location: use a selected location instead of the location set in the template", false) + const locationInput = Minimap.createMiniMap().SetClass("block w-full") + const searchField = new SearchAndGo({leafletMap: locationInput.leafletMap}) + const selectLocation = + new Combine([ + new Toggle(new Combine([new Title("Select override location"), searchField]).SetClass("flex"), undefined, overrideMapLocation.GetValue()), + new Toggle(locationInput.SetStyle("height: 20rem"), undefined, overrideMapLocation.GetValue()).SetStyle("height: 20rem") + ]).SetClass("block").SetStyle("height: 25rem") + super([new Title("Select options"), + dummy, + overrideMapLocation, + selectLocation + ]); + this.Value = dummy.GetValue().map((disableMaps) => { + return { + pages, + title, + options: { + disableMaps, + getFreeDiv, + overrideLocation: overrideMapLocation.GetValue().data ? locationInput.location.data : undefined + } + } + }, [overrideMapLocation.GetValue(), locationInput.location]) + this.IsValid = new ImmutableStore(true) + } + +} + +class PreparePdf extends Combine implements FlowStep<{ svgToPdf: SvgToPdf, languages: string[] }> { + readonly IsValid: Store; + readonly Value: Store<{ svgToPdf: SvgToPdf, languages: string[] }>; + + constructor(title: string, pages: string[], options: SvgToPdfOptions) { + const svgToPdf = new SvgToPdf(title, pages, options) + const languageOptions = [ + new FixedInputElement("Nederlands", "nl"), + new FixedInputElement("English", "en") + ] + const languages = new CheckBoxes(languageOptions) + + const isPrepared = UIEventSource.FromPromiseWithErr(svgToPdf.Prepare()) + + super([ + new Title("Select languages..."), + languages, + new Toggle( + new Loading("Preparing maps..."), + undefined, + isPrepared.map(p => p === undefined) + ) + ]); + this.Value = isPrepared.map(isPrepped => { + if (isPrepped === undefined) { + return undefined + } + if (isPrepped["success"] !== undefined) { + const svgToPdf = isPrepped["success"] + const langs = languages.GetValue().data.map(i => languageOptions[i].GetValue().data) + if (langs.length === 0) { + return undefined + } + return {svgToPdf, languages: langs} + } + return undefined; + }, [languages.GetValue()]) + this.IsValid = this.Value.map(v => v !== undefined) + } + +} + + +class SavePdf extends Combine { + + constructor(svgToPdf: SvgToPdf, languages: string[]) { + + super([ + new Title("Generating your pdfs..."), + new List(languages.map(lng => new Toggle( + lng + " is done!", + new Loading("Creating pdf for " + lng), + UIEventSource.FromPromiseWithErr(svgToPdf.ConvertSvg(lng).then(() => true)) + .map(x => x !== undefined && x["success"] === true) + ))) + ]); + } +} + +export class PdfExportGui extends LeftIndex { + + + constructor(freeDivId: string) { + + let i = 0 + const createDiv = (): string => { + const div = document.createElement("div") + div.id = "freediv-" + (i++) + document.getElementById(freeDivId).append(div) + return div.id + } + + Constants.defaultOverpassUrls.splice(0, 1) + const {flow, furthestStep, titles} = FlowPanelFactory.start( + new Title("Select template"), new SelectTemplate() + ).then(new Title("Select options"), ({title, pages}) => new SelectPdfOptions(title, pages, createDiv)) + .then("Generate maps...", ({title, pages, options}) => new PreparePdf(title, pages, options)) + .finish("Generating...", ({svgToPdf, languages}) => new SavePdf(svgToPdf, languages)) + + + const toc = new List( + titles.map( + (title, i) => + new VariableUiElement( + furthestStep.map((currentStep) => { + if (i > currentStep) { + return new Combine([title]).SetClass("subtle") + } + if (i == currentStep) { + return new Combine([title]).SetClass("font-bold") + } + if (i < currentStep) { + return title + } + }) + ) + ), + true + ) + + const leftContents: BaseUIElement[] = [ + toc + ].map((el) => el?.SetClass("pl-4")) + + super(leftContents, flow) + } +} diff --git a/Utils/pngMapCreator.ts b/Utils/pngMapCreator.ts index f17066c29..d8f45b620 100644 --- a/Utils/pngMapCreator.ts +++ b/Utils/pngMapCreator.ts @@ -96,12 +96,10 @@ export class PngMapCreator { }) await Utils.waitFor(2000) } - minimap.TakeScreenshot(format).then(result => { + minimap.TakeScreenshot(format).then(async result => { const divId = this._options.divId - window.setTimeout(() => { + await Utils.waitFor(250) document.getElementById(divId).removeChild(/*Will fetch the cached htmlelement:*/minimap.ConstructElement()) - - }, 500) return resolve(result); }).catch(failreason => { console.error("Could no make a screenshot due to ",failreason) diff --git a/Utils/svgToPdf.ts b/Utils/svgToPdf.ts index 0ab5e1a36..9ade36468 100644 --- a/Utils/svgToPdf.ts +++ b/Utils/svgToPdf.ts @@ -13,6 +13,7 @@ import {Utils} from "../Utils"; import Locale from "../UI/i18n/Locale"; import Constants from "../Models/Constants"; import Hash from "../Logic/Web/Hash"; +import {QueryParameters} from "../Logic/Web/QueryParameters"; class SvgToPdfInternals { private readonly doc: jsPDF; @@ -213,7 +214,7 @@ class SvgToPdfInternals { private extractTranslation(text: string) { if(text === "$version"){ - return new Date().toISOString().substring("2022-01-02THH:MM".length )+" - v"+Constants.vNumber + return new Date().toISOString().substring(0, "2022-01-02THH:MM".length )+" - v"+Constants.vNumber } const pathPart = text.match(/\$(([_a-zA-Z0-9?]+\.)+[_a-zA-Z0-9?]+)(.*)/) if (pathPart === null) { @@ -289,33 +290,57 @@ class SvgToPdfInternals { this.doc.setFontSize(fontsize * 2.5) let textTemplate = tspan.textContent.split(" ") - let result: string[] = [] + let result: string = "" + let addSpace = false for (let text of textTemplate) { - - if (!text.startsWith("$")) { - result.push(text) + if(text === "\\n"){ + result += "\n" + addSpace = false continue } - if (text.startsWith("$list(")) { - text = text.substring("$list(".length, text.length - ")".length) - result.push("\n") - let r = this.extractTranslation("$" + text + "0"); - let i = 0 - while (r !== undefined && i < 100) { - result.push("• " + r + "\n") - i++ - r = this.extractTranslation("$" + text + i); + if(text === "\\n\\n"){ + result += "\n\n" + addSpace = false + continue + } + + if (!text.startsWith("$")) { + if(addSpace){ + result += " " } + result += text + addSpace = true + continue + } + const list = text.match(/\$list\(([a-zA-Z0-9_.-]+)\)/) + if (list) { + const key = list[1] + console.log("Generating a list with key" + key) + let r = this.extractTranslation("$" + key + "0"); + let i = 0 + result += "\n" + while (r !== undefined && i < 100) { + result += "• " + r + "\n" + i++ + r = this.extractTranslation("$" + key + i); + } + result += "\n" + addSpace = false } else { const found = this.extractTranslation(text) ?? text - result.push(found) + if(addSpace){ + result += " " + } + result += found + addSpace = true } } - this.doc.text(result.join(" "), x, y, { + this.doc.text(result, x, y, { maxWidth, }, this.currentMatrix) + } private drawSvgViaCanvas(element: Element): void { @@ -649,7 +674,7 @@ export class SvgToPdfPage { console.error("Could not show map with parameters", params) throw "Theme not found:" + params["theme"] + ". Use theme: to define which theme to use. " } - layout.widenFactor = 0 + layout.widenFactor = 0 layout.overpassTimeout = 600 layout.defaultBackgroundId = params["background"] ?? layout.defaultBackgroundId for (const paramsKey in params) { @@ -669,7 +694,7 @@ export class SvgToPdfPage { const zoom = Number(params["zoom"] ?? params["z"] ?? 14); Hash.hash.setData(undefined) - history.replaceState(null, "", "") + // QueryParameters.ClearAll() const state = new FeaturePipelineState(layout) state.locationControl.setData({ @@ -678,6 +703,8 @@ export class SvgToPdfPage { lon: this.options?.overrideLocation?.lon ?? Number(params["lon"] ?? 3.717842) }) + console.log("Params are", params, params["layers"]==="none") + const fl = state.filteredLayers.data for (const filteredLayer of fl) { if (params["layer-" + filteredLayer.layerDef.id] !== undefined) { @@ -783,7 +810,7 @@ export class SvgToPdf { public static readonly templates : Record= { flyer_a4:{pages: ["/assets/templates/MapComplete-flyer.svg","/assets/templates/MapComplete-flyer.back.svg"], description: Translations.t.flyer.description}, - poster_a2: {pages: ["/assets/templates/MapComplete-poster-a2.svg"], description: "A basic A2 poster (similar to the flyer)"} + poster_a3: {pages: ["/assets/templates/MapComplete-poster-a3.svg"], description: "A basic A3 poster (similar to the flyer)"} } private readonly _title: string; diff --git a/assets/templates/MapComplete-flyer.back.svg b/assets/templates/MapComplete-flyer.back.svg index 5aa27cfe0..b4af661f9 100644 --- a/assets/templates/MapComplete-flyer.back.svg +++ b/assets/templates/MapComplete-flyer.back.svg @@ -26,9 +26,9 @@ showgrid="false" showguides="true" inkscape:guide-bbox="true" - inkscape:zoom="2.0156476" - inkscape:cx="420.46041" - inkscape:cy="393.42195" + inkscape:zoom="3.2396737" + inkscape:cx="234.12852" + inkscape:cy="507.61285" inkscape:window-width="1920" inkscape:window-height="1007" inkscape:window-x="0" @@ -416,6 +416,12 @@ width="296.10439" height="80.904815" id="rect28360" /> + $map(theme:aed,z:14,lat:$map(theme:aed,z:14,lat:51.2098,lon:3.2284) + id="tspan32789">51.2098,lon:3.2284) $flyer.toerisme_vlaanderen + id="tspan32793">$flyer.toerisme_vlaanderen + style="display:inline;fill:#deadff;fill-opacity:1;stroke:#000000;stroke-width:0.0224792;stroke-linecap:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + id="rect11121-7" + width="88.887001" + height="79.970268" + x="5.4906716" + y="123.10045" /> $map(theme:toerisme_vlaandere$map(theme:toerisme_vlaanderen,lat:n,layers:none,layer-50.8552,lon:4.3156, z:10,layers:none, +charging_station_ebikes:force,lat:50.8552,lon:4.3156, z:10) + y="706.32213" + id="tspan32809">layer-charging_station_ebikes:force) $map(theme:cyclofix,z:14,lat:51.05016,lon:$map(theme:cyclofix,z:14,lat:51.05016,lon:3.717842,layers:none,layer-3.717842,layers:none,layer-bike_repair_station:true,layer-bike_repair_station:true,layer-drinking_water:true,layer-drinking_water:true,layer-bike_cafe:true, bike_cafe:true,layer-layer-charging_station_ebikes:false,layer-bicycle_tube_vending_machine: true) + id="tspan32833">bicycle_tube_vending_machine: true) $map(theme:artwork,z:15,lat:51.2098,lon:$map(theme:artwork,z:15,lat:51.2098,lon:3.2284,background:AGIV) + id="tspan32841">3.2284,background:AGIV) $map(theme:cyclestreets,$map(theme:cyclestreets,z:15,lat:51.02702,lon:z:15,lat:51.02702,lon:4.48029, scaling:3) + id="tspan32853">4.48029, scaling:3) @@ -637,14 +639,14 @@ style="font-style:normal;font-weight:normal;font-size:40px;line-height:1.25;font-family:sans-serif;white-space:pre;shape-inside:url(#rect890);display:inline;fill:#000000;fill-opacity:1;stroke:#000000;stroke-width:0.264848;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1">$map(theme:benches,z:14,lat:51.2098,lon:$map(theme:benches,z:14,lat:51.2098,lon:3.2284, layers:none, layer-bench:force) + id="tspan32861">3.2284, layers:none, layer-bench:force) $flyer.aerial + id="tspan32865">$flyer.aerial $flyer.examples + id="tspan32869">$flyer.examples @@ -732,9 +734,9 @@ style="font-style:normal;font-weight:normal;font-size:40px;line-height:1.25;font-family:sans-serif;white-space:pre;shape-inside:url(#rect5917);fill:#000000;fill-opacity:1;stroke:none">$flyer.lines_too + id="tspan32873">$flyer.lines_too $map(theme:onwheels,z:19,lat:50.86622,lon:$map(theme:onwheels,z:19,lat:50.86622,lon:4.350124.35012,layer-governments:false,layer-,layer-governments:false,layer-parking:false,layer-toilet:false,layer-parking:false,layer-toilet:false,layer-cafe_pub:false,layer-food:false,scaling:1.5) + id="tspan32891">cafe_pub:false,layer-food:false,scaling:1.5) $flyer.onwheels + id="tspan32895">$flyer.onwheels @@ -811,9 +813,9 @@ style="font-style:normal;font-weight:normal;font-size:40px;line-height:0.25;font-family:sans-serif;white-space:pre;shape-inside:url(#rect28360);fill:#000000;fill-opacity:1;stroke:none">$flyer.cyclofix + id="tspan32899">$flyer.cyclofix $version + id="tspan32903">$version @@ -883,9 +885,9 @@ style="font-style:normal;font-weight:normal;font-size:40px;line-height:1.25;font-family:sans-serif;white-space:pre;shape-inside:url(#rect135032);fill:#000000;fill-opacity:1;stroke:none">$flyer.title + id="tspan32907">$flyer.title $flyer.frontParagraph + id="tspan7257">$flyer.frontParagraph $flyer.tagline + id="tspan7261">$flyer.tagline $flyer.title + id="tspan7265">$flyer.title $flyer.whatIsOsm + id="tspan7267">$flyer.whatIsOsm $flyer.mapcomplete.title + id="tspan7271">$flyer.mapcomplete.title $flyer.mapcomplete.intro + id="tspan7275">$fmc.intro \n $list(fmc.li) + id="tspan7279"> $list(flyer.mapcomplete.li) + id="tspan7283"> + id="tspan7287"> + id="tspan7291"> + id="tspan7295"> + id="tspan7299"> + id="tspan7303"> + id="tspan7307"> + id="tspan7311"> $flyer.mapcomplete.customize + id="tspan7315">$fmc.customize $flyer.callToAction$flyer.callToAction + id="tspan7321"> + id="tspan7325"> $flyer.osm + id="tspan7329">$flyer.osm $flyer.editing.title + id="tspan7333">$flyer.editing.title $flyer.editing.intro + id="tspan7337">$flyer.editing.intro \n\n $flyer.editing.ex $import layer.nature_reserve as nr + id="tspan7341">$import layer.nature_reserve as nr $import layer.nature_reserve.tagRenderings.Dogs? as nrd + id="tspan7345">$import layer.nature_reserve.tagRenderings.Dogs? as nrd $import layer.nature_reserve.tagRenderings.Access tag as nra + id="tspan7349">$import layer.nature_reserve.tagRenderings.Access tag as nra $import layer.nature_reserve.tagRenderings.Surface area as nrsa + id="tspan7353">$import layer.nature_reserve.tagRenderings.Surface area as nrsa $import flyer.fakeui as fui + id="tspan7357">$import flyer.fakeui as fui $import general as g + id="tspan7361">$import flyer.mapcomplete as fmc $set(_surface:ha, 13.2) + id="tspan7365">$import general as g +$set(_surface:ha, 13.2) $image.addPicture + id="tspan7373">$image.addPicture $fui.add_images + id="tspan7377">$fui.add_images $fui.see_images + id="tspan7381">$fui.see_images $fui.wikipedia + id="tspan7385">$fui.wikipedia $nra.mappings.0.then + id="tspan7389">$nra.mappings.0.then $nrsa.render + id="tspan7393">$nrsa.render $fui.attributes + id="tspan7397">$fui.attributes $fui.edit + id="tspan7401">$fui.edit www.example.org/nature + id="tspan7405">www.example.org/nature $g.wikipedia.fromWikipedia + id="tspan7409">$g.wikipedia.fromWikipedia $fui.question + id="tspan7413">$fui.question $general.save + id="tspan7417">$general.save $nrd.question + id="tspan7421">$nrd.question $nrd.mappings.0.then + id="tspan7425">$nrd.mappings.0.then $nrd.mappings.1.then + id="tspan7429">$nrd.mappings.1.then $nrd.mappings.2.then + id="tspan7433">$nrd.mappings.2.then Skip + id="tspan7437">Skip + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + $flyer.frontParagraph + + + $flyer.title + + + + + + + + + + + + +$flyer.mapcomplete.customize + $fmc.intro $list(fmc.li) + + + + + + + + $flyer.callToAction + + $flyer.osm + $flyer.editing.title + + + + + $flyer.toerisme_vlaanderen + + + + + + + + + + + $flyer.examples + $version + + $flyer.whatIsOsm + + + + + + $map(theme:onwheels,z:19,lat:50.86622,lon:4.35012,layer-governments:false,layer-parking:false,layer-toilet:false,layer-cafe_pub:false,layer-food:false,scaling:1.5) + $flyer.onwheels + + + + + + + + + + $map(theme:cyclofix,z:14,lat:51.05016,lon:3.717842,layers:none,layer-bike_repair_station:true,layer-drinking_water:true,layer-bike_cafe:true,layer-bicycle_tube_vending_machine: true, +layer-charging_station_ebikes:false) + $flyer.cyclofix + + + + + + + + + + $map(theme:benches,z:14,lat:51.2098,lon:3.2284, layers:none, layer-bench:force) + $map(theme:cyclestreets,z:15,lat:51.02702,lon:4.48029, scaling:3) + + + + + + + + + + + $flyer.lines_too + $map(theme:aed,z:14,lat:51.2098,lon:3.2284, background:AGIV) + $flyer.aerial + + + + + + + + + + $flyer.tagline + $flyer.mapcomplete.title + $map(theme:cyclofix,lat:50.8552,lon:4.3156, z:10,layers:none, +layer-charging_station_ebikes:force) + + + + + + + $nr.title.render + $import layer.nature_reserve as nr +$import layer.nature_reserve.tagRenderings.Dogs? as nrd +$import layer.nature_reserve.tagRenderings.Access tag as nra +$import layer.nature_reserve.tagRenderings.Surface area as nrsa +$import flyer.fakeui as fui +$import flyer.mapcomplete as fmc +$import general as g +$set(_surface:ha, 13.2) + + + + + $image.addPicture + + Pieter Vander Vennet + $fui.add_images + + + + + + + + + + $fui.see_images + + + + $fui.wikipedia + CC0 + $nra.mappings.0.then + + + $nrsa.render + $fui.attributes + $fui.edit + + www.example.org/nature + + + + + + + + $g.wikipedia.fromWikipedia + + + + + + + + + + + + + + + + $fui.question + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + $general.save + + $nrd.question + $nrd.mappings.0.then +$nrd.mappings.1.then +$nrd.mappings.2.then + Skip + + + + diff --git a/langs/en.json b/langs/en.json index 2e03dc469..1305d905e 100644 --- a/langs/en.json +++ b/langs/en.json @@ -45,7 +45,8 @@ "cyclofix": "Bicycle pumps, repair stations, drinking water and cycle shops are on cyclofix", "description": "An A4-landscape flyer to promote MapComplete", "editing": { - "intro": "The user is greeted by a map with feature. Upon selecting one, the information about that feature is shown. A simplified example of what this looks like for a nature reserve is shown below.", + "ex": "A simplified example of what this looks like for a nature reserve is shown below.", + "intro": "The user is greeted by a map with feature. Upon selecting one, the information about that feature is shown.", "title": "What does the interface look like?" }, "examples": "There are many thematic maps available of which a few are printed here.\n\nThere are many more thematic maps online: about healthcare, indoor navigation, wheelchair accessibility, waste facilities, public bookcases, pedestrian crossings with a rainbow-painting,... Discover them all on mapcomplete.osm.be ", @@ -61,7 +62,7 @@ "lines_too": "Lines and polygons are shown too. Attributes and images can be added and updated on those objects as well.", "mapcomplete": { "customize": "MapComplete can be tailored to your needs, with new map layers, new functionalities or styled with your organisations colours and font.\nWe also have experience with starting campaigns to crowdsource geodata.\nContact pietervdvn@posteo.net for a quote.", - "intro": "MapComplete is a website which has {mapCount} interactive maps. Every single map allows to add or update information.", + "intro": "MapComplete is a website which has {mapCount} interactive maps. Every single map allows to add or update information. It has many features:", "li0": "Show where POI are", "li1": "Add new points and update information on existing points", "li2": "Add contact information and opening hours easily", diff --git a/langs/nl.json b/langs/nl.json index 6f73b483c..bc8cfc008 100644 --- a/langs/nl.json +++ b/langs/nl.json @@ -44,7 +44,8 @@ "callToAction": "Probeer het uit op mapcomplete.osm.be", "cyclofix": "Fietspompen, -winkels, -bandenautomaten en drinkwaterkraantjes vind je op Cyclofix", "editing": { - "intro": "De gebruiker krijgt eerst een kaart met interesspunten te zien. Klik je op een punt, dan wordt de interface met informatie geopend.\nEen (vereenvoudigd) voorbeeld voor een natuurgebied wordt hieronder getoond:", + "ex": "Een (vereenvoudigd) voorbeeld voor een natuurgebied wordt hieronder getoond:", + "intro": "De gebruiker krijgt eerst een kaart met interesspunten te zien. Klik je op een punt, dan wordt de interface met informatie geopend.", "title": "Hoe ziet de interface eruit?" }, "examples": "Online zijn verschillende kaarten met diverse thema's beschikbaar, zoals gezondheidszorg, binnenruimtes, rolstoeltoegankelijkheid, afvalcontainers, boekenruilkasten, regenboog-zebrapaden,...\nEnkele zijn hier afgeprint.\n\nOntdek ze allemaal op mapcomplete.osm.be", @@ -56,7 +57,7 @@ "see_images": "Toont afbeeldingen van eerdere bijdragers, Wikipedia, Mapillary, ...", "wikipedia": "Gelinkte Wikipedia-artikelen worden getoond" }, - "frontParagraph": "MapComplete is een web-applicatie om OpenStreetMap-data te tonen en aan te passen op basis van thematische kaarten. Het maakt het mogelijk om open geodata te crowdsourcen en te managen op een makkelijke manier.\n\nNieuwe categorieën en attributen kunnen op vraag worden toegevoegd.", + "frontParagraph": "MapComplete is een web-applicatie om OpenStreetMap-data te tonen en aan te passen op thematische kaarten.\nHet maakt het mogelijk om open geodata te crowdsourcen en te managen op een makkelijke manier.\n\nNieuwe categorieën en attributen kunnen op vraag worden toegevoegd.", "lines_too": "Lijnobjecten en polygonen worden ook getoond. Afbeeldingen en attributen daarvan kunnen toegevoegd en aangepast worden.", "mapcomplete": { "customize": "Wil je een versie op maat?\nWil je een versie in jullie huisstijl?\nWil je een nieuwe kaartlaag of functionaliteit?\nWil je een crowdsourcing-campagne opzetten?\nNeem contact op met pietervdvn@posteo.net voor meer info.", diff --git a/test.ts b/test.ts index 4d7f51e6d..a21df4f36 100644 --- a/test.ts +++ b/test.ts @@ -1,226 +1,6 @@ import MinimapImplementation from "./UI/Base/MinimapImplementation"; -import {Utils} from "./Utils"; -import {FlowPanelFactory, FlowStep} from "./UI/ImportFlow/FlowStep"; -import Title from "./UI/Base/Title"; -import Combine from "./UI/Base/Combine"; -import {ImmutableStore, Store, UIEventSource} from "./Logic/UIEventSource"; -import {RadioButton} from "./UI/Input/RadioButton"; -import {InputElement} from "./UI/Input/InputElement"; -import {FixedInputElement} from "./UI/Input/FixedInputElement"; -import List from "./UI/Base/List"; -import {VariableUiElement} from "./UI/Base/VariableUIElement"; -import BaseUIElement from "./UI/BaseUIElement"; -import LeftIndex from "./UI/Base/LeftIndex"; -import {SvgToPdf, SvgToPdfOptions} from "./Utils/svgToPdf"; -import Img from "./UI/Base/Img"; -import Toggle from "./UI/Input/Toggle"; -import CheckBoxes, {CheckBox} from "./UI/Input/Checkboxes"; -import Loading from "./UI/Base/Loading"; -import Minimap from "./UI/Base/Minimap"; -import {FixedUiElement} from "./UI/Base/FixedUiElement"; -import SearchAndGo from "./UI/BigComponents/SearchAndGo"; -import {SubtleButton} from "./UI/Base/SubtleButton"; +import {PdfExportGui} from "./UI/BigComponents/PdfExportGui"; MinimapImplementation.initialize() -let i = 0 - -function createElement(): string { - const div = document.createElement("div") - div.id = "freediv-" + (i++) - document.getElementById("extradiv").append(div) - return div.id -} - -class SelectTemplate extends Combine implements FlowStep<{ title: string, pages: string[] }> { - readonly IsValid: Store; - readonly Value: Store<{ title: string, pages: string[] }>; - - constructor() { - const elements: InputElement<{ templateName: string, pages: string[] }>[] = [] - for (const templateName in SvgToPdf.templates) { - const template = SvgToPdf.templates[templateName] - elements.push(new FixedInputElement( - new Combine([new FixedUiElement(templateName).SetClass("font-bold pr-2"), - template.description - ]) - , new UIEventSource({templateName, pages: template.pages}))) - } - const radio = new RadioButton(elements, {selectFirstAsDefault: true}) - - const loaded: Store<{ success: { title: string, pages: string[] } } | { error: any }> = radio.GetValue().bind(template => { - if (template === undefined) { - return undefined - } - const urls = template.pages.map(p => SelectTemplate.ToUrl(p)) - const dloadAll: Promise<{ title: string, pages: string[] }> = Promise.all(urls.map(url => Utils.download(url))).then(pages => ({ - pages, - title: template.templateName - })) - - return UIEventSource.FromPromiseWithErr(dloadAll) - }) - const preview = new VariableUiElement( - loaded.map(pages => { - if (pages === undefined) { - return new Loading() - } - if (pages["err"] !== undefined) { - return new FixedUiElement("Loading preview failed: " + pages["err"]).SetClass("alert") - } - const els: BaseUIElement[] = [] - for (const pageSrc of pages["success"].pages) { - const el = new Img(pageSrc, true) - .SetClass("w-96 m-2 border-black border-2") - els.push(el) - } - return new Combine(els).SetClass("flex border border-subtle rounded-xl"); - }) - ) - - super([ - new Title("Select template"), - radio, - new Title("Preview"), - preview - ]); - this.Value = loaded.map(l => l === undefined ? undefined : l["success"]) - this.IsValid = this.Value.map(v => v !== undefined) - } - - public static ToUrl(spec: string) { - if (spec.startsWith("http")) { - return spec - } - return window.location.protocol + "//" + window.location.host + "/" + spec - } - -} - -class SelectPdfOptions extends Combine implements FlowStep<{ title: string, pages: string[], options: SvgToPdfOptions }> { - readonly IsValid: Store; - readonly Value: Store<{ title: string, pages: string[], options: SvgToPdfOptions }>; - - constructor(title: string, pages: string[], getFreeDiv: () => string) { - const dummy = new CheckBox("Don't add data to the map (to quickly preview the PDF)", false) - const overrideMapLocation = new CheckBox("Override map location: use a selected location instead of the location set in the template", false) - const locationInput = Minimap.createMiniMap().SetClass("block w-full") - const searchField = new SearchAndGo({leafletMap: locationInput.leafletMap}) - const selectLocation = - new Combine([ - new Toggle(new Combine([new Title("Select override location"), searchField]).SetClass("flex"), undefined, overrideMapLocation.GetValue()), - new Toggle(locationInput.SetStyle("height: 20rem"), undefined, overrideMapLocation.GetValue()).SetStyle("height: 20rem") - ]).SetClass("block").SetStyle("height: 25rem") - super([new Title("Select options"), - dummy, - overrideMapLocation, - selectLocation - ]); - this.Value = dummy.GetValue().map((disableMaps) => { - return { - pages, - title, - options: { - disableMaps, - getFreeDiv, - overrideLocation: overrideMapLocation.GetValue().data ? locationInput.location.data : undefined - } - } - }, [overrideMapLocation.GetValue(), locationInput.location]) - this.IsValid = new ImmutableStore(true) - } - -} - -class PreparePdf extends Combine implements FlowStep<{ svgToPdf: SvgToPdf, languages: string[] }> { - readonly IsValid: Store; - readonly Value: Store<{ svgToPdf: SvgToPdf, languages: string[] }>; - - constructor(title: string, pages: string[], options: SvgToPdfOptions) { - const svgToPdf = new SvgToPdf(title, pages, options) - const languageOptions = [ - new FixedInputElement("Nederlands", "nl"), - new FixedInputElement("English", "en") - ] - const languages = new CheckBoxes(languageOptions) - - const isPrepared = UIEventSource.FromPromiseWithErr(svgToPdf.Prepare()) - - super([ - new Title("Select languages..."), - languages, - new Toggle( - new Loading("Preparing maps..."), - undefined, - isPrepared.map(p => p === undefined) - ) - ]); - this.Value = isPrepared.map(isPrepped => { - if (isPrepped === undefined) { - return undefined - } - if (isPrepped["success"] !== undefined) { - const svgToPdf = isPrepped["success"] - const langs = languages.GetValue().data.map(i => languageOptions[i].GetValue().data) - if (langs.length === 0) { - return undefined - } - return {svgToPdf, languages: langs} - } - return undefined; - }, [languages.GetValue()]) - this.IsValid = this.Value.map(v => v !== undefined) - } - -} - - -class SavePdf extends Combine { - - constructor(svgToPdf: SvgToPdf, languages: string[]) { - - super([ - new Title("Generating your pdfs..."), - new List(languages.map(lng => new Toggle( - lng + " is done!", - new Loading("Creating pdf for " + lng), - UIEventSource.FromPromiseWithErr(svgToPdf.ConvertSvg(lng).then(() => true)) - .map(x => x !== undefined && x["success"] === true) - ))) - ]); - } -} - -const {flow, furthestStep, titles} = FlowPanelFactory.start( - new Title("Select template"), new SelectTemplate() -).then(new Title("Select options"), ({title, pages}) => new SelectPdfOptions(title, pages, createElement)) - .then("Generate maps...", ({title, pages, options}) => new PreparePdf(title, pages, options)) - .finish("Generating...", ({svgToPdf, languages}) => new SavePdf(svgToPdf, languages)) - - -const toc = new List( - titles.map( - (title, i) => - new VariableUiElement( - furthestStep.map((currentStep) => { - if (i > currentStep) { - return new Combine([title]).SetClass("subtle") - } - if (i == currentStep) { - return new Combine([title]).SetClass("font-bold") - } - if (i < currentStep) { - return title - } - }) - ) - ), - true -) - -const leftContents: BaseUIElement[] = [ - toc -].map((el) => el?.SetClass("pl-4")) - -new LeftIndex(leftContents, flow).AttachTo("maindiv") -// main().then(() => console.log("Done!")) +new PdfExportGui("extradiv").AttachTo("maindiv")