import { Utils } from "../../../../src/Utils"
import LayoutConfig from "../../../../src/Models/ThemeConfig/LayoutConfig"
import { BBox } from "../../../../src/Logic/BBox"
import ReplaceGeometryAction from "../../../../src/Logic/Osm/Actions/ReplaceGeometryAction"
import { describe, expect, it } from "vitest"
import { OsmConnection } from "../../../../src/Logic/Osm/OsmConnection"
import { ImmutableStore } from "../../../../src/Logic/UIEventSource"
import { Changes } from "../../../../src/Logic/Osm/Changes"
import FullNodeDatabaseSource from "../../../../src/Logic/FeatureSource/TiledFeatureSource/FullNodeDatabaseSource"
describe("ReplaceGeometryAction", () => {
const grbStripped = {
id: "grb",
title: {
nl: "GRB import helper",
},
description: "Smaller version of the GRB theme",
language: ["nl", "en"],
socialImage: "img.jpg",
version: "0",
startLat: 51.0249,
startLon: 4.026489,
startZoom: 9,
clustering: false,
overrideAll: {
minzoom: 19,
},
layers: [
{
id: "type_node",
source: {
osmTags: "type=node",
},
pointRendering: null,
lineRendering: [{}],
override: {
calculatedTags: [
"_is_part_of_building=feat.get('parent_ways')?.some(p => p.building !== undefined && p.building !== '') ?? false",
"_is_part_of_grb_building=feat.get('parent_ways')?.some(p => p['source:geometry:ref'] !== undefined) ?? false",
"_is_part_of_building_passage=feat.get('parent_ways')?.some(p => p.tunnel === 'building_passage') ?? false",
"_is_part_of_highway=!feat.get('is_part_of_building_passage') && (feat.get('parent_ways')?.some(p => p.highway !== undefined && p.highway !== '') ?? false)",
"_is_part_of_landuse=feat.get('parent_ways')?.some(p => (p.landuse !== undefined && p.landuse !== '') || (p.natural !== undefined && p.natural !== '')) ?? false",
"_moveable=feat.get('_is_part_of_building') && !feat.get('_is_part_of_grb_building')",
],
pointRendering: [
{
marker: [
{
icon: "square",
color: "#cc0",
},
],
iconSize: "5,5",
location: ["point"],
},
],
passAllFeatures: true,
},
},
{
id: "osm-buildings",
name: "All OSM-buildings",
source: {
osmTags: "building~*",
maxCacheAge: 0,
},
calculatedTags: ["_surface:strict:=feat.get('_surface')"],
lineRendering: [
{
width: {
render: "2",
mappings: [
{
if: "fixme~*",
then: "5",
},
],
},
color: {
render: "#00c",
mappings: [
{
if: "fixme~*",
then: "#ff00ff",
},
{
if: "building=house",
then: "#a00",
},
{
if: "building=shed",
then: "#563e02",
},
{
if: {
or: ["building=garage", "building=garages"],
},
then: "#f9bfbb",
},
{
if: "building=yes",
then: "#0774f2",
},
],
},
},
],
title: "OSM-gebouw",
tagRenderings: [
{
id: "building type",
freeform: {
key: "building",
},
render: "The building type is {building}",
question: {
en: "What kind of building is this?",
},
mappings: [
{
if: "building=house",
then: "A normal house",
},
{
if: "building=detached",
then: "A house detached from other building",
},
{
if: "building=semidetached_house",
then: "A house sharing only one wall with another house",
},
{
if: "building=apartments",
then: "An apartment building - highrise for living",
},
{
if: "building=office",
then: "An office building - highrise for work",
},
{
if: "building=apartments",
then: "An apartment building",
},
{
if: "building=shed",
then: "A small shed, e.g. in a garden",
},
{
if: "building=garage",
then: "A single garage to park a car",
},
{
if: "building=garages",
then: "A building containing only garages; typically they are all identical",
},
{
if: "building=yes",
then: "A building - no specification",
},
],
},
{
id: "grb-housenumber",
render: {
nl: "Het huisnummer is {addr:housenumber}",
},
question: {
nl: "Wat is het huisnummer?",
},
freeform: {
key: "addr:housenumber",
},
mappings: [
{
if: {
and: ["not:addr:housenumber=yes", "addr:housenumber="],
},
then: {
nl: "Geen huisnummer",
},
},
],
},
{
id: "grb-unit",
question: "Wat is de wooneenheid-aanduiding?",
render: {
nl: "De wooneenheid-aanduiding is {addr:unit} ",
},
freeform: {
key: "addr:unit",
},
mappings: [
{
if: "addr:unit=",
then: "Geen wooneenheid-nummer",
},
],
},
{
id: "grb-street",
render: {
nl: "De straat is {addr:street}",
},
freeform: {
key: "addr:street",
},
question: {
nl: "Wat is de straat?",
},
},
{
id: "grb-fixme",
render: {
nl: "De fixme is {fixme}",
},
question: {
nl: "Wat zegt de fixme?",
},
freeform: {
key: "fixme",
},
mappings: [
{
if: {
and: ["fixme="],
},
then: {
nl: "Geen fixme",
},
},
],
},
{
id: "grb-min-level",
render: {
nl: "Dit gebouw begint maar op de {building:min_level} verdieping",
},
question: {
nl: "Hoeveel verdiepingen ontbreken?",
},
freeform: {
key: "building:min_level",
type: "pnat",
},
},
"all_tags",
],
filter: [
{
id: "has-fixme",
options: [
{
osmTags: "fixme~*",
question: "Heeft een FIXME",
},
],
},
],
},
{
id: "grb",
description: "Geometry which comes from GRB with tools to import them",
source: {
osmTags: {
and: ["HUISNR~*", "man_made!=mast"],
},
geoJson:
"https://betadata.grbosm.site/grb?bbox={x_min},{y_min},{x_max},{y_max}",
geoJsonZoomLevel: 18,
mercatorCrs: true,
maxCacheAge: 0,
},
name: "GRB geometries",
title: "GRB outline",
calculatedTags: [
"_overlaps_with_buildings=feat.overlapWith('osm-buildings').filter(f => f.feat.properties.id.indexOf('-') < 0)",
"_overlaps_with=feat.get('_overlaps_with_buildings').filter(f => f.overlap > 1 /* square meter */ )[0] ?? ''",
"_osm_obj:source:ref=feat.get('_overlaps_with')?.feat?.properties['source:geometry:ref']",
"_osm_obj:id=feat.get('_overlaps_with')?.feat?.properties?.id",
"_osm_obj:source:date=feat.get('_overlaps_with')?.feat?.properties['source:geometry:date'].replace(/\\//g, '-')",
"_osm_obj:building=feat.get('_overlaps_with')?.feat?.properties?.building",
"_osm_obj:addr:street=(feat.get('_overlaps_with')?.feat?.properties ?? {})['addr:street']",
"_osm_obj:addr:housenumber=(feat.get('_overlaps_with')?.feat?.properties ?? {})['addr:housenumber']",
"_osm_obj:surface=(feat.get('_overlaps_with')?.feat?.properties ?? {})['_surface:strict']",
"_overlap_absolute=feat.get('_overlaps_with')?.overlap",
"_reverse_overlap_percentage=Math.round(100 * feat.get('_overlap_absolute') / feat.get('_surface'))",
"_overlap_percentage=Math.round(100 * feat.get('_overlap_absolute') / feat.get('_osm_obj:surface'))",
"_grb_ref=feat.properties['source:geometry:entity'] + '/' + feat.properties['source:geometry:oidn']",
"_imported_osm_object_found= feat.properties['_osm_obj:source:ref'] == feat.properties._grb_ref",
"_grb_date=feat.properties['source:geometry:date'].replace(/\\//g,'-')",
"_imported_osm_still_fresh= feat.properties['_osm_obj:source:date'] == feat.properties._grb_date",
"_target_building_type=feat.properties['_osm_obj:building'] === 'yes' ? feat.properties.building : (feat.properties['_osm_obj:building'] ?? feat.properties.building)",
"_building:min_level= feat.properties['fixme']?.startsWith('verdieping, correct the building tag, add building:level and building:min_level before upload in JOSM!') ? '1' : ''",
"_intersects_with_other_features=feat.intersectionsWith('generic_osm_object').map(f => \"\" + f.feat.properties.id + \"\").join(', ')",
],
tagRenderings: [],
pointRendering: [
{
marker: [
{
icon: "./assets/themes/grb/housenumber_blank.svg",
},
],
iconSize: "50,50",
location: ["point", "centroid"],
},
],
},
],
}
const coordinates = <[number, number][]>[
[3.216690793633461, 51.21474084112525],
[3.2167256623506546, 51.214696737309964],
[3.2169999182224274, 51.214768983537674],
[3.2169650495052338, 51.21480720678671],
[3.2169368863105774, 51.21480090625335],
[3.2169489562511444, 51.21478074454077],
[3.216886594891548, 51.214765203214625],
[3.2168812304735184, 51.21477192378873],
[3.2168644666671753, 51.214768983537674],
[3.2168537378311157, 51.21478746511261],
[3.216690793633461, 51.21474084112525],
]
const targetFeature = {
type: "Feature",
properties: {},
geometry: {
type: "Polygon",
coordinates: [coordinates],
},
}
const wayId = "way/160909312"
Utils.injectJsonDownloadForTests(
"https://www.openstreetmap.org/api/0.6/map.json?bbox=3.2166673243045807,51.21467321525788,3.217007964849472,51.21482442824023",
{
version: "0.6",
generator: "CGImap 0.8.6 (1549677 spike-06.openstreetmap.org)",
copyright: "OpenStreetMap and contributors",
attribution: "http://www.openstreetmap.org/copyright",
license: "http://opendatacommons.org/licenses/odbl/1-0/",
bounds: { minlat: 51.2146732, minlon: 3.2166673, maxlat: 51.2148244, maxlon: 3.217008 },
elements: [
{
type: "node",
id: 1612385157,
lat: 51.2148016,
lon: 3.2168453,
timestamp: "2018-04-30T12:26:00Z",
version: 3,
changeset: 58553478,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 1728816256,
lat: 51.2147111,
lon: 3.2170233,
timestamp: "2017-07-18T22:52:44Z",
version: 2,
changeset: 50391526,
user: "catweazle67",
uid: 1976209,
},
{
type: "node",
id: 1728816287,
lat: 51.2146408,
lon: 3.2167601,
timestamp: "2021-10-29T16:24:43Z",
version: 3,
changeset: 113131915,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 1728823481,
lat: 51.2146968,
lon: 3.2167242,
timestamp: "2021-11-02T23:37:11Z",
version: 5,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 1728823499,
lat: 51.2147127,
lon: 3.2170302,
timestamp: "2017-07-18T22:52:45Z",
version: 2,
changeset: 50391526,
user: "catweazle67",
uid: 1976209,
},
{
type: "node",
id: 1728823501,
lat: 51.2148696,
lon: 3.2168941,
timestamp: "2017-07-18T22:52:45Z",
version: 2,
changeset: 50391526,
user: "catweazle67",
uid: 1976209,
},
{
type: "node",
id: 1728823514,
lat: 51.2147863,
lon: 3.2168551,
timestamp: "2021-11-02T23:37:11Z",
version: 5,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 1728823522,
lat: 51.2148489,
lon: 3.2169012,
timestamp: "2017-07-18T22:52:45Z",
version: 2,
changeset: 50391526,
user: "catweazle67",
uid: 1976209,
},
{
type: "node",
id: 1728823523,
lat: 51.2147578,
lon: 3.2169995,
timestamp: "2017-07-18T22:52:45Z",
version: 2,
changeset: 50391526,
user: "catweazle67",
uid: 1976209,
},
{
type: "node",
id: 1728823543,
lat: 51.2148075,
lon: 3.2166445,
timestamp: "2017-07-18T22:52:46Z",
version: 3,
changeset: 50391526,
user: "catweazle67",
uid: 1976209,
},
{
type: "node",
id: 1728823544,
lat: 51.2148553,
lon: 3.2169315,
timestamp: "2017-07-18T22:52:46Z",
version: 2,
changeset: 50391526,
user: "catweazle67",
uid: 1976209,
},
{
type: "node",
id: 1728823549,
lat: 51.2147401,
lon: 3.2168877,
timestamp: "2021-11-02T23:37:11Z",
version: 5,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 4978288376,
lat: 51.2147306,
lon: 3.2168928,
timestamp: "2017-07-18T22:52:21Z",
version: 1,
changeset: 50391526,
user: "catweazle67",
uid: 1976209,
},
{
type: "node",
id: 4978288381,
lat: 51.2147638,
lon: 3.2168856,
timestamp: "2021-11-02T23:37:11Z",
version: 4,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 4978288382,
lat: 51.2148189,
lon: 3.216912,
timestamp: "2017-07-18T22:52:21Z",
version: 1,
changeset: 50391526,
user: "catweazle67",
uid: 1976209,
},
{
type: "node",
id: 4978288385,
lat: 51.2148835,
lon: 3.2170623,
timestamp: "2017-07-18T22:52:21Z",
version: 1,
changeset: 50391526,
user: "catweazle67",
uid: 1976209,
},
{
type: "node",
id: 4978288387,
lat: 51.2148904,
lon: 3.2171037,
timestamp: "2017-07-18T22:52:21Z",
version: 1,
changeset: 50391526,
user: "catweazle67",
uid: 1976209,
},
{
type: "node",
id: 4978289383,
lat: 51.2147678,
lon: 3.2169969,
timestamp: "2021-11-02T23:37:11Z",
version: 4,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 4978289384,
lat: 51.2147684,
lon: 3.2168674,
timestamp: "2021-11-02T23:37:11Z",
version: 4,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 4978289386,
lat: 51.2147716,
lon: 3.2168811,
timestamp: "2021-11-02T23:37:11Z",
version: 4,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 4978289388,
lat: 51.2148115,
lon: 3.216966,
timestamp: "2021-11-02T23:38:13Z",
version: 7,
changeset: 113306325,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 4978289391,
lat: 51.2148019,
lon: 3.2169194,
timestamp: "2017-07-18T22:52:21Z",
version: 1,
changeset: 50391526,
user: "catweazle67",
uid: 1976209,
},
{
type: "node",
id: 9219974337,
lat: 51.2148449,
lon: 3.2171278,
timestamp: "2021-11-02T23:40:52Z",
version: 1,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 9219979643,
lat: 51.2147405,
lon: 3.216693,
timestamp: "2021-11-02T23:37:11Z",
version: 1,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 9219979646,
lat: 51.2148043,
lon: 3.2169312,
timestamp: "2021-11-02T23:38:13Z",
version: 2,
changeset: 113306325,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 9219979647,
lat: 51.2147792,
lon: 3.2169466,
timestamp: "2021-11-02T23:37:11Z",
version: 1,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "way",
id: 160909311,
timestamp: "2021-12-23T12:03:37Z",
version: 6,
changeset: 115295690,
user: "s8evq",
uid: 3710738,
nodes: [
1728823481, 1728823549, 4978288376, 1728823523, 1728823499, 1728816256,
1728816287, 1728823481,
],
tags: {
"addr:city": "Brugge",
"addr:country": "BE",
"addr:housenumber": "106",
"addr:postcode": "8000",
"addr:street": "Ezelstraat",
building: "house",
"source:geometry:date": "2015-07-09",
"source:geometry:ref": "Gbg/2391617",
},
},
{
type: "way",
id: 160909312,
timestamp: "2021-11-02T23:38:13Z",
version: 4,
changeset: 113306325,
user: "Pieter Vander Vennet",
uid: 3818858,
nodes: [
9219979643, 1728823481, 1728823549, 4978289383, 4978289388, 9219979646,
9219979647, 4978288381, 4978289386, 4978289384, 1728823514, 9219979643,
],
tags: {
"addr:city": "Brugge",
"addr:country": "BE",
"addr:housenumber": "108",
"addr:postcode": "8000",
"addr:street": "Ezelstraat",
building: "house",
"source:geometry:date": "2018-10-02",
"source:geometry:ref": "Gbg/5926383",
},
},
{
type: "way",
id: 160909315,
timestamp: "2021-12-23T12:03:37Z",
version: 8,
changeset: 115295690,
user: "s8evq",
uid: 3710738,
nodes: [
1728823543, 1728823501, 1728823522, 4978288382, 1612385157, 1728823514,
9219979643, 1728823543,
],
tags: {
"addr:city": "Brugge",
"addr:country": "BE",
"addr:housenumber": "110",
"addr:postcode": "8000",
"addr:street": "Ezelstraat",
building: "house",
name: "La Style",
shop: "hairdresser",
"source:geometry:date": "2015-07-09",
"source:geometry:ref": "Gbg/5260837",
},
},
{
type: "way",
id: 508533816,
timestamp: "2021-12-23T12:03:37Z",
version: 7,
changeset: 115295690,
user: "s8evq",
uid: 3710738,
nodes: [
4978288387, 4978288385, 1728823544, 1728823522, 4978288382, 4978289391,
9219979646, 4978289388, 9219974337, 4978288387,
],
tags: {
building: "yes",
"source:geometry:date": "2015-07-09",
"source:geometry:ref": "Gbg/5260790",
},
},
],
}
)
Utils.injectJsonDownloadForTests("https://www.openstreetmap.org/api/0.6/way/160909312/full", {
version: "0.6",
generator: "CGImap 0.8.6 (2407324 spike-06.openstreetmap.org)",
copyright: "OpenStreetMap and contributors",
attribution: "http://www.openstreetmap.org/copyright",
license: "http://opendatacommons.org/licenses/odbl/1-0/",
elements: [
{
type: "node",
id: 1728823481,
lat: 51.2146968,
lon: 3.2167242,
timestamp: "2021-11-02T23:37:11Z",
version: 5,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 1728823514,
lat: 51.2147863,
lon: 3.2168551,
timestamp: "2021-11-02T23:37:11Z",
version: 5,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 1728823549,
lat: 51.2147401,
lon: 3.2168877,
timestamp: "2021-11-02T23:37:11Z",
version: 5,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 4978288381,
lat: 51.2147638,
lon: 3.2168856,
timestamp: "2021-11-02T23:37:11Z",
version: 4,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 4978289383,
lat: 51.2147678,
lon: 3.2169969,
timestamp: "2021-11-02T23:37:11Z",
version: 4,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 4978289384,
lat: 51.2147684,
lon: 3.2168674,
timestamp: "2021-11-02T23:37:11Z",
version: 4,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 4978289386,
lat: 51.2147716,
lon: 3.2168811,
timestamp: "2021-11-02T23:37:11Z",
version: 4,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 4978289388,
lat: 51.2148115,
lon: 3.216966,
timestamp: "2021-11-02T23:38:13Z",
version: 7,
changeset: 113306325,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 9219979643,
lat: 51.2147405,
lon: 3.216693,
timestamp: "2021-11-02T23:37:11Z",
version: 1,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 9219979646,
lat: 51.2148043,
lon: 3.2169312,
timestamp: "2021-11-02T23:38:13Z",
version: 2,
changeset: 113306325,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "node",
id: 9219979647,
lat: 51.2147792,
lon: 3.2169466,
timestamp: "2021-11-02T23:37:11Z",
version: 1,
changeset: 113305401,
user: "Pieter Vander Vennet",
uid: 3818858,
},
{
type: "way",
id: 160909312,
timestamp: "2021-11-02T23:38:13Z",
version: 4,
changeset: 113306325,
user: "Pieter Vander Vennet",
uid: 3818858,
nodes: [
9219979643, 1728823481, 1728823549, 4978289383, 4978289388, 9219979646,
9219979647, 4978288381, 4978289386, 4978289384, 1728823514, 9219979643,
],
tags: {
"addr:city": "Brugge",
"addr:country": "BE",
"addr:housenumber": "108",
"addr:postcode": "8000",
"addr:street": "Ezelstraat",
building: "house",
"source:geometry:date": "2018-10-02",
"source:geometry:ref": "Gbg/5926383",
},
},
],
})
Utils.injectJsonDownloadForTests(
"https://raw.githubusercontent.com/pietervdvn/MapComplete-data/main/latlon2country/0.0.0.json",
"be"
)
it("should move nodes accordingly", async () => {
const layout = new LayoutConfig(grbStripped)
const bbox = new BBox([
[3.2166673243045807, 51.21467321525788],
[3.217007964849472, 51.21482442824023],
])
const url = `https://www.openstreetmap.org/api/0.6/map.json?bbox=${bbox.minLon},${bbox.minLat},${bbox.maxLon},${bbox.maxLat}`
const data = await Utils.downloadJson(url)
const fullNodeDatabase = new FullNodeDatabaseSource()
fullNodeDatabase.handleOsmJson(data, 0, 0, 0)
const changes = new Changes({
dryRun: new ImmutableStore(true),
osmConnection: new OsmConnection(),
})
const osmConnection = new OsmConnection({
dryRun: new ImmutableStore(true),
})
const action = new ReplaceGeometryAction(
{ osmConnection, fullNodeDatabase },
targetFeature,
wayId,
{
theme: "test",
}
)
const closestIds = await action.GetClosestIds()
expect(closestIds.closestIds).toEqual([
9219979643,
1728823481,
4978289383,
4978289388,
9219979646,
9219979647,
4978288381,
4978289386,
4978289384,
1728823514,
undefined,
])
expect(closestIds.reprojectedNodes.size).toEqual(1)
const reproj = closestIds.reprojectedNodes.get(1728823549)
expect(reproj.projectAfterIndex).toEqual(1)
expect(reproj.newLon).toEqual(3.2168880864669203)
expect(reproj.newLat).toEqual(51.214739524104694)
expect(closestIds.detachedNodes.size).toEqual(0)
const changed = await action.Perform(changes)
expect(changed[11].changes["coordinates"]).toEqual([
[3.216690793633461, 51.21474084112525],
[3.2167256623506546, 51.214696737309964],
[3.2168880864669203, 51.214739524104694],
[3.2169999182224274, 51.214768983537674],
[3.2169650495052338, 51.21480720678671],
[3.2169368863105774, 51.21480090625335],
[3.2169489562511444, 51.21478074454077],
[3.216886594891548, 51.214765203214625],
[3.2168812304735184, 51.21477192378873],
[3.2168644666671753, 51.214768983537674],
[3.2168537378311157, 51.21478746511261],
[3.216690793633461, 51.21474084112525],
])
})
})