forked from MapComplete/MapComplete
Add module to fetch data (via a proxy) from the website with jsonld
This commit is contained in:
parent
1b06eee15b
commit
352414b29d
17 changed files with 388 additions and 351 deletions
|
@ -90,11 +90,8 @@
|
||||||
"id": "show-data-velopark",
|
"id": "show-data-velopark",
|
||||||
"render": {
|
"render": {
|
||||||
"special": {
|
"special": {
|
||||||
"type": "compare_data",
|
"type": "linked_data_from_website",
|
||||||
"url": "ref:velopark",
|
"key": "ref:velopark"
|
||||||
"host": "https://data.velopark.be",
|
|
||||||
"postprocessing": "velopark",
|
|
||||||
"readonly": "yes"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -338,10 +335,8 @@
|
||||||
},
|
},
|
||||||
"render": {
|
"render": {
|
||||||
"special": {
|
"special": {
|
||||||
"type": "compare_data",
|
"type": "linked_data_from_website",
|
||||||
"url": "ref:velopark",
|
"key": "ref:velopark"
|
||||||
"host": "https://data.velopark.be",
|
|
||||||
"postprocessing": "velopark"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -94,7 +94,9 @@
|
||||||
"weblate-fix-heavy": "git fetch weblate-hosted-layers; git fetch weblate-hosted-core; git merge weblate-hosted-layers/master weblate-hosted-core/master ",
|
"weblate-fix-heavy": "git fetch weblate-hosted-layers; git fetch weblate-hosted-core; git merge weblate-hosted-layers/master weblate-hosted-core/master ",
|
||||||
"housekeeping": "git pull && npx update-browserslist-db@latest && npm run weblate-fix-heavy && npm run generate && npm run generate:docs && npm run generate:contributor-list && vite-node scripts/fetchLanguages.ts && npm run format && git add assets/ langs/ Docs/ **/*.ts Docs/* src/* && git commit -m 'chore: automated housekeeping...'",
|
"housekeeping": "git pull && npx update-browserslist-db@latest && npm run weblate-fix-heavy && npm run generate && npm run generate:docs && npm run generate:contributor-list && vite-node scripts/fetchLanguages.ts && npm run format && git add assets/ langs/ Docs/ **/*.ts Docs/* src/* && git commit -m 'chore: automated housekeeping...'",
|
||||||
"reuse-compliance": "reuse lint",
|
"reuse-compliance": "reuse lint",
|
||||||
"backup:images": "vite-node scripts/generateImageAnalysis.ts -- ~/data/imgur-image-backup/"
|
"backup:images": "vite-node scripts/generateImageAnalysis.ts -- ~/data/imgur-image-backup/",
|
||||||
|
"dloadVelopark": "vite-node scripts/velopark/veloParkToGeojson.ts ",
|
||||||
|
"scrapeWebsites": "vite-node scripts/importscripts/compareWebsiteData.ts -- ~/Downloads/ShopsWithWebsiteNodes.csv ~/data/scraped_websites/\n"
|
||||||
},
|
},
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"OpenStreetMap",
|
"OpenStreetMap",
|
||||||
|
|
|
@ -148,7 +148,16 @@ export default class ScriptUtils {
|
||||||
const data = await ScriptUtils.Download(url, headers)
|
const data = await ScriptUtils.Download(url, headers)
|
||||||
return JSON.parse(data["content"])
|
return JSON.parse(data["content"])
|
||||||
}
|
}
|
||||||
|
public static async DownloadFetch(
|
||||||
|
url: string,
|
||||||
|
headers?: any
|
||||||
|
): Promise<{ content: string } | { redirect: string }> {
|
||||||
|
console.log("Fetching", url)
|
||||||
|
const req = await fetch(url, {headers})
|
||||||
|
const data= await req.text()
|
||||||
|
console.log("Fetched", url,data)
|
||||||
|
return {content: data}
|
||||||
|
}
|
||||||
public static Download(
|
public static Download(
|
||||||
url: string,
|
url: string,
|
||||||
headers?: any
|
headers?: any
|
||||||
|
|
80
scripts/importscripts/compareWebsiteData.ts
Normal file
80
scripts/importscripts/compareWebsiteData.ts
Normal file
|
@ -0,0 +1,80 @@
|
||||||
|
import fs from "fs"
|
||||||
|
// import readline from "readline"
|
||||||
|
import Script from "../Script"
|
||||||
|
import LinkedDataLoader from "../../src/Logic/Web/LinkedDataLoader"
|
||||||
|
import UrlValidator from "../../src/UI/InputElement/Validators/UrlValidator"
|
||||||
|
// vite-node scripts/importscripts/compareWebsiteData.ts -- ~/Downloads/ShopsWithWebsiteNodes.csv ~/data/scraped_websites/
|
||||||
|
/*
|
||||||
|
class CompareWebsiteData extends Script {
|
||||||
|
constructor() {
|
||||||
|
super("Given a csv file with 'id', 'tags' and 'website', attempts to fetch jsonld and compares the attributes. Usage: csv-file datadir")
|
||||||
|
}
|
||||||
|
|
||||||
|
private readonly urlFormatter = new UrlValidator()
|
||||||
|
async getWithCache(cachedir : string, url: string): Promise<any>{
|
||||||
|
const filename= cachedir+"/"+encodeURIComponent(url)
|
||||||
|
if(fs.existsSync(filename)){
|
||||||
|
return JSON.parse(fs.readFileSync(filename, "utf-8"))
|
||||||
|
}
|
||||||
|
const jsonLd = await LinkedDataLoader.fetchJsonLdWithProxy(url)
|
||||||
|
console.log("Got:", jsonLd)
|
||||||
|
fs.writeFileSync(filename, JSON.stringify(jsonLd))
|
||||||
|
return jsonLd
|
||||||
|
}
|
||||||
|
async handleEntry(line: string, cachedir: string, targetfile: string) : Promise<boolean>{
|
||||||
|
const id = JSON.parse(line.split(",")[0])
|
||||||
|
let tags = line.substring(line.indexOf("{") - 1)
|
||||||
|
tags = tags.substring(1, tags.length - 1)
|
||||||
|
tags = tags.replace(/""/g, "\"")
|
||||||
|
const data = JSON.parse(tags)
|
||||||
|
|
||||||
|
const website = data.website //this.urlFormatter.reformat(data.website)
|
||||||
|
if(!website.startsWith("https://stores.delhaize.be")){
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
console.log(website)
|
||||||
|
const jsonld = await this.getWithCache(cachedir, website)
|
||||||
|
console.log(jsonld)
|
||||||
|
if(Object.keys(jsonld).length === 0){
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
const diff = LinkedDataLoader.removeDuplicateData(jsonld, data)
|
||||||
|
fs.appendFileSync(targetfile, id +", "+ JSON.stringify(diff)+"\n")
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
async main(args: string[]): Promise<void> {
|
||||||
|
if (args.length < 2) {
|
||||||
|
throw "Not enough arguments"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const readInterface = readline.createInterface({
|
||||||
|
input: fs.createReadStream(args[0]),
|
||||||
|
})
|
||||||
|
|
||||||
|
let handled = 0
|
||||||
|
let diffed = 0
|
||||||
|
const targetfile = "diff.csv"
|
||||||
|
fs.writeFileSync(targetfile, "id, diff-json\n")
|
||||||
|
for await (const line of readInterface) {
|
||||||
|
try {
|
||||||
|
if(line.startsWith("\"id\"")){
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
const madeComparison = await this.handleEntry(line, args[1], targetfile)
|
||||||
|
handled ++
|
||||||
|
diffed = diffed + (madeComparison ? 1 : 0)
|
||||||
|
if(handled % 1000 == 0){
|
||||||
|
// console.log("Handled ",handled," got ",diffed,"diff results")
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
// console.error(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
new CompareWebsiteData().run()
|
||||||
|
*/
|
0
scripts/scrapeOsm.ts
Normal file
0
scripts/scrapeOsm.ts
Normal file
|
@ -15,6 +15,7 @@ class ServerLdScrape extends Script {
|
||||||
mimetype: "application/ld+json",
|
mimetype: "application/ld+json",
|
||||||
async handle(content, searchParams: URLSearchParams) {
|
async handle(content, searchParams: URLSearchParams) {
|
||||||
const url = searchParams.get("url")
|
const url = searchParams.get("url")
|
||||||
|
console.log("Fetching", url)
|
||||||
if (cache[url]) {
|
if (cache[url]) {
|
||||||
return JSON.stringify(cache[url])
|
return JSON.stringify(cache[url])
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,39 +1,42 @@
|
||||||
import Script from "../Script"
|
import Script from "../Script"
|
||||||
import { Utils } from "../../src/Utils"
|
|
||||||
import VeloparkLoader, { VeloparkData } from "../../src/Logic/Web/VeloparkLoader"
|
|
||||||
import fs from "fs"
|
import fs from "fs"
|
||||||
import { Overpass } from "../../src/Logic/Osm/Overpass"
|
import { Overpass } from "../../src/Logic/Osm/Overpass"
|
||||||
import { RegexTag } from "../../src/Logic/Tags/RegexTag"
|
import { RegexTag } from "../../src/Logic/Tags/RegexTag"
|
||||||
import Constants from "../../src/Models/Constants"
|
import Constants from "../../src/Models/Constants"
|
||||||
import { ImmutableStore } from "../../src/Logic/UIEventSource"
|
import { ImmutableStore } from "../../src/Logic/UIEventSource"
|
||||||
import { BBox } from "../../src/Logic/BBox"
|
import { BBox } from "../../src/Logic/BBox"
|
||||||
|
import LinkedDataLoader from "../../src/Logic/Web/LinkedDataLoader"
|
||||||
|
|
||||||
class VeloParkToGeojson extends Script {
|
class VeloParkToGeojson extends Script {
|
||||||
constructor() {
|
constructor() {
|
||||||
super(
|
super(
|
||||||
"Downloads the latest Velopark data and converts it to a geojson, which will be saved at the current directory"
|
"Downloads the latest Velopark data and converts it to a geojson, which will be saved at the current directory",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
exportTo(filename: string, features) {
|
exportTo(filename: string, features) {
|
||||||
fs.writeFileSync(
|
features = features.slice(0,25) // TODO REMOVE
|
||||||
filename + "_" + new Date().toISOString() + ".geojson",
|
const file = filename + "_" + /*new Date().toISOString() + */".geojson"
|
||||||
|
fs.writeFileSync(file,
|
||||||
JSON.stringify(
|
JSON.stringify(
|
||||||
{
|
{
|
||||||
type: "FeatureCollection",
|
type: "FeatureCollection",
|
||||||
|
"#":"Only 25 features are shown!", // TODO REMOVE
|
||||||
features,
|
features,
|
||||||
},
|
},
|
||||||
null,
|
null,
|
||||||
" "
|
" ",
|
||||||
)
|
),
|
||||||
)
|
)
|
||||||
|
console.log("Written",file)
|
||||||
}
|
}
|
||||||
|
|
||||||
async main(args: string[]): Promise<void> {
|
async main(args: string[]): Promise<void> {
|
||||||
console.log("Downloading velopark data")
|
console.log("Downloading velopark data")
|
||||||
// Download data for NIS-code 1000. 1000 means: all of belgium
|
// Download data for NIS-code 1000. 1000 means: all of belgium
|
||||||
const url = "https://www.velopark.be/api/parkings/1000"
|
const url = "https://www.velopark.be/api/parkings/1000"
|
||||||
const data = <VeloparkData[]>await Utils.downloadJson(url)
|
const allVelopark = await LinkedDataLoader.fetchJsonLd(url, { country: "be" })
|
||||||
|
this.exportTo("velopark_all", allVelopark)
|
||||||
|
|
||||||
const bboxBelgium = new BBox([
|
const bboxBelgium = new BBox([
|
||||||
[2.51357303225, 49.5294835476],
|
[2.51357303225, 49.5294835476],
|
||||||
|
@ -44,15 +47,13 @@ class VeloParkToGeojson extends Script {
|
||||||
[],
|
[],
|
||||||
Constants.defaultOverpassUrls[0],
|
Constants.defaultOverpassUrls[0],
|
||||||
new ImmutableStore(60 * 5),
|
new ImmutableStore(60 * 5),
|
||||||
false
|
false,
|
||||||
)
|
)
|
||||||
const alreadyLinkedFeatures = await alreadyLinkedQuery.queryGeoJson(bboxBelgium)
|
const alreadyLinkedFeatures = await alreadyLinkedQuery.queryGeoJson(bboxBelgium)
|
||||||
const seenIds = new Set<string>(
|
const seenIds = new Set<string>(
|
||||||
alreadyLinkedFeatures[0].features.map((f) => f.properties["ref:velopark"])
|
alreadyLinkedFeatures[0].features.map((f) => f.properties["ref:velopark"]),
|
||||||
)
|
)
|
||||||
console.log("OpenStreetMap contains", seenIds.size, "bicycle parkings with a velopark ref")
|
console.log("OpenStreetMap contains", seenIds.size, "bicycle parkings with a velopark ref")
|
||||||
const allVelopark = data.map((f) => VeloparkLoader.convert(f))
|
|
||||||
this.exportTo("velopark_all", allVelopark)
|
|
||||||
|
|
||||||
const features = allVelopark.filter((f) => !seenIds.has(f.properties["ref:velopark"]))
|
const features = allVelopark.filter((f) => !seenIds.has(f.properties["ref:velopark"]))
|
||||||
|
|
||||||
|
|
|
@ -237,8 +237,11 @@ export abstract class Store<T> implements Readable<T> {
|
||||||
|
|
||||||
public bindD<X>(f: (t: Exclude<T, undefined | null>) => Store<X>): Store<X> {
|
public bindD<X>(f: (t: Exclude<T, undefined | null>) => Store<X>): Store<X> {
|
||||||
return this.bind((t) => {
|
return this.bind((t) => {
|
||||||
if (t === undefined || t === null) {
|
if(t=== null){
|
||||||
return <undefined | null>t
|
return null
|
||||||
|
}
|
||||||
|
if (t === undefined ) {
|
||||||
|
return undefined
|
||||||
}
|
}
|
||||||
return f(<Exclude<T, undefined | null>>t)
|
return f(<Exclude<T, undefined | null>>t)
|
||||||
})
|
})
|
||||||
|
|
|
@ -6,7 +6,11 @@ import PhoneValidator from "../../UI/InputElement/Validators/PhoneValidator"
|
||||||
import EmailValidator from "../../UI/InputElement/Validators/EmailValidator"
|
import EmailValidator from "../../UI/InputElement/Validators/EmailValidator"
|
||||||
import { Validator } from "../../UI/InputElement/Validator"
|
import { Validator } from "../../UI/InputElement/Validator"
|
||||||
import UrlValidator from "../../UI/InputElement/Validators/UrlValidator"
|
import UrlValidator from "../../UI/InputElement/Validators/UrlValidator"
|
||||||
|
import Constants from "../../Models/Constants"
|
||||||
|
|
||||||
|
interface JsonLdLoaderOptions {
|
||||||
|
country?: string
|
||||||
|
}
|
||||||
export default class LinkedDataLoader {
|
export default class LinkedDataLoader {
|
||||||
private static readonly COMPACTING_CONTEXT = {
|
private static readonly COMPACTING_CONTEXT = {
|
||||||
name: "http://schema.org/name",
|
name: "http://schema.org/name",
|
||||||
|
@ -43,6 +47,10 @@ export default class LinkedDataLoader {
|
||||||
"http://schema.org/contactPoint",
|
"http://schema.org/contactPoint",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
private static ignoreTypes = [
|
||||||
|
"Breadcrumblist"
|
||||||
|
]
|
||||||
|
|
||||||
static async geoToGeometry(geo): Promise<Geometry> {
|
static async geoToGeometry(geo): Promise<Geometry> {
|
||||||
const context = {
|
const context = {
|
||||||
lat: {
|
lat: {
|
||||||
|
@ -102,15 +110,30 @@ export default class LinkedDataLoader {
|
||||||
return OH.ToString(OH.MergeTimes(allRules))
|
return OH.ToString(OH.MergeTimes(allRules))
|
||||||
}
|
}
|
||||||
|
|
||||||
static async fetchJsonLd(url: string, country?: string): Promise<Record<string, any>> {
|
static async fetchJsonLdWithProxy(url: string, options?: JsonLdLoaderOptions): Promise<any> {
|
||||||
const proxy = "http://127.0.0.1:2346/extractgraph" // "https://cache.mapcomplete.org/extractgraph"
|
const urlWithProxy = Constants.linkedDataProxy.replace("{url}", encodeURIComponent(url))
|
||||||
const data = await Utils.downloadJson(`${proxy}?url=${url}`)
|
return await this.fetchJsonLd(urlWithProxy, options)
|
||||||
const compacted = await jsonld.compact(data, LinkedDataLoader.COMPACTING_CONTEXT)
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* {
|
||||||
|
* "content": "{\"@context\":\"http://schema.org\",\"@type\":\"LocalBusiness\",\"@id\":\"http://stores.delhaize.be/nl/ad-delhaize-munsterbilzen\",\"name\":\"AD Delhaize Munsterbilzen\",\"url\":\"http://stores.delhaize.be/nl/ad-delhaize-munsterbilzen\",\"logo\":\"https://stores.delhaize.be/build/images/web/shop/delhaize-be/favicon.ico\",\"image\":\"http://stores.delhaize.be/image/mobilosoft-testing?apiPath=rehab/delhaize-be/images/location/ad%20delhaize%20image%20ge%CC%81ne%CC%81rale%20%281%29%201652787176865&imageSize=h_500\",\"email\":\"\",\"telephone\":\"+3289413520\",\"address\":{\"@type\":\"PostalAddress\",\"streetAddress\":\"Waterstraat, 18\",\"addressLocality\":\"Bilzen\",\"postalCode\":\"3740\",\"addressCountry\":\"BE\"},\"geo\":{\"@type\":\"GeoCoordinates\",\"latitude\":50.8906898,\"longitude\":5.5260586},\"openingHoursSpecification\":[{\"@type\":\"OpeningHoursSpecification\",\"dayOfWeek\":\"Tuesday\",\"opens\":\"08:00\",\"closes\":\"18:30\"},{\"@type\":\"OpeningHoursSpecification\",\"dayOfWeek\":\"Wednesday\",\"opens\":\"08:00\",\"closes\":\"18:30\"},{\"@type\":\"OpeningHoursSpecification\",\"dayOfWeek\":\"Thursday\",\"opens\":\"08:00\",\"closes\":\"18:30\"},{\"@type\":\"OpeningHoursSpecification\",\"dayOfWeek\":\"Friday\",\"opens\":\"08:00\",\"closes\":\"18:30\"},{\"@type\":\"OpeningHoursSpecification\",\"dayOfWeek\":\"Saturday\",\"opens\":\"08:00\",\"closes\":\"18:30\"},{\"@type\":\"OpeningHoursSpecification\",\"dayOfWeek\":\"Sunday\",\"opens\":\"08:00\",\"closes\":\"12:00\"},{\"@type\":\"OpeningHoursSpecification\",\"dayOfWeek\":\"Monday\",\"opens\":\"12:00\",\"closes\":\"18:30\"}],\"@base\":\"https://stores.delhaize.be/nl/ad-delhaize-munsterbilzen\"}"
|
||||||
|
* }
|
||||||
|
*/
|
||||||
|
private static async compact(data: any, options?: JsonLdLoaderOptions): Promise<any>{
|
||||||
|
console.log("Compacting",data)
|
||||||
|
if(Array.isArray(data)) {
|
||||||
|
return await Promise.all(data.map(d => LinkedDataLoader.compact(d)))
|
||||||
|
}
|
||||||
|
const country = options?.country
|
||||||
|
const compacted = await jsonld.compact(data, <any> LinkedDataLoader.COMPACTING_CONTEXT)
|
||||||
compacted["opening_hours"] = await LinkedDataLoader.ohToOsmFormat(
|
compacted["opening_hours"] = await LinkedDataLoader.ohToOsmFormat(
|
||||||
compacted["opening_hours"]
|
compacted["opening_hours"]
|
||||||
)
|
)
|
||||||
if (compacted["openingHours"]) {
|
if (compacted["openingHours"]) {
|
||||||
const ohspec: string[] = compacted["openingHours"]
|
const ohspec: string[] = <any> compacted["openingHours"]
|
||||||
compacted["opening_hours"] = OH.simplify(
|
compacted["opening_hours"] = OH.simplify(
|
||||||
ohspec.map((r) => LinkedDataLoader.ohStringToOsmFormat(r)).join("; ")
|
ohspec.map((r) => LinkedDataLoader.ohStringToOsmFormat(r)).join("; ")
|
||||||
)
|
)
|
||||||
|
@ -138,5 +161,39 @@ export default class LinkedDataLoader {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return <any>compacted
|
return <any>compacted
|
||||||
|
|
||||||
|
}
|
||||||
|
static async fetchJsonLd(url: string, options?: JsonLdLoaderOptions): Promise<any> {
|
||||||
|
const data = await Utils.downloadJson(url)
|
||||||
|
return await LinkedDataLoader.compact(data, options)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Only returns different items
|
||||||
|
* @param externalData
|
||||||
|
* @param currentData
|
||||||
|
*/
|
||||||
|
static removeDuplicateData(externalData: Record<string, string>, currentData: Record<string, string>) : Record<string, string>{
|
||||||
|
const d = { ...externalData }
|
||||||
|
delete d["@context"]
|
||||||
|
for (const k in d) {
|
||||||
|
const v = currentData[k]
|
||||||
|
if (!v) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if (k === "opening_hours") {
|
||||||
|
const oh = [].concat(...v.split(";").map(r => OH.ParseRule(r) ?? []))
|
||||||
|
const merged = OH.ToString(OH.MergeTimes(oh ?? []))
|
||||||
|
if (merged === d[k]) {
|
||||||
|
delete d[k]
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (v === d[k]) {
|
||||||
|
delete d[k]
|
||||||
|
}
|
||||||
|
delete d.geo
|
||||||
|
}
|
||||||
|
return d
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,48 +1,51 @@
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { UIEventSource } from "../../Logic/UIEventSource"
|
import { UIEventSource } from "../../Logic/UIEventSource"
|
||||||
import type { OsmTags } from "../../Models/OsmFeature"
|
import type { OsmTags } from "../../Models/OsmFeature"
|
||||||
import type { SpecialVisualizationState } from "../SpecialVisualization"
|
import type { SpecialVisualizationState } from "../SpecialVisualization"
|
||||||
import type { Feature } from "geojson"
|
import type { Feature } from "geojson"
|
||||||
import LayerConfig from "../../Models/ThemeConfig/LayerConfig"
|
import LayerConfig from "../../Models/ThemeConfig/LayerConfig"
|
||||||
import ChangeTagAction from "../../Logic/Osm/Actions/ChangeTagAction"
|
import ChangeTagAction from "../../Logic/Osm/Actions/ChangeTagAction"
|
||||||
import { Tag } from "../../Logic/Tags/Tag"
|
import { Tag } from "../../Logic/Tags/Tag"
|
||||||
import Loading from "../Base/Loading.svelte"
|
import Loading from "../Base/Loading.svelte"
|
||||||
|
|
||||||
export let key: string
|
export let key: string
|
||||||
export let externalProperties: Record<string, string>
|
export let externalProperties: Record<string, string>
|
||||||
|
|
||||||
export let tags: UIEventSource<OsmTags>
|
export let tags: UIEventSource<OsmTags>
|
||||||
export let state: SpecialVisualizationState
|
export let state: SpecialVisualizationState
|
||||||
export let feature: Feature
|
export let feature: Feature
|
||||||
export let layer: LayerConfig
|
export let layer: LayerConfig
|
||||||
|
|
||||||
export let readonly = false
|
export let readonly = false
|
||||||
|
|
||||||
let currentStep: "init" | "applying" | "done" = "init"
|
let currentStep: "init" | "applying" | "done" = "init"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Copy the given key into OSM
|
* Copy the given key into OSM
|
||||||
* @param key
|
* @param key
|
||||||
*/
|
*/
|
||||||
async function apply(key: string) {
|
async function apply(key: string) {
|
||||||
currentStep = "applying"
|
currentStep = "applying"
|
||||||
const change = new ChangeTagAction(
|
const change = new ChangeTagAction(
|
||||||
tags.data.id,
|
tags.data.id,
|
||||||
new Tag(key, externalProperties[key]),
|
new Tag(key, externalProperties[key]),
|
||||||
tags.data,
|
tags.data,
|
||||||
{
|
{
|
||||||
theme: state.layout.id,
|
theme: state.layout.id,
|
||||||
changeType: "import",
|
changeType: "import",
|
||||||
}
|
},
|
||||||
)
|
)
|
||||||
await state.changes.applyChanges(await change.CreateChangeDescriptions())
|
await state.changes.applyChanges(await change.CreateChangeDescriptions())
|
||||||
currentStep = "done"
|
currentStep = "done"
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
<td><b>{key}</b></td>
|
<td><b>{key}</b></td>
|
||||||
|
|
||||||
|
{#if $tags[key]}
|
||||||
|
{$tags[key]}
|
||||||
|
{/if}
|
||||||
<td>
|
<td>
|
||||||
{#if externalProperties[key].startsWith("http")}
|
{#if externalProperties[key].startsWith("http")}
|
||||||
<a href={externalProperties[key]} target="_blank">
|
<a href={externalProperties[key]} target="_blank">
|
||||||
|
|
|
@ -26,23 +26,21 @@
|
||||||
let externalKeys: string[] = Object.keys(externalProperties).sort()
|
let externalKeys: string[] = Object.keys(externalProperties).sort()
|
||||||
|
|
||||||
const imageKeyRegex = /image|image:[0-9]+/
|
const imageKeyRegex = /image|image:[0-9]+/
|
||||||
console.log("Calculating knwon images")
|
|
||||||
let knownImages = new Set(
|
let knownImages = new Set(
|
||||||
Object.keys(osmProperties)
|
Object.keys(osmProperties)
|
||||||
.filter((k) => k.match(imageKeyRegex))
|
.filter((k) => k.match(imageKeyRegex))
|
||||||
.map((k) => osmProperties[k])
|
.map((k) => osmProperties[k])
|
||||||
)
|
)
|
||||||
console.log("Known images are:", knownImages)
|
|
||||||
let unknownImages = externalKeys
|
let unknownImages = externalKeys
|
||||||
.filter((k) => k.match(imageKeyRegex))
|
.filter((k) => k.match(imageKeyRegex))
|
||||||
.map((k) => externalProperties[k])
|
.map((k) => externalProperties[k])
|
||||||
.filter((i) => !knownImages.has(i))
|
.filter((i) => !knownImages.has(i))
|
||||||
|
|
||||||
let propertyKeysExternal = externalKeys.filter((k) => k.match(imageKeyRegex) === null)
|
let propertyKeysExternal = externalKeys.filter((k) => k.match(imageKeyRegex) === null)
|
||||||
let missing = propertyKeysExternal.filter((k) => osmProperties[k] === undefined)
|
let missing = propertyKeysExternal.filter((k) => osmProperties[k] === undefined && typeof externalProperties[k] === "string")
|
||||||
let same = propertyKeysExternal.filter((key) => osmProperties[key] === externalProperties[key])
|
let same = propertyKeysExternal.filter((key) => osmProperties[key] === externalProperties[key])
|
||||||
let different = propertyKeysExternal.filter(
|
let different = propertyKeysExternal.filter(
|
||||||
(key) => osmProperties[key] !== undefined && osmProperties[key] !== externalProperties[key]
|
(key) => osmProperties[key] !== undefined && osmProperties[key] !== externalProperties[key] && typeof externalProperties[key] === "string"
|
||||||
)
|
)
|
||||||
|
|
||||||
let currentStep: "init" | "applying_all" | "all_applied" = "init"
|
let currentStep: "init" | "applying_all" | "all_applied" = "init"
|
||||||
|
@ -68,11 +66,7 @@
|
||||||
<th>External</th>
|
<th>External</th>
|
||||||
</tr>
|
</tr>
|
||||||
{#each different as key}
|
{#each different as key}
|
||||||
<tr>
|
<ComparisonAction {key} {state} {tags} {externalProperties} {layer} {feature} {readonly} />
|
||||||
<td>{key}</td>
|
|
||||||
<td>{osmProperties[key]}</td>
|
|
||||||
<td>{externalProperties[key]}</td>
|
|
||||||
</tr>
|
|
||||||
{/each}
|
{/each}
|
||||||
</table>
|
</table>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
|
@ -1,65 +1,34 @@
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
/**
|
/**
|
||||||
* The comparison tool loads json-data from a speficied URL, eventually post-processes it
|
* The comparison tool loads json-data from a speficied URL, eventually post-processes it
|
||||||
* and compares it with the current object
|
* and compares it with the current object
|
||||||
*/
|
*/
|
||||||
import { onMount } from "svelte"
|
import Loading from "../Base/Loading.svelte"
|
||||||
import { Utils } from "../../Utils"
|
import type { SpecialVisualizationState } from "../SpecialVisualization"
|
||||||
import VeloparkLoader from "../../Logic/Web/VeloparkLoader"
|
import { Store, UIEventSource } from "../../Logic/UIEventSource"
|
||||||
import Loading from "../Base/Loading.svelte"
|
import ComparisonTable from "./ComparisonTable.svelte"
|
||||||
import type { SpecialVisualizationState } from "../SpecialVisualization"
|
import LayerConfig from "../../Models/ThemeConfig/LayerConfig"
|
||||||
import { UIEventSource } from "../../Logic/UIEventSource"
|
import type { Feature } from "geojson"
|
||||||
import ComparisonTable from "./ComparisonTable.svelte"
|
import type { OsmTags } from "../../Models/OsmFeature"
|
||||||
import LayerConfig from "../../Models/ThemeConfig/LayerConfig"
|
|
||||||
import type { Feature } from "geojson"
|
|
||||||
import type { OsmTags } from "../../Models/OsmFeature"
|
|
||||||
|
|
||||||
export let url: string
|
export let externalData: Store<{ success: {content: Record<string, string> } } | { error: string } | undefined | null /* null if no URL is found, undefined if loading*/>
|
||||||
export let postprocessVelopark: boolean
|
export let state: SpecialVisualizationState
|
||||||
export let state: SpecialVisualizationState
|
export let tags: UIEventSource<OsmTags>
|
||||||
export let tags: UIEventSource<OsmTags>
|
export let layer: LayerConfig
|
||||||
export let layer: LayerConfig
|
export let feature: Feature
|
||||||
export let feature: Feature
|
export let readonly = false
|
||||||
export let readonly = false
|
|
||||||
let data: any = undefined
|
|
||||||
let error: any = undefined
|
|
||||||
|
|
||||||
onMount(async () => {
|
|
||||||
const _url = tags.data[url]
|
|
||||||
if (!_url) {
|
|
||||||
error = "No URL found in attribute" + url
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
console.log("Attempting to download", _url)
|
|
||||||
const downloaded = await Utils.downloadJsonAdvanced(_url)
|
|
||||||
if (downloaded["error"]) {
|
|
||||||
console.error(downloaded)
|
|
||||||
error = downloaded["error"]
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if (postprocessVelopark) {
|
|
||||||
data = VeloparkLoader.convert(downloaded["content"])
|
|
||||||
return
|
|
||||||
}
|
|
||||||
data = downloaded["content"]
|
|
||||||
} catch (e) {
|
|
||||||
console.error(e)
|
|
||||||
error = "" + e
|
|
||||||
}
|
|
||||||
})
|
|
||||||
</script>
|
</script>
|
||||||
|
{#if $externalData === null}
|
||||||
{#if error !== undefined}
|
<!-- empty block -->
|
||||||
|
{:else if $externalData === undefined}
|
||||||
|
<Loading>{$externalData}</Loading>
|
||||||
|
{:else if $externalData["error"] !== undefined}
|
||||||
<div class="alert">
|
<div class="alert">
|
||||||
Something went wrong: {error}
|
Something went wrong: {$externalData["error"]}
|
||||||
</div>
|
</div>
|
||||||
{:else if data === undefined}
|
{:else if $externalData["success"] !== undefined}
|
||||||
<Loading>
|
|
||||||
Loading {$tags[url]}
|
|
||||||
</Loading>
|
|
||||||
{:else if data.properties !== undefined}
|
|
||||||
<ComparisonTable
|
<ComparisonTable
|
||||||
externalProperties={data.properties}
|
externalProperties={$externalData["success"]}
|
||||||
osmProperties={$tags}
|
osmProperties={$tags}
|
||||||
{state}
|
{state}
|
||||||
{feature}
|
{feature}
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
import { Mapillary } from "../../Logic/ImageProviders/Mapillary"
|
import { Mapillary } from "../../Logic/ImageProviders/Mapillary"
|
||||||
import { UIEventSource } from "../../Logic/UIEventSource"
|
import { UIEventSource } from "../../Logic/UIEventSource"
|
||||||
|
|
||||||
export let image: ProvidedImage
|
export let image: Partial<ProvidedImage>
|
||||||
let fallbackImage: string = undefined
|
let fallbackImage: string = undefined
|
||||||
if (image.provider === Mapillary.singleton) {
|
if (image.provider === Mapillary.singleton) {
|
||||||
fallbackImage = "./assets/svg/blocked.svg"
|
fallbackImage = "./assets/svg/blocked.svg"
|
||||||
|
|
|
@ -41,6 +41,11 @@ export default class UrlValidator extends Validator {
|
||||||
"AdGroup",
|
"AdGroup",
|
||||||
"TargetId",
|
"TargetId",
|
||||||
"msclkid",
|
"msclkid",
|
||||||
|
"pk_source",
|
||||||
|
"pk_medium",
|
||||||
|
"pk_campaign",
|
||||||
|
"pk_content",
|
||||||
|
"pk_kwd"
|
||||||
]
|
]
|
||||||
for (const dontLike of blacklistedTrackingParams) {
|
for (const dontLike of blacklistedTrackingParams) {
|
||||||
url.searchParams.delete(dontLike.toLowerCase())
|
url.searchParams.delete(dontLike.toLowerCase())
|
||||||
|
|
|
@ -1,90 +0,0 @@
|
||||||
<script lang="ts">
|
|
||||||
/**
|
|
||||||
* Shows attributes that are loaded via linked data and which are suitable for import
|
|
||||||
*/
|
|
||||||
import type { SpecialVisualizationState } from "./SpecialVisualization"
|
|
||||||
import type { Store } from "../Logic/UIEventSource"
|
|
||||||
import { Stores, UIEventSource } from "../Logic/UIEventSource"
|
|
||||||
|
|
||||||
import type { Feature, Geometry } from "geojson"
|
|
||||||
import LayerConfig from "../Models/ThemeConfig/LayerConfig"
|
|
||||||
import LinkedDataLoader from "../Logic/Web/LinkedDataLoader"
|
|
||||||
import Loading from "./Base/Loading.svelte"
|
|
||||||
import { GeoOperations } from "../Logic/GeoOperations"
|
|
||||||
import { OH } from "./OpeningHours/OpeningHours"
|
|
||||||
|
|
||||||
export let state: SpecialVisualizationState
|
|
||||||
export let tagsSource: UIEventSource<Record<string, string>>
|
|
||||||
export let argument: string[]
|
|
||||||
export let feature: Feature
|
|
||||||
export let layer: LayerConfig
|
|
||||||
export let key: string
|
|
||||||
|
|
||||||
|
|
||||||
let url = tagsSource.mapD(tags => {
|
|
||||||
if (!tags._country || !tags[key] || tags[key] === "undefined") {
|
|
||||||
return undefined
|
|
||||||
}
|
|
||||||
return ({ url: tags[key], country: tags._country })
|
|
||||||
})
|
|
||||||
let dataWithErr = url.bindD(({ url, country }) => {
|
|
||||||
return Stores.FromPromiseWithErr(LinkedDataLoader.fetchJsonLd(url, country))
|
|
||||||
})
|
|
||||||
|
|
||||||
let error = dataWithErr.mapD(d => d["error"])
|
|
||||||
let data = dataWithErr.mapD(d => d["success"])
|
|
||||||
|
|
||||||
let distanceToFeature: Store<string> = data.mapD(d => d.geo).mapD(geo => {
|
|
||||||
const dist = Math.round(GeoOperations.distanceBetween(
|
|
||||||
GeoOperations.centerpointCoordinates(<Geometry>geo), GeoOperations.centerpointCoordinates(feature)))
|
|
||||||
return dist + "m"
|
|
||||||
})
|
|
||||||
let dataCleaned = data.mapD(d => {
|
|
||||||
const featureTags = tagsSource.data
|
|
||||||
console.log("Downloaded data is", d)
|
|
||||||
d = { ...d }
|
|
||||||
delete d["@context"]
|
|
||||||
for (const k in d) {
|
|
||||||
const v = featureTags[k]
|
|
||||||
if (!v) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if (k === "opening_hours") {
|
|
||||||
const oh = [].concat(...v.split(";").map(r => OH.ParseRule(r) ?? []))
|
|
||||||
const merged = OH.ToString(OH.MergeTimes(oh ?? []))
|
|
||||||
if (merged === d[k]) {
|
|
||||||
delete d[k]
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (featureTags[k] === d[k]) {
|
|
||||||
delete d[k]
|
|
||||||
}
|
|
||||||
delete d.geo
|
|
||||||
}
|
|
||||||
return d
|
|
||||||
}, [tagsSource])
|
|
||||||
|
|
||||||
</script>
|
|
||||||
{#if $error}
|
|
||||||
<div class="alert">
|
|
||||||
{$error}
|
|
||||||
</div>
|
|
||||||
{:else if $url}
|
|
||||||
<div class="flex flex-col border border-dashed border-gray-500 p-1">
|
|
||||||
{#if $dataCleaned !== undefined && Object.keys($dataCleaned).length === 0}
|
|
||||||
No new data from website
|
|
||||||
{:else if !$data}
|
|
||||||
<Loading />
|
|
||||||
{:else}
|
|
||||||
{$distanceToFeature}
|
|
||||||
<ul>
|
|
||||||
{#each Object.keys($dataCleaned) as k}
|
|
||||||
<li>
|
|
||||||
<b>{k}</b>: {JSON.stringify($dataCleaned[k])} {$tagsSource[k]} {($dataCleaned[k]) === $tagsSource[k]}
|
|
||||||
</li>
|
|
||||||
{/each}
|
|
||||||
</ul>
|
|
||||||
{/if}
|
|
||||||
</div>
|
|
||||||
{/if}
|
|
|
@ -459,7 +459,7 @@ class LineRenderingLayer {
|
||||||
} else {
|
} else {
|
||||||
const tags = this._fetchStore(id)
|
const tags = this._fetchStore(id)
|
||||||
this._listenerInstalledOn.add(id)
|
this._listenerInstalledOn.add(id)
|
||||||
tags.addCallbackAndRunD((properties) => {
|
tags?.addCallbackAndRunD((properties) => {
|
||||||
// Make sure to use 'getSource' here, the layer names are different!
|
// Make sure to use 'getSource' here, the layer names are different!
|
||||||
try {
|
try {
|
||||||
if (map.getSource(this._layername) === undefined) {
|
if (map.getSource(this._layername) === undefined) {
|
||||||
|
|
|
@ -3,11 +3,7 @@ import { FixedUiElement } from "./Base/FixedUiElement"
|
||||||
import BaseUIElement from "./BaseUIElement"
|
import BaseUIElement from "./BaseUIElement"
|
||||||
import Title from "./Base/Title"
|
import Title from "./Base/Title"
|
||||||
import Table from "./Base/Table"
|
import Table from "./Base/Table"
|
||||||
import {
|
import { RenderingSpecification, SpecialVisualization, SpecialVisualizationState } from "./SpecialVisualization"
|
||||||
RenderingSpecification,
|
|
||||||
SpecialVisualization,
|
|
||||||
SpecialVisualizationState,
|
|
||||||
} from "./SpecialVisualization"
|
|
||||||
import { HistogramViz } from "./Popup/HistogramViz"
|
import { HistogramViz } from "./Popup/HistogramViz"
|
||||||
import { MinimapViz } from "./Popup/MinimapViz"
|
import { MinimapViz } from "./Popup/MinimapViz"
|
||||||
import { ShareLinkViz } from "./Popup/ShareLinkViz"
|
import { ShareLinkViz } from "./Popup/ShareLinkViz"
|
||||||
|
@ -93,7 +89,7 @@ import SpecialVisualisationUtils from "./SpecialVisualisationUtils"
|
||||||
import LoginButton from "./Base/LoginButton.svelte"
|
import LoginButton from "./Base/LoginButton.svelte"
|
||||||
import Toggle from "./Input/Toggle"
|
import Toggle from "./Input/Toggle"
|
||||||
import ImportReviewIdentity from "./Reviews/ImportReviewIdentity.svelte"
|
import ImportReviewIdentity from "./Reviews/ImportReviewIdentity.svelte"
|
||||||
import LinkedDataDisplay from "./LinkedDataDisplay.svelte"
|
import LinkedDataLoader from "../Logic/Web/LinkedDataLoader"
|
||||||
|
|
||||||
class NearbyImageVis implements SpecialVisualization {
|
class NearbyImageVis implements SpecialVisualization {
|
||||||
// Class must be in SpecialVisualisations due to weird cyclical import that breaks the tests
|
// Class must be in SpecialVisualisations due to weird cyclical import that breaks the tests
|
||||||
|
@ -120,7 +116,7 @@ class NearbyImageVis implements SpecialVisualization {
|
||||||
tags: UIEventSource<Record<string, string>>,
|
tags: UIEventSource<Record<string, string>>,
|
||||||
args: string[],
|
args: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
const isOpen = args[0] === "open"
|
const isOpen = args[0] === "open"
|
||||||
const readonly = args[1] === "readonly"
|
const readonly = args[1] === "readonly"
|
||||||
|
@ -187,7 +183,7 @@ class StealViz implements SpecialVisualization {
|
||||||
selectedElement: otherFeature,
|
selectedElement: otherFeature,
|
||||||
state,
|
state,
|
||||||
layer,
|
layer,
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
if (elements.length === 1) {
|
if (elements.length === 1) {
|
||||||
|
@ -195,8 +191,8 @@ class StealViz implements SpecialVisualization {
|
||||||
}
|
}
|
||||||
return new Combine(elements).SetClass("flex flex-col")
|
return new Combine(elements).SetClass("flex flex-col")
|
||||||
},
|
},
|
||||||
[state.indexedFeatures.featuresById]
|
[state.indexedFeatures.featuresById],
|
||||||
)
|
),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -235,7 +231,7 @@ export class QuestionViz implements SpecialVisualization {
|
||||||
tags: UIEventSource<Record<string, string>>,
|
tags: UIEventSource<Record<string, string>>,
|
||||||
args: string[],
|
args: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
const labels = args[0]
|
const labels = args[0]
|
||||||
?.split(";")
|
?.split(";")
|
||||||
|
@ -271,38 +267,38 @@ export default class SpecialVisualizations {
|
||||||
viz.docs,
|
viz.docs,
|
||||||
viz.args.length > 0
|
viz.args.length > 0
|
||||||
? new Table(
|
? new Table(
|
||||||
["name", "default", "description"],
|
["name", "default", "description"],
|
||||||
viz.args.map((arg) => {
|
viz.args.map((arg) => {
|
||||||
let defaultArg = arg.defaultValue ?? "_undefined_"
|
let defaultArg = arg.defaultValue ?? "_undefined_"
|
||||||
if (defaultArg == "") {
|
if (defaultArg == "") {
|
||||||
defaultArg = "_empty string_"
|
defaultArg = "_empty string_"
|
||||||
}
|
}
|
||||||
return [arg.name, defaultArg, arg.doc]
|
return [arg.name, defaultArg, arg.doc]
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
: undefined,
|
: undefined,
|
||||||
new Title("Example usage of " + viz.funcName, 4),
|
new Title("Example usage of " + viz.funcName, 4),
|
||||||
new FixedUiElement(
|
new FixedUiElement(
|
||||||
viz.example ??
|
viz.example ??
|
||||||
"`{" +
|
"`{" +
|
||||||
viz.funcName +
|
viz.funcName +
|
||||||
"(" +
|
"(" +
|
||||||
viz.args.map((arg) => arg.defaultValue).join(",") +
|
viz.args.map((arg) => arg.defaultValue).join(",") +
|
||||||
")}`"
|
")}`",
|
||||||
).SetClass("literal-code"),
|
).SetClass("literal-code"),
|
||||||
])
|
])
|
||||||
}
|
}
|
||||||
|
|
||||||
public static constructSpecification(
|
public static constructSpecification(
|
||||||
template: string,
|
template: string,
|
||||||
extraMappings: SpecialVisualization[] = []
|
extraMappings: SpecialVisualization[] = [],
|
||||||
): RenderingSpecification[] {
|
): RenderingSpecification[] {
|
||||||
return SpecialVisualisationUtils.constructSpecification(template, extraMappings)
|
return SpecialVisualisationUtils.constructSpecification(template, extraMappings)
|
||||||
}
|
}
|
||||||
|
|
||||||
public static HelpMessage() {
|
public static HelpMessage() {
|
||||||
const helpTexts = SpecialVisualizations.specialVisualizations.map((viz) =>
|
const helpTexts = SpecialVisualizations.specialVisualizations.map((viz) =>
|
||||||
SpecialVisualizations.DocumentationFor(viz)
|
SpecialVisualizations.DocumentationFor(viz),
|
||||||
)
|
)
|
||||||
|
|
||||||
return new Combine([
|
return new Combine([
|
||||||
|
@ -336,10 +332,10 @@ export default class SpecialVisualizations {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
null,
|
null,
|
||||||
" "
|
" ",
|
||||||
)
|
),
|
||||||
).SetClass("code"),
|
).SetClass("code"),
|
||||||
'In other words: use `{ "before": ..., "after": ..., "special": {"type": ..., "argname": ...argvalue...}`. The args are in the `special` block; an argvalue can be a string, a translation or another value. (Refer to class `RewriteSpecial` in case of problems)',
|
"In other words: use `{ \"before\": ..., \"after\": ..., \"special\": {\"type\": ..., \"argname\": ...argvalue...}`. The args are in the `special` block; an argvalue can be a string, a translation or another value. (Refer to class `RewriteSpecial` in case of problems)",
|
||||||
]).SetClass("flex flex-col"),
|
]).SetClass("flex flex-col"),
|
||||||
...helpTexts,
|
...helpTexts,
|
||||||
]).SetClass("flex flex-col")
|
]).SetClass("flex flex-col")
|
||||||
|
@ -348,20 +344,20 @@ export default class SpecialVisualizations {
|
||||||
// noinspection JSUnusedGlobalSymbols
|
// noinspection JSUnusedGlobalSymbols
|
||||||
public static renderExampleOfSpecial(
|
public static renderExampleOfSpecial(
|
||||||
state: SpecialVisualizationState,
|
state: SpecialVisualizationState,
|
||||||
s: SpecialVisualization
|
s: SpecialVisualization,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
const examples =
|
const examples =
|
||||||
s.structuredExamples === undefined
|
s.structuredExamples === undefined
|
||||||
? []
|
? []
|
||||||
: s.structuredExamples().map((e) => {
|
: s.structuredExamples().map((e) => {
|
||||||
return s.constr(
|
return s.constr(
|
||||||
state,
|
state,
|
||||||
new UIEventSource<Record<string, string>>(e.feature.properties),
|
new UIEventSource<Record<string, string>>(e.feature.properties),
|
||||||
e.args,
|
e.args,
|
||||||
e.feature,
|
e.feature,
|
||||||
undefined
|
undefined,
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
return new Combine([new Title(s.funcName), s.docs, ...examples])
|
return new Combine([new Title(s.funcName), s.docs, ...examples])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -401,7 +397,7 @@ export default class SpecialVisualizations {
|
||||||
assignTo: state.userRelatedState.language,
|
assignTo: state.userRelatedState.language,
|
||||||
availableLanguages: state.layout.language,
|
availableLanguages: state.layout.language,
|
||||||
preferredLanguages: state.osmConnection.userDetails.map(
|
preferredLanguages: state.osmConnection.userDetails.map(
|
||||||
(ud) => ud.languages
|
(ud) => ud.languages,
|
||||||
),
|
),
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
@ -426,7 +422,7 @@ export default class SpecialVisualizations {
|
||||||
|
|
||||||
constr(
|
constr(
|
||||||
state: SpecialVisualizationState,
|
state: SpecialVisualizationState,
|
||||||
tagSource: UIEventSource<Record<string, string>>
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
return new VariableUiElement(
|
return new VariableUiElement(
|
||||||
tagSource
|
tagSource
|
||||||
|
@ -436,7 +432,7 @@ export default class SpecialVisualizations {
|
||||||
return new SplitRoadWizard(<WayId>id, state)
|
return new SplitRoadWizard(<WayId>id, state)
|
||||||
}
|
}
|
||||||
return undefined
|
return undefined
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -450,7 +446,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
if (feature.geometry.type !== "Point") {
|
if (feature.geometry.type !== "Point") {
|
||||||
return undefined
|
return undefined
|
||||||
|
@ -473,7 +469,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
if (!layer.deletion) {
|
if (!layer.deletion) {
|
||||||
return undefined
|
return undefined
|
||||||
|
@ -501,7 +497,7 @@ export default class SpecialVisualizations {
|
||||||
state: SpecialVisualizationState,
|
state: SpecialVisualizationState,
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature
|
feature: Feature,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
const [lon, lat] = GeoOperations.centerpointCoordinates(feature)
|
const [lon, lat] = GeoOperations.centerpointCoordinates(feature)
|
||||||
return new SvelteUIElement(CreateNewNote, {
|
return new SvelteUIElement(CreateNewNote, {
|
||||||
|
@ -565,7 +561,7 @@ export default class SpecialVisualizations {
|
||||||
.map((tags) => tags[args[0]])
|
.map((tags) => tags[args[0]])
|
||||||
.map((wikidata) => {
|
.map((wikidata) => {
|
||||||
wikidata = Utils.NoEmpty(
|
wikidata = Utils.NoEmpty(
|
||||||
wikidata?.split(";")?.map((wd) => wd.trim()) ?? []
|
wikidata?.split(";")?.map((wd) => wd.trim()) ?? [],
|
||||||
)[0]
|
)[0]
|
||||||
const entry = Wikidata.LoadWikidataEntry(wikidata)
|
const entry = Wikidata.LoadWikidataEntry(wikidata)
|
||||||
return new VariableUiElement(
|
return new VariableUiElement(
|
||||||
|
@ -575,9 +571,9 @@ export default class SpecialVisualizations {
|
||||||
}
|
}
|
||||||
const response = <WikidataResponse>e["success"]
|
const response = <WikidataResponse>e["success"]
|
||||||
return Translation.fromMap(response.labels)
|
return Translation.fromMap(response.labels)
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
})
|
}),
|
||||||
),
|
),
|
||||||
},
|
},
|
||||||
new MapillaryLinkVis(),
|
new MapillaryLinkVis(),
|
||||||
|
@ -591,7 +587,7 @@ export default class SpecialVisualizations {
|
||||||
tags: UIEventSource<Record<string, string>>,
|
tags: UIEventSource<Record<string, string>>,
|
||||||
_,
|
_,
|
||||||
__,
|
__,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
) => new SvelteUIElement(AllTagsPanel, { tags, layer }),
|
) => new SvelteUIElement(AllTagsPanel, { tags, layer }),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -613,7 +609,7 @@ export default class SpecialVisualizations {
|
||||||
return new ImageCarousel(
|
return new ImageCarousel(
|
||||||
AllImageProviders.LoadImagesFor(tags, imagePrefixes),
|
AllImageProviders.LoadImagesFor(tags, imagePrefixes),
|
||||||
tags,
|
tags,
|
||||||
state
|
state,
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -669,7 +665,7 @@ export default class SpecialVisualizations {
|
||||||
{
|
{
|
||||||
nameKey: nameKey,
|
nameKey: nameKey,
|
||||||
fallbackName,
|
fallbackName,
|
||||||
}
|
},
|
||||||
)
|
)
|
||||||
return new SvelteUIElement(StarsBarIcon, {
|
return new SvelteUIElement(StarsBarIcon, {
|
||||||
score: reviews.average,
|
score: reviews.average,
|
||||||
|
@ -702,7 +698,7 @@ export default class SpecialVisualizations {
|
||||||
{
|
{
|
||||||
nameKey: nameKey,
|
nameKey: nameKey,
|
||||||
fallbackName,
|
fallbackName,
|
||||||
}
|
},
|
||||||
)
|
)
|
||||||
return new SvelteUIElement(ReviewForm, { reviews, state, tags, feature, layer })
|
return new SvelteUIElement(ReviewForm, { reviews, state, tags, feature, layer })
|
||||||
},
|
},
|
||||||
|
@ -734,7 +730,7 @@ export default class SpecialVisualizations {
|
||||||
{
|
{
|
||||||
nameKey: nameKey,
|
nameKey: nameKey,
|
||||||
fallbackName,
|
fallbackName,
|
||||||
}
|
},
|
||||||
)
|
)
|
||||||
return new SvelteUIElement(AllReviews, { reviews, state, tags, feature, layer })
|
return new SvelteUIElement(AllReviews, { reviews, state, tags, feature, layer })
|
||||||
},
|
},
|
||||||
|
@ -754,7 +750,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
const [text] = argument
|
const [text] = argument
|
||||||
return new SvelteUIElement(ImportReviewIdentity, { state, text })
|
return new SvelteUIElement(ImportReviewIdentity, { state, text })
|
||||||
|
@ -813,7 +809,7 @@ export default class SpecialVisualizations {
|
||||||
tags: UIEventSource<Record<string, string>>,
|
tags: UIEventSource<Record<string, string>>,
|
||||||
args: string[],
|
args: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): SvelteUIElement {
|
): SvelteUIElement {
|
||||||
const keyToUse = args[0]
|
const keyToUse = args[0]
|
||||||
const prefix = args[1]
|
const prefix = args[1]
|
||||||
|
@ -850,17 +846,17 @@ export default class SpecialVisualizations {
|
||||||
return undefined
|
return undefined
|
||||||
}
|
}
|
||||||
const allUnits: Unit[] = [].concat(
|
const allUnits: Unit[] = [].concat(
|
||||||
...(state?.layout?.layers?.map((lyr) => lyr.units) ?? [])
|
...(state?.layout?.layers?.map((lyr) => lyr.units) ?? []),
|
||||||
)
|
)
|
||||||
const unit = allUnits.filter((unit) =>
|
const unit = allUnits.filter((unit) =>
|
||||||
unit.isApplicableToKey(key)
|
unit.isApplicableToKey(key),
|
||||||
)[0]
|
)[0]
|
||||||
if (unit === undefined) {
|
if (unit === undefined) {
|
||||||
return value
|
return value
|
||||||
}
|
}
|
||||||
const getCountry = () => tagSource.data._country
|
const getCountry = () => tagSource.data._country
|
||||||
return unit.asHumanLongValue(value, getCountry)
|
return unit.asHumanLongValue(value, getCountry)
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -877,7 +873,7 @@ export default class SpecialVisualizations {
|
||||||
new Combine([
|
new Combine([
|
||||||
t.downloadFeatureAsGeojson.SetClass("font-bold text-lg"),
|
t.downloadFeatureAsGeojson.SetClass("font-bold text-lg"),
|
||||||
t.downloadGeoJsonHelper.SetClass("subtle"),
|
t.downloadGeoJsonHelper.SetClass("subtle"),
|
||||||
]).SetClass("flex flex-col")
|
]).SetClass("flex flex-col"),
|
||||||
)
|
)
|
||||||
.onClick(() => {
|
.onClick(() => {
|
||||||
console.log("Exporting as Geojson")
|
console.log("Exporting as Geojson")
|
||||||
|
@ -890,7 +886,7 @@ export default class SpecialVisualizations {
|
||||||
title + "_mapcomplete_export.geojson",
|
title + "_mapcomplete_export.geojson",
|
||||||
{
|
{
|
||||||
mimetype: "application/vnd.geo+json",
|
mimetype: "application/vnd.geo+json",
|
||||||
}
|
},
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
.SetClass("w-full")
|
.SetClass("w-full")
|
||||||
|
@ -926,7 +922,7 @@ export default class SpecialVisualizations {
|
||||||
constr: (state) => {
|
constr: (state) => {
|
||||||
return new SubtleButton(
|
return new SubtleButton(
|
||||||
Svg.delete_icon_svg().SetStyle("height: 1.5rem"),
|
Svg.delete_icon_svg().SetStyle("height: 1.5rem"),
|
||||||
Translations.t.general.removeLocationHistory
|
Translations.t.general.removeLocationHistory,
|
||||||
).onClick(() => {
|
).onClick(() => {
|
||||||
state.historicalUserLocations.features.setData([])
|
state.historicalUserLocations.features.setData([])
|
||||||
state.selectedElement.setData(undefined)
|
state.selectedElement.setData(undefined)
|
||||||
|
@ -964,10 +960,10 @@ export default class SpecialVisualizations {
|
||||||
.filter((c) => c.text !== "")
|
.filter((c) => c.text !== "")
|
||||||
.map(
|
.map(
|
||||||
(c, i) =>
|
(c, i) =>
|
||||||
new NoteCommentElement(c, state, i, comments.length)
|
new NoteCommentElement(c, state, i, comments.length),
|
||||||
)
|
),
|
||||||
).SetClass("flex flex-col")
|
).SetClass("flex flex-col")
|
||||||
})
|
}),
|
||||||
),
|
),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -1001,7 +997,7 @@ export default class SpecialVisualizations {
|
||||||
tagsSource: UIEventSource<Record<string, string>>,
|
tagsSource: UIEventSource<Record<string, string>>,
|
||||||
_: string[],
|
_: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
) =>
|
) =>
|
||||||
new VariableUiElement(
|
new VariableUiElement(
|
||||||
tagsSource.map((tags) => {
|
tagsSource.map((tags) => {
|
||||||
|
@ -1019,7 +1015,7 @@ export default class SpecialVisualizations {
|
||||||
feature,
|
feature,
|
||||||
layer,
|
layer,
|
||||||
}).SetClass("px-1")
|
}).SetClass("px-1")
|
||||||
})
|
}),
|
||||||
),
|
),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -1035,8 +1031,8 @@ export default class SpecialVisualizations {
|
||||||
let challenge = Stores.FromPromise(
|
let challenge = Stores.FromPromise(
|
||||||
Utils.downloadJsonCached(
|
Utils.downloadJsonCached(
|
||||||
`${Maproulette.defaultEndpoint}/challenge/${parentId}`,
|
`${Maproulette.defaultEndpoint}/challenge/${parentId}`,
|
||||||
24 * 60 * 60 * 1000
|
24 * 60 * 60 * 1000,
|
||||||
)
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
return new VariableUiElement(
|
return new VariableUiElement(
|
||||||
|
@ -1061,7 +1057,7 @@ export default class SpecialVisualizations {
|
||||||
} else {
|
} else {
|
||||||
return [title, new List(listItems)]
|
return [title, new List(listItems)]
|
||||||
}
|
}
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
docs: "Fetches the metadata of MapRoulette campaign that this task is part of and shows those details (namely `title`, `description` and `instruction`).\n\nThis reads the property `mr_challengeId` to detect the parent campaign.",
|
docs: "Fetches the metadata of MapRoulette campaign that this task is part of and shows those details (namely `title`, `description` and `instruction`).\n\nThis reads the property `mr_challengeId` to detect the parent campaign.",
|
||||||
|
@ -1075,15 +1071,15 @@ export default class SpecialVisualizations {
|
||||||
"\n" +
|
"\n" +
|
||||||
"```json\n" +
|
"```json\n" +
|
||||||
"{\n" +
|
"{\n" +
|
||||||
' "id": "mark_duplicate",\n' +
|
" \"id\": \"mark_duplicate\",\n" +
|
||||||
' "render": {\n' +
|
" \"render\": {\n" +
|
||||||
' "special": {\n' +
|
" \"special\": {\n" +
|
||||||
' "type": "maproulette_set_status",\n' +
|
" \"type\": \"maproulette_set_status\",\n" +
|
||||||
' "message": {\n' +
|
" \"message\": {\n" +
|
||||||
' "en": "Mark as not found or false positive"\n' +
|
" \"en\": \"Mark as not found or false positive\"\n" +
|
||||||
" },\n" +
|
" },\n" +
|
||||||
' "status": "2",\n' +
|
" \"status\": \"2\",\n" +
|
||||||
' "image": "close"\n' +
|
" \"image\": \"close\"\n" +
|
||||||
" }\n" +
|
" }\n" +
|
||||||
" }\n" +
|
" }\n" +
|
||||||
"}\n" +
|
"}\n" +
|
||||||
|
@ -1163,8 +1159,8 @@ export default class SpecialVisualizations {
|
||||||
const fsBboxed = new BBoxFeatureSourceForLayer(fs, bbox)
|
const fsBboxed = new BBoxFeatureSourceForLayer(fs, bbox)
|
||||||
return new StatisticsPanel(fsBboxed)
|
return new StatisticsPanel(fsBboxed)
|
||||||
},
|
},
|
||||||
[state.mapProperties.bounds]
|
[state.mapProperties.bounds],
|
||||||
)
|
),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1230,7 +1226,7 @@ export default class SpecialVisualizations {
|
||||||
constr(
|
constr(
|
||||||
state: SpecialVisualizationState,
|
state: SpecialVisualizationState,
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
args: string[]
|
args: string[],
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
let [text, href, classnames, download, ariaLabel] = args
|
let [text, href, classnames, download, ariaLabel] = args
|
||||||
if (download === "") {
|
if (download === "") {
|
||||||
|
@ -1244,14 +1240,14 @@ export default class SpecialVisualizations {
|
||||||
text: Utils.SubstituteKeys(text, tags),
|
text: Utils.SubstituteKeys(text, tags),
|
||||||
href: Utils.SubstituteKeys(href, tags).replaceAll(
|
href: Utils.SubstituteKeys(href, tags).replaceAll(
|
||||||
/ /g,
|
/ /g,
|
||||||
"%20"
|
"%20",
|
||||||
) /* Chromium based browsers eat the spaces */,
|
) /* Chromium based browsers eat the spaces */,
|
||||||
classnames,
|
classnames,
|
||||||
download: Utils.SubstituteKeys(download, tags),
|
download: Utils.SubstituteKeys(download, tags),
|
||||||
ariaLabel: Utils.SubstituteKeys(ariaLabel, tags),
|
ariaLabel: Utils.SubstituteKeys(ariaLabel, tags),
|
||||||
newTab,
|
newTab,
|
||||||
})
|
}),
|
||||||
)
|
),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1273,7 +1269,7 @@ export default class SpecialVisualizations {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
null,
|
null,
|
||||||
" "
|
" ",
|
||||||
) +
|
) +
|
||||||
"\n```",
|
"\n```",
|
||||||
args: [
|
args: [
|
||||||
|
@ -1297,7 +1293,7 @@ export default class SpecialVisualizations {
|
||||||
featureTags: UIEventSource<Record<string, string>>,
|
featureTags: UIEventSource<Record<string, string>>,
|
||||||
args: string[],
|
args: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
) {
|
) {
|
||||||
const [key, tr, classesRaw] = args
|
const [key, tr, classesRaw] = args
|
||||||
let classes = classesRaw ?? ""
|
let classes = classesRaw ?? ""
|
||||||
|
@ -1322,7 +1318,7 @@ export default class SpecialVisualizations {
|
||||||
elements.push(subsTr)
|
elements.push(subsTr)
|
||||||
}
|
}
|
||||||
return elements
|
return elements
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1342,7 +1338,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
return new VariableUiElement(
|
return new VariableUiElement(
|
||||||
tagSource.map((tags) => {
|
tagSource.map((tags) => {
|
||||||
|
@ -1354,7 +1350,7 @@ export default class SpecialVisualizations {
|
||||||
console.error("Cannot create a translation for", v, "due to", e)
|
console.error("Cannot create a translation for", v, "due to", e)
|
||||||
return JSON.stringify(v)
|
return JSON.stringify(v)
|
||||||
}
|
}
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1374,7 +1370,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
const key = argument[0]
|
const key = argument[0]
|
||||||
const validator = new FediverseValidator()
|
const validator = new FediverseValidator()
|
||||||
|
@ -1384,7 +1380,7 @@ export default class SpecialVisualizations {
|
||||||
.map((fediAccount) => {
|
.map((fediAccount) => {
|
||||||
fediAccount = validator.reformat(fediAccount)
|
fediAccount = validator.reformat(fediAccount)
|
||||||
const [_, username, host] = fediAccount.match(
|
const [_, username, host] = fediAccount.match(
|
||||||
FediverseValidator.usernameAtServer
|
FediverseValidator.usernameAtServer,
|
||||||
)
|
)
|
||||||
|
|
||||||
const normalLink = new SvelteUIElement(Link, {
|
const normalLink = new SvelteUIElement(Link, {
|
||||||
|
@ -1396,10 +1392,10 @@ export default class SpecialVisualizations {
|
||||||
const loggedInContributorMastodon =
|
const loggedInContributorMastodon =
|
||||||
state.userRelatedState?.preferencesAsTags?.data?.[
|
state.userRelatedState?.preferencesAsTags?.data?.[
|
||||||
"_mastodon_link"
|
"_mastodon_link"
|
||||||
]
|
]
|
||||||
console.log(
|
console.log(
|
||||||
"LoggedinContributorMastodon",
|
"LoggedinContributorMastodon",
|
||||||
loggedInContributorMastodon
|
loggedInContributorMastodon,
|
||||||
)
|
)
|
||||||
if (!loggedInContributorMastodon) {
|
if (!loggedInContributorMastodon) {
|
||||||
return normalLink
|
return normalLink
|
||||||
|
@ -1415,7 +1411,7 @@ export default class SpecialVisualizations {
|
||||||
newTab: true,
|
newTab: true,
|
||||||
}).SetClass("button"),
|
}).SetClass("button"),
|
||||||
])
|
])
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1435,7 +1431,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
args: string[],
|
args: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
return new FixedUiElement("{" + args[0] + "}")
|
return new FixedUiElement("{" + args[0] + "}")
|
||||||
},
|
},
|
||||||
|
@ -1456,7 +1452,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
const key = argument[0] ?? "value"
|
const key = argument[0] ?? "value"
|
||||||
return new VariableUiElement(
|
return new VariableUiElement(
|
||||||
|
@ -1474,12 +1470,12 @@ export default class SpecialVisualizations {
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
return new FixedUiElement(
|
return new FixedUiElement(
|
||||||
"Could not parse this tag: " +
|
"Could not parse this tag: " +
|
||||||
JSON.stringify(value) +
|
JSON.stringify(value) +
|
||||||
" due to " +
|
" due to " +
|
||||||
e
|
e,
|
||||||
).SetClass("alert")
|
).SetClass("alert")
|
||||||
}
|
}
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1500,7 +1496,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
const giggityUrl = argument[0]
|
const giggityUrl = argument[0]
|
||||||
return new SvelteUIElement(Giggity, { tags: tagSource, state, giggityUrl })
|
return new SvelteUIElement(Giggity, { tags: tagSource, state, giggityUrl })
|
||||||
|
@ -1516,12 +1512,12 @@ export default class SpecialVisualizations {
|
||||||
_: UIEventSource<Record<string, string>>,
|
_: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
const tags = (<ThemeViewState>(
|
const tags = (<ThemeViewState>(
|
||||||
state
|
state
|
||||||
)).geolocation.currentUserLocation.features.map(
|
)).geolocation.currentUserLocation.features.map(
|
||||||
(features) => features[0]?.properties
|
(features) => features[0]?.properties,
|
||||||
)
|
)
|
||||||
return new Combine([
|
return new Combine([
|
||||||
new SvelteUIElement(OrientationDebugPanel, {}),
|
new SvelteUIElement(OrientationDebugPanel, {}),
|
||||||
|
@ -1543,7 +1539,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
return new SvelteUIElement(MarkAsFavourite, {
|
return new SvelteUIElement(MarkAsFavourite, {
|
||||||
tags: tagSource,
|
tags: tagSource,
|
||||||
|
@ -1563,7 +1559,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
return new SvelteUIElement(MarkAsFavouriteMini, {
|
return new SvelteUIElement(MarkAsFavouriteMini, {
|
||||||
tags: tagSource,
|
tags: tagSource,
|
||||||
|
@ -1583,7 +1579,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
return new SvelteUIElement(DirectionIndicator, { state, feature })
|
return new SvelteUIElement(DirectionIndicator, { state, feature })
|
||||||
},
|
},
|
||||||
|
@ -1598,7 +1594,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
return new VariableUiElement(
|
return new VariableUiElement(
|
||||||
tagSource
|
tagSource
|
||||||
|
@ -1620,9 +1616,9 @@ export default class SpecialVisualizations {
|
||||||
`${window.location.protocol}//${window.location.host}${window.location.pathname}?${layout}lat=${lat}&lon=${lon}&z=15` +
|
`${window.location.protocol}//${window.location.host}${window.location.pathname}?${layout}lat=${lat}&lon=${lon}&z=15` +
|
||||||
`#${id}`
|
`#${id}`
|
||||||
return new Img(new Qr(url).toImageElement(75)).SetStyle(
|
return new Img(new Qr(url).toImageElement(75)).SetStyle(
|
||||||
"width: 75px"
|
"width: 75px",
|
||||||
)
|
)
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1642,7 +1638,7 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
args: string[],
|
args: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
const key = args[0] === "" ? "_direction:centerpoint" : args[0]
|
const key = args[0] === "" ? "_direction:centerpoint" : args[0]
|
||||||
return new VariableUiElement(
|
return new VariableUiElement(
|
||||||
|
@ -1653,11 +1649,11 @@ export default class SpecialVisualizations {
|
||||||
})
|
})
|
||||||
.mapD((value) => {
|
.mapD((value) => {
|
||||||
const dir = GeoOperations.bearingToHuman(
|
const dir = GeoOperations.bearingToHuman(
|
||||||
GeoOperations.parseBearing(value)
|
GeoOperations.parseBearing(value),
|
||||||
)
|
)
|
||||||
console.log("Human dir", dir)
|
console.log("Human dir", dir)
|
||||||
return Translations.t.general.visualFeedback.directionsAbsolute[dir]
|
return Translations.t.general.visualFeedback.directionsAbsolute[dir]
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1675,11 +1671,6 @@ export default class SpecialVisualizations {
|
||||||
required: true,
|
required: true,
|
||||||
doc: "The domain name(s) where data might be fetched from - this is needed to set the CSP. A domain must include 'https', e.g. 'https://example.com'. For multiple domains, separate them with ';'. If you don't know the possible domains, use '*'. ",
|
doc: "The domain name(s) where data might be fetched from - this is needed to set the CSP. A domain must include 'https', e.g. 'https://example.com'. For multiple domains, separate them with ';'. If you don't know the possible domains, use '*'. ",
|
||||||
},
|
},
|
||||||
{
|
|
||||||
name: "postprocessing",
|
|
||||||
required: false,
|
|
||||||
doc: "Apply some postprocessing. Currently, only 'velopark' is allowed as value",
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
name: "readonly",
|
name: "readonly",
|
||||||
required: false,
|
required: false,
|
||||||
|
@ -1692,19 +1683,19 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
args: string[],
|
args: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
const url = args[0]
|
const url = args[0]
|
||||||
const postprocessVelopark = args[2] === "velopark"
|
|
||||||
const readonly = args[3] === "yes"
|
const readonly = args[3] === "yes"
|
||||||
|
const externalData = Stores.FromPromiseWithErr(Utils.downloadJson(url))
|
||||||
return new SvelteUIElement(ComparisonTool, {
|
return new SvelteUIElement(ComparisonTool, {
|
||||||
url,
|
url,
|
||||||
postprocessVelopark,
|
|
||||||
state,
|
state,
|
||||||
tags: tagSource,
|
tags: tagSource,
|
||||||
layer,
|
layer,
|
||||||
feature,
|
feature,
|
||||||
readonly,
|
readonly,
|
||||||
|
externalData,
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1718,12 +1709,12 @@ export default class SpecialVisualizations {
|
||||||
tagSource: UIEventSource<Record<string, string>>,
|
tagSource: UIEventSource<Record<string, string>>,
|
||||||
args: string[],
|
args: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
return new Toggle(
|
return new Toggle(
|
||||||
undefined,
|
undefined,
|
||||||
new SvelteUIElement(LoginButton),
|
new SvelteUIElement(LoginButton),
|
||||||
state.osmConnection.isLoggedIn
|
state.osmConnection.isLoggedIn,
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1740,19 +1731,36 @@ export default class SpecialVisualizations {
|
||||||
needsUrls: [Constants.linkedDataProxy],
|
needsUrls: [Constants.linkedDataProxy],
|
||||||
constr(
|
constr(
|
||||||
state: SpecialVisualizationState,
|
state: SpecialVisualizationState,
|
||||||
tagsSource: UIEventSource<Record<string, string>>,
|
tags: UIEventSource<Record<string, string>>,
|
||||||
argument: string[],
|
argument: string[],
|
||||||
feature: Feature,
|
feature: Feature,
|
||||||
layer: LayerConfig
|
layer: LayerConfig,
|
||||||
): BaseUIElement {
|
): BaseUIElement {
|
||||||
const key = argument[0] ?? "website"
|
const key = argument[0] ?? "website"
|
||||||
return new SvelteUIElement(LinkedDataDisplay, {
|
let url = tags.mapD(tags => {
|
||||||
feature,
|
if (!tags._country || !tags[key] || tags[key] === "undefined") {
|
||||||
state,
|
return null
|
||||||
tagsSource,
|
}
|
||||||
key,
|
return ({ url: tags[key], country: tags._country })
|
||||||
layer,
|
|
||||||
})
|
})
|
||||||
|
const externalData: Store<{ success: { content: any } } | {
|
||||||
|
error: string
|
||||||
|
} | undefined | null> = url.bindD(({
|
||||||
|
url,
|
||||||
|
country,
|
||||||
|
}) => Stores.FromPromiseWithErr(LinkedDataLoader.fetchJsonLdWithProxy(url, { country })))
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
return new Toggle(
|
||||||
|
new SvelteUIElement(ComparisonTool, {
|
||||||
|
feature,
|
||||||
|
state,
|
||||||
|
tags,
|
||||||
|
layer,
|
||||||
|
externalData,
|
||||||
|
}), undefined, url.map(url => !!url),
|
||||||
|
)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
|
@ -1766,7 +1774,7 @@ export default class SpecialVisualizations {
|
||||||
throw (
|
throw (
|
||||||
"Invalid special visualisation found: funcName is undefined for " +
|
"Invalid special visualisation found: funcName is undefined for " +
|
||||||
invalid.map((sp) => sp.i).join(", ") +
|
invalid.map((sp) => sp.i).join(", ") +
|
||||||
'. Did you perhaps type \n funcName: "funcname" // type declaration uses COLON\ninstead of:\n funcName = "funcName" // value definition uses EQUAL'
|
". Did you perhaps type \n funcName: \"funcname\" // type declaration uses COLON\ninstead of:\n funcName = \"funcName\" // value definition uses EQUAL"
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue