WME Assist UA

Check and fix street names for POI and segments. UA fork of original WME Assist

当前为 2024-05-31 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         WME Assist UA
// @description  Check and fix street names for POI and segments. UA fork of original WME Assist
// @version      2024.05.31.001
// @namespace    https://greasyfork.org/uk/users/160654-waze-ukraine
// @author       borman84 (Boris Molodenkov), madnut, turbopirate + (add yourself here)
// @grant        GM_xmlhttpRequest
// @connect      google.com
// @connect      script.googleusercontent.com
// @match        https://beta.waze.com/*editor*
// @match        https://www.waze.com/*editor*
// @exclude      https://www.waze.com/*user/*editor/*
// @require      https://rawcdn.githack.com/nextapps-de/winbox/0.2.82/dist/winbox.bundle.min.js#sha256=dfd7e8cc105863d51558637b3671460bee60d3a84af2dd4676ea73fab21258e7
// @icon         
// ==/UserScript==

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

(function () {

    const scriptName = GM_info.script.name;

    function debug(message) {
        if (typeof message === 'string') {
            console.debug(scriptName + " DEBUG: " + message);
        } else {
            console.debug(scriptName + " DEBUG: ", message);
        }
    }

    function info(message) {
        if (typeof message === 'string') {
            console.log(scriptName + " INFO: " + message);
        } else {
            console.log(scriptName + " INFO: ", message);
        }
    }

    function warning(message) {
        if (typeof message === 'string') {
            console.warn(scriptName + " WARN: " + message);
        } else {
            console.warn(scriptName + " WARN: ", message);
        }
    }

    function series(array, start, action, alldone) {
        var helper = function (i) {
            if (i < array.length) {
                action(array[i], function () {
                    helper(i + 1);
                });
            } else {
                if (alldone) {
                    alldone();
                }
            }
        };

        helper(start);
    }

    function run_wme_assist() {
        const supportedRulesVersion = "1.1";
        const requestsTimeout = 20000; // in ms
        const rulesHash = "AKfycbyCR85UB-OexWIcN2pkTV1828bf0M6hUXkfHmu79M50PW3LMjpXkZ4ynRUzf2AOJqQqBA";
        let rulesDB = {};

        function displayHtmlPage(res) {
            if (res.responseText.match(/Authorization needed/) || res.responseText.match(/ServiceLogin/)) {
                alert(scriptName + ":\n" +
                    "Authorization is required for using this script. This is one time action.\n" +
                    "Now you will be redirected to the authorization page, where you'll need to approve request.\n" +
                    "After confirmation, please close the page and reload WME.");
            }
            let w = window.open();
            w.document.open();
            w.document.write(res.responseText);
            w.document.close();
            w.location = res.finalUrl;
        }

        function validateHTTPResponse(res) {
            let result = false,
            displayError = true;
            if (res) {
                switch (res.status) {
                case 200:
                    displayError = false;
                    if (res.responseHeaders.match(/content-type: application\/json/i)) {
                        result = true;
                    } else if (res.responseHeaders.match(/content-type: text\/html/i)) {
                        displayHtmlPage(res);
                    }
                    break;
                default:
                    displayError = false;
                    alert(scriptName + " Error: unsupported status code - " + res.status);
                    info(res.responseHeaders);
                    info(res.responseText);
                    break;
                }
            } else {
                displayError = false;
                alert(scriptName + " error: Response is empty!");
            }

            if (displayError) {
                alert(scriptName + ": Error processing request. Response: " + res.responseText);
            }
            return result;
        }

        function requestRules(callbackFunc) {
            GM_xmlhttpRequest({
                url: 'https://script.google.com/macros/s/' + rulesHash + '/exec?func=getStreetRules&user=' + W.loginManager.user.getUsername(),
                method: 'GET',
                timeout: requestsTimeout,
                onload: function (res) {
                    if (validateHTTPResponse(res)) {
                        let out = JSON.parse(res.responseText);
                        if (out.result == "success") {
                            info("Rules format version: " + out.version);
                            if (out.version == supportedRulesVersion) {
                                rulesDB = out.rules;
                            } else {
                                alert(scriptName + ": Table rules format version is not supported!\nPlease, update Assist script to newer version.");
                            }
                        } else {
                            alert(scriptName + ": Error getting rules!");
                        }
                    }
                    callbackFunc();
                },
                ontimeout: function (res) {
                    alert(scriptName + ": Sorry, request timeout!");
                },
                onerror: function (res) {
                    alert(scriptName + ": Sorry, request error!");
                }
            });
        }

        var Rule = function (comment, func, variant) {
            this.comment = comment;
            this.correct = func;
            this.variant = variant;
        };

        var CustomRule = function (oldname, newname) {
            var title = '/' + oldname + '/ ➤ ' + newname;
            this.oldname = oldname;
            this.newname = newname;
            this.custom = true;
            $.extend(this, new Rule(title, function (text) {
                    return text.replace(new RegExp(oldname), newname);
                }));
        };

        var ExperimentalRule = function (comment, func) {
            this.comment = comment;
            this.correct = func;
            this.experimental = true;
        };

        var Rules = function () {
            var rules_basicCommon = function () {
                return [
                    new Rule('Unbreak space in street name', function (text) {
                        return text.replace(/\s+/g, ' ');
                    }),
                    new Rule('ACUTE ACCENT in street name', function (text) {
                        return text.replace(/\u0301|\u0300/g, '');
                    }),
                    new Rule('Dash in street name', function (text) {
                        return text.replace(/\u2010|\u2011|\u2012|\u2013|\u2014|\u2015|\u2043|\u2212|\u2796/g, '-');
                    }),
                    new Rule('No space after the word', function (text) {
                        return text.replace(/\.(?!\s)(.+)/g, '. $1');
                    }),
                    new Rule('No space after the >', function (text) {
                        return text.replace(/>(?!\s)/g, '> ');
                    }),
                    new Rule('Garbage dot', function (text) {
                        return text.replace(/(^|\s+)\./g, '$1');
                    }),
                ];
            };

            var rules_UA = function () {
                var hasCyrillic = function (s) {
                    return s.search(/[а-яіїєґ]/i) != -1;
                };
                var hasShortStatus = function (s) {
                    return s.search(/( |^)(вул\.|просп\.|мкрн\.|наб\.|пров\.|ст\.|пр\.|б-р|р-н)( |$)/i) != -1;
                };
                var hasLongStatus = function (s) {
                    return s.search(/( |^)(площа|алея|шосе|тракт|узвіз|тупик|міст|в\'їзд|виїзд|виїзд|розворот|трамвай|залізниця|майдан|заїзд|траса|дорог[аи]|шляхопровід|шлях|завулок|квартал|автомагістраль)( |$)/i) != -1;
                };
                var hasSpecialStatus = function (s) {
                    return s.search(/( |^)([РНТМ](-[0-9]+)+|[EОС][0-9]+)|~|>|\/( |$)|^(|до|на|>) /i) != -1;
                };
                var hasInternationalName = function (s) {
                    return s.search(/^E[0-9]+$/i) != -1;
                };
                var hasStatus = function (s) {
                    return (hasShortStatus(s) || hasLongStatus(s) || hasSpecialStatus(s));
                };

                var hasAdjName = function (s) {
                    var adjRegex = new RegExp(
                            '( |^)(Балтійська|Кропивницька|Бориславська|Овочева|Спортивна|Дорогобицька|Зарічна|Привокзальна|Клубна|Запречистська|Заставська|Глибока|Японська' +
                            '|Київська|Городоцька|Зелена|Судова|Замкнена|Стрийська|Козельницька|Снопківська|Волоська|Турецька|Скельна|Грецька|Кубанська|Кримська|Водогінна' +
                            '|Аральська|Студентська|Переяславська|Дунайська|Дністерська|Тернопільська|Зубрівська|Сихівська|Райдужна|Вулецька|Соняшникова|Коломийська' +
                            '|Садибна|Демнянська|Наукова|Жасминова|Білоцерківська|Орлина|Кульпарківська|Вітряна|Молдавська|Виноградна|Холодноярська|Керамічна|Кишинівська' +
                            '|Львівська|Урожайна|Садова|Гіпсова|Окружна|Зв\'язкова|Житомирська|Повстанська|Збиральна|Авіаційна|Кондукторська|Полева|Дублянська|Вокзальна' +
                            '|Галицька|Любінська|Спокійна|Народна|Залізнична|Личаківська|Сполучна|Тернова|Конюшинна|Яворівська|Західна|Суховольська|Світла|Озерна|Ряшівська' +
                            '|Коротка|Сосновська|Весняна|Січова|Вузька|Журавлина|Рудненська|Чернівецька|Стародубська|Хотинська|Одеська|Стрілецька|Замарстинівська|Топольна' +
                            '|Інструментальна|Господарська|Волошкова|Сріблиста|Торф\'яна|Городницька|Сінна|Покутська|Заповітна|Малинова|Вербова|Перекопська|Квітова|Корінна' +
                            '|Східна|Крута|Реміснича|Узбецька|Технічна|Половинна|Хімічна|Жовківська|Лемківська|Сорочинська|Джерельна|Батуринська|Замкова|Клепарівська' +
                            '|Смерекова|Золота|Чорноморська|Вугільна|Сянська|Мулярська|Весела|Мукачівська|Ужгородська|Пильникарська|Базарна|Водна|Вагова|Таманська' +
                            '|Театральна|Вірменська|Університетська|Вічева|Руська|Друкарська|Сербська|Ставропігійська|Стара|Насипна|Рівна|Шевська|Староєврейська|Архівна' +
                            '|Підвальна|Валова|Гуцульська|Банківська|Пекарська|Севастопольська|Тиха|Лісна|Слободна|Харківська|Мала|Круп\'ярська|Таджицька|Кутова|Грибова' +
                            '|Ярова|Букова|Ромоданівська|Зимова|Долішня|Яричівська|Копальна|Казахська|Низова|Міжгірна|Грушева|Ялтинська|Чумацька|Богданівська|Глиняна' +
                            '|Переможна|Поетична|Приязна|Визвольна|Бігова|Наступальна|Пластова|Польова|Ковельська|Врізана|Ігорева|Корейська|Теребовлянська|Черкаська' +
                            '|Белзька|Молочна|Корецька|Крайня|Милятинська|Горіхова|Юнацька|Трависта|Бродівська|Старознесенська|Почаївська|Пинська|Миргородська|Поворотна' +
                            '|Потелицька|Новознесеньська|Волинська|Промислова|Опришківська|Механічна|Донецька|Льняна|Полтв\'яна|Селянська|Космічна|Купальська|Кукурудзяна' +
                            '|Бузька|Тарасівська|Бескидська|Лазнева|Підмурна|Рибна|Тролейбусна|Північна|Лугова|Лісова|Сигнальна|Таллінська|Ливарна|Левандівська|Повітряна' +
                            '|Тісна|Кочегарська|Естонська|Олешківська|Ясна|Щекавицька|Алмазна|Слюсарська|Папоротна|Ботанічна|Заболотівська|Мирна|Скромна|Пропелерна' +
                            '|Загородня|Моторна|Широка|Холмська|Лисеницька|Довга|Пасічна|Хлібна|Китайська|Садівнича|Каштанова|Медова|Околична|Відкрита|Бойківська' +
                            '|Куликівська|Червона|Мила|Сарненська|Природна|Перемиська|Моршинська|Конотопська|Похила|Художня|Вишнева|Молодіжна|Дивізійна|Поштова|Тунельна' +
                            '|Білоруська|Яблунева|Творча|Пільна|Шпитальна|Винниківська|Поліська|Загірна|Нагірна|Мурована|Нова|Архітекторська|Грюнвальдська|Політехнічна' +
                            '|Професорська|Бібліотечна|Болгарська|Випасова|Малоголосківська|Монгольська|Скісна|Резедова|Простинна|Бузинова|Порічкова|Осикова|Нарцисова' +
                            '|Розлога|Ряснянська|Паралельна|Південна|Комарнівська|Перемишльська|Заводська|Соборна|Тупікова|Горішня|Шкільна|Українська|Сонячна|Артищівська' +
                            '|Паркова|Равська|Старомостівська|Головна|Травнева|Клюсовська|Сокальська|Крива|Святоюрська|Завадівська|Центральна|Жовтнева|Колгоспна|Больнична' +
                            '|Радянська|Ювілейна|Степова|Порохова|Робітнича|Очеретяна|Жнивна|Буковинська|Луганська|Абхазька|Лижв\'ярська|Гайдамацька|Грабова|Полунична' +
                            '|Томашівська|Каховська|Гіацинтова|Дальня|Дозвільна|Лютнева|Корсунська|Підгаєцька|Дубнівська|Дрогобицька|Мисливська|Бакінська|Чуваська' +
                            '|Скнилівська|Щирецька|Санітарна|Лікувальна|Баштанна|Мостова|Паровозна|Вагонна|Проста|Суха|Фабрична|Солов\'[яї]на|Хорватська|Вільна|Затишна' +
                            '|Крехівська|Сходова|Спадиста|Туркменська|Олійна|Рослинна|Албанська|Азовська|Карпатська|Листопадна|Віденська|Енергетична|Соколина|Латвійська' +
                            '|Земельна|Трускавецька|Росиста|Рядова|Сусідня|Рахівська|Розбіжна|Рівнинна|Керченська|Піскова|Ніжинська|Кошова|Козацька|Гранітна|Дубова' +
                            '|Полуднева|Лебедина|Навколишня|Січнева|Горівська|Поморянська|Кінцева|Курінна|Новознесенська|Міртова|Шполянська|Грунтова|Ґрунтова|Варшавська)( |$)',
                            'i');
                    return s.search(adjRegex) != -1;
                };

                // ATTENTION: Rule order is important!
                return rules_basicCommon().concat([
                        new Rule('Check with rules from Google Sheet', function (text, city) {
                            let ruleKey = text + '_' + city;
                            if (rulesDB[ruleKey]) {
                                let matchCity = rulesDB[ruleKey].city ? rulesDB[ruleKey].city == city : true;
                                if (matchCity) {
                                    return rulesDB[ruleKey].new_name;
                                }
                            }
                            return text;
                        }, 'GSheets'),

                        new Rule('Fix English characters in name', function (t) {
                            return !hasCyrillic(t) || hasInternationalName(t) ? t : t.replace(/[AaBCcEeHIiKkMOoPpTXxYy]/g, function (c) {
                                return {
                                    'A': 'А',
                                    'a': 'а',
                                    'B': 'В',
                                    'C': 'С',
                                    'c': 'с',
                                    'E': 'Е',
                                    'e': 'е',
                                    'H': 'Н',
                                    'I': 'І',
                                    'i': 'і',
                                    'K': 'К',
                                    'k': 'к',
                                    'M': 'М',
                                    'O': 'О',
                                    'o': 'о',
                                    'P': 'Р',
                                    'p': 'р',
                                    'T': 'Т',
                                    'X': 'Х',
                                    'x': 'х',
                                    'Y': 'У',
                                    'y': 'у'
                                }
                                [c];
                            });
                        }),
                        new Rule('Delete space in initials', function (text) {
                            return text.replace(/(^| +)([А-ЯІЇЄҐ]\.) ([А-ЯІЇЄҐ]\.)/, '$1$2$3');
                        }),
                        new Rule('Incorrect characters in street name', function (t) {
                            // This rule should be before renaming rules or they couldn't see some errors
                            return t
                            .replace(/[@#№$,^!:;*"?<]/g, ' ').replace(/ {2,}/, ' ')
                            .replace(/[`\u02bc]/g, '\''); // replace incorrect apostrophes (`’)
                        }),
                        /*
                        new Rule('Incorrect language', function (t) {
                        // Translate full Russian names to full Ukrainian
                        // and next rules will shorten them if necessary
                        return t
                        .replace(/(^| )в?улица( |$)/i, '$1вулиця$2')
                        .replace(/(^| )спуск( |$)/i, '$1узвіз$2')
                        .replace(/(^| )(т)расса( |$)/i, '$1$2раса$3')
                        .replace(/(^| )(п)ереулок( |$)/i, '$1$2ровулок$3')
                        .replace(/(^| )(п)роезд( |$)/i, '$1$2роїзд$3')
                        .replace(/(^| )(п)лощадь( |$)/i, '$1$2лоща$3')
                        .replace(/(^| )(ш)оссе( |$)/i, '$1$2осе$3')
                        .replace(/(^| )(с)танция( |$)/i, '$1$2танція$3')
                        .replace(/(^| )(а)ллея( |$)/i, '$1$2лея$3')
                        .replace(/(^| )(н)абережная( |$)/i, '$1$2абережна$3')
                        .replace(/(^| )(м)икрорайон( |$)/i, '$1$2ікрорайон$3')
                        .replace(/(^| )(л)иния( |$)/i, '$1$2інія$3')
                        .replace(/(^| )(а)кадемика( |$)/i, '$1$2кадеміка$3')
                        .replace(/(^| )(а)дмирала( |$)/i, '$1$2дмірала$3')
                        .replace(/ и /i, ' та ');
                        }),
                         */
                        new Rule('Mistake in short status', function (t) {
                            return t
                            .replace(/(^| )(буль?в?\.?|б-р\.)( |$)/i, '$1б-р$3')
                            .replace(/(^| )(?:пр-к?т|п(?:р|о)?сп)\.?( |$)/i, '$1просп.$2')
                            .replace(/(^| )пр-з?д\.?( |$)/i, '$1пр.$2')
                            .replace(/(^| )ул\.?( |$)/i, '$1вул.$2')
                            .replace(/(^| )р-н\.( |$)/i, '$1р-н$2')
                            .replace(/(^| )пер\.?( |$)/i, '$1пров.$2')
                            .replace(/(^| )(пров|просп|пр|вул|ст|мкрн|наб|дор)( |$)/i, '$1$2.$3');
                        }),
                        new Rule('Rules for back status', function (t) {
                            // Якщо закінчується на "-ний; -ський", переносимо статус в кінець (сміливе рішення)
                            // !!! Сміливе рішення :)
                            return t
                                .replace(/(^)(пров\.|пр\.|тупик|узвіз)( )(.*ний$)/i, '$4 $2')
                                .replace(/(^)(пров\.|пр\.|тупик|узвіз)( )(.*ський$)/i, '$4 $2')
                        }),
                        new Rule('Long status must be short', function (t) {
                            // Do short status only if there no other shorten statuses in name
                            return hasShortStatus(t) ? t : t
                            .replace(/(^| )район( |$)/i, '$1р-н$2')
                            .replace(/(^| )бульвар( |$)/i, '$1б-р$2')
                            .replace(/(^| )провулок( |$)/i, '$1пров.$2')
                            .replace(/(^| )проспект( |$)/i, '$1просп.$2')
                            .replace(/(^| )вулиця( |$)/i, '$1вул.$2')
                            .replace(/(^| )станція( |$)/i, '$1ст.$2')
                            .replace(/(^| )мікрорайон( |$)/i, '$1мкрн.$2')
                            .replace(/(^| )набережна( |$)/i, '$1наб.$2');
                        }),
                        new Rule('Shorten street name or status must be long', function (t) {
                            return t
                            .replace(/(^| )туп\.?( |$)/i, '$1тупик$2')
                            .replace(/(^| )тр-т\.?( |$)/i, '$1тракт$2')
                            .replace(/(^| )(сп\.?|узв\.?|узвоз)( |$)/i, '$1узвіз$3')
                            .replace(/(^| )пр\.( |$)/i, '$1проїзд$2')
                            .replace(/(^| )пл\.?( |$)/i, '$1площа$2')
                            .replace(/(^| )ал\.?( |$)/i, '$1алея$2')
                            .replace(/(^| )ш\.?( |$)/i, '$1шосе$2')
                            .replace(/(^|а )дор\.?( |$)/i, '$1дорога$2')
                            .replace(/(ї )дор\.?( |$)/i, '$1дороги$2')
                            .replace(/(^| )ген\.?( |$)/i, '$1Генерала$2')
                            .replace(/(^| )див\.?( |$)/i, '$1Дивізії$2')
                            .replace(/(^| )ак\.?( |$)/i, '$1Академіка$2')
                            .replace(/(^| )марш\.?( |$)/i, '$1Маршала$2')
                            .replace(/(^| )адм\.?( |$)/i, '$1Адмірала$2');
                        }),
                        new Rule('Incorrect number ending', function (t) {
                            return t
                            .replace(/-[гштм]а/, '-а')
                            .replace(/-[ыоиі]й/, '-й')
                            .replace(/-тя/, '-я')
                            .replace(/-ая/, '-а');
                        }),
                        new Rule('Incorrect highway name', function (text) {
                            return text.replace(/([РрНнМмPpHM])[-\s]*([0-9]{2})/, function (a, p1, p2) {
                                p1 = p1
                                    .replace('р', 'Р')
                                    .replace('н', 'Н')
                                    .replace('м', 'М')
                                    .replace('P', 'Р')
                                    .replace('p', 'Р')
                                    .replace('H', 'Н')
                                    .replace('M', 'М');

                                return p1 + '-' + p2;
                            });
                        }),
                        new Rule('Incorrect local street name', function (text) {
                            return text.replace(/([ТтT])[-\s]*([0-9]{2})[-\s]*([0-9]{2})/, function (a, p1, p2, p3) {
                                p1 = p1
                                    .replace('т', 'Т')
                                    .replace('T', 'Т');

                                return p1 + '-' + p2 + '-' + p3;
                            });
                        }),
                        new Rule('Incorrect international highway name', function (text) {
                            return text.replace(/^ *[eе][- ]*([0-9]+)/i, 'E$1');
                        }),
                        new Rule('Incorrect local road name', function (text) {
                            return text.replace(/([OoCcОоСс])[-\s]*([0-9]+)[-\s]*([0-9]+)[-\s]*([0-9]+)/, function (a, p1, p2, p3, p4) {
                                p1 = p1
                                    .replace('o', 'О')
                                    .replace('O', 'О')
                                    .replace('c', 'С')
                                    .replace('C', 'С');

                                return p1 + p2 + p3 + p4;
                            });
                        }),

                        new Rule('Fix status', function (t) {
                            return hasStatus(t) ? t : 'вул. ' + t;
                        }, 'Ukraine'),

                        new Rule('Detect status absense or incorrect placement', function (t) {
                            return hasStatus(t) ? (hasAdjName(t) ? t.replace(/(.*)(вул\.)(.*)/, '$1 $3 $2') : t) : (hasAdjName(t) ? t + ' вул.' : '');
                        }, 'Lviv'),

                        new Rule('Move status to begin of name', function (text) {
                            if (!hasSpecialStatus(text)) {
                                return text.replace(/(.*)(вул\.)(.*)/, '$2 $1 $3');
                            }
                            return text;
                        }, 'Ukraine'),
                    ]);
            };

            var getCountryRules = function () {
                var commonRules = [
                    // Following rules must be at the end because
                    // previous rules might insert additional spaces
                    new Rule('Redundant space in street name', function (text) {
                        return text.replace(/[ ]+/g, ' ');
                    }),
                    new Rule('Space at the begin of street name', function (text) {
                        return text.replace(/^[ ]*/, '');
                    }),
                    new Rule('Space at the end of street name', function (text) {
                        return text.replace(/[ ]*$/, '');
                    }),
                ];
                //var countryName = W.model.getTopCountry().getName();
                //info('Get rules for country: ' + countryName);
                var countryRules = rules_UA();

                return countryRules.concat(commonRules);
            };

            var rules = [];
            var customRulesNumber = 0;

            var onAdd = function (rule) {};
            var onEdit = function (index, rule) {};
            var onDelete = function (index) {};

            this.onAdd = function (cb) {
                onAdd = cb;
            };
            this.onEdit = function (cb) {
                onEdit = cb;
            };
            this.onDelete = function (cb) {
                onDelete = cb;
            };

            //this.onCountryChange = function () {
            //    info('Country was changed. Reloading rules...');
            //    rules.splice(customRulesNumber, rules.length - customRulesNumber);
            //    rules = rules.concat(getCountryRules());
            //};

            this.get = function (index) {
                return rules[index];
            };

            this.correct = function (variant, text, city) {
                var newtext = text;
                var experimental = false;
                var custom_enabled = localStorage.getItem('assist_enable_custom_rules') == 'true';

                for (var i = 0; i < rules.length; ++i) {
                    var rule = rules[i];

                    if (rule.custom && !custom_enabled)
                        continue;

                    if (rule.experimental && !this.experimental)
                        continue;

                    if (rule.variant && rule.variant != variant)
                        continue;

                    var previous = newtext;
                    newtext = rule.correct(newtext, city);
                    var changed = (previous != newtext);
                    if (rule.experimental && previous != newtext) {
                        experimental = true;
                    }
                    previous = newtext;
                    // if (rule.custom && changed) {
                    //     // prevent result overwriting by common rules
                    //     break;
                    // }
                }

                return {
                    value: newtext,
                    experimental: experimental
                };
            };

            var save = function (rules) {
                if (localStorage) {
                    localStorage.setItem('assistRulesKey', JSON.stringify(rules.slice(0, customRulesNumber)));
                }
            };

            this.load = function () {
                if (localStorage) {
                    var str = localStorage.getItem('assistRulesKey');
                    if (str) {
                        var arr = JSON.parse(str);
                        for (var i = 0; i < arr.length; ++i) {
                            var rule = arr[i];
                            this.push(rule.oldname, rule.newname);
                        }
                    }
                }

                rules = rules.concat(getCountryRules());
            };

            this.push = function (oldname, newname) {
                var rule = new CustomRule(oldname, newname);
                rules.splice(customRulesNumber++, 0, rule);
                onAdd(rule);

                save(rules);
            };

            this.update = function (index, oldname, newname) {
                var rule = new CustomRule(oldname, newname);
                rules[index] = rule;
                onEdit(index, rule);

                save(rules);
            };

            this.remove = function (index) {
                rules.splice(index, 1);
                --customRulesNumber;
                onDelete(index);

                save(rules);
            };
        };

        var ActionHelper = function () {
            var WazeActionAddAlternateStreet = require("Waze/Action/AddAlternateStreet");
            var WazeActionUpdateFeatureAddress = require("Waze/Action/UpdateFeatureAddress");
            var WazeActionUpdateObject = require("Waze/Action/UpdateObject");

            var ui;

            var type2repo = function (type) {
                var map = {
                    'venue': W.model.venues,
                    'segment': W.model.segments
                };
                return map[type];
            };

            this.setUi = function (u) {
                ui = u;
            };

            this.Select = function (id, type, center, zoom) {
                var attemptNum = 10;

                var select = function () {
                    info('select: ' + id);

                    var obj = type2repo(type).getObjectById(id);

                    W.model.events.unregister('mergeend', null, select);

                    if (obj) {
                        W.selectionManager.setSelectedModels([obj]);
                    } else if (--attemptNum > 0) {
                        W.model.events.register('mergeend', null, select);
                    }

                    debug("Attempt number left: " + attemptNum);

                    W.map.setCenter(center, zoom);
                };

                return select;
            };

            this.fixProblem = function (problem) {
                var deferred = $.Deferred();
                var attemptNum = 10; // after that we decide that object was removed
                var setOld2Alt = localStorage.getItem('assist_move_old_to_alt') == 'true';

                var fix = function () {
                    var uniqueId = problem.object.id + '_' + problem.streetID;
                    var obj = type2repo(problem.object.type).getObjectById(problem.object.id);
                    W.model.events.unregister('mergeend', null, fix);

                    if (obj) {
                        var addr = obj.getAddress().attributes;
                        var attr = {
                            countryID: addr.country.attributes.id,
                            stateID: addr.state.attributes.id,
                            cityName: addr.city.attributes.name,
                            emptyCity: addr.city.attributes.name === null || addr.city.attributes.name === '',
                            streetName: problem.newStreetName,
                            emptyStreet: problem.isEmpty
                        };

                        // check if alternative name
                        if (problem.attrName == 'streetIDs') {
                            // check if still exist
                            if (obj.attributes.streetIDs.indexOf(problem.streetID) > -1) {
                                // remove old street and keep other ones
                                var streets2keep = [];
                                obj.attributes.streetIDs.forEach(function (sid) {
                                    if (problem.streetID !== sid) {
                                        streets2keep.push(sid);
                                    } else {
                                        var altStreet = W.model.streets.getObjectById(sid);
                                        var city = W.model.cities.getObjectById(altStreet.attributes.cityID);
                                        attr.cityName = city.attributes.name;
                                        attr.emptyCity = city.hasName() ? null : true;
                                    }
                                });
                                W.model.actionManager.add(new WazeActionUpdateObject(obj, {
                                        streetIDs: streets2keep
                                    }));

                                // add new street
                                W.model.actionManager.add(new WazeActionAddAlternateStreet(obj, attr, {
                                        streetIDField: problem.attrName
                                    }));
                            } else {
                                ui.updateProblem(uniqueId, '(not found. Deleted?)');
                            }
                        } else {
                            // protect user manual fix
                            if (problem.reason == addr.street.attributes.name) {
                                W.model.actionManager.add(new WazeActionUpdateFeatureAddress(obj, attr, {
                                        streetIDField: problem.attrName
                                    }));
                                // move old name to alt street, if option enabled
                                if (setOld2Alt && obj.type == 'segment') {
                                    var altAttr = {
                                        countryID: addr.country.attributes.id,
                                        stateID: addr.state.attributes.id,
                                        cityName: addr.city.attributes.name,
                                        emptyCity: addr.city.attributes.name === null || addr.city.attributes.name === '',
                                        streetName: problem.reason,
                                        emptyStreet: false //problem.isEmpty
                                    };
                                    W.model.actionManager.add(new WazeActionAddAlternateStreet(obj, altAttr, {
                                            streetIDField: problem.attrName
                                        }));
                                }
                            } else {
                                ui.updateProblem(uniqueId, '(user fix: ' + addr.street.attributes.name + ')');
                            }
                        }
                        deferred.resolve(uniqueId);
                    } else if (--attemptNum <= 0) {
                        ui.updateProblem(uniqueId, '(was not fixed. Deleted?)');
                        deferred.resolve(uniqueId);
                    } else {
                        W.model.events.register('mergeend', null, fix);
                        W.map.setCenter(problem.detectPos, problem.zoom);
                    }

                    debug('Attempt number left: ' + attemptNum);
                };

                fix();

                return deferred.promise();
            };
        };

        var Ui = function () {
            // load main window size and position
            var wndW = localStorage.getItem('assist_window_w');
            var wndH = localStorage.getItem('assist_window_h');
            var wndX = localStorage.getItem('assist_window_x');
            var wndY = localStorage.getItem('assist_window_y');
            // main window default size and position
            var defaultW = 500;
            var defaultH = 500;
            var defaultX = "right";
            var defaultY = "center";
            // define main window limits
            var minH = 100;
            var minW = 200;
            var maxH = 800;
            var maxW = 1024;
            // workaround for bug with window minimize detection
            var saveAllowed = false;

            var addon = document.createElement('div');

            addon.innerHTML = '<wz-overline>' + scriptName + ' v' + GM_info.script.version + '</wz-overline>';

            var section = document.createElement('div');
            section.id = "assist_options";
            section.className = "form-group";
            section.innerHTML = '<wz-label>Options</wz-label>' +
                '<wz-checkbox name="assist_enabled" id="assist_enabled" value="on">Enable/disable</wz-checkbox>' +
                '<wz-checkbox name="assist_skip_alt" id="assist_skip_alt" value="on">Skip checking alternative names</wz-checkbox>' +
                '<wz-checkbox name="assist_move_old_to_alt" id="assist_move_old_to_alt" value="on">Move old name to alternative</wz-checkbox>' +
                '<wz-button name="assist_reset_window" id="assist_reset_window" color="text" size="sm">Reset window size and position</wz-button>';
            addon.appendChild(section);

            var variant = document.createElement('div');
            variant.id = 'variant_options';
            variant.className = "form-group";
            variant.innerHTML = '<wz-label>Naming Rules <a href="https://wazeopedia.waze.com/wiki/Ukraine/Як_називати_вулиці" target="_blank"><span class="fa fa-question-circle"></span></a></wz-label>' +
                '<wz-radio-button name="assist_variant" value="Ukraine" checked="">Ukraine (Classic)</wz-radio-button>' +
                '<wz-radio-button name="assist_variant" value="Lviv">🦁 Lviv (Alternative)</wz-radio-button>';
            if (!$.isEmptyObject(rulesDB)) {
                console.log("WME Assist UA INFO: Downloaded " + Object.keys(rulesDB).length + " rules from Google Sheet");
                variant.innerHTML += '<wz-radio-button name="assist_variant" value="GSheets">Rules from Google Sheet (' + Object.keys(rulesDB).length + ')</wz-radio-button>';
            }
            addon.appendChild(variant);

            section = document.createElement('div');
            section.id = "assist_custom_rules";
            section.className = "form-group";
            $(section)
            .append($('<wz-label>Custom Rules</wz-label>'))
            .append($('<wz-checkbox name="assist_enable_custom_rules" id="assist_enable_custom_rules" value="on">Enable custom rules</wz-checkbox>'))
            .append($('<div>').addClass('btn-toolbar').css({
                    "margin-bottom": "4px"
                })
                .append($('<button>').prop('id', 'assist_add_custom_rule').addClass('btn btn-default btn-primary').text('Add'))
                .append($('<button>').prop('id', 'assist_edit_custom_rule').addClass('btn btn-default').text('Edit'))
                .append($('<button>').prop('id', 'assist_del_custom_rule').addClass('btn btn-default btn-warning').text('Del')))
            .append($('<ul>').addClass('issue-tracker').css({
                    "height": "250px",
                    "overflow": "auto",
                    "padding": "4px",
                    "border": "1px solid lightgray"
                }));
            addon.appendChild(section);

            section = document.createElement('div');
            section.id = "assist_exceptions";
            section.className = "form-group";
            $(section)
            .append($('<wz-label title="Right click on error in list to add">').text('Exceptions'))
            .append($('<ul>').addClass('issue-tracker').css({
                    "height": "250px",
                    "overflow": "auto",
                    "padding": "4px",
                    "border": "1px solid lightgray"
                }));
            addon.appendChild(section);

            const { tabLabel, tabPane } = W.userscripts.registerSidebarTab("sidepanel-assist");

            tabLabel.innerText = scriptName;
            tabLabel.title = scriptName;

            tabPane.innerHTML = addon.innerHTML;

            //tabPane.addEventListener("element-connected", () => {
            //    alert("connected");
            //}, { once: false });

            //tabPane.addEventListener("element-disconnected", () => {
            //    alert("disconnected");
            //}, { once: false });

            var selectedCustomRule = -1;

            this.selectedCustomRule = function () {
                return selectedCustomRule;
            };

            this.addCustomRule = function (title) {
                var thisrule = $('<li>').addClass('list-item-card').click(function () {
                    selectedCustomRule = $('#assist_custom_rules li.list-item-card').index(thisrule);
                    info('index: ' + selectedCustomRule);
                    $('#assist_custom_rules li.list-item-card').css({
                        'background-color': ''
                    });
                    $('#assist_custom_rules li.list-item-card').removeClass('active');
                    $(this).css({
                        'background-color': 'lightblue'
                    });
                    $(this).addClass('active');
                }).hover(function () {
                    $(this).css({
                        cursor: 'pointer',
                        'background-color': 'lightblue'
                    });
                }, function () {
                    $(this).css({
                        cursor: 'auto'
                    });
                    if (!$(this).hasClass('active')) {
                        $(this).css({
                            'background-color': ''
                        });
                    }
                })
                    .append($('<p>').addClass('additional-info clearfix').text(title))
                    .appendTo($('#assist_custom_rules ul.issue-tracker'));
            };

            this.updateCustomRule = function (index, title) {
                $('#assist_custom_rules li.list-item-card').eq(index).find('p.additional-info').text(title);
            };

            this.removeCustomRule = function (index) {
                $('#assist_custom_rules li.list-item-card').eq(index).remove();
                selectedCustomRule = -1;
            };

            this.addException = function (name, del) {
                var thisrule = $('<li>').addClass('list-item-card').click(function () {
                    var index = $('#assist_exceptions li.list-item-card').index(thisrule);
                    del(index);
                }).hover(function () {
                    $(this).css({
                        cursor: 'pointer',
                        'background-color': 'lightblue'
                    });
                }, function () {
                    $(this).css({
                        cursor: 'auto'
                    });
                    if (!$(this).hasClass('active')) {
                        $(this).css({
                            'background-color': ''
                        });
                    }
                })
                    .append($('<p>').addClass('additional-info clearfix').text(name))
                    .appendTo($('#assist_exceptions ul.issue-tracker'));
            };

            this.removeException = function (index) {
                $('#assist_exceptions li.list-item-card').eq(index).remove();
            };

            this.showMainWindow = function () {
                localStorage.setItem('assist_enabled', true);
                mainWindow[0].winbox.show();
                info('enabled');
            };

            this.hideMainWindow = function () {
                localStorage.setItem('assist_enabled', false);
                $('#assist_clearall_btn').click();
                mainWindow[0].winbox.hide();
                info('disabled');
            };

            var wazeMap = $('#WazeMap');

            // ==== Main Window
            $('<div>').prop('id', 'assist_main_window_content')
            .append($('<div>').css({
                    padding: 10
                })
                .append($('<div class="btn-toolbar">')
                    .append($('<button id="assist_fixall_btn" class="btn waze-btn waze-btn-small waze-btn-red">Fix all</button>'))
                    .append($('<button id="assist_fixselected_btn" class="btn waze-btn waze-btn-small waze-btn-red">Fix selected</button>'))
                    .append($('<button id="assist_scanarea_btn" class="btn waze-btn waze-btn-small waze-btn-blue">Scan area</button>'))
                    .append($('<button id="assist_clearfixed_btn" class="btn waze-btn waze-btn-small waze-btn-green">Clear fixed</button>'))
                    .append($('<button id="assist_clearall_btn" class="btn waze-btn waze-btn-small waze-btn-grey" title="Clear all results"><i class="fa fa-close"></i></button>')))
                .append($('<h2><input id="assist_select_all_chk" type="checkbox" />Unresolved issues</h2>').css({
                        'font-size': '100%',
                        'font-weight': 'bold',
                    }))
                .append($('<ol id="assist_unresolved_list"></ol>').css({
                        border: '1px solid lightgrey',
                        'padding-top': 2,
                        'padding-bottom': 2,
                    })))
            .append($('<div>').css({
                    padding: 10,
                })
                .append($('<h2>Fixed issues</h2>').css({
                        'font-size': '100%',
                        'font-weight': 'bold',
                    }))
                .append($('<ol id="assist_fixed_list"></ol>').css({
                        border: '1px solid lightgrey',
                        'padding-top': 2,
                        'padding-bottom': 2,
                    })))
            .appendTo(wazeMap);

            new WinBox(scriptName, {
                id: "assist_main_window",
                class: [ "no-full" ],
                hidden: true,
                x: wndX ? wndX : defaultX,
                y: wndY ? wndY : defaultY,
                width: wndW ? wndW : defaultW,
                height: wndH ? wndH : defaultH,
                minheight: minH,
                minwidth: minW,
                maxheight: maxH,
                maxwidth: maxW,
                background: 'lightblue',
                border: 4,
                mount: document.getElementById("assist_main_window_content"),
                onminimize: function(){
                    this.focus();
                },
                onresize: function(w, h) {
                    if (!this.hidden && !this.min && !this.max &&
                        w > minW && h > minH && w < maxW && h < maxH) {
                        saveAllowed = true;
                        localStorage.setItem('assist_window_w', w);
                        localStorage.setItem('assist_window_h', h);
                    } else {
                        saveAllowed = false;
                    }
                },
                onmove: function(x, y) {
                    if (!this.hidden && !this.min && !this.max &&
                        saveAllowed) {
                        localStorage.setItem('assist_window_x', x);
                        localStorage.setItem('assist_window_y', y);
                    }
                },
                onclose: function(force) {
                    $('#assist_enabled').click();
                    return true;
                }
            });
            var mainWindow = $('#assist_main_window');

            mainWindow.find('.wb-title').css({
                'font-weight': 'bold',
                color: 'black'
            });
            mainWindow.find('.wb-title').append($('<span> - </span>'));
            mainWindow.find('.wb-title')
            .append($('<span>', {
                    id: 'assist-error-num',
                    title: 'Number of unresolved issues',
                    text: 0,
                }).css({
                    color: 'red'
                }));
            mainWindow.find('.wb-title').append($('<span> / </span>'));
            mainWindow.find('.wb-title')
            .append($('<span>', {
                    id: 'assist-fixed-num',
                    title: 'Number of fixed issues',
                    text: 0,
                }).css({
                    color: 'green'
                }));
            /*
            mainWindow.find('.wb-title').append($('<span> - </span>'));
            mainWindow.find('.wb-title')
            .append($('<span>', {
                    id: 'assist-scan-progress',
                    title: 'Scan progress',
                    text: 0,
                }).css({
                    color: 'blue'
                }));
            */
            $("#assist_reset_window").click(function () {
                mainWindow[0].winbox.resize(defaultW, defaultH).move(defaultX, defaultY);
            });

            // ==== Custom Rule Dialog
            $('<div>').prop('id', 'assist_custom_rule_dialog_content')
            .append($('<div>').css({
                    padding: 10
                })
                .append($('<p>All form fields are required</p>'))
                    .append($('<fieldset>')
                        .append($('<label>').prop('for', 'oldname').text('RegExp'))
                        .append($('<input>', {
                            type: 'text',
                            name: 'oldname',
                            id: 'oldname',
                        }))
                    .append($('<label>').prop('for', 'newname').text('Replace text'))
                    .append($('<input>', {
                            type: 'text',
                            name: 'newname',
                            id: 'newname',
                        }))))
            .append($('<div>').css({
                    padding: 10
                })
                .append($('<div class="btn-toolbar">')
                    .append($('<button id="assist_custom_submit_btn" class="btn waze-btn waze-btn-small waze-btn-green">Submit</button>'))
                    .append($('<button id="assist_custom_cancel_btn" class="btn waze-btn waze-btn-small waze-btn-red">Cancel</button>'))
                ))
            .appendTo(wazeMap);

            $('#assist_custom_rule_dialog_content label').css({
                display: 'block'
            });
            $('#assist_custom_rule_dialog_content input').css({
                display: 'block',
                width: '100%'
            });

            new WinBox("Add Custom Rule", {
                id: "assist_custom_rule_dialog",
                class: [ "no-full", "no-min", "no-max", "no-resize" ],
                hidden: true,
                x: "center",
                y: "center",
                width: "300px",
                height: "250px",
                background: 'lightblue',
                border: 4,
                mount: document.getElementById("assist_custom_rule_dialog_content"),
                onclose: function(force){
                    this.hide();
                    return true;
                }
            });
            var customRuleDialog = $('#assist_custom_rule_dialog');

            $("#assist_custom_submit_btn").click(function () {
                customRuleDialog_Ok();
                customRuleDialog[0].winbox.hide();
            });
            $("#assist_custom_cancel_btn").click(function () {
                customRuleDialog[0].winbox.hide();
            });

            var self = this;

            this.addProblem = function (id, text, selectFunc, editFunc, exception, experimental) {
                var problem = $('<li>')
                    .prop('id', 'issue-' + id)
                    .append($('<input>', {
                            value: id,
                            type: "checkbox"
                        }))
                    .append($('<a>', {
                            href: "javascript:void(0)",
                            text: text,
                            click: function (event) {
                                selectFunc(event);
                            },
                            contextmenu: function (event) {
                                exception(event);
                                event.preventDefault();
                                event.stopPropagation();
                            },
                        }))
                    .append('&nbsp;')
                    .append($('<span>', {
                            title: "Add custom rule for this problem",
                            class: "fa fa-edit",
                            style: "cursor: pointer;",
                            click: function (event) {
                                editFunc(event);
                            }
                        }))
                    .appendTo($('#assist_unresolved_list'));

                if (experimental) {
                    problem.children().css({
                        color: 'red'
                    }).prop('title', 'Experimental rule');
                }
            };

            this.getCheckedItemsList = function () {
                var itemsList = [];
                $('#assist_unresolved_list').find('input').each(function () {
                    if (this.checked) {
                        itemsList.push(this.value);
                    }
                });
                return itemsList;
            };

            this.updateProblem = function (id, text) {
                var a = $('li#issue-' + escapeId(id) + ' > a');
                a.text(a.text() + ' ' + text);
            };

            this.setUnresolvedErrorNum = function (text) {
                $('#assist-error-num').text(text);
            };

            this.setFixedErrorNum = function (text) {
                $('#assist-fixed-num').text(text);
            };

            this.setScanProgress = function (text) {
                //$('#assist-scan-progress').text(text);
            };

            var escapeId = function (id) {
                return String(id).replace(/\./g, "\\.");
            };

            this.moveToFixedList = function (id) {
                $("#issue-" + escapeId(id)).appendTo($('#assist_fixed_list')).find("span").remove();
                $("#issue-" + escapeId(id)).find("input").remove();
            };

            this.removeError = function (id) {
                $("#issue-" + escapeId(id)).remove();
            };

            var fixAllBtn = $('#assist_fixall_btn');
            var fixSelectedBtn = $('#assist_fixselected_btn');
            var scanAreaBtn = $('#assist_scanarea_btn');
            var clearFixedBtn = $('#assist_clearfixed_btn');
            var clearAllBtn = $('#assist_clearall_btn');

            var selectAllChk = $('#assist_select_all_chk');

            var unresolvedList = $('#assist_unresolved_list');
            var fixedList = $('#assist_fixed_list');

            var enableCheckbox = $('#assist_enabled');
            var skipAltCheckbox = $('#assist_skip_alt');
            var moveOld2AltCheckbox = $('#assist_move_old_to_alt');
            var enableCustomRulesCheckbox = $('#assist_enable_custom_rules');

            var addCustomRuleBtn = $('#assist_add_custom_rule');
            var editCustomRuleBtn = $('#assist_edit_custom_rule');
            var delCustomRuleBtn = $('#assist_del_custom_rule');

            this.fixAllBtn = function () {
                return fixAllBtn;
            };
            this.fixSelectedBtn = function () {
                return fixSelectedBtn;
            };
            this.scanAreaBtn = function () {
                return scanAreaBtn;
            };
            this.clearFixedBtn = function () {
                return clearFixedBtn;
            };
            this.clearAllBtn = function () {
                return clearAllBtn;
            };

            this.selectAllChk = function () {
                return selectAllChk;
            };

            this.unresolvedList = function () {
                return unresolvedList;
            };
            this.fixedList = function () {
                return fixedList;
            };

            this.enableCheckbox = function () {
                return enableCheckbox;
            };
            this.skipAltCheckbox = function () {
                return skipAltCheckbox;
            };
            this.moveOld2AltCheckbox = function () {
                return moveOld2AltCheckbox;
            };
            this.enableCustomRulesCheckbox = function () {
                return enableCustomRulesCheckbox;
            };
            this.variantRadio = function (value) {
                if (!value) {
                    return $('[name=assist_variant]');
                }

                return $('[name=assist_variant][value=' + value + ']');
            };

            this.addCustomRuleBtn = function () {
                return addCustomRuleBtn;
            };
            this.editCustomRuleBtn = function () {
                return editCustomRuleBtn;
            };
            this.delCustomRuleBtn = function () {
                return delCustomRuleBtn;
            };
            this.customRuleDialog = function (title, params) {
                var deferred = $.Deferred();

                if (params) {
                    customRuleDialog.find('#oldname').val(params.oldname);
                    customRuleDialog.find('#newname').val(params.newname);
                }

                customRuleDialog_Ok = function () {
                    deferred.resolve({
                        oldname: customRuleDialog.find('#oldname').val(),
                        newname: customRuleDialog.find('#newname').val(),
                    });
                };

                customRuleDialog[0].winbox.setTitle(title);
                customRuleDialog[0].winbox.show();

                return deferred.promise();
            };
            //this.variant = function () {
            //    return $('[name=assist_variant][checked]')[0].value;
            //};
        };

        var Scanner = function () {
            var ROAD_TYPE = {
                STREET: 1,
                PRIMARY_STREET: 2,
                FREEWAY: 3,
                RAMP: 4,
                WALKING_TRAIL: 5,
                MAJOR_HIGHWAY: 6,
                MINOR_HIGHWAY: 7,
                OFF_ROAD: 8,
                WALKWAY: 9,
                PEDESTRIAN_BOARDWALK: 10,
                FERRY: 15,
                STAIRWAY: 16,
                PRIVATE_ROAD: 17,
                RAILROAD: 18,
                RUNWAY_TAXIWAY: 19,
                PARKING_LOT_ROAD: 20,
                ALLEY: 22
            };

            var zoomToRoadType = function (e) {
                if (e < 14) {
                    return [];
                }
                switch (e) {
                case 14:
                    return [ROAD_TYPE.PRIMARY_STREET, ROAD_TYPE.FREEWAY, ROAD_TYPE.RAMP, ROAD_TYPE.MAJOR_HIGHWAY, ROAD_TYPE.MINOR_HIGHWAY, ROAD_TYPE.FERRY];
                case 15:
                    return [ROAD_TYPE.PRIMARY_STREET, ROAD_TYPE.FREEWAY, ROAD_TYPE.RAMP, ROAD_TYPE.MAJOR_HIGHWAY, ROAD_TYPE.MINOR_HIGHWAY, ROAD_TYPE.OFF_ROAD, ROAD_TYPE.WALKWAY, ROAD_TYPE.PEDESTRIAN_BOARDWALK, ROAD_TYPE.FERRY, ROAD_TYPE.STAIRWAY, ROAD_TYPE.PRIVATE_ROAD, ROAD_TYPE.RAILROAD, ROAD_TYPE.RUNWAY_TAXIWAY, ROAD_TYPE.PARKING_LOT_ROAD, ROAD_TYPE.ALLEY];
                default:
                    return Object.values(ROAD_TYPE);
                }
            };
            var zoomToVenueLevel = function (e) {
                switch (e) {
                case 12:
                    return 1;
                case 13:
                    return 2;
                case 14:
                case 15:
                case 16:
                    return 3;
                case 17:
                case 18:
                case 19:
                case 20:
                case 21:
                case 22:
                    return 4;
                default:
                    return null;
                }
            };

            var getData = function (e, cb) {
                //debug(e);
                $.get(W.Config.paths.features, e).done(cb);
            };

            var splitExtent = function (extent, zoom) {
                var result = [];

                var ratio = 1; //map.getResolution() / map.getResolutionForZoom(zoom); //FIXME: temporary commented, because getResolutionForZoom() is gone
                var dx = extent.getWidth() / ratio;
                var dy = extent.getHeight() / ratio;

                var x,
                y;
                for (x = extent.left; x < extent.right; x += dx) {
                    for (y = extent.bottom; y < extent.top; y += dy) {
                        var bounds = new OpenLayers.Bounds();
                        bounds.extend(new OpenLayers.LonLat(x, y));
                        bounds.extend(new OpenLayers.LonLat(x + dx, y + dy));

                        result.push(bounds);
                    }
                }

                return result;
            };

            this.scan = function (bounds, zoom, analyze, progress) {
                if (localStorage.getItem('assist_enabled') != 'true') {
                    return;
                }
                var boundsArray = splitExtent(bounds, zoom);
                var completed = 0;

                if (boundsArray.length > 20 && !confirm('Script will scan ' + boundsArray.length + ' pieces. Are you OK?')) {
                    return;
                }

                progress = progress || function () {};

                series(boundsArray, 0, function (bounds, next) {
                    var piece = bounds.transform(W.map.getProjectionObject(), 'EPSG:4326');

                    var e = {
                        bbox: piece.toBBOX(),
                        language: I18n.locale,
                        venueFilter: '3',
                        venueLevel: zoomToVenueLevel(zoom),
                    };
                    var z = {
                        roadTypes: zoomToRoadType(zoom).toString()
                    };
                    OpenLayers.Util.extend(e, z);

                    getData(e, function (data) {
                        analyze(piece, zoom, data);
                        progress(++completed * 100 / boundsArray.length);
                        next();
                    });
                });
            };
        };

        var Analyzer = function () {
            var Exceptions = function () {
                var exceptions = [];

                var onAdd = function (name) {};
                var onDelete = function (index) {};

                var save = function (exceptions) {
                    if (localStorage) {
                        localStorage.setItem('assistExceptionsKey', JSON.stringify(exceptions));
                    }
                };

                this.load = function () {
                    if (localStorage) {
                        var str = localStorage.getItem('assistExceptionsKey');
                        if (str) {
                            var arr = JSON.parse(str);
                            for (var i = 0; i < arr.length; ++i) {
                                var exception = arr[i];
                                this.add(exception);
                            }
                        }
                    }
                };

                this.contains = function (name) {
                    if (exceptions.indexOf(name) == -1)
                        return false;
                    return true;
                };

                this.add = function (name) {
                    exceptions.push(name);
                    save(exceptions);
                    onAdd(name);
                };

                this.remove = function (index) {
                    exceptions.splice(index, 1);
                    save(exceptions);
                    onDelete(index);
                };

                this.onAdd = function (cb) {
                    onAdd = cb;
                };
                this.onDelete = function (cb) {
                    onDelete = cb;
                };
            };

            var analyzedIds = [];
            var problems = [];
            var unresolvedIdx = 0;
            var skippedErrors = 0;
            var variant;
            var exceptions = new Exceptions();
            var rules;
            var action;

            var getUnresolvedErrorNum = function () {
                return problems.length - unresolvedIdx - skippedErrors;
            };

            var getFixedErrorNum = function () {
                return unresolvedIdx;
            };

            this.unresolvedErrorNum = getUnresolvedErrorNum;
            this.fixedErrorNum = getFixedErrorNum;

            this.setRules = function (r) {
                rules = r;
            };

            this.setActionHelper = function (a) {
                action = a;
            };

            this.loadExceptions = function () {
                exceptions.load();
            };

            this.onExceptionAdd = function (cb) {
                exceptions.onAdd(cb);
            };

            this.onExceptionDelete = function (cb) {
                exceptions.onDelete(cb);
            };

            this.addException = function (reason, cb) {
                exceptions.add(reason);

                var i;
                for (i = 0; i < problems.length; ++i) {
                    var problem = problems[i];
                    if (problem.reason == reason) {
                        problem.skip = true;
                        ++skippedErrors;

                        cb(problem.object.id);
                    }
                }
            };

            this.removeException = function (i) {
                exceptions.remove(i);
            };

            this.setVariant = function (v) {
                variant = v;
            };

            this.reset = function () {
                analyzedIds = [];
                problems = [];
                unresolvedIdx = 0;
                skippedErrors = 0;
            };

            this.fixAll = function (oneFixed, allFixed) {
                series(problems, unresolvedIdx, function (p, next) {
                    if (p.skip) {
                        next();
                        return;
                    }

                    action.fixProblem(p).done(function (id) {
                        ++unresolvedIdx;
                        oneFixed(id);

                        setTimeout(next, 0);
                    });
                }, allFixed);
            };

            this.fixSelected = function (listToFix, oneFixed, allFixed) {
                series(problems, unresolvedIdx, function (p, next) {
                    if (listToFix.indexOf(p.object.id + '_' + p.streetID) == -1) {
                        next();
                        return;
                    }
                    if (p.skip) {
                        next();
                        return;
                    }

                    action.fixProblem(p).done(function (id) {
                        ++unresolvedIdx;
                        oneFixed(id);

                        setTimeout(next, 0);
                    });
                }, allFixed);
            };

            var checkStreet = function (bounds, zoom, streetID, obj, attrName, onProblemDetected) {
                var userlevel = W.loginManager.getUserRank() + 1;
                var street = W.model.streets.getObjectById(streetID);

                if (!street)
                    return;

                var detected = false;
                var skip = false;
                var title = '';
                var reason;
                var newStreetName;

                if (!street.attributes.isEmpty) {
                    let streetName = street.attributes.name;
                    if (!exceptions.contains(streetName)) {
                        try {
                            var city = W.model.cities.getObjectById(street.attributes.cityID);
                            var result = rules.correct(variant, streetName, city.attributes.name);
                            newStreetName = result.value;
                            detected = (newStreetName != streetName);
                            if (obj.type == 'venue') {
                                title = 'POI: ';
                            }
                            // alternative names
                            if (attrName == 'streetIDs') {
                                title = 'ALT: ';
                            }
                            // if user has lower rank, just show the segment, but no fix allowed
                            if (obj.lockRank && obj.lockRank >= userlevel) {
                                title = '(L' + (obj.lockRank + 1) + ') ' + title;
                                skip = true;
                            }
                            // show segments with closures, but lock them from fixing
                            if (obj.hasClosures) {
                                title = '(🚧) ' + title;
                                skip = true;
                            }
                            title = title + streetName.replace(/\u00A0/g, '■').replace(/^\s|\s$/, '■');
                            // for "detect only rules" we have no replacement to show
                            if (!newStreetName) {
                                skip = true;
                            } else {
                                title = title + ' ➤ ' + newStreetName;
                            }
                            if (skip) {
                                title = '🔒 ' + title;
                            }
                            reason = streetName;
                        } catch (err) {
                            warning('Street name "' + streetName + '" causes error in rules');
                            return;
                        }
                    }
                }

                if (detected) {
                    var gj = new OpenLayers.Format.GeoJSON();
                    var geometry = gj.parseGeometry(obj.geometry);
                    var objCenter = geometry.getBounds().getCenterLonLat().transform(W.Config.map.projection.remote, W.map.getProjectionObject());
                    var boundsCenter = bounds.clone().getCenterLonLat().transform(W.Config.map.projection.remote, W.map.getProjectionObject());
                    obj.center = objCenter;

                    problems.push({
                        object: obj,
                        reason: reason,
                        attrName: attrName,
                        detectPos: boundsCenter,
                        zoom: zoom,
                        newStreetName: newStreetName,
                        isEmpty: street.attributes.isEmpty,
                        cityId: street.attributes.cityID,
                        streetID: streetID,
                        experimental: false,
                        skip: skip,
                    });

                    onProblemDetected(obj.id + '_' + streetID, obj, title, reason);
                }
            };

            this.analyze = function (bounds, zoom, data, onProblemDetected) {
                var startTime = new Date().getTime();
                var analyzeAlt = true;

                info('start analyze');

                var subjects = {
                    'segment': {
                        attr: 'primaryStreetID',
                        name: 'segments'
                    },
                    'venue': {
                        attr: 'streetID',
                        name: 'venues'
                    }
                };

                if (localStorage) {
                    if (localStorage.getItem('assist_skip_alt') == 'true') {
                        analyzeAlt = false;
                    }
                }

                for (var k in subjects) {
                    var subject = subjects[k];
                    var subjectData = data[subject.name];

                    if (!subjectData)
                        continue;

                    var objects = subjectData.objects;

                    for (var i = 0; i < objects.length; ++i) {
                        var obj = objects[i];
                        var id = obj.id;

                        obj.type = k;

                        if (analyzedIds.indexOf(id) >= 0)
                            continue;

                        if (typeof obj.approved != 'undefined' && !obj.approved)
                            continue;

                        checkStreet(bounds, zoom, obj[subject.attr], obj, subject.attr, onProblemDetected);

                        // support for alternative names
                        if (subject.name == 'segments' && analyzeAlt) {
                            for (var j = 0, n = obj.streetIDs.length; j < n; j++) {
                                checkStreet(bounds, zoom, obj.streetIDs[j], obj, 'streetIDs', onProblemDetected);
                            }
                        }
                        analyzedIds.push(id);
                    }
                }

                info('end analyze: ' + (new Date().getTime() - startTime) + 'ms');
            };
        };

        var Application = function () {
            var scanner = new Scanner();
            var analyzer = new Analyzer();

            var FULL_ZOOM_LEVEL = 17;

            var scanForZoom = function (zoom) {
                scanner.scan(W.map.olMap.calculateBounds(), zoom, function (bounds, zoom, data) {
                    //debug(data);
                    //var w = window.open();
                    //w.document.open();
                    //w.document.write(JSON.stringify(data));
                    //w.document.close();

                    analyzer.analyze(bounds, zoom, data, function (id, obj, title, reason) {
                        ui.addProblem(id, title,
                            action.Select(obj.id, obj.type, obj.center, zoom),
                            function () {
                            ui.customRuleDialog('Add custom rule', {
                                oldname: '(.*)' + reason + '(.*)',
                                newname: reason
                            }).done(function (response) {
                                rules.push(response.oldname, response.newname);
                                ui.scanAreaBtn().click();
                            });
                        },
                            function () {
                            analyzer.addException(reason, function (id) {
                                ui.removeError(id);
                                ui.setUnresolvedErrorNum(analyzer.unresolvedErrorNum());
                            });
                        }, false);

                        ui.setUnresolvedErrorNum(analyzer.unresolvedErrorNum());
                    });
                }, function (progress) {
                    ui.setScanProgress(Math.round(progress) + '%');
                });
            };

            var fullscan = function () {
                scanForZoom(FULL_ZOOM_LEVEL);
            };

            var scan = function () {
                scanForZoom(W.map.getZoom());
            };

            var action = new ActionHelper();
            var rules = new Rules();
            var ui = new Ui();

            analyzer.setRules(rules);
            analyzer.setActionHelper(action);

            action.setUi(ui);

            analyzer.onExceptionAdd(function (name) {
                ui.addException(name, function (index) {
                    if (confirm('Delete exception for ' + name + '?')) {
                        analyzer.removeException(index);
                    }
                });
            });

            analyzer.onExceptionDelete(function (index) {
                ui.removeException(index);
            });

            //        rules.experimental = true;

            rules.onAdd(function (rule) {
                ui.addCustomRule(rule.comment);
            });

            rules.onEdit(function (index, rule) {
                ui.updateCustomRule(index, rule.comment);
            });

            rules.onDelete(function (index) {
                ui.removeCustomRule(index);
            });

            //W.model.events.register('mergeend', null, function () {
            //    var name = W.model.getTopCountry().getName();
            //    if (name != currentCountry) {
            //        rules.onCountryChange(name);
            //        currentCountry = name;
            //    }
            //});

            analyzer.loadExceptions();
            rules.load();

            this.start = function () {
                ui.enableCheckbox().change(function () {
                    if (this.checked) {
                        ui.showMainWindow();

                        var savedVariant = localStorage.getItem('assist_variant');
                        if (savedVariant !== null) {
                            ui.variantRadio(savedVariant).prop('checked', true);
                            analyzer.setVariant(savedVariant);
                        }

                        scan();
                        W.model.events.register('mergeend', null, scan);
                    } else {
                        W.model.events.unregister('mergeend', null, scan);
                        ui.hideMainWindow();
                    }
                });

                ui.skipAltCheckbox().change(function () {
                    localStorage.setItem('assist_skip_alt', this.checked);
                    ui.scanAreaBtn().click();
                });

                ui.moveOld2AltCheckbox().change(function () {
                    localStorage.setItem('assist_move_old_to_alt', this.checked);
                    if (this.checked) {
                        // force enable skip alt option
                        localStorage.setItem('assist_skip_alt', true);
                        ui.skipAltCheckbox().prop('checked', true);
                        ui.skipAltCheckbox().prop('disabled', true);
                    } else {
                        // unblock skip alt option
                        ui.skipAltCheckbox().prop('disabled', false);
                    }
                });

                ui.enableCustomRulesCheckbox().change(function () {
                    localStorage.setItem('assist_enable_custom_rules', this.checked);
                    ui.scanAreaBtn().click();
                });

                ui.variantRadio().change(function (e) {
                    if (e.currentTarget.checked) {
                        localStorage.setItem('assist_variant', this.value);

                        analyzer.setVariant(this.value);
                        ui.scanAreaBtn().click();
                    }
                });

                if (localStorage.getItem('assist_enabled') == 'true') {
                    ui.enableCheckbox().click();
                }
                if (localStorage.getItem('assist_skip_alt') == 'true') {
                    ui.skipAltCheckbox().click();
                }
                if (localStorage.getItem('assist_move_old_to_alt') == 'true') {
                    ui.moveOld2AltCheckbox().click();
                }
                if (localStorage.getItem('assist_enable_custom_rules') == 'true') {
                    ui.enableCustomRulesCheckbox().click();
                }

                ui.fixAllBtn().click(function () {
                    ui.fixAllBtn().hide();
                    ui.fixSelectedBtn().hide();
                    ui.scanAreaBtn().hide();
                    ui.clearFixedBtn().hide();
                    ui.clearAllBtn().hide();

                    W.model.events.unregister('mergeend', null, scan);

                    setTimeout(function () {
                        analyzer.fixAll(function (id) {
                            ui.setUnresolvedErrorNum(analyzer.unresolvedErrorNum());
                            ui.setFixedErrorNum(analyzer.fixedErrorNum());
                            ui.moveToFixedList(id);
                        }, function () {
                            ui.fixAllBtn().show();
                            ui.fixSelectedBtn().show();
                            ui.scanAreaBtn().show();
                            ui.clearFixedBtn().show();
                            ui.clearAllBtn().show();

                            W.model.events.register('mergeend', null, scan);
                        });
                    }, 0);
                });

                ui.fixSelectedBtn().click(function () {
                    ui.fixAllBtn().hide();
                    ui.fixSelectedBtn().hide();
                    ui.scanAreaBtn().hide();
                    ui.clearFixedBtn().hide();
                    ui.clearAllBtn().hide();

                    W.model.events.unregister('mergeend', null, scan);

                    var listToFix = ui.getCheckedItemsList();

                    setTimeout(function () {
                        analyzer.fixSelected(listToFix, function (id) {
                            ui.setUnresolvedErrorNum(analyzer.unresolvedErrorNum());
                            ui.setFixedErrorNum(analyzer.fixedErrorNum());
                            ui.moveToFixedList(id);
                        }, function () {
                            ui.fixAllBtn().show();
                            ui.fixSelectedBtn().show();
                            ui.scanAreaBtn().show();
                            ui.clearFixedBtn().show();
                            ui.clearAllBtn().show();

                            W.model.events.register('mergeend', null, scan);
                        });
                    }, 0);
                });

                ui.clearFixedBtn().click(function () {
                    ui.fixedList().empty();
                });

                ui.clearAllBtn().click(function () {
                    ui.fixedList().empty();
                    ui.unresolvedList().empty();

                    analyzer.reset();

                    ui.setUnresolvedErrorNum(0);
                    ui.setFixedErrorNum(0);
                });

                ui.selectAllChk().change(function () {
                    var allChecked = this.checked;
                    ui.unresolvedList().find('input').each(function () {
                        this.checked = allChecked;
                    });
                });

                ui.scanAreaBtn().click(function () {
                    ui.fixedList().empty();
                    ui.unresolvedList().empty();

                    analyzer.reset();

                    ui.setUnresolvedErrorNum(0);
                    ui.setFixedErrorNum(0);

                    fullscan();
                });

                ui.addCustomRuleBtn().click(function () {
                    ui.customRuleDialog('Add', {
                        oldname: '',
                        newname: ''
                    }).done(function (response) {
                        rules.push(response.oldname, response.newname);
                    });
                });

                ui.editCustomRuleBtn().click(function () {
                    var id = ui.selectedCustomRule();
                    if (id >= 0) {
                        ui.customRuleDialog('Edit', {
                            oldname: rules.get(id).oldname,
                            newname: rules.get(id).newname
                        }).done(function (response) {
                            rules.update(id, response.oldname, response.newname);
                        });
                    } else {
                        alert('Custom rule is not selected');
                    }
                });

                ui.delCustomRuleBtn().click(function () {
                    var id = ui.selectedCustomRule();
                    if (id >= 0) {
                        rules.remove(id);
                    } else {
                        alert('Custom rule is not selected');
                    }
                });

                window.assist = this;
            };
        };

        function readyFunc() {
            requestRules(function () {
                info("Ready to work!");
                var app = new Application();
                app.start();
            });
        }

        if (W?.userscripts?.state.isReady) {
            readyFunc();
        } else {
            document.addEventListener("wme-ready", readyFunc, {
                once: true,
            });
        }

    }

    run_wme_assist();

})();