import {LayerDefinition} from "../LayerDefinition"; import {And, Or, RegexTag, Tag} from "../../Logic/Tags"; import {TagRenderingOptions} from "../TagRenderingOptions"; import {FromJSON} from "../JSON/FromJSON"; export class Widths extends LayerDefinition { private readonly cyclistWidth: number; private readonly carWidth: number; private readonly pedestrianWidth: number; private readonly _bothSideParking = new Tag("parking:lane:both", "parallel"); private readonly _noSideParking = new Tag("parking:lane:both", "no_parking"); private readonly _otherParkingMode = new Or([ new Tag("parking:lane:both", "perpendicular"), new Tag("parking:lane:left", "perpendicular"), new Tag("parking:lane:right", "perpendicular"), new Tag("parking:lane:both", "diagonal"), new Tag("parking:lane:left", "diagonal"), new Tag("parking:lane:right", "diagonal"), ]) private readonly _leftSideParking = new And([new Tag("parking:lane:left", "parallel"), new Tag("parking:lane:right", "no_parking")]); private readonly _rightSideParking = new And([new Tag("parking:lane:right", "parallel"), new Tag("parking:lane:left", "no_parking")]); private _sidewalkBoth = new Tag("sidewalk", "both"); private _sidewalkLeft = new Tag("sidewalk", "left"); private _sidewalkRight = new Tag("sidewalk", "right"); private _sidewalkNone = new Tag("sidewalk", "none"); private readonly _oneSideParking = new Or([this._leftSideParking, this._rightSideParking]); private readonly _notCarfree = FromJSON.Tag({"and":[ "highway!~pedestrian|living_street", "access!~destination", "motor_vehicle!~destination|no" ]}); private calcProps(properties) { let parkingStateKnown = true; let parallelParkingCount = 0; if (this._oneSideParking.matchesProperties(properties)) { parallelParkingCount = 1; } else if (this._bothSideParking.matchesProperties(properties)) { parallelParkingCount = 2; } else if (this._noSideParking.matchesProperties(properties)) { parallelParkingCount = 0; } else if (this._otherParkingMode.matchesProperties(properties)) { parallelParkingCount = 0; } else { parkingStateKnown = false; console.log("No parking data for ", properties.name, properties.id, properties) } let pedestrianFlowNeeded; if (this._sidewalkBoth.matchesProperties(properties)) { pedestrianFlowNeeded = 0; } else if (this._sidewalkNone.matchesProperties(properties)) { pedestrianFlowNeeded = 2; } else if (this._sidewalkLeft.matchesProperties(properties) || this._sidewalkRight.matches(properties)) { pedestrianFlowNeeded = 1; } else { pedestrianFlowNeeded = -1; } let onewayCar = properties.oneway === "yes"; let onewayBike = properties["oneway:bicycle"] === "yes" || (onewayCar && properties["oneway:bicycle"] === undefined) let cyclingAllowed = !(properties.bicycle === "use_sidepath" || properties.bicycle === "no"); let carWidth = (onewayCar ? 1 : 2) * this.carWidth; let cyclistWidth = 0; if (cyclingAllowed) { cyclistWidth = (onewayBike ? 1 : 2) * this.cyclistWidth; } const width = parseFloat(properties["width:carriageway"]); const targetWidthIgnoringPedestrians = carWidth + cyclistWidth + parallelParkingCount * this.carWidth; const targetWidth = targetWidthIgnoringPedestrians + Math.max(0, pedestrianFlowNeeded) * this.pedestrianWidth; return { parkingLanes: parallelParkingCount, parkingStateKnown: parkingStateKnown, width: width, targetWidth: targetWidth, targetWidthIgnoringPedestrians: targetWidthIgnoringPedestrians, onewayBike: onewayBike, pedestrianFlowNeeded: pedestrianFlowNeeded, cyclingAllowed: cyclingAllowed } } constructor(carWidth: number, cyclistWidth: number, pedestrianWidth: number) { super("width"); this.carWidth = carWidth; this.cyclistWidth = cyclistWidth; this.pedestrianWidth = pedestrianWidth; this.minzoom = 12; function r(n: number) { const pre = Math.floor(n); const post = Math.floor((n * 10) % 10); return "" + pre + "." + post; } this.name = "widths"; this.overpassFilter = new RegexTag("width:carriageway", /.*/); this.title = new TagRenderingOptions({ freeform: { renderTemplate: "{name}", template: "$$$", key: "name" } }) console.log("Not car free: ", this._notCarfree) const self = this; this.style = (properties) => { let c = "#f00"; const props = self.calcProps(properties); if (props.width >= props.targetWidthIgnoringPedestrians) { c = "#fa0" } if (props.width >= props.targetWidth || !props.cyclingAllowed) { c = "#0c0"; } if (!props.parkingStateKnown && properties["note:width:carriageway"] === undefined) { c = "#f0f" } if (!this._notCarfree.matchesProperties(properties)) { c = "#aaa"; } // Mark probably wrong data if (props.width > 15) { c = "#f0f" } let dashArray = undefined; if (props.onewayBike) { dashArray = [20, 8] } return { icon: null, color: c, weight: 9, dashArray: dashArray } } this.elementsToShow = [ new TagRenderingOptions({ mappings: [ { k: this._bothSideParking, txt: "Auto's kunnen langs beide zijden parkeren.Dit gebruikt " + r(this.carWidth * 2) + "m
" }, { k: this._oneSideParking, txt: "Auto's kunnen langs één kant parkeren.
Dit gebruikt " + r(this.carWidth) + "m
" }, { k: this._otherParkingMode, txt: "Deze straat heeft dwarsparkeren of diagonaalparkeren aan minstens één zijde. Deze parkeerruimte is niet opgenomen in de straatbreedte." }, {k: this._noSideParking, txt: "Auto's mogen hier niet parkeren"}, ], freeform: { key: "note:width:carriageway", renderTemplate: "{note:width:carriageway}", template: "$$$", } }).OnlyShowIf(this._notCarfree), new TagRenderingOptions({ mappings: [ { k: this._sidewalkNone, txt: "Deze straat heeft geen voetpaden. Voetgangers hebben hier " + r(this.pedestrianWidth * 2) + "m nodig" }, { k: new Or([this._sidewalkLeft, this._sidewalkRight]), txt: "Deze straat heeft een voetpad aan één kant. Voetgangers hebben hier " + r(this.pedestrianWidth) + "m nodig" }, {k: this._sidewalkBoth, txt: "Deze straat heeft voetpad aan beide zijden."}, ], freeform: { key: "note:width:carriageway", renderTemplate: "{note:width:carriageway}", template: "$$$", } }).OnlyShowIf(this._notCarfree), new TagRenderingOptions({ mappings: [ { k: new Tag("bicycle", "use_sidepath"), txt: "Er is een afgescheiden, verplicht te gebruiken fietspad. Fietsen op dit wegsegment hoeft dus niet" }, { k: new Tag("bicycle", "no"), txt: "Fietsen is hier niet toegestaan" }, { k: new Tag("oneway:bicycle", "yes"), txt: "Eenrichtingsverkeer, óók voor fietsers. Dit gebruikt " + r(this.carWidth + this.cyclistWidth) + "m" }, { k: new And([new Tag("oneway", "yes"), new Tag("oneway:bicycle", "no")]), txt: "Tweerichtingverkeer voor fietsers, eenrichting voor auto's Dit gebruikt " + r(this.carWidth + 2 * this.cyclistWidth) + "m" }, { k: new Tag("oneway", "yes"), txt: "Eenrichtingsverkeer voor iedereen. Dit gebruikt " + (this.carWidth + this.cyclistWidth) + "m" }, { k: null, txt: "Tweerichtingsverkeer voor iedereen. Dit gebruikt " + r(2 * this.carWidth + 2 * this.cyclistWidth) + "m" } ] }).OnlyShowIf(this._notCarfree), new TagRenderingOptions( { tagsPreprocessor: (tags) => { const props = self.calcProps(tags); tags.targetWidth = r(props.targetWidth); tags.short = ""; if (props.width < props.targetWidth) { tags.short = r(props.targetWidth - props.width) } }, mappings:[ {k: new Tag("short","*"), txt: "De totale nodige ruimte voor vlot en veilig verkeer is dus {targetWidth}m
" + "Er is dus {short}m te weinig", substitute: true}, {k: new Tag("short",""), txt: "De totale nodige ruimte voor vlot en veilig verkeer is dus {targetWidth}m"} ] } ).OnlyShowIf(this._notCarfree), new TagRenderingOptions({ mappings: [ {k:new Tag("highway","living_street"),txt: "Dit is een woonerf"}, {k:new Tag("highway","pedestrian"),txt: "Deze weg is autovrij"} ] }), new TagRenderingOptions({ mappings: [ { k: new Tag("sidewalk", "none"), txt: "De afstand van huis tot huis is {width:carriageway}m" }, { k: new Tag("sidewalk", "left"), txt: "De afstand van huis tot voetpad is {width:carriageway}m" }, { k: new Tag("sidewalk", "right"), txt: "De afstand van huis tot voetpad is {width:carriageway}m" }, { k: new Tag("sidewalk", "both"), txt: "De afstand van voetpad tot voetpad is {width:carriageway}m" }, { k: new Tag("sidewalk", ""), txt: "De straatbreedte is {width:carriageway}m" } ] }) ] } }