NCC City Mapper

Displays city names in the WME map

// ==UserScript==
// @name         NCC City Mapper
// @namespace    https://www.waze.com/user/editor/B4ckTrace
// @version      1.1
// @description  Displays city names in the WME map
// @author       B4ckTrace
// @include      /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/
// @grant        none
// ==/UserScript==

/* global W */
/* global OL */
/* global $ */
/* global WazeWrap */

var nccmapperIcon = '';

(function() {
    'use strict';

    var settings = {};
    var wmeNccLayer;
	var blDraw = false;

    function bootstrap(tries) {
        tries = tries || 1;

        if (W &&
			W.map &&
            W.model &&
			W.loginManager.user && $ 
			) {
            init();
        } else if (tries < 1000)
            setTimeout(function () {bootstrap(tries++);}, 200);
    }

    function init()
    {
        console.log("WME NCC City Mapper Init");
        loadSettings();
		debugger;
		var oldTogglers = document.querySelector('.togglers');
/* 		if (oldTogglers.querySelector('.layer-switcher-group_scripts') === null)
		{
            var newScriptsToggler = document.createElement('li');
            newScriptsToggler.className = 'group';
            newScriptsToggler.innerHTML = '<div class="controls-container main toggler">\
																		          <input class="layer-switcher-group_scripts toggle" id="layer-switcher-group_scripts" type="checkbox">\
																		          <label for="layer-switcher-group_scripts">\
																		          <span class="label-text">Scripts</span>\
																		          </label>\
																			      </div>\
																			      <ul class="children">\
										                        </ul>';
            oldTogglers.appendChild(newScriptsToggler);
        } */
		
		// Create toggler
        var newToggler = document.createElement('li');
        newToggler.innerHTML = '<div class="wz-checkbox">\
                                        <input class="layer-switcher-item_ncc_mapper toggle" id="layer-switcher-item_ncc_mapper" type="checkbox">\
                                        <label for="layer-switcher-item_ncc_mapper">\
                                          '+ 'مکانهای پایگاه ملی' +'\
                                        </label>\
                                      </div>';


        var groupDisplay = document.querySelector('.collapsible-GROUP_DISPLAY');
        groupDisplay.appendChild(newToggler);
		  
		var toggler = getId('layer-switcher-item_ncc_mapper');
		toggler.checked = settings.Enabled;
		
		// Add new tab item
		var userTabs = getId('user-info');
		var navTabs = getElementsByClassName('nav-tabs', userTabs)[0];
		var tabContent = getElementsByClassName('tab-content', userTabs)[0];
		var newtab = document.createElement('li');
		newtab.innerHTML = '<a href="#sidepanel-ncccitymapper" data-toggle="tab"><img style="padding-bottom: 6px;width: 18px;padding-right: 1px;" id="nccmapperIcon" class="nccmapperIconClass" src="' + nccmapperIcon + '">NCC Mapper</a>';
		var addon = document.createElement('section');
		addon.id = "sidepanel-ncccitymapper";
		addon.className = "tab-pane";
		var section = document.createElement('p');
		section.style.paddingTop = "0px";
		section.id = "ncccitymapper-box";
		section.className = 'input';
		section.innerHTML  = '<div><fieldset style="border: 1px solid silver; padding: 8px; border-radius: 4px;">'
							 + '<legend style="margin-bottom: 0px; border-bottom-style:none; width: auto;"><h4>Paste csv content here:</h4></legend>'
							 + '<textarea id="ncccitymapper-textarea" style="width:265px; height:120px;" placeholder="Paste here..."></textarea></br>'
							 + '<button id="ncccitymapper-NCCLOADBTN">Find cities</button>'
							 + '</fieldset></div>'
							 + '</div>'
		
		addon.appendChild(section);
		navTabs.appendChild(newtab);
		tabContent.appendChild(addon);
		
		
		onLayerCheckboxChanged(settings.Enabled);
		
		// Button handler
		getId('ncccitymapper-NCCLOADBTN').onclick = LOADBTN_Click;
		toggler.addEventListener('click', function(e) {
              onLayerCheckboxChanged(e.target.checked);
			  drawLines();
        });
    }
	
    function onLayerCheckboxChanged(checked) {
        settings.Enabled = checked;
        if (wmeNccLayer) {
            wmeNccLayer.setVisibility(settings.Enabled);
        }
        if (settings.Enabled)
        {
            if (!wmeNccLayer) {
                wmeNccLayer = new OL.Layer.Vector("wmeNccLayer", {uniqueName: "__wmeNccLayer"});
                W.map.addLayer(wmeNccLayer);
				blDraw = false;
            }
            W.map.events.register("moveend", W.map, drawLines);
            //drawLines();
        } else {
            W.map.events.unregister("moveend", W.map, drawLines);
            if (wmeNccLayer) {
                wmeNccLayer.removeAllFeatures();
                W.map.removeLayer(wmeNccLayer);
                wmeNccLayer = null;
            }
        }
        saveSettings();
    }
	
	function LOADBTN_Click()
	{
		if (wmeNccLayer) {
			blDraw = false;
			drawLines();
		}
	}

    function drawLines()
    {
		if (blDraw)
			return;
		
        wmeNccLayer.removeAllFeatures();

        var lineWidth = 2;
        var lineColor = 'gray';
        if (W.map.getZoom() <= 1)
        {
            lineWidth = 1;
        }
        else if (W.map.getZoom() >= 3)
        {
            lineColor = '#EDEDED';
        }
		
		var myVillages = getId('ncccitymapper-textarea');
		if (myVillages == null)
			return;
		
		myVillages = myVillages.value;
		if (myVillages.length<10)
			return;
		
		myVillages = myVillages.trim();
		var lines = myVillages.split('\n');
		for(var line = 0; line < lines.length; line++){
			var elements = lines[line].split(',');
			var type = elements[elements.length-3];
			if (type == 'شهر' || type == 'محله' || type == 'روستا، آبادی' || type == 'بخش' || type == 'دهستان' || type == 'شهرك' || type == 'شهرستان')
			{
				var dashStyle = 'solid';
				switch(type) {
				  case 'شهرستان':
					dashStyle = 'solid'
					break;
				  case 'بخش':
				    lineWidth = 3;
					dashStyle = 'dot'
					break;
				  case 'شهر':				    
					dashStyle = 'longdashdot'
					break;
				  case 'شهرك':
					dashStyle = 'dot'
					break;
				  case 'محله':
					dashStyle = 'dashdot'
					break;
				  case 'روستا، آبادی':
					dashStyle = 'dash'
					break;
				  case 'دهستان':
					dashStyle = 'longdash'
					break;
				  default:
				    lineWidth = 2;
					dashStyle = 'solid';
				}

				var lon = elements[elements.length-1];
				var lat = elements[elements.length-2];
				
				var ColorCode = '#' + (0x1000000+(Math.random())*0xffffff).toString(16).substr(1,6);
				lineColor =  invertColor(ColorCode, true);
				
				var pointEnd = convertCoordToPoint([lon, lat]);
				var pointFeature = new OL.Feature.Vector(pointEnd); // End point
				pointFeature.style = {
						pointRadius: 6,
						fillColor: 'white',
						fillOpacity: 1,
						strokeColor: ColorCode,
						strokeWidth: lineWidth+1,
						strokeLinecap: 'round',
						strokeDashstyle: 'dash',
						fontFamily: "arial, monospace",
						fontSize: '12pt',
						fontWeight: "bold",
						fontColor: ColorCode,
						labelOutlineColor: lineColor,
						labelOutlineWidth: lineWidth,
						label : elements[1] + '(' + (line) + ')\n' + type,
						labelAlign: "rt"//set to top right
					};
				wmeNccLayer.addFeatures([pointFeature]);
			} // End of if
		} // End of for
		
		blDraw = true;
    }
	
	function getElementsByClassName(classname, node) {
	  if(!node) node = document.getElementsByTagName("body")[0];
	  var a = [];
	  var re = new RegExp('\\b' + classname + '\\b');
	  var els = node.getElementsByTagName("*");
	  for (var i=0,j=els.length; i<j; i++)
		if (re.test(els[i].className)) a.push(els[i]);
	  return a;
	};
	function getId(node) {
	  return document.getElementById(node);
	};

    function saveSettings() {
        if (localStorage) {
            var localsettings = {
                Enabled: settings.Enabled
            };

            localStorage.setItem("wmeNcc_Settings", JSON.stringify(localsettings));
        }
    }

    function loadSettings() {
        var loadedSettings = $.parseJSON(localStorage.getItem("wmeNcc_Settings"));
        var defaultSettings = {
            Enabled: true
        };
        settings = loadedSettings ? loadedSettings : defaultSettings;
        for (var prop in defaultSettings) {
            if (!settings.hasOwnProperty(prop))
                settings[prop] = defaultSettings[prop];
        }
    }
	
	function invertColor(hex, bw) {
		if (hex.indexOf('#') === 0) {
			hex = hex.slice(1);
		}
		// convert 3-digit hex to 6-digits.
		if (hex.length === 3) {
			hex = hex[0] + hex[0] + hex[1] + hex[1] + hex[2] + hex[2];
		}
		if (hex.length !== 6) {
			throw new Error('Invalid HEX color.');
		}
		var r = parseInt(hex.slice(0, 2), 16),
			g = parseInt(hex.slice(2, 4), 16),
			b = parseInt(hex.slice(4, 6), 16);
		if (bw) {
			return (r * 0.299 + g * 0.587 + b * 0.114) > 186
				? '#000000'
				: '#FFFFFF';
		}
		// invert color components
		r = (255 - r).toString(16);
		g = (255 - g).toString(16);
		b = (255 - b).toString(16);
		// pad each with zeros and return
		return "#" + padZero(r) + padZero(g) + padZero(b);
	}
	function padZero(str, len) {
		len = len || 2;
		var zeros = new Array(len).join('0');
		return (zeros + str).slice(-len);
	}
	
	function convertCoordToPoint(coordinates) {
		return window.OL.Projection.transform(
			new window.OL.Geometry.Point(
				coordinates[0],
				coordinates[1]
			),
			"EPSG:4326",
			wmeNccLayer.projection.projCode
		);
	}


    bootstrap();
})();