forked from MapComplete/MapComplete
Cleanup of changeset handler, prep for #2082
This commit is contained in:
parent
e3a0a1dbcb
commit
90916cdd32
5 changed files with 161 additions and 138 deletions
|
@ -106,7 +106,7 @@ class HandleErrors extends Script {
|
||||||
deletedObjects: OsmObject[]
|
deletedObjects: OsmObject[]
|
||||||
} = changesObj.CreateChangesetObjects(toUpload, objects)
|
} = changesObj.CreateChangesetObjects(toUpload, objects)
|
||||||
|
|
||||||
const changeset = Changes.createChangesetFor("", changes)
|
const changeset = Changes.buildChangesetXML("", changes)
|
||||||
const path =
|
const path =
|
||||||
"error_changeset_" + parsed.index + "_" + e.layout + "_" + e.username + ".osc"
|
"error_changeset_" + parsed.index + "_" + e.layout + "_" + e.username + ".osc"
|
||||||
if (
|
if (
|
||||||
|
|
|
@ -313,7 +313,7 @@ export default class CleanRepair extends Script {
|
||||||
|
|
||||||
const changedObjects = changes.CreateChangesetObjects(changesToMake, objects)
|
const changedObjects = changes.CreateChangesetObjects(changesToMake, objects)
|
||||||
|
|
||||||
const osc = Changes.createChangesetFor("", changedObjects)
|
const osc = Changes.buildChangesetXML("", changedObjects)
|
||||||
|
|
||||||
writeFileSync("Cleanup.osc", osc, "utf8")
|
writeFileSync("Cleanup.osc", osc, "utf8")
|
||||||
}
|
}
|
||||||
|
|
|
@ -41,7 +41,7 @@ export class Changes {
|
||||||
private readonly previouslyCreated: OsmObject[] = []
|
private readonly previouslyCreated: OsmObject[] = []
|
||||||
private readonly _leftRightSensitive: boolean
|
private readonly _leftRightSensitive: boolean
|
||||||
public readonly _changesetHandler: ChangesetHandler
|
public readonly _changesetHandler: ChangesetHandler
|
||||||
private readonly _reportError?: (string: string | Error) => void
|
private readonly _reportError?: (string: string | Error, extramessage?: string) => void
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
state: {
|
state: {
|
||||||
|
@ -53,7 +53,7 @@ export class Changes {
|
||||||
featureSwitches?: FeatureSwitchState
|
featureSwitches?: FeatureSwitchState
|
||||||
},
|
},
|
||||||
leftRightSensitive: boolean = false,
|
leftRightSensitive: boolean = false,
|
||||||
reportError?: (string: string | Error) => void
|
reportError?: (string: string | Error, extramessage?: string) => void
|
||||||
) {
|
) {
|
||||||
this._leftRightSensitive = leftRightSensitive
|
this._leftRightSensitive = leftRightSensitive
|
||||||
// We keep track of all changes just as well
|
// We keep track of all changes just as well
|
||||||
|
@ -68,7 +68,7 @@ export class Changes {
|
||||||
state.osmConnection,
|
state.osmConnection,
|
||||||
state.featurePropertiesStore,
|
state.featurePropertiesStore,
|
||||||
this,
|
this,
|
||||||
(e) => this._reportError(e)
|
(e, extramessage: string) => this._reportError(e, extramessage)
|
||||||
)
|
)
|
||||||
this.historicalUserLocations = state.historicalUserLocations
|
this.historicalUserLocations = state.historicalUserLocations
|
||||||
|
|
||||||
|
@ -76,7 +76,7 @@ export class Changes {
|
||||||
// This doesn't matter however, as the '-1' is per piecewise upload, not global per changeset
|
// This doesn't matter however, as the '-1' is per piecewise upload, not global per changeset
|
||||||
}
|
}
|
||||||
|
|
||||||
static createChangesetFor(
|
static buildChangesetXML(
|
||||||
csId: string,
|
csId: string,
|
||||||
allChanges: {
|
allChanges: {
|
||||||
modifiedObjects: OsmObject[]
|
modifiedObjects: OsmObject[]
|
||||||
|
@ -618,14 +618,15 @@ export class Changes {
|
||||||
openChangeset: UIEventSource<number>
|
openChangeset: UIEventSource<number>
|
||||||
): Promise<ChangeDescription[]> {
|
): Promise<ChangeDescription[]> {
|
||||||
const neededIds = Changes.GetNeededIds(pending)
|
const neededIds = Changes.GetNeededIds(pending)
|
||||||
// We _do not_ pass in the Changes object itself - we want the data from OSM directly in order to apply the changes
|
/* Download the latest version of the OSM-objects
|
||||||
|
* We _do not_ pass in the Changes object itself - we want the data from OSM directly in order to apply the changes
|
||||||
|
*/
|
||||||
const downloader = new OsmObjectDownloader(this.backend, undefined)
|
const downloader = new OsmObjectDownloader(this.backend, undefined)
|
||||||
let osmObjects = await Promise.all<{ id: string; osmObj: OsmObject | "deleted" }>(
|
const osmObjects = Utils.NoNull(await Promise.all<{ id: string; osmObj: OsmObject | "deleted" }>(
|
||||||
neededIds.map((id) => this.getOsmObject(id, downloader))
|
neededIds.map((id) => this.getOsmObject(id, downloader))
|
||||||
)
|
))
|
||||||
|
|
||||||
osmObjects = Utils.NoNull(osmObjects)
|
|
||||||
|
|
||||||
|
// Drop changes to deleted items
|
||||||
for (const { osmObj, id } of osmObjects) {
|
for (const { osmObj, id } of osmObjects) {
|
||||||
if (osmObj === "deleted") {
|
if (osmObj === "deleted") {
|
||||||
pending = pending.filter((ch) => ch.type + "/" + ch.id !== id)
|
pending = pending.filter((ch) => ch.type + "/" + ch.id !== id)
|
||||||
|
@ -645,20 +646,56 @@ export class Changes {
|
||||||
return undefined
|
return undefined
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const metatags = this.buildChangesetTags(pending)
|
||||||
|
|
||||||
|
let { toUpload, refused } = this.fragmentChanges(pending, objects)
|
||||||
|
|
||||||
|
if (toUpload.length === 0) {
|
||||||
|
return refused
|
||||||
|
}
|
||||||
|
await this._changesetHandler.UploadChangeset(
|
||||||
|
(csId, remappings) => {
|
||||||
|
if (remappings.size > 0) {
|
||||||
|
toUpload = toUpload.map((ch) =>
|
||||||
|
ChangeDescriptionTools.rewriteIds(ch, remappings)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const changes: {
|
||||||
|
newObjects: OsmObject[]
|
||||||
|
modifiedObjects: OsmObject[]
|
||||||
|
deletedObjects: OsmObject[]
|
||||||
|
} = this.CreateChangesetObjects(toUpload, objects)
|
||||||
|
|
||||||
|
return Changes.buildChangesetXML("" + csId, changes)
|
||||||
|
},
|
||||||
|
metatags,
|
||||||
|
openChangeset
|
||||||
|
)
|
||||||
|
|
||||||
|
console.log("Upload successful! Refused changes are", refused)
|
||||||
|
return refused
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Builds all the changeset tags, such as `theme=cyclofix; answer=42; add-image: 5`, ...
|
||||||
|
*/
|
||||||
|
private buildChangesetTags(pending: ChangeDescription[]) {
|
||||||
|
// Build statistics for the changeset tags
|
||||||
const perType = Array.from(
|
const perType = Array.from(
|
||||||
Utils.Hist(
|
Utils.Hist(
|
||||||
pending
|
pending
|
||||||
.filter(
|
.filter(
|
||||||
(descr) =>
|
(descr) =>
|
||||||
descr.meta.changeType !== undefined && descr.meta.changeType !== null
|
descr.meta.changeType !== undefined && descr.meta.changeType !== null,
|
||||||
)
|
)
|
||||||
.map((descr) => descr.meta.changeType)
|
.map((descr) => descr.meta.changeType),
|
||||||
),
|
),
|
||||||
([key, count]) => ({
|
([key, count]) => ({
|
||||||
key: key,
|
key: key,
|
||||||
value: count,
|
value: count,
|
||||||
aggregate: true,
|
aggregate: true,
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
const motivations = pending
|
const motivations = pending
|
||||||
.filter((descr) => descr.meta.specialMotivation !== undefined)
|
.filter((descr) => descr.meta.specialMotivation !== undefined)
|
||||||
|
@ -697,7 +734,7 @@ export class Changes {
|
||||||
value: count,
|
value: count,
|
||||||
aggregate: true,
|
aggregate: true,
|
||||||
}
|
}
|
||||||
})
|
}),
|
||||||
)
|
)
|
||||||
|
|
||||||
// This method is only called with changedescriptions for this theme
|
// This method is only called with changedescriptions for this theme
|
||||||
|
@ -720,34 +757,7 @@ export class Changes {
|
||||||
...motivations,
|
...motivations,
|
||||||
...perBinMessage,
|
...perBinMessage,
|
||||||
]
|
]
|
||||||
|
return metatags
|
||||||
let { toUpload, refused } = this.fragmentChanges(pending, objects)
|
|
||||||
|
|
||||||
if (toUpload.length === 0) {
|
|
||||||
return refused
|
|
||||||
}
|
|
||||||
await this._changesetHandler.UploadChangeset(
|
|
||||||
(csId, remappings) => {
|
|
||||||
if (remappings.size > 0) {
|
|
||||||
toUpload = toUpload.map((ch) =>
|
|
||||||
ChangeDescriptionTools.rewriteIds(ch, remappings)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
const changes: {
|
|
||||||
newObjects: OsmObject[]
|
|
||||||
modifiedObjects: OsmObject[]
|
|
||||||
deletedObjects: OsmObject[]
|
|
||||||
} = this.CreateChangesetObjects(toUpload, objects)
|
|
||||||
|
|
||||||
return Changes.createChangesetFor("" + csId, changes)
|
|
||||||
},
|
|
||||||
metatags,
|
|
||||||
openChangeset
|
|
||||||
)
|
|
||||||
|
|
||||||
console.log("Upload successful! Refused changes are", refused)
|
|
||||||
return refused
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async flushChangesAsync(): Promise<void> {
|
private async flushChangesAsync(): Promise<void> {
|
||||||
|
|
|
@ -13,6 +13,19 @@ export interface ChangesetTag {
|
||||||
aggregate?: boolean
|
aggregate?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type ChangesetMetadata = {
|
||||||
|
type: "changeset"
|
||||||
|
id: number
|
||||||
|
created_at: string
|
||||||
|
open: boolean
|
||||||
|
uid: number
|
||||||
|
user: string
|
||||||
|
changes_count: number
|
||||||
|
tags: Record<string, string>,
|
||||||
|
minlat: number, minlon: number, maxlat: number, maxlon: number
|
||||||
|
comments_count: number
|
||||||
|
}
|
||||||
|
|
||||||
export class ChangesetHandler {
|
export class ChangesetHandler {
|
||||||
private readonly allElements: FeaturePropertiesStore
|
private readonly allElements: FeaturePropertiesStore
|
||||||
private osmConnection: OsmConnection
|
private osmConnection: OsmConnection
|
||||||
|
@ -26,7 +39,7 @@ export class ChangesetHandler {
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
public readonly _remappings = new Map<string, string>()
|
public readonly _remappings = new Map<string, string>()
|
||||||
private readonly _reportError: (e: string | Error) => void
|
private readonly _reportError: (e: string | Error, extramsg: string) => void
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
dryRun: Store<boolean>,
|
dryRun: Store<boolean>,
|
||||||
|
@ -36,7 +49,7 @@ export class ChangesetHandler {
|
||||||
| { addAlias: (id0: string, id1: string) => void }
|
| { addAlias: (id0: string, id1: string) => void }
|
||||||
| undefined,
|
| undefined,
|
||||||
changes: Changes,
|
changes: Changes,
|
||||||
reportError: (e: string | Error) => void
|
reportError: (e: string | Error, extramessage: string) => void,
|
||||||
) {
|
) {
|
||||||
this.osmConnection = osmConnection
|
this.osmConnection = osmConnection
|
||||||
this._reportError = reportError
|
this._reportError = reportError
|
||||||
|
@ -94,6 +107,27 @@ export class ChangesetHandler {
|
||||||
return hasChange
|
return hasChange
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async UploadWithNew(generateChangeXML: (csid: number, remappings: Map<string, string>) => string, openChangeset: UIEventSource<number>, extraMetaTags: ChangesetTag[]) {
|
||||||
|
const csId = await this.OpenChangeset(extraMetaTags)
|
||||||
|
openChangeset.setData(csId)
|
||||||
|
const changeset = generateChangeXML(csId, this._remappings)
|
||||||
|
console.log(
|
||||||
|
"Opened a new changeset (openChangeset.data is undefined):",
|
||||||
|
changeset,
|
||||||
|
extraMetaTags,
|
||||||
|
)
|
||||||
|
const changes = await this.UploadChange(csId, changeset)
|
||||||
|
const hasSpecialMotivationChanges = ChangesetHandler.rewriteMetaTags(
|
||||||
|
extraMetaTags,
|
||||||
|
changes,
|
||||||
|
)
|
||||||
|
if (hasSpecialMotivationChanges) {
|
||||||
|
// At this point, 'extraMetaTags' will have changed - we need to set the tags again
|
||||||
|
await this.UpdateTags(csId, extraMetaTags)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The full logic to upload a change to one or more elements.
|
* The full logic to upload a change to one or more elements.
|
||||||
*
|
*
|
||||||
|
@ -107,7 +141,7 @@ export class ChangesetHandler {
|
||||||
public async UploadChangeset(
|
public async UploadChangeset(
|
||||||
generateChangeXML: (csid: number, remappings: Map<string, string>) => string,
|
generateChangeXML: (csid: number, remappings: Map<string, string>) => string,
|
||||||
extraMetaTags: ChangesetTag[],
|
extraMetaTags: ChangesetTag[],
|
||||||
openChangeset: UIEventSource<number>
|
openChangeset: UIEventSource<number>,
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
if (
|
if (
|
||||||
!extraMetaTags.some((tag) => tag.key === "comment") ||
|
!extraMetaTags.some((tag) => tag.key === "comment") ||
|
||||||
|
@ -130,29 +164,44 @@ export class ChangesetHandler {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if (openChangeset.data === undefined) {
|
if (openChangeset.data) {
|
||||||
// We have to open a new changeset
|
|
||||||
try {
|
try {
|
||||||
const csId = await this.OpenChangeset(extraMetaTags)
|
const csId = openChangeset.data
|
||||||
openChangeset.setData(csId)
|
const oldChangesetMeta = await this.GetChangesetMeta(csId)
|
||||||
const changeset = generateChangeXML(csId, this._remappings)
|
if (oldChangesetMeta.open) {
|
||||||
console.log(
|
// We can hopefully reuse the changeset
|
||||||
"Opened a new changeset (openChangeset.data is undefined):",
|
|
||||||
changeset,
|
try {
|
||||||
extraMetaTags
|
|
||||||
|
const rewritings = await this.UploadChange(
|
||||||
|
csId,
|
||||||
|
generateChangeXML(csId, this._remappings),
|
||||||
)
|
)
|
||||||
const changes = await this.UploadChange(csId, changeset)
|
|
||||||
const hasSpecialMotivationChanges = ChangesetHandler.rewriteMetaTags(
|
const rewrittenTags = this.RewriteTagsOf(
|
||||||
extraMetaTags,
|
extraMetaTags,
|
||||||
changes
|
rewritings,
|
||||||
|
oldChangesetMeta,
|
||||||
)
|
)
|
||||||
if (hasSpecialMotivationChanges) {
|
await this.UpdateTags(csId, rewrittenTags)
|
||||||
// At this point, 'extraMetaTags' will have changed - we need to set the tags again
|
return // We are done!
|
||||||
await this.UpdateTags(csId, extraMetaTags)
|
} catch (e) {
|
||||||
|
this._reportError(e, "While reusing a changeset " + openChangeset.data)
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
} catch (e) {
|
||||||
|
this._reportError(e, "While getting metadata from a changeset " + openChangeset.data)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// We have to open a new changeset
|
||||||
|
try {
|
||||||
|
return await this.UploadWithNew(generateChangeXML, openChangeset, extraMetaTags)
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (this._reportError) {
|
if (this._reportError) {
|
||||||
this._reportError(e)
|
this._reportError(e, "While opening a new changeset")
|
||||||
}
|
}
|
||||||
if ((<XMLHttpRequest>e).status === 400) {
|
if ((<XMLHttpRequest>e).status === 400) {
|
||||||
// This request is invalid. We simply drop the changes and hope that someone will analyze what went wrong with it in the upload; we pretend everything went fine
|
// This request is invalid. We simply drop the changes and hope that someone will analyze what went wrong with it in the upload; we pretend everything went fine
|
||||||
|
@ -161,52 +210,12 @@ export class ChangesetHandler {
|
||||||
console.warn(
|
console.warn(
|
||||||
"Could not open/upload changeset due to ",
|
"Could not open/upload changeset due to ",
|
||||||
e,
|
e,
|
||||||
"trying again with a another fresh changeset "
|
"trying again with a another fresh changeset ",
|
||||||
)
|
)
|
||||||
openChangeset.setData(undefined)
|
openChangeset.setData(undefined)
|
||||||
|
|
||||||
throw e
|
throw e
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
// There still exists an open changeset (or at least we hope so)
|
|
||||||
// Let's check!
|
|
||||||
const csId = openChangeset.data
|
|
||||||
try {
|
|
||||||
const oldChangesetMeta = await this.GetChangesetMeta(csId)
|
|
||||||
if (!oldChangesetMeta.open) {
|
|
||||||
// Mark the CS as closed...
|
|
||||||
console.log("Could not fetch the metadata from the already open changeset")
|
|
||||||
openChangeset.setData(undefined)
|
|
||||||
// ... and try again. As the cs is closed, no recursive loop can exist
|
|
||||||
await this.UploadChangeset(generateChangeXML, extraMetaTags, openChangeset)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const rewritings = await this.UploadChange(
|
|
||||||
csId,
|
|
||||||
generateChangeXML(csId, this._remappings)
|
|
||||||
)
|
|
||||||
|
|
||||||
const rewrittenTags = this.RewriteTagsOf(
|
|
||||||
extraMetaTags,
|
|
||||||
rewritings,
|
|
||||||
oldChangesetMeta
|
|
||||||
)
|
|
||||||
await this.UpdateTags(csId, rewrittenTags)
|
|
||||||
} catch (e) {
|
|
||||||
if (this._reportError) {
|
|
||||||
this._reportError(
|
|
||||||
"Could not reuse changeset " +
|
|
||||||
csId +
|
|
||||||
", might be closed: " +
|
|
||||||
(e.stacktrace ?? e.status ?? "" + e)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
console.warn("Could not upload, changeset is probably closed: ", e)
|
|
||||||
openChangeset.setData(undefined)
|
|
||||||
throw e
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -227,7 +236,7 @@ export class ChangesetHandler {
|
||||||
uid: number // User ID
|
uid: number // User ID
|
||||||
changes_count: number
|
changes_count: number
|
||||||
tags: any
|
tags: any
|
||||||
}
|
},
|
||||||
): ChangesetTag[] {
|
): ChangesetTag[] {
|
||||||
// Note: extraMetaTags is where all the tags are collected into
|
// Note: extraMetaTags is where all the tags are collected into
|
||||||
|
|
||||||
|
@ -346,15 +355,9 @@ export class ChangesetHandler {
|
||||||
console.log("Closed changeset ", changesetId)
|
console.log("Closed changeset ", changesetId)
|
||||||
}
|
}
|
||||||
|
|
||||||
private async GetChangesetMeta(csId: number): Promise<{
|
private async GetChangesetMeta(csId: number): Promise<ChangesetMetadata> {
|
||||||
id: number
|
|
||||||
open: boolean
|
|
||||||
uid: number
|
|
||||||
changes_count: number
|
|
||||||
tags: any
|
|
||||||
}> {
|
|
||||||
const url = `${this.backend}/api/0.6/changeset/${csId}`
|
const url = `${this.backend}/api/0.6/changeset/${csId}`
|
||||||
const csData = await Utils.downloadJson(url)
|
const csData = await Utils.downloadJson<{ elements: ChangesetMetadata[] }>(url)
|
||||||
return csData.elements[0]
|
return csData.elements[0]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -370,7 +373,7 @@ export class ChangesetHandler {
|
||||||
tag.key !== undefined &&
|
tag.key !== undefined &&
|
||||||
tag.value !== undefined &&
|
tag.value !== undefined &&
|
||||||
tag.key !== "" &&
|
tag.key !== "" &&
|
||||||
tag.value !== ""
|
tag.value !== "",
|
||||||
)
|
)
|
||||||
const metadata = tags.map((kv) => `<tag k="${kv.key}" v="${escapeHtml(kv.value)}"/>`)
|
const metadata = tags.map((kv) => `<tag k="${kv.key}" v="${escapeHtml(kv.value)}"/>`)
|
||||||
const content = [`<osm><changeset>`, metadata, `</changeset></osm>`].join("")
|
const content = [`<osm><changeset>`, metadata, `</changeset></osm>`].join("")
|
||||||
|
@ -410,7 +413,7 @@ export class ChangesetHandler {
|
||||||
const csId = await this.osmConnection.put(
|
const csId = await this.osmConnection.put(
|
||||||
"changeset/create",
|
"changeset/create",
|
||||||
[`<osm><changeset>`, metadata, `</changeset></osm>`].join(""),
|
[`<osm><changeset>`, metadata, `</changeset></osm>`].join(""),
|
||||||
{ "Content-Type": "text/xml" }
|
{ "Content-Type": "text/xml" },
|
||||||
)
|
)
|
||||||
return Number(csId)
|
return Number(csId)
|
||||||
}
|
}
|
||||||
|
@ -420,12 +423,12 @@ export class ChangesetHandler {
|
||||||
*/
|
*/
|
||||||
private async UploadChange(
|
private async UploadChange(
|
||||||
changesetId: number,
|
changesetId: number,
|
||||||
changesetXML: string
|
changesetXML: string,
|
||||||
): Promise<Map<string, string>> {
|
): Promise<Map<string, string>> {
|
||||||
const response = await this.osmConnection.post<XMLDocument>(
|
const response = await this.osmConnection.post<XMLDocument>(
|
||||||
"changeset/" + changesetId + "/upload",
|
"changeset/" + changesetId + "/upload",
|
||||||
changesetXML,
|
changesetXML,
|
||||||
{ "Content-Type": "text/xml" }
|
{ "Content-Type": "text/xml" },
|
||||||
)
|
)
|
||||||
const changes = this.parseUploadChangesetResponse(response)
|
const changes = this.parseUploadChangesetResponse(response)
|
||||||
console.log("Uploaded changeset ", changesetId)
|
console.log("Uploaded changeset ", changesetId)
|
||||||
|
|
|
@ -278,7 +278,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
|
||||||
featureSwitches: this.featureSwitches,
|
featureSwitches: this.featureSwitches,
|
||||||
},
|
},
|
||||||
layout?.isLeftRightSensitive() ?? false,
|
layout?.isLeftRightSensitive() ?? false,
|
||||||
(e) => this.reportError(e),
|
(e, extraMsg) => this.reportError(e, extraMsg),
|
||||||
)
|
)
|
||||||
this.historicalUserLocations = this.geolocation.historicalUserLocations
|
this.historicalUserLocations = this.geolocation.historicalUserLocations
|
||||||
this.newFeatures = new NewGeometryFromChangesFeatureSource(
|
this.newFeatures = new NewGeometryFromChangesFeatureSource(
|
||||||
|
@ -650,9 +650,9 @@ export default class ThemeViewState implements SpecialVisualizationState {
|
||||||
available,
|
available,
|
||||||
category,
|
category,
|
||||||
current.data,
|
current.data,
|
||||||
skipLayers
|
skipLayers,
|
||||||
)
|
)
|
||||||
if(!best){
|
if (!best) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
console.log("Best layer for category", category, "is", best?.properties?.id)
|
console.log("Best layer for category", category, "is", best?.properties?.id)
|
||||||
|
@ -680,19 +680,19 @@ export default class ThemeViewState implements SpecialVisualizationState {
|
||||||
Hotkeys.RegisterHotkey(
|
Hotkeys.RegisterHotkey(
|
||||||
{ shift: "O" },
|
{ shift: "O" },
|
||||||
Translations.t.hotkeyDocumentation.selectOsmbasedmap,
|
Translations.t.hotkeyDocumentation.selectOsmbasedmap,
|
||||||
() => setLayerCategory("osmbasedmap",2),
|
() => setLayerCategory("osmbasedmap", 2),
|
||||||
)
|
)
|
||||||
|
|
||||||
Hotkeys.RegisterHotkey(
|
Hotkeys.RegisterHotkey(
|
||||||
{ shift: "M" },
|
{ shift: "M" },
|
||||||
Translations.t.hotkeyDocumentation.selectMap,
|
Translations.t.hotkeyDocumentation.selectMap,
|
||||||
() => setLayerCategory("map",2),
|
() => setLayerCategory("map", 2),
|
||||||
)
|
)
|
||||||
|
|
||||||
Hotkeys.RegisterHotkey(
|
Hotkeys.RegisterHotkey(
|
||||||
{ shift: "P" },
|
{ shift: "P" },
|
||||||
Translations.t.hotkeyDocumentation.selectAerial,
|
Translations.t.hotkeyDocumentation.selectAerial,
|
||||||
() => setLayerCategory("photo",2),
|
() => setLayerCategory("photo", 2),
|
||||||
)
|
)
|
||||||
Hotkeys.RegisterHotkey(
|
Hotkeys.RegisterHotkey(
|
||||||
{ nomod: "L" },
|
{ nomod: "L" },
|
||||||
|
@ -907,7 +907,7 @@ export default class ThemeViewState implements SpecialVisualizationState {
|
||||||
this.selectedElement.setData(this.currentView.features?.data?.[0])
|
this.selectedElement.setData(this.currentView.features?.data?.[0])
|
||||||
}
|
}
|
||||||
|
|
||||||
public async reportError(message: string | Error | XMLHttpRequest) {
|
public async reportError(message: string | Error | XMLHttpRequest, extramessage: string = "") {
|
||||||
const isTesting = this.featureSwitchIsTesting.data
|
const isTesting = this.featureSwitchIsTesting.data
|
||||||
console.log(
|
console.log(
|
||||||
isTesting
|
isTesting
|
||||||
|
@ -922,7 +922,17 @@ export default class ThemeViewState implements SpecialVisualizationState {
|
||||||
|
|
||||||
if ("" + message === "[object XMLHttpRequest]") {
|
if ("" + message === "[object XMLHttpRequest]") {
|
||||||
const req = <XMLHttpRequest>message
|
const req = <XMLHttpRequest>message
|
||||||
message = "XMLHttpRequest with status code " + req.status + ", " + req.statusText
|
let body = ""
|
||||||
|
try {
|
||||||
|
body = req.responseText
|
||||||
|
} catch (e) {
|
||||||
|
// pass
|
||||||
|
}
|
||||||
|
message = "XMLHttpRequest with status code " + req.status + ", " + req.statusText + ", received: " + body
|
||||||
|
}
|
||||||
|
|
||||||
|
if (extramessage) {
|
||||||
|
message += "(" + extramessage + ")"
|
||||||
}
|
}
|
||||||
|
|
||||||
const stacktrace: string = new Error().stack
|
const stacktrace: string = new Error().stack
|
||||||
|
|
Loading…
Reference in a new issue