Save, and load, restrictions from local storage.
/* global $ */ /* global I18n */ /* global W */ // ==UserScript== // @name Restriction Manager // @version 1.6 // @description Save, and load, restrictions from local storage. // @namespace mailto:[email protected] // @include https://www.waze.com/editor // @include https://www.waze.com/editor* // @include https://www.waze.com/*/editor* // @include https://beta.waze.com/* // @exclude https://www.waze.com/user/* // @exclude https://www.waze.com/*/user/* // @icon  // @resource jqUI_CSS https://ajax.googleapis.com/ajax/libs/jqueryui/1.11.4/themes/smoothness/jquery-ui.css // @grant none // @copyright 2018, kjg53 // @author kjg53 // @license MIT // ==/UserScript== (function() { var initialized = false; var lsPrefix = "rtmgr:"; // Map the css class used to identify the three restriction blocks to the direction constants used in the data. var classToDirection={"forward-restrictions-summary": "FWD", "reverse-restrictions-summary": "REV", "bidi-restrictions-summary": "BOTH"}; // Convert the segment's default type to the driving modality that implied the type. Creating a Toll Free restriction // implies that the segment is otherwise (i.e. defaults) to tolled which is what is then stored in the model. // Finding the tolled default thereby implies that the current restriction is specifying a toll free rule. var defaultType2drivingModality = {"TOLL": "DRIVING_TOLL_FREE", "FREE":"DRIVING_BLOCKED", "BLOCKED":"DRIVING_ALLOWED"}; // Map the single bit constants used in the weekdays property to the integer numbers encoded in each week days HTML display. var weekdayBit2Idx = { 1:1, 2:2, 4:3, 8:4, 16:5, 32:6, 64:0 }; var clipboardTarget = "*Clipboard*"; // Get a sorted list of saved restrictions found in local storage. function allSavedRestrictions() { var all = []; for(var i = 0; i < localStorage.length; i++) { var key = localStorage.key(i); if (key.indexOf(lsPrefix) == 0) { key = key.substring(lsPrefix.length); all.push(key); } } all.sort(); return all; } function extractRestrictions() { var extracted = {}; for(var i = 0; i < localStorage.length; i++) { var key = localStorage.key(i); if (key.indexOf(lsPrefix) == 0) { var name = key.substring(lsPrefix.length); extracted[name] = localStorage.getItem(key); } } return extracted; } // Convert list of saved restrictions into a string of HTML option elements. function allSavedRestrictionAsOptions() { var all = allSavedRestrictions(); return all.length == 0 ? "" : "<option selected></option><option>" + all.join("</option><option>") + "</option>"; } // Update all restriction selectors to display the saved restrictions returned by allSavedRestrictions function updateSavedRestrictionSelectors(root) { $("div.rtmgr div.name select", root).html(allSavedRestrictionAsOptions()).each(resizeDivName); } // The content of the div.name element are positioned relative to its location. As a r###lt, the // div normally collapses to a point in the screen layout. This function expands the div to enclose // its contents such that other elements are laid out around them. function resizeDivName(idx, child) { var div = $(child).parents("div.name").first(); var height = 0; var width = 0; div.children().each(function(idx, child) { child = $(child); height = Math.max(height, child.height()); width = Math.max(width, child.width()); }); div.width(width).height(height); } // Identify the direction of the restrictions associated with the specified button. function direction(btn) { var classes = btn.parents("div.restriction-summary-group").attr('class').split(' '); while(classes.length) { var cls = classes.pop(); var dir = classToDirection[cls]; if (dir) { return dir; } } } function setValue(selector, model, value) { if (value != null) { var sel = $(selector, model); var oldValue = sel.val(); if (oldValue != value) { sel.val(value); sel.change(); } } } function setCheck(selector, model, value) { if (value != null) { value = !!value; var sel = $(selector, model); var oldValue = sel.prop('checked'); if (oldValue != value) { sel.prop('checked', value); sel.change(); } } } function setSelector(name, model, value) { setValue('select[name="' + name + '"]', model, value); } function lastFaPlus(modal) { return $("i.fa-plus", modal).last(); } function clearMessages() { $("div.modal-header-messages div.rtmgr").remove(); } function addMessage(text, icon, color) { var rvr = $("div.modal-header-messages"); if (icon) { icon = '<i class="fa fa-' + icon + '"/> '; } else { icon = ""; } if (color) { color = ' style="color: ' + color + '"'; } else { color = ""; } rvr.append('<div class="modal-header-message"' + color + '>' + icon + text + '</div>'); } var timeRegexp = /(\d\d?):(\d\d?)/ function time2Int(time, ifNull) { var m = timeRegexp.exec(time); return m == null ? -1 : (m[1] * 60) + m[2]; } function compareTimeFrames(a, b) { if (a == null) { return (b == null ? 0 : -1); } else if (b == null) { return 1; } else { a = a[0]; b = b[0]; var c = time2Int(a.fromTime, -1) - time2Int(b.fromTime, -1); if (c == 0) { c = time2Int(a.toTime, 1440) - time2Int(b.toTime, 1440); } return c; } } function compareRestrictions(a, b) { var c = compareTimeFrames(a.timeFrames, b.timeFrames); if (c == 0) { c = a.defaultType.localeCompare(b.defaultType); } return c; } function handleImportDragOver(evt) { evt.stopPropagation(); evt.preventDefault(); evt.dataTransfer.dropEffect = "copy"; } function handleImportFile(file) { var reader = new FileReader(); reader.onload = function(e) { var restrictions = JSON.parse(e.target.r###lt); for(var restriction in restrictions) { localStorage.setItem(lsPrefix + restriction, restrictions[restriction]); } }; reader.readAsText(file); } function handleImportDrop(evt) { evt.stopPropagation(); evt.preventDefault(); var i, f, item, seen = {}; for(i = 0; item = evt.dataTransfer.items[i]; i++) { if (item.kind === 'file') { f = item.getAsFile(); handleImportFile(f); seen[f.name] = true; } } for(i = 0; f = evt.dataTransfer.files[i]; i++) { if (seen[f.name] !== true) { handleImportFile(f); } } } function initializeRestrictionManager() { if (initialized) { return; } var observerTarget = document.getElementById("dialog-region"); if (!observerTarget) { window.console.log("Restriction Manager: waiting for WME..."); setTimeout(initializeRestrictionManager, 1015); } // Inject my stylesheet into the head var sheet = $('head').append('<style type="text/css"/>').children('style').last(); sheet.append('div.rtmgr-column {display: flex; flex-direction: column; align-items: center}'); sheet.append('div.rtmgr-row {display: flex; flex-direction: row; justify-content: space-around}'); sheet.append('div.rtmgr button.btn {margin-top: 5px; border-radius: 40%}'); sheet.append('div.rtmgr div.name input {width: 250px; position: absolute; left: 0px; top: 0px; z-index: 1}'); sheet.append('div.rtmgr div.name select {width: 275px; position: absolute; left: 0px; top: 0px}'); sheet.append('div.rtmgr div.name {width: 275px; position: relative; left: 0px; top: 0px}'); sheet.append('div.modal-dialog div.modal-header .modal-title img.icon {float:right;height:20px;width:20px}'); sheet.append('dialog.rtmgr span.cmd {text-decoration: underline}'); sheet.append('dialog.rtmgr .import, dialog.rtmgr .export {width:30px; height: 30px}'); // create an observer instance var observer = new MutationObserver(function(mutations) { var si = W.selectionManager.getSelectedFeatures(); mutations.forEach(function(mutation) { if("childList" == mutation.type && mutation.addedNodes.length) { var restrictionsModal = $("div.modal-dialog.restrictions-modal", observerTarget); if (restrictionsModal) { var modalTitle = $(restrictionsModal).find("div.modal-header .modal-title").first(); var title = modalTitle.text().trim(); if (I18n.translations[I18n.locale].restrictions.modal_headers.restriction_summary == title) { if (modalTitle.data('rtmgr') === undefined) { // Flag this modal as having already augmented modalTitle.data('rtmgr', true); modalTitle.append("<img src='' class='icon'>" + "<dialog class='rtmgr'>" + "<p style='text-align: center; font-weight: bold; font-size: 1.3em'>Restriction Manager</p>" + "<p style='font-style: italic; font-size: .7em'>Stores restrictions in your browser's local storage so that they may be easily applied to other segments.</p>" + "<p style='padding-left: 3em; text-indent: -3em;'>" + "<span class='cmd'>Save</span>: Saves these restrictions to the selected key.<br>" + "<span style='font-style: italic; font-size: .7em'>Note: Newly edited restrictions must be applied to the segment before the manager can save them.</span>" + "</p>" + "<p>" + "<span class='cmd'>Apply</span>: Replaces the current restrictions with the restrictions associated with the selected key.</p>" + "<p>" + "<span class='cmd'>Delete</span>: Delete the selected key from your browser's local storage.</p>" + "</dialog>"); if (window.File && window.FileReader && window.FileList && window.Blob) { $('dialog.rtmgr', modalTitle) .append("<p>Offline Storage: " + "<a download='restrictions.txt'><img title='Click here to export all saved restrictions' src='' class='export' download='restrictions.txt'></a>" + "<img title='Drag file here to import restrictions' src='' class='import'>" + "</p>"); } $('dialog.rtmgr', modalTitle).append("<p style='text-align: right; font-style: italic; font-size: .5em'>Click anywhere to close.</p>"); $("img.icon", modalTitle).click(function(evt) { $("dialog.rtmgr", modalTitle)[0].showModal(); }); $("dialog.rtmgr", modalTitle).click(function(evt) { evt.currentTarget.close(); }); if (window.File && window.FileReader && window.FileList && window.Blob) { $("img.export", modalTitle).click(function(evt) { var data = JSON.stringify(extractRestrictions()); evt.target.parentNode.href = "data:text/plain;base64," + btoa(data); }); var imgImport = $("img.import", modalTitle)[0]; imgImport.addEventListener('dragover', handleImportDragOver, false); imgImport.addEventListener('drop', handleImportDrop, false); } // Add the UI elements to the modal // Original, single, jquery statement split up after it stopped working. var restrictionSummaryGroups = $("div.restriction-summary-group div.restriction-summary-title", restrictionsModal); restrictionSummaryGroups.before ("<div class='rtmgr rtmgr-column'>" + "<div class='name'>" + "<input type='text'/>" + "<select/>" + "</div>" + "</div>"); var rtmgrColumns = restrictionSummaryGroups.parent().children("div.rtmgr-column"); rtmgrColumns.append ("<div class='rtmgr-row'>" + "<button class='btn save'>Save</button>" + "<button class='btn apply'>Apply</button>" + "<button class='btn delete'>Delete</button>" + "</div>"); // Initialize the saved restriction selectors updateSavedRestrictionSelectors(restrictionsModal); // When a selection is made copy it to the overlapping input element to make it visible. $("div.rtmgr select").change(function(evt) { var tgt = evt.target; var txt = $(tgt).parent().children("input"); var opt = tgt.options[tgt.selectedIndex]; txt.val(opt.text); $(opt).prop('selected', false); $(opt).parent().children("option:first").prop("selected", "selected"); }); // Delete action $("div.rtmgr button.delete", restrictionsModal).click(function(evt) { var tgt = $(evt.target); var inp = tgt.parents('div.rtmgr').find("input"); var name = inp.val(); if (name == "") { addMessage("Specify the name of the restrictions being deleted.", 'ban', 'red'); } else { localStorage.removeItem(lsPrefix + name); updateSavedRestrictionSelectors(restrictionsModal); inp.val(""); } }); // Save action (only one segment currently selected) if (si.length == 1) { $("div.rtmgr button.save", restrictionsModal).click(function(evt) { var tgt = $(evt.target); var input = tgt.parents('div.rtmgr').find("input"); var name = input.val(); if (name == "") { addMessage("The restrictions require a name before they can be saved.", 'ban', 'red'); } else { var dir = direction(tgt); var attrs = si[0].model.attributes; var src = attrs.restrictions; // Checking for pending updates to the selected segment's restrictions. If found, save a copy of them. // This is a convenience feature that enables an editor to Apply a restriction change to a segment and then store it for re-use without first having to save it on the original segment. var actions = W.model.actionManager.getActions(); for(var i = actions.length; i-- > 0;) { var action = actions[i]; if (action.model.hasOwnProperty('subActions') && action.subActions[0].attributes.id == si[0].model.attributes.id && action.subActions[0].newAttributes.hasOwnProperty('restrictions')) { src = action.subActions[0].newAttributes.restrictions; break; } } var restrictions = []; for (i = 0; i< src.length; i++) { var restriction = src[i]; if (restriction._direction == dir) { restrictions.push(restriction); } } restrictions = JSON.stringify(restrictions); clearMessages(); if (clipboardTarget == name) { input.val(restrictions).select(); document.execCommand('copy'); input.val(clipboardTarget).blur(); addMessage("Restrictions copied to clipboard"); } else { localStorage.setItem(lsPrefix + name, restrictions); addMessage("Restrictions saved to " + name); } input.val(""); updateSavedRestrictionSelectors(restrictionsModal); } }); } else { $("div.rtmgr button.save", restrictionsModal).click(function(evt) { clearMessages(); addMessage("Save is only enabled when displaying the restrictions for a SINGLE segment", 'ban', 'red'); }); } // Apply saved restrictions to the current segment $("div.rtmgr button.apply", restrictionsModal).click(function(evt) { var tgt = $(evt.target); var input = tgt.parents('div.rtmgr').find("input"); var name = input.val().trim(); if (name == "") { addMessage("Specify the name of the restrictions being applied.", 'ban', 'red'); } else { var restrictions; input.val(""); if (name.startsWith('[') & name.endsWith(']')) { restrictions = name; } else { restrictions = localStorage.getItem(lsPrefix + name); } restrictions = JSON.parse(restrictions).sort(compareRestrictions); var rsg = $(evt.target).parents("div.restriction-summary-group").first(); var classes = rsg.attr('class').split(' '); classes.splice(classes.indexOf('restriction-summary-group'), 1); // Delete all current restrictions associated with the action's direction while (true) { var doDelete = "." + classes[0] + " .restriction-editing-actions i.do-delete"; var deleteRestrictions = $(doDelete, restrictionsModal); if (deleteRestrictions.length == 0) { break; } deleteRestrictions.eq(0).click(); } // Create new restrictions while (restrictions.length) { var restriction = restrictions.shift(); $("." + classes[0] + " button.do-create", restrictionsModal).click(); setSelector('disposition', restrictionsModal, restriction.disposition); setSelector('laneType', restrictionsModal, restriction.laneType); setValue('textarea[name="description"]', restrictionsModal, restriction.description); if (restriction.timeFrames != null && restriction.timeFrames.length != 0) { var weekdays = restriction.timeFrames[0].weekdays; var bit = 1; for(var idx = 0; idx < 7; idx++) { var set = weekdays & bit; set = (set != 0); setCheck('input#day-ordinal-' + weekdayBit2Idx[bit] + '-checkbox', restrictionsModal, set); bit <<= 1; } if (restriction.timeFrames[0].fromTime && restriction.timeFrames[0].toTime) { setCheck("input#is-all-day-checkbox", restrictionsModal, false); setValue("input.timepicker-from-time", restrictionsModal, restriction.timeFrames[0].fromTime); setValue("input.timepicker-to-time", restrictionsModal, restriction.timeFrames[0].toTime); } if (restriction.timeFrames[0].startDate && restriction.timeFrames[0].endDate) { setCheck("input#is-during-dates-on-radio", restrictionsModal, true); // Ref: http://www.daterangepicker.com/ var drp = $('input.btn.datepicker', restrictionsModal).data('daterangepicker'); var re = /(\d{4})-(\d{2})-(\d{2})/; var match = re.exec(restriction.timeFrames[0].startDate); var startDate = match[2] + "/" + match[3] + "/" + match[1]; match = re.exec(restriction.timeFrames[0].endDate); var endDate = match[2] + "/" + match[3] + "/" + match[1]; // WME's callback is fired by drp.hide(). drp.show(); drp.setStartDate(startDate); drp.setEndDate(endDate); drp.hide(); setCheck("input#repeat-yearly-checkbox", restrictionsModal, restriction.timeFrames[0].repeatYearly); } } var drivingModality; // if ALL vehicles are blocked then the default type is simply BLOCKED and the modality is blocked. if ("BLOCKED" == restriction.defaultType && !restriction.driveProfiles.hasOwnProperty("FREE") && !restriction.driveProfiles.hasOwnProperty("BLOCKED")) { drivingModality = "DRIVING_BLOCKED"; } else { drivingModality = defaultType2drivingModality[restriction.defaultType]; } setValue("select.do-change-driving-modality", restrictionsModal, drivingModality); var driveProfiles, driveProfile, i, j, vehicleType, plus, driveProfileItem, subscription; if (restriction.driveProfiles.hasOwnProperty("FREE")) { driveProfiles = restriction.driveProfiles.FREE; for(i = 0; i < driveProfiles.length; i++) { driveProfile = driveProfiles[i]; var numDriveProfileItems = Math.max(1, driveProfile.vehicleTypes.length); for(j = 0; j < numDriveProfileItems; j++) { $("div.add-drive-profile-item.do-add-item", restrictionsModal).click(); vehicleType = driveProfile.vehicleTypes[j]; if (vehicleType !== undefined) { plus = lastFaPlus(restrictionsModal); plus.click(); driveProfileItem = plus.parents("div.drive-profile-item"); $("div.btn-group.open a.do-init-vehicle-type", driveProfileItem).click(); $("div.vehicle-type span.restriction-chip-content", driveProfileItem).click(); $('a.do-set-vehicle-type[data-value="' + vehicleType + '"]', driveProfileItem).click(); } if (driveProfile.numPassengers > 0) { plus = lastFaPlus(restrictionsModal); plus.click(); driveProfileItem = plus.parents("div.drive-profile-item"); $("div.btn-group.open a.do-init-num-passengers", driveProfileItem).click(); if (driveProfile.numPassengers > 2) { $("a.do-set-num-passengers[data-value='" + driveProfile.numPassengers + "']", driveProfileItem).click(); } } for(var k = 0; k < driveProfile.subscriptions.length; k++) { subscription = driveProfile.subscriptions[k]; plus = lastFaPlus(restrictionsModal); plus.click(); driveProfileItem = plus.parents("div.drive-profile-item"); $("div.btn-group.open a.do-init-subscription", driveProfileItem).click(); $("div.subscription span.restriction-chip-content", driveProfileItem).click(); $('a.do-set-subscription[data-value="' + subscription + '"]', driveProfileItem).click(); } } } } else if (restriction.driveProfiles.hasOwnProperty("BLOCKED")) { driveProfiles = restriction.driveProfiles.BLOCKED; for(i = 0; i < driveProfiles.length; i++) { driveProfile = driveProfiles[i]; if (driveProfile.vehicleTypes.length > 0) { setCheck('input#all-vehicles-off-radio', restrictionsModal, true); for(j = 0; j < driveProfile.vehicleTypes.length; j++) { vehicleType = driveProfile.vehicleTypes[j]; setCheck('input#vehicle-type-' + vehicleType + '-checkbox', restrictionsModal, true); } } } } $("div.modal-footer button.do-create", restrictionsModal).click(); } clearMessages(); addMessage("Restrictions from " + name + " applied to current selection."); } }); } } } } }); }); // configuration of the observer: var config = { attributes: false, childList: true, characterData: false, subtree: true }; // pass in the target node, as well as the observer options observer.observe(observerTarget, config); initialized = true; } setTimeout(initializeRestrictionManager, 1000); })();