WME Street to River PLUS (mod)

This script create a new river landmark in waze map editor (WME). It transforms the the geometry of a new unsaved street to a polygon.

// ==UserScript==
// @name            WME Street to River PLUS (mod)
// @description     This script create a new river landmark in waze map editor (WME). It transforms the the geometry of a new unsaved street to a polygon.
// @namespace       https://greasyfork.org/users/160654-waze-ukraine
// @grant           none
// @version         2024.01.25.001
// @match           https://beta.waze.com/*editor*
// @match           https://www.waze.com/*editor*
// @exclude         https://www.waze.com/*user/*editor/*
// @icon 
// @require         https://greasyfork.org/scripts/450160-wme-bootstrap/code/WME-Bootstrap.js
// ==/UserScript==

// Based on WME Street to river

// Mini howto:
// 1) install this script as greasemonkey script or chrome extension
// 2) draw a new street but do not save the street
// 3) add and apply a street name to define the rivers name and the the width of the river
//    Example: "20m Spree" creates a 20 meters width river named "Spree"
// 4) Select the helper street
// 5) Click the "Street to river" button
// 4) Delete the helper street
// 5) Edit the new landmark as you like
//
// Updated by: Eduardo Carvajal

/* jshint esversion: 11 */
/* global W */
/* global I18n */
/* global OpenLayers */
/* global $ */
/* global require */

console.warn('Remove this line, when WME-Bootstrap will fix its syntax. now it causes script error on load, details https://stackoverflow.com/questions/42036349/uncaught-typeerror-intermediate-value-is-not-a-function');

(function () {
    const version = GM_info.script.version;

    const idMeters = 0;
    const idWidth = 1;
    const idTitle = 2;
    const idStreetToRiver = 3;
    const idUnlimitedSize = 4;
    const idNoUsavedStreet = 5;
    const idAllSegmentsInside = 6;
    const idMultipleSegmentsInside = 7;
    const idStreetToOther = 8;
    const idStreetToForest = 9;
    const idDeleteSegment = 10;

    function streetToRiver_bootstrap() {
        $(document)
        .on('bootstrap.wme', function () {
            streetToRiver_init()
        })
    }

    function streetToRiver_init() {

        const defaultWidth = 15;
        var scriptLanguage = "us";
        var langText;{
            var Config = [{
                    handler: 'WME-Street-to-River_other',
                    title: "Other",
                    func: function (ev) {
                        doPOI(ev, "OTHER");
                    },
                    key: -1,
                    arg: {
                        type: "OTHER"
                    }
                },
            ];

            for (var i = 0; i < Config.length; ++i) {
                WMEKSRegisterKeyboardShortcut('WME-Street-to-River', 'WME-Street-to-River', Config[i].handler, Config[i].title, Config[i].func, Config[i].key, Config[i].arg);
            }

            WMEKSLoadKeyboardShortcuts('WME-Street-to-River');

            window.addEventListener("beforeunload", function () {
                WMEKSSaveKeyboardShortcuts('WME-Street-to-River');
            }, false);
        }

        function insertButtons() {

            if (W.selectionManager.getSelectedFeatures().length === 0)
                return;

            var btn0 = $('<wz-button size="sm" color="submit" title="' + getString(idTitle) + '">' + getString(idStreetToOther) + '</wz-button>');
            btn0.click(doOther);

            var btn1 = $('<wz-button size="sm" color="submit" title="' + getString(idTitle) + '">' + getString(idStreetToRiver) + '</wz-button>');
            btn1.click(doRiver);

            var btn2 = $('<wz-button size="sm" color="submit" title="' + getString(idTitle) + '">' + getString(idStreetToForest) + '</wz-button>');
            btn2.click(doForest);

            var strMeters = getString(idMeters);

            var selRiverWidth = $('<wz-select name="riverWidth" />');
            selRiverWidth.append($('<wz-option value="5"> 5 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="8"> 8 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="10">10 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="11">11 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="12">12 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="13">13 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="15">15 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="17">17 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="20">20 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="25">25 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="30">30 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="40">40 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="50">50 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="80">80 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="100">100 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="120">120 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="150">150 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="180">180 ' + strMeters + '</wz-option>'));
            selRiverWidth.append($('<wz-option value="200">200 ' + strMeters + '</wz-option>'));
            selRiverWidth.change(function () {
                setLastRiverWidth(this.value);
            });

            var lastRiverWidth = getLastRiverWidth(defaultWidth);
            console_log("Last river width: " + lastRiverWidth);
            selRiverWidth.val(lastRiverWidth);

            var chk = $('<wz-checkbox title="' + getString(idUnlimitedSize) + '" name="_isUnlimitedSize" >' + getString(idUnlimitedSize) + '</wz-checkbox>');
            chk.prop("checked", getLastIsUnlimitedSize(false));
            chk.change(function () {
                setLastIsUnlimitedSize(this.checked);
            });
            var chkDel = $('<wz-checkbox name="_isDeleteSegment" >' + getString(idDeleteSegment) + '</wz-checkbox>');
            chkDel.prop("checked", getLastIsDeleteSegment(true));
            chkDel.change(function () {
                setLastIsDeleteSegment(this.checked);
            });

            var cnt = $('<div class="form-group" />');
            var label = $('<wz-label><a href="https://github.com/waze-ua/wme-street-to-river-plus-mod" target="_blank">Street to River+ (Mod) v' + version + '</a></wz-label>');
            cnt.append(label);

            var divGroup1 = $('<div class="controls-container" />');
            divGroup1.append($('<wz-label html-for>' + getString(idWidth) + '</wz-label>'));
            divGroup1.append(selRiverWidth);

            var divGroup2 = $('<div class="controls-container" />');
            divGroup2.append(chk);
            divGroup2.append(chkDel);

            var divGroup3 = $('<div class="controls-container" />');
            divGroup3.append(btn0);
            divGroup3.append(btn1);
            divGroup3.append(btn2);

            cnt.append(divGroup1);
            cnt.append(divGroup2);
            cnt.append(divGroup3);

            $("#edit-panel .more-actions").after(cnt);

            console_log("Street to River Language: " + scriptLanguage);
            console_log("Street to river PLUS initialized");
        }

        function doPOI(ev, typ) {
            var convertOK;
            var foundSelectedSegment = false;

            var isUnlimitedSize = getLastIsUnlimitedSize(false);
            var isDeleteSegment = getLastIsDeleteSegment(true);

            // 2014-01-09: Search for helper street. If found create or expand a river
            for (var s = W.selectionManager.getSelectedFeatures().length - 1; s >= 0; s--) {
                var sel = W.selectionManager.getSelectedFeatures()[s]._wmeObject;
                if (sel.type == "segment") {
                    // found segment
                    foundSelectedSegment = true;
                    convertOK = convertToLandmark(sel, typ, isUnlimitedSize);
                    if (convertOK && isDeleteSegment) {
                        var wazeActionDeleteSegment = require("Waze/Action/DeleteSegment");
                        W.model.actionManager.add(new wazeActionDeleteSegment(sel));
                    }
                }
            }
            if (!foundSelectedSegment) {
                alert(getString(idNoUsavedStreet));
            }

        }

        function doRiver(ev) {
            doPOI(ev, "RIVER_STREAM");
        }

        function doForest(ev) {
            doPOI(ev, "FOREST_GROVE");
        }
        function doOther(ev) {
            doPOI(ev, "OTHER");
        }

        function CalcRL(components) {
            var count = components.length;
            var j = count - 1;
            var area = 0;

            for (var i = 0; i < count; ++i) {
                area += (components[i].y * components[j].x) - (components[i].x * components[j].y);
                j = i;
            }
            return area < 0 ? 1 : -1; // 1 - по часовой, -1 - против часовой
        }

        function uniq(a) {
            var seen = {};
            return a.filter(function (item) {
                return seen.hasOwnProperty(item) ? false : (seen[item] = true);
            });
        }

        // 2014-01-09: Base on selected helper street creates or expand an existing river/railway
        function convertToLandmark(sel, lmtype, isUnlimitedSize) {
            var i;
            var leftPa,
            rightPa,
            leftPb,
            rightPb;
            var prevLeftEq,
            prevRightEq;
            var street = getStreet(sel);

            var displacement = getDisplacement(street);

            // create place with a minimum area 100m2
            // for simple segments only (A-B)
            if (sel.geometry.components.length === 2) {
                var segLength = 0;
                var minArea = 100;
                var pt = [];
                pt[0] = sel.geometry.components[0];
                pt[1] = sel.geometry.components[1];

                var seg = new OpenLayers.Geometry.LineString(pt);
                segLength = seg.getGeodesicLength(W.map.getProjectionObject());

                // if small area is expected
                if (minArea / displacement > segLength) {
                    if (segLength <= Math.sqrt(minArea)) {
                        // create a minimum square
                        var line = Math.sqrt(minArea);
                        var segScale = line / segLength;
                        displacement = line / 1.18;
                        pt[1].resize(segScale, pt[0], 1);
                    } else {
                        // adjust displacement (width)
                        displacement = minArea / segLength;
                    }
                }
            }

            var streetVertices = sel.geometry.getVertices();
            var polyPoints = null;
            var firstPolyPoint = null;
            var secondPolyPoint = null;

            var wazeActionUpdateFeatureGeometry = require("Waze/Action/UpdateFeatureGeometry");
            var wazefeatureVectorLandmark = require("Waze/Feature/Vector/Landmark");
            var wazeActionAddLandmark = require("Waze/Action/AddLandmark");
            var wazeActionDeleteObject = require("Waze/Action/DeleteObject");
            var wazeActionUpdateFeatureAddress = require("Waze/Action/UpdateFeatureAddress");

            console_log("Street vertices: " + streetVertices.length);

            // 2013-10-13: Is new street inside an existing river?
            var bAddNew = !0;
            var riverLandmark = null;
            var repo = W.model.venues;

            var rrr,
            donorLandmark = null;
            for (var t in repo.objects) {
                riverLandmark = repo.objects[t];
                if (riverLandmark.attributes.categories[0] === lmtype) {
                    console_log("riverLandmark.attributes.id=" + riverLandmark.attributes.id);
                    console_log("streetVertices.length=" + streetVertices.length);
                    console_log("streetVertices[0]=" + streetVertices[0]);
                    console_log("streetVertices[streetVertices.length - 1]=" + streetVertices[streetVertices.length - 1]);

                    // 2014-06-27: Verify if the landmark object has containsPoint function
                    if ("function" === typeof riverLandmark.geometry.containsPoint) {
                        if (riverLandmark.geometry.containsPoint(streetVertices[0])) {
                            bAddNew = false; // Street is inside an existing river
                            console_log("rrr=" + riverLandmark.attributes.id);
                            rrr = riverLandmark;
                            //             break;
                        }
                        if (riverLandmark.geometry.containsPoint(streetVertices[streetVertices.length - 1])) {
                            //                        bAddNew = false;    // Street is inside an existing river
                            console_log("donorLandmark=" + riverLandmark.attributes.id);
                            donorLandmark = riverLandmark;
                            //             break;
                        }

                    }
                }
            }
            riverLandmark = rrr;

            // 2013-10-13: Ignore vertices inside river
            var bIsOneVerticeStreet = false;
            var firstStreetVerticeOutside = 0;
            if (!bAddNew) {
                console_log("Expanding an existing river");
                while (firstStreetVerticeOutside < streetVertices.length) {
                    if (!riverLandmark.geometry.containsPoint(streetVertices[firstStreetVerticeOutside]))
                        break;
                    firstStreetVerticeOutside += 1;
                }
                if (firstStreetVerticeOutside === streetVertices.length) {
                    alert(getString(idAllSegmentsInside));
                    return false;
                }
                bIsOneVerticeStreet = firstStreetVerticeOutside === (streetVertices.length - 1);
                if (bIsOneVerticeStreet) {
                    console_log("It's one vertice street");
                }
                if (firstStreetVerticeOutside > 1) {
                    alert(getString(idMultipleSegmentsInside));
                    return false;
                }
                console_log("First street vertice outside river:" + firstStreetVerticeOutside);
            }

            // 2013-10-13: Add to polyPoints river polygon
            console_log("River polygon: Create");
            var first;
            if (bAddNew)
                first = 0;
            else
                first = firstStreetVerticeOutside - 1;

            for (i = first; i < streetVertices.length - 1; i++) {
                var pa = streetVertices[i];
                var pb = streetVertices[i + 1];

                // fix for incorrect scale calculation, as distanceTo() returns units, but displacement is in meters
                // old:
                //var scale = (pa.distanceTo(pb) + displacement) / pa.distanceTo(pb);
                // new:
                //TODO optimize this, convert displacement into map units for easier scale calculation
                var points = [pa, pb];
                var ls = new OpenLayers.Geometry.LineString(points);
                var len = ls.getGeodesicLength(W.map.getProjectionObject());
                var scale = (len + displacement / 2) / len;

                leftPa = pa.clone();
                leftPa.resize(scale, pb, 1);
                rightPa = leftPa.clone();
                leftPa.rotate(90, pa);
                rightPa.rotate(-90, pa);

                leftPb = pb.clone();
                leftPb.resize(scale, pa, 1);
                rightPb = leftPb.clone();
                leftPb.rotate(-90, pb);
                rightPb.rotate(90, pb);

                var leftEq = getEquation({
                    'x1': leftPa.x,
                    'y1': leftPa.y,
                    'x2': leftPb.x,
                    'y2': leftPb.y
                });
                var rightEq = getEquation({
                    'x1': rightPa.x,
                    'y1': rightPa.y,
                    'x2': rightPb.x,
                    'y2': rightPb.y
                });
                if (polyPoints === null) {
                    polyPoints = [leftPa, rightPa];
                } else {
                    var li = intersectX(leftEq, prevLeftEq);
                    var ri = intersectX(rightEq, prevRightEq);
                    if (li && ri) {
                        // 2013-10-17: Is point outside river?
                        if (i >= firstStreetVerticeOutside) {
                            polyPoints.unshift(li);
                            polyPoints.push(ri);

                            // 2013-10-17: Is first point outside river? -> Save it for later use
                            if (i == firstStreetVerticeOutside) {
                                firstPolyPoint = li.clone();
                                secondPolyPoint = ri.clone();
                                polyPoints = [li, ri];
                            }
                        }
                    } else {
                        // 2013-10-17: Is point outside river?
                        if (i >= firstStreetVerticeOutside) {
                            polyPoints.unshift(leftPb.clone());
                            polyPoints.push(rightPb.clone());

                            // 2013-10-17: Is first point outside river? -> Save it for later use
                            if (i == firstStreetVerticeOutside) {
                                firstPolyPoint = leftPb.clone();
                                secondPolyPoint = rightPb.clone();
                                polyPoints = [leftPb, rightPb];
                            }
                        }
                    }
                }

                prevLeftEq = leftEq;
                prevRightEq = rightEq;

                // 2013-06-03: Is Waze limit reached?
                if ((polyPoints.length > 50) && !isUnlimitedSize) {
                    break;
                }
            }

            if (bIsOneVerticeStreet) {
                firstPolyPoint = leftPb.clone();
                secondPolyPoint = rightPb.clone();
                polyPoints = [leftPb, rightPb];
                console_log("One vertice river:" + polyPoints.length);
            } else {
                polyPoints.push(rightPb);
                polyPoints.push(leftPb);
            }
            console_log("River polygon: done");

            // 2014-01-09: Create or expand an existing river?
            if (bAddNew) {
                // 2014-01-09: Add new river
                // 2014-01-09: Create new river's Polygon
                var polygon = new OpenLayers.Geometry.Polygon(new OpenLayers.Geometry.LinearRing(polyPoints));

                // 2014-10-08: Creates river's Landmark

                //FIX (?????? Start)
                //                riverLandmark = new wazefeatureVectorLandmark();
                var ldk = {};
                ldk.geoJSONGeometry = W.userscripts.toGeoJSONGeometry (polygon);
                riverLandmark = new wazefeatureVectorLandmark(ldk);
                //FIX (?????? End)

                riverLandmark.geometry = polygon;
                riverLandmark.attributes.categories.push(lmtype);

                // 2014-01-09: Add river's name base on Street Name
                if (street && street.attributes.name) {
                    riverLandmark.attributes.name = street.attributes.name.replace(/^\d+(m|м|ft)\s*/, ''); // TODO make localizable
                }

                // 2014-10-08: Add new Landmark to Waze Editor
                var riverLandmark_o = new wazeActionAddLandmark(riverLandmark);
                W.model.actionManager.add(riverLandmark_o);
                try {
                    W.selectionManager.setSelectedModels([riverLandmark]);
                } catch (err) {
                    // Ignore error:
                    // Uncaught TypeError: Cannot read properties of undefined (reading 'children')
                    // at Object.WMETB_FPnewSelectionAvailable (FancyPermalink.min.js:216:184)
                    // at v (third_party-45fe9aba9d649e1fe91a.js.gz:2:1169484)
                    console_log(err)
                }

                if (lmtype !== "OTHER") {
                    console_log("bAddNew");
                    let address = riverLandmark.getAddress().attributes;
                    console_log(address);
                    let newAddressAtts = {
                        streetName: null,
                        emptyStreet: true,
                        cityName: null,
                        emptyCity: true,
                        stateID: address.state.attributes.id,
                        countryID: address.country.attributes.id
                    };
                    W.model.actionManager.add(new wazeActionUpdateFeatureAddress(riverLandmark, newAddressAtts, {
                            streetIDField: 'streetID'
                        }));
                }

            } else {
                // 2014-01-09: Expand an existing river
                var originalGeometry = riverLandmark.geometry.clone();

                if (donorLandmark) // если есть донор
                {
                    let undoGeometry = riverLandmark.geometry.clone();
                    var undoGeometryDonor = donorLandmark.geometry.clone();
                    var components = riverLandmark.geometry.components[0].components;
                    var componentsDonor = donorLandmark.geometry.components[0].components;

                    //window.donorLandmark=donorLandmark;
                    //window.riverLandmark=riverLandmark;

                    // куда закручен массив?
                    var componentsRL = CalcRL(components);
                    var componentsDonorRL = CalcRL(componentsDonor);
                    console_log("src=" + componentsRL + ", donor=" + componentsDonorRL);
                    // найти индекс ближайшей точки к началу сегмента
                    var dist = 1000000000;
                    var p1 = [0, 0],
                    p2 = [0, 0]; // индексы ближайших точек
                    for (let i1 = 0; i1 < components.length; i1++) {
                        var d1 = Math.sqrt(Math.pow(Math.abs(components[i1].x - streetVertices[0].x), 2) + Math.pow(Math.abs(components[i1].y - streetVertices[0].y), 2));
                        if (d1 < dist) {
                            dist = d1;
                            p1[0] = i1;
                            if (componentsRL > 0)
                                p1[1] = i1 === 0 ? components.length - 1 : i1 - 1;
                            else
                                p1[1] = i1 == components.length - 1 ? 0 : i1 + 1;
                        }
                    }

                    console_log("p1=" + p1 + ", dist=" + dist);
                    // ищем индекс во втором ПОИ, откуда начинать вставку.
                    dist = 1000000000;
                    for (let i1 = 0; i1 < componentsDonor.length; i1++) {
                        let d1 = Math.sqrt(Math.pow(Math.abs(componentsDonor[i1].x - streetVertices[streetVertices.length - 1].x), 2) + Math.pow(Math.abs(componentsDonor[i1].y - streetVertices[streetVertices.length - 1].y), 2));
                        if (d1 < dist) {
                            dist = d1;
                            p2[0] = i1;
                            if (componentsDonorRL > 0)
                                p2[1] = i1 === 0 ? componentsDonor.length - 1 : i1 - 1;
                            else
                                p2[1] = i1 == componentsDonor.length - 1 ? 0 : i1 + 1;
                        }
                    }
                    console_log("p2=" + p2 + ", dist=" + dist);

                    var componentsNew = components.slice();
                    componentsNew.length = 0;

                    // добавляем источник
                    for (let i1 = 0; i1 <= p1[0]; ++i1)
                        componentsNew.push(components[i1]);

                    // добавляем донора
                    if (componentsRL < 0) {
                        if (componentsDonorRL < 0) {
                            // добавляем донора по кругу
                            for (let i1 = p2[0]; i1 < componentsDonor.length; ++i1)
                                componentsNew.push(componentsDonor[i1]);

                            // ...остаток донора
                            for (let i1 = 0; i1 < p2[0]; ++i1)
                                componentsNew.push(componentsDonor[i1]);
                        } else {
                            // добавляем донора по кругу
                            for (let i1 = p2[0]; i1 >= 0; --i1)
                                componentsNew.push(componentsDonor[i1]);

                            // ...остаток донора
                            for (let i1 = componentsDonor.length - 1; i1 > p2[0]; --i1)
                                componentsNew.push(componentsDonor[i1]);
                        }
                    } else {
                        if (componentsDonorRL < 0) {
                            // добавляем донора по кругу
                            for (let i1 = p2[0]; i1 >= 0; --i1)
                                componentsNew.push(componentsDonor[i1]);

                            // ...остаток донора
                            for (let i1 = componentsDonor.length - 1; i1 > p2[0]; --i1)
                                componentsNew.push(componentsDonor[i1]);
                        } else {
                            // добавляем донора по кругу
                            for (let i1 = p2[0]; i1 < componentsDonor.length; ++i1)
                                componentsNew.push(componentsDonor[i1]);

                            // ...остаток донора
                            for (let i1 = 0; i1 < p2[0]; ++i1)
                                componentsNew.push(componentsDonor[i1]);
                        }
                    }

                    // добавляем источник
                    for (let i1 = p1[0] + 1; i1 < components.length; ++i1)
                        componentsNew.push(components[i1]);

                    //window.componentsNew=componentsNew
                    // обновляемся
                    riverLandmark.geometry.components[0].components = uniq(componentsNew);

                    W.model.actionManager.add(new wazeActionUpdateFeatureGeometry(riverLandmark, W.model.venues, W.userscripts.toGeoJSONGeometry(undoGeometry), W.userscripts.toGeoJSONGeometry(riverLandmark.geometry)));
                    W.model.actionManager.add(new wazeActionDeleteObject(donorLandmark, W.model.venues, W.userscripts.toGeoJSONGeometry(undoGeometryDonor), W.userscripts.toGeoJSONGeometry(donorLandmark.geometry)));

                    return true;
                }
                var riverVertices = riverLandmark.geometry.getVertices();
                console_log("Total river vertices:" + riverVertices.length);

                // 2013-06-01: Adjust first street vertice in case of a 2 vertice river
                if (firstStreetVerticeOutside === 0)
                    firstStreetVerticeOutside = 1;

                // 2013-06-01: Find on selected river, the nearest point from the beginning of road

                var distance = 0;
                var minDistance = 100000;
                var indexNearestPolyPoint = 0;
                for (i = 0; i < polyPoints.length; i++) {
                    distance = polyPoints[i].distanceTo(streetVertices[firstStreetVerticeOutside]);
                    if (distance < minDistance) {
                        minDistance = distance;
                        indexNearestPolyPoint = i;
                    }
                }
                console_log("polyPoints.length: " + polyPoints.length);
                console_log("indexNearestPolyPoint: " + indexNearestPolyPoint);

                var indexNearestRiverVertice = 0;
                var nextIndex;
                minDistance = 100000;
                for (i = 0; i < riverVertices.length; i++) {
                    nextIndex = getNextIndex(i, riverVertices.length, +1);
                    if (isIntersectingLines(riverVertices[i], riverVertices[nextIndex], streetVertices[0], streetVertices[1])) {
                        distance = polyPoints[indexNearestPolyPoint].distanceTo(riverVertices[i]);
                        if (distance < minDistance) {
                            minDistance = distance;
                            indexNearestRiverVertice = i;
                        }
                    }
                }
                console_log("indexNearestRiverVertice: " + indexNearestRiverVertice);
                var nextRiverVertice = getNextIndex(indexNearestRiverVertice, riverVertices.length, 1);

                // 2013-06-01: Is river's Polygon clockwise or counter-clockwise?

                console_log("nextRiverVertice: " + nextRiverVertice);

                console_log("firstPolyPoint:" + firstPolyPoint);
                console_log("secondPolyPoint:" + secondPolyPoint);

                var inc = 1;
                var incIndex = 0;
                if (isIntersectingLines(riverVertices[indexNearestRiverVertice], firstPolyPoint, riverVertices[nextRiverVertice], secondPolyPoint)) {
                    //inc = -1;
                    console_log("Lines intersect: clockwise polygon");
                    inc = +1;
                    incIndex = 1;
                } else {
                    inc = +1;
                    console_log("Lines doesn't intersect: counter-clockwise polygon");
                }

                // 2013-06-03: Update river's polygon (add new vertices)
                //var indexLastPolyPoint = getNextIndex(index, polyPoints.length, -inc);
                var indexNextVertice = 1;
                var index = polyPoints.length / 2 - 1;

                if (bIsOneVerticeStreet)
                    index += 1;

                for (i = 0; i < polyPoints.length; i++) {
                    if (!originalGeometry.containsPoint(polyPoints[index])) {

                        // 2014-01-09: Save's old Landmark
                        let undoGeometry = riverLandmark.geometry.clone();

                        // 2014-01-09: Add a new point to existing river landmark
                        riverLandmark.geometry.components[0].addComponent(polyPoints[index], indexNearestRiverVertice + indexNextVertice);

                        // 2014-01-09: Update river landmark on Waze editor
                        // 2014-09-30: Gets UptdateFeatureGeometry
                        W.model.actionManager.add(new wazeActionUpdateFeatureGeometry(riverLandmark, W.model.venues, W.userscripts.toGeoJSONGeometry(undoGeometry), W.userscripts.toGeoJSONGeometry(riverLandmark.geometry)));
                        //delete undoGeometry;

                        console_log("Added: " + index);
                        indexNextVertice += incIndex;
                    }
                    index = getNextIndex(index, polyPoints.length, inc);
                }

                // 2013-06-03: Notify Waze that current river's geometry change.
                //Waze.model.actionManager.add(new Waze.Action.UpdateFeatureGeometry(riverLandmark,Waze.model.landmarks,originalGeometry,riverLandmark.geometry));
                //delete originalGeometry;

                if (lmtype !== "OTHER") {
                    console_log("!bAddNew");
                    let address = riverLandmark.getAddress().attributes;
                    console_log(address);
                    let newAddressAtts = {
                        streetName: null,
                        emptyStreet: true,
                        cityName: null,
                        emptyCity: true,
                        stateID: address.state.attributes.id,
                        countryID: address.country.attributes.id
                    };
                    W.model.actionManager.add(new wazeActionUpdateFeatureAddress(riverLandmark, newAddressAtts, {
                            streetIDField: 'streetID'
                        }));
                }

            }
            return true;
        }

        // 2013-06-02: Returns TRUE if line1 intersects lines2
        function isIntersectingLines(pointLine1From, pointLine1To, pointLine2From, pointLine2To) {
            var segment1;
            var segment2;

            // 2013-06-02: OpenLayers.Geometry.segmentsIntersect requires that start and end are ordered so that x1 < x2.
            if (pointLine1From.x <= pointLine1To.x)
                segment1 = {
                    'x1': pointLine1From.x,
                    'y1': pointLine1From.y,
                    'x2': pointLine1To.x,
                    'y2': pointLine1To.y
                };
            else
                segment1 = {
                    'x1': pointLine1To.x,
                    'y1': pointLine1To.y,
                    'x2': pointLine1From.x,
                    'y2': pointLine1From.y
                };

            if (pointLine2From.x <= pointLine2To.x)
                segment2 = {
                    'x1': pointLine2From.x,
                    'y1': pointLine2From.y,
                    'x2': pointLine2To.x,
                    'y2': pointLine2To.y
                };
            else
                segment2 = {
                    'x1': pointLine2To.x,
                    'y1': pointLine2To.y,
                    'x2': pointLine2From.x,
                    'y2': pointLine2From.y
                };

            return OpenLayers.Geometry.segmentsIntersect(segment1, segment2, !1);
        }

        // 2013-06-02: Returns TRUE if polygon's direction is clockwise. FALSE -> counter-clockwise
        // Based on: http://stackoverflow.com/questions/1165647/how-to-determine-if-a-list-of-polygon-points-are-in-clockwise-order
        /*
        function isClockwise(vertices,index,count){
        var total=0;
        var nextIndex;

        if(count > vertices.length)
        count = vertices.length;


        for(var i=0; i < vertices.length-1; i++){
        nextIndex = getNextIndex(index,vertices.length,+1);
        total += (vertices[nextIndex].x-vertices[index].x) * (vertices[nextIndex].y+vertices[index].y);
        index = nextIndex;
        }
        return total>=0;
        }
         */

        // 2013-06-01: Increment/decrement index by 1
        function getNextIndex(index, length, inc) {
            var next = index + inc;
            if (next == length)
                next = 0;
            if (next < 0)
                next = length - 1;
            return next;
        }

        function getEquation(segment) {
            if (segment.x2 == segment.x1)
                return {
                    'x': segment.x1
                };

            var slope = (segment.y2 - segment.y1) / (segment.x2 - segment.x1);
            var offset = segment.y1 - (slope * segment.x1);
            return {
                'slope': slope,
                'offset': offset
            };
        }

        //
        // line A: y = ax + b
        // line B: y = cx + b
        //
        // x = (d - b) / (a - c)
        function intersectX(eqa, eqb) {
            if ("number" == typeof eqa.slope && "number" == typeof eqb.slope) {
                if (eqa.slope == eqb.slope)
                    return null;

                var ix = (eqb.offset - eqa.offset) / (eqa.slope - eqb.slope);
                var iy = eqa.slope * ix + eqa.offset;
                return new OpenLayers.Geometry.Point(ix, iy);
            } else if ("number" == typeof eqa.x) {
                return new OpenLayers.Geometry.Point(eqa.x, eqb.slope * eqa.x + eqb.offset);
            } else if ("number" == typeof eqb.y) {
                return new OpenLayers.Geometry.Point(eqb.x, eqa.slope * eqb.x + eqa.offset);
            }
            return null;
        }

        function getStreet(segment) {
            if (!segment.attributes.primaryStreetID)
                return null;
            var street = segment.model.streets.getObjectById(segment.attributes.primaryStreetID);
            return street;
        }

        function getDisplacement(street) {
            if (!street)
                return getLastRiverWidth(defaultWidth);
            if (!street.attributes.name)
                return getLastRiverWidth(defaultWidth);
            if (street.attributes.name.match(/^(\d+)(m|м)\b/)) // TODO make localizable
                return parseInt(RegExp.$1);
            if (street.attributes.name.match(/^(\d+)ft\b/)) // TODO make localizable
                return parseInt(RegExp.$1) * 0.3048;
            return getLastRiverWidth(defaultWidth);
        }

        // 2013-06-09: Save current river Width
        function setLastRiverWidth(riverWidth) {
            if (typeof(Storage) !== "undefined") {
                // 2013-06-09: Yes! localStorage and sessionStorage support!
                sessionStorage.riverWidth = Number(riverWidth);
            } else {
                // Sorry! No web storage support..
                console_log("No web storage support");
            }
        }

        // 2013-06-09: Returns last saved river width
        function getLastRiverWidth(defaultRiverWidth) {
            if (typeof(Storage) !== "undefined") {
                // 2013-06-09: Yes! localStorage and sessionStorage support!
                if (sessionStorage.riverWidth)
                    return Number(sessionStorage.riverWidth);
                else
                    return Number(defaultRiverWidth); // Default river width
            } else {
                // Sorry! No web storage support..
                return Number(defaultRiverWidth); // Default river width
            }
        }

        // 2013-10-20: Save current unlimited size preference
        function setLastIsUnlimitedSize(isUnlimitedSize) {
            if (typeof(Storage) !== "undefined") {
                // 2013-06-09: Yes! localStorage and sessionStorage support!
                sessionStorage.isUnlimitedSize = Number(isUnlimitedSize);
            } else {
                // Sorry! No web storage support..
                console_log("No web storage support");
            }
        }

        // 2013-10-20: Returns last saved unlimited size preference
        function getLastIsUnlimitedSize(defaultValue) {
            if (typeof(Storage) !== "undefined") {
                // 2013-10-20: Yes! localStorage and sessionStorage support!
                if (sessionStorage.isUnlimitedSize)
                    return Number(sessionStorage.isUnlimitedSize);
                else
                    return Number(defaultValue); // Default preference
            } else {
                // Sorry! No web storage support..
                return Number(defaultValue); // Default preference
            }
        }

        // 2013-10-20: Save current unlimited size preference
        function setLastIsDeleteSegment(isDeleteSegment) {
            if (typeof(Storage) !== "undefined") {
                // 2013-06-09: Yes! localStorage and sessionStorage support!
                sessionStorage.isDeleteSegment = Number(isDeleteSegment);
            } else {
                // Sorry! No web storage support..
                console_log("No web storage support");
            }
        }

        // 2013-10-20: Returns last saved unlimited size preference
        function getLastIsDeleteSegment(defaultValue) {
            if (typeof(Storage) !== "undefined") {
                // 2013-10-20: Yes! localStorage and sessionStorage support!
                if (sessionStorage.isDeleteSegment)
                    return Number(sessionStorage.isDeleteSegment);
                else
                    return Number(defaultValue); // Default preference
            } else {
                // Sorry! No web storage support..
                return Number(defaultValue); // Default preference
            }
        }

        // 2014-06-05: Returns WME interface language
        function getLanguage() {
            var wmeLanguage;
            var urlParts;

            urlParts = location.pathname.split("/");
            wmeLanguage = urlParts[1].toLowerCase();
            if (wmeLanguage === "editor")
                wmeLanguage = "us";

            return wmeLanguage;

        }

        // 2014-06-05: Returns WME interface language
        /*
        function isBetaEditor(){
        var wmeEditor = location.host.toLowerCase();

        return wmeEditor==="editor-beta.waze.com";
        }
         */

        // 2014-06-05: Translate text to different languages
        function intLanguageStrings() {
            switch (getLanguage()) {
            case "es": // 2014-06-05: Spanish
            case "es-419":
                langText = new Array("metros", "Ancho", "Cree una nueva calle, selecciónela y oprima este botón.", "Calle a Río", "Tamaño ilimitado",
                        "¡No se encontró una calle sin guardar!", "Todos los segmentos de la calle adentro del río. No se puede continuar.",
                        "Múltiples segmentos de la calle dentro del río. No se puede continuar", "Other", "Forest", "Delete segment");
                break;
            case "fr": // 2014-06-05: French
                langText = new Array("mètres", "Largura", "Crie uma nova rua, a selecione e clique neste botão.", "Rue á rivière", "Taille illimitée (dangereux)",
                        "Pas de nouvelle rue non enregistré trouvée!", "Tous les segments de la rue dans la rivière. Vous ne pouvez pas continuer.",
                        "Plusieurs segments de rues à l'intérieur de la rivière. Vous ne pouvez pas continuer.", "Other", "Forest", "Delete segment");
                break;
            case "ru": // 2014-06-05: Russian
                langText = new Array("метров", "Ширина", "Создайте новую дорогу (не сохраняйте), выберите ее и нажмите эту кнопку.", "Река", "Вся длина",
                        "Не выделено ни одной не сохраненной дороги!", "Все сегменты дороги находятся внутри реки. Преобразование невозможно.",
                        "Слишком много сегментов дороги находится внутри реки. Преобразование невозможно.", "Контур", "Лес", "Удалить сегмент");
                break;
            case "uk": // 2018-05-03: Ukrainian
                langText = new Array("метрів", "Ширина", "Створіть нову дорогу (не зберігайте і не знімайте виділення) та натисніть цю кнопку.", "Ріка", "Безлімітна довжина (небезпечно)",
                        "Не виділено жодної збереженої дороги!", "Усі сегменти дороги знаходяться всередині ріки. Перетворення неможливе.",
                        "Занадто багато сегментів дороги знаходяться всередині ріки. Перетворення неможливе.", "Контур", "Ліс", "Видалити сегмент");
                break;
            case "hu": // 2014-07-02: Hungarian
                langText = new Array("méter", "Szélesség", "Hozzon létre egy új utcát, válassza ki, majd kattintson erre a gombra.", "Utcából folyó", "Korlátlan méretű (nem biztonságos)",
                        "Nem található nem mentett és kiválasztott új utca!", "Az útszakasz a folyón belül található! Nem lehet folytatni.",
                        "Minden útszakasz a folyón belül található! Nem lehet folytatni.", "Other", "Forest", "Delete segment");
                break;
            case "cs": // 2014-07-03: Czech
                langText = new Array("metrů", "Šířka", "Vytvořte osu řeky, vyberte segment a stiskněte toto tlačítko.", "Silnice na řeku", "Neomezená šířka (nebezpečné)",
                        "Nebyly vybrány žádné neuložené segmenty!", "Všechny segmenty jsou uvnitř řeky! Nelze pokračovat.",
                        "Uvnitř řeky je více segmentů! Nelze pokračovat.", "Other", "Forest", "Delete segment");
                break;
            case "pl": // 2014-11-08: Polish - By Zniwek
                langText = new Array("metrów", "Szerokość", "Stwórz ulicę, wybierz ją i kliknij ten przycisk.", "Ulica w Rzekę", "Nieskończony rozmiar (niebezpieczne)",
                        "Nie znaleziono nowej i niezapisanej ulicy!", "Wszystkie segmenty ulicy wewnątrz rzeki. Nie mogę kontynuować.",
                        "Wiele segmentów ulicy wewnątrz rzeki. Nie mogę kontynuować.", "Other", "Forest", "Delete segment");
                break;
            case "pt-br": // 2015-04-05: Portuguese - By esmota
                langText = new Array("metros", "Largura", "Criar uma nova rua, selecione e clique neste botão.", "Rua para Rio", "Comprimento ilimitado (instável)",
                        "Nenhuma nova rua, sem salvar, selecionada!", "Todos os segmentos de rua estão dentro de um rio. Nada a fazer.",
                        "Múltiplos segmentos de rua dentro de um rio. Impossível continuar.", "Other", "Forest", "Delete segment");
                break;
            default: // 2014-06-05: English
                langText = new Array("meters", "Width", "Create a new street, select and click this button.", "River", "Unlimited size (unsafe)",
                        "No unsaved and selected new street found!", "All street segments inside river. Cannot continue.",
                        "Multiple street segments inside river. Cannot continue.", "Other", "Forest", "Delete segment");
            }
        }

        // 2014-06-05: Returns the translated  string to current language, if the language is not recognized assumes English
        function getString(stringID) {
            return langText[stringID];
        }

        function console_log(msg) {
            //if (console.log)
            // 2013-05-19: Alternate method to validate console object
            if (typeof console != "undefined")
                console.log(msg);
        }

        // 2014-06-05: Get interface language
        scriptLanguage = getLanguage();
        intLanguageStrings();

        $(document)
        .on('segment.wme', (event, element, model) => {
            insertButtons()
        })
    }

    streetToRiver_bootstrap();

    // from: https://greasyfork.org/ru/scripts/16071-wme-keyboard-shortcuts (modify)
    /*
    when adding shortcuts each shortcut will need a unique name
    the command to add links is WMERegisterKeyboardShortcut(ScriptName, ShortcutsHeader, NewShortcut, ShortcutDescription, FunctionToCall, ShortcutKeysObj) {
    ScriptName: This is the name of your script used to track all of your shortcuts on load and save.
    ScriptName: replace 'WMEAwesome' with your scripts name such as 'SomeOtherScript'
    ShortcutsHeader: this is the header that will show up in the keyboard editor
    NewShortcut: This is the name of the shortcut and needs to be unique from all of the other shortcuts, from other scripts, and WME
    ShortcutDescription: This will show up as the text next to your shortcut
    FunctionToCall: this is the name of your function that will be called when the keyboard shortcut is presses
    ShortcutKeysObj: the is the object representing the keys watched set this to '-1' to let the users specify their own shortcuts.
    ShortcutKeysObj: The alt, shift, and ctrl keys are A=alt, S=shift, C=ctrl. for short cut to use "alt shift ctrl and l" the object would be 'ASC+l'
     */
    function WMEKSRegisterKeyboardShortcut(a, b, c, d, e, f, g) {
        try {
            I18n.translations[I18n.locale].keyboard_shortcuts.groups[a].members.length
        } catch (c) {
            W.accelerators.Groups[a] = [],
            W.accelerators.Groups[a].members = [],
            I18n.translations[I18n.locale].keyboard_shortcuts.groups[a] = [],
            I18n.translations[I18n.locale].keyboard_shortcuts.groups[a].description = b,
            I18n.translations[I18n.locale].keyboard_shortcuts.groups[a].members = []
        }
        if (e && "function" == typeof e) {
            I18n.translations[I18n.locale].keyboard_shortcuts.groups[a].members[c] = d,
            W.accelerators.addAction(c, {
                group: a
            });
            var i = "-1",
            j = {};
            j[i] = c,
            W.accelerators._registerShortcuts(j),
            null !== f && (j = {}, j[f] = c, W.accelerators._registerShortcuts(j)),
            W.accelerators.events.register(c, null, function () {
                e(g)
            })
        } else
            alert("The function " + e + " has not been declared")
    }
    function WMEKSLoadKeyboardShortcuts(a) {
        if (console.log("WMEKSLoadKeyboardShortcuts(" + a + ")"), localStorage[a + "KBS"])
            for (var b = JSON.parse(localStorage[a + "KBS"]), c = 0; c < b.length; c++)
                try {
                    W.accelerators._registerShortcuts(b[c])
                } catch (a) {
                    console.log(a)
                }
    }
    function WMEKSSaveKeyboardShortcuts(a) {
        console.log("WMEKSSaveKeyboardShortcuts(" + a + ")");
        var b = [];
        for (var c in W.accelerators.Actions) {
            var d = "";
            if (W.accelerators.Actions[c].group == a) {
                W.accelerators.Actions[c].shortcut ? (W.accelerators.Actions[c].shortcut.altKey === !0 && (d += "A"), W.accelerators.Actions[c].shortcut.shiftKey === !0 && (d += "S"), W.accelerators.Actions[c].shortcut.ctrlKey === !0 && (d += "C"), "" !== d && (d += "+"), W.accelerators.Actions[c].shortcut.keyCode && (d += W.accelerators.Actions[c].shortcut.keyCode)) : d = "-1";
                var e = {};
                e[d] = W.accelerators.Actions[c].id,
                b[b.length] = e
            }
        }
        localStorage[a + "KBS"] = JSON.stringify(b)
    }
    /* ********************************************************** */
})();