// ==UserScript==
// @name WME URComments-Enhanced
// @namespace https://greasyfork.org/users/166843
// @version 2024.11.26.02
// eslint-disable-next-line max-len
// @description URComments-Enhanced (URC-E) allows Waze editors to handle WME update requests more quickly and efficiently. Also adds many UR filtering options, ability to change the markers, plus much, much, more!
// @grant GM_xmlhttpRequest
// @match *://*.waze.com/*editor*
// @exclude *://*.waze.com/user/editor*
// @require https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @author dBsooner
// @license MIT/BSD/X11
// @connect greasyfork.org
// @connect sheets.googleapis.com
// @icon 
// @contributionURL https://github.com/WazeDev/Thank-The-Authors
// ==/UserScript==
/* global unsafeWindow, GM_info, GM_xmlhttpRequest, I18n, OpenLayers, trustedTypes W, WazeWrap */
/*
* Original concept and code for URComments (URC) was written by rickzabel and licensed under MIT/BSD/X11.
* This script is a ground-up rewrite of URC. Special thanks is definitely given to rickzabel for his hard
* work and dedication to the original script. You can reach him at rickzabel@gmail.com.
*
*/
/*
* Portions of this script were inspired by URO+ written by Twister-UK. Credit is given to him and his team
* for development of that script. Where code was directly copied, function name or variable name was retained
* and a comment was placed before the code. The code that was inspired by was a complete rewrite using URO+
* as a reference for the logic. URO+ is located at: https://greasyfork.org/en/scripts/1952-uroverview-plus-uro
*
*/
(function () {
'use strict';
let _settings = {},
_selUr = {
doubleClick: false,
handling: false,
newStatus: undefined,
urId: -1,
urOpen: false
},
_restrictionsEnforce = {},
_restrictionsEnforcedTitle,
_commentList = [],
_commentListLoaded = false,
_customReplaceVars = [],
// 2024.03.21: Now that the UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085), don't have time to fix marker (un)stacking. Removing for now.
// _markerStackArray = [],
_currentCommentList = null,
_filtersAppliedOnZoom = false,
_initialUrLayerScan = false,
_markerCountOnInit = -1,
_mousedOverMarkerId = null,
_mouseIsDown = false,
_needTranslation = false,
// 2024.03.21: Now that the UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085), don't have time to fix marker (un)stacking. Removing for now.
// _unstackedMasterId = null,
_restoreZoom = null,
_restoreDrawerTab,
_restoreTab,
_restoreTabPosition,
_wmeUserId = null,
_initUrIdInUrlObserver,
// 2024.03.21: Let's just remove the needUrId safety net for now. This will keep the script "working" if something else breaks when looking for the URID.
// _needUrId = false,
_mapUpdateRequests = {},
_initError = false,
_policyTrustedHTML,
_policyUntrustedHTML;
// eslint-disable-next-line no-nested-ternary
const _SCRIPT_SHORT_NAME = `URC-E${(GM_info.script.name.includes('beta') ? ' β' : GM_info.script.name.includes('(DEV)') ? ' Ω' : '')}`,
_SCRIPT_LONG_NAME = GM_info.script.name,
_IS_ALPHA_VERSION = _SCRIPT_SHORT_NAME.includes('Ω'),
_IS_BETA_VERSION = _SCRIPT_SHORT_NAME.includes('β'),
_SCRIPT_AUTHOR = GM_info.script.author,
_PROD_DL_URL = 'https://greasyfork.org/scripts/375430-wme-urcomments-enhanced/code/WME-URComments-Enhanced.user.js',
_FORUM_URL = 'https://www.waze.com/forum/viewtopic.php?f=819&t=275608',
_SETTINGS_STORE_NAME = 'WME_URC-E',
_BETA_DL_URL = 'YUhSMGNITTZMeTluY21WaGMzbG1iM0pyTG05eVp5OXpZM0pwY0hSekx6TTNOelEyTkMxM2JXVXRkWEpqYjIxdFpXNTBjeTFsYm1oaGJtTmxaQzFpWlhSaEwyTnZaR1V2VjAxRkxWVlNRMjl0YldWdWRITXRSVzVvWVc1alpXUXVkWE5sY2k1cWN3PT0=',
_ALERT_UPDATE = true,
_SCRIPT_VERSION = GM_info.script.version.toString(),
_SCRIPT_VERSION_CHANGES = ['CHANGE: WME release vv2.261-6-gce1dd85be compatibility.'],
_MIN_VERSION_AUTOSWITCH = '2019.01.11.01',
_MIN_VERSION_COMMENTLISTS = '2018.01.01.01',
_MIN_VERSION_COMMENTS = '2019.03.01.01',
_MIN_VERSION_RESTRICTIONS = '2018.03.08.01',
_MIN_VERSION_TRANSLATIONS = '2019.08.16.01',
_DEBUG = _SCRIPT_SHORT_NAME.includes('β') || _SCRIPT_SHORT_NAME.includes('Ω') || false,
_LOAD_BEGIN_TIME = performance.now(),
_STATIC_ONLY_USERS = ['itzwolf'],
_URCE_API_KEY = 'UVVsNllWTjVRVEo0VDJWVlptOXdSSEZvUWpoeU9HVnpSV0V5UVMxSE1GZzJORlZOY2pGag==',
_URCE_SPREADSHEET_ID = 'TVdGV1MwSlBkMnBaYlU4NE9IZzVObVpKU0hSSlVXZEJkMDFoUTFaZlRtWnJiSFpRY1dZd1NqQndlbEU9',
_autoSwitch = {},
_commentLists = [],
_currentArea = { country: undefined, state: undefined },
_defaultComments = {
dr: { commentNum: null, urNum: 98 }, // Default reminder
dc: { commentNum: null, urNum: 99 }, // Default closed / not identified
it: { commentNum: null, urNum: 6 }, // Incorrect turn
ia: { commentNum: null, urNum: 7 }, // Incorrect address
ir: { commentNum: null, urNum: 8 }, // Incorrect route
mra: { commentNum: null, urNum: 9 }, // Missing roundabout
ge: { commentNum: null, urNum: 10 }, // General error
tna: { commentNum: null, urNum: 11 }, // Turn not allowed
ij: { commentNum: null, urNum: 12 }, // Incorrect junction
mbo: { commentNum: null, urNum: 13 }, // Missing bridge overpass
wdd: { commentNum: null, urNum: 14 }, // Wrong driving direction
me: { commentNum: null, urNum: 15 }, // Missing exit
mr: { commentNum: null, urNum: 16 }, // Missing road
ml: { commentNum: null, urNum: 18 }, // Missing landmark
br: { commentNum: null, urNum: 19 }, // Blocked road
msn: { commentNum: null, urNum: 21 }, // Missing street name
isps: { commentNum: null, urNum: 22 }, // Incorrect street prefix or suffix
sl: { commentNum: null, urNum: 23 } // Speed Limit
},
_elems = {
a: document.createElement('a'),
br: document.createElement('br'),
button: document.createElement('button'),
div: document.createElement('div'),
hr: document.createElement('hr'),
i: document.createElement('i'),
img: document.createElement('img'),
input: document.createElement('input'),
fieldset: document.createElement('fieldset'),
label: document.createElement('label'),
legend: document.createElement('legend'),
li: document.createElement('li'),
ol: document.createElement('ol'),
option: document.createElement('option'),
p: document.createElement('p'),
select: document.createElement('select'),
span: document.createElement('span'),
style: document.createElement('style'),
textarea: document.createElement('textarea'),
ul: document.createElement('ul')
},
_overflowUrsUrls = [],
_restrictions = {},
_spinners = {
buildCommentList: false,
changeCommentList: false,
checkMarkerStacking: false,
checkRestrictions: false,
handleAfterCommentMutation: false,
handleClickedShortcut: false,
handleUpdateRequestContainer: false,
handleUrLayer: false,
handleUrOverflow: false,
init: false,
markerMouseDown: false,
postUrComment: false
},
_timeouts = {
areTranslationsReady: undefined,
autoCloseUrPanel: undefined,
autoScrollComments: undefined,
checkForStaticListArray: undefined,
checkRestrictions: {},
getMapUrsAsync: {},
getOverflowUrsFromUrl: {},
initUrIdInUrl: undefined,
isDomElementReady: {},
isPanelReady: {},
onWmeReady: undefined,
popup: undefined,
popupDelay: undefined,
saveSettingsToStorage: undefined
},
_saveButtonObserver = new MutationObserver((mutations) => {
if ((W.model.actionManager._redoStack.length === 0)
&& mutations.some((mutation) => ((mutation.attributeName === 'disabled') && (mutation.oldValue === 'false') && (mutation.target.attributes.disabled.value === 'true')))
)
handleAfterSave();
}),
_urDataStateObserver = new MutationObserver((mutations) => {
if (_selUr.handling) {
const dataStateMutations = mutations.filter((mutation) => mutation.attributeName === 'data-state');
if (dataStateMutations.length > 0) {
const newDataState = dataStateMutations[0].target.attributes['data-state'].nodeValue.replaceAll('-', '');
if ((newDataState === 'open') || (newDataState === 'solved') || (newDataState === 'notidentified'))
_selUr.newStatus = newDataState;
else
logWarning(`INVALID DATA STATE CHANGE: ${dataStateMutations[0].target.attributes['data-state'].nodeValue}`);
}
}
}),
_urCommentsObserver = new MutationObserver((mutations) => {
if (_selUr.handling && (mutations[0].addedNodes.length > 0))
handleAfterCommentMutation(mutations[0].addedNodes[0]);
}),
_urPanelContainerObserver = new MutationObserver((mutations) => {
/**
* 2024.11.26
* Swapped the if and else if so that handleUpdateContainer goes first if it finds that show exists in the same mutation set as the removed item.
*/
if (mutations
.filter((mutation) => (mutation.type === 'attributes'))
.filter((mutation) => (mutation.oldValue?.includes('panel') && mutation.target.classList.contains('show')))
.length > 0
)
handleUpdateRequestContainer();
else if (_selUr.handling
&& (mutations
.filter((mutation) => (mutation.removedNodes.length > 0) && mutation.target.matches('#panel-container'))
.filter((removedChild) => removedChild.removedNodes[0].classList.contains('show') && removedChild.removedNodes[0].classList.contains('panel'))
.length > 0
)
)
handleAfterCloseUpdateContainer();
}),
_urMarkerObserver = new MutationObserver((mutations) => {
mutations.filter((mutation) => (mutation.addedNodes.length > 0) && !mutation.addedNodes[0].id.includes('-text')).forEach((newMarker) => {
const urceCounter = newMarker.addedNodes[0].id.includes('urceCounters');
if (!urceCounter && !newMarker.addedNodes[0].dataset.urceHasListeners) {
newMarker.addedNodes[0].addEventListener('mouseover', markerMouseOver);
newMarker.addedNodes[0].addEventListener('mouseout', markerMouseOut);
newMarker.addedNodes[0].dataset.urceHasListeners = true;
}
});
/** 2024.03.21: Let's just remove the needUrId safety net for now. This will keep the script "working" if something else breaks when looking for the URID.
* This mutation observer would not be efficient, even if it can do it at all, in finding the UR ID as the ID no longer exists in the element itself. You
* might be able to use the functions in W.userscripts to find the data model, but that would be expensive.
*
if (_needUrId) {
const urId = mutations.filter(
(mutation) => ((mutation.attributeName === 'class') && !mutation.oldValue.includes('marker-selected') && mutation.target.classList.contains('marker-selected'))
)[0]?.target.attributes['data-id']?.value;
if (urId > 0) {
_needUrId = false;
_selUr.urId = +urId;
logDebug(`Selected UR from marker mutation: ${_selUr.urId}`);
maskBoxes(undefined, true, 'needUrId', true);
handleUpdateRequestContainer();
}
}
*/
}),
_urceSidepanelContentObserver = new MutationObserver((mutations) => {
if (mutations.some((mutation) => mutation.target.classList.contains('active')))
checkSidebarHeight();
}),
_userscriptsApiDocsLinkIntersectionObserver = new IntersectionObserver((entries) => {
if (entries.some((entry) => entry.isIntersecting)) {
checkSidebarHeight();
_userscriptsApiDocsLinkIntersectionObserver.disconnect();
_userscriptsApiDocsLinkIntersectionObserver.isObserving = false;
}
}, { root: document.documentElement }),
_userInfoTabContentObserver = new MutationObserver((mutations) => {
const checkNodes = (node) => {
if (node.classList.contains('userscripts-api-docs-link-container') && !_userscriptsApiDocsLinkIntersectionObserver.isObserving) {
_userscriptsApiDocsLinkIntersectionObserver.observing = true;
_userscriptsApiDocsLinkIntersectionObserver.observe(node);
}
};
mutations.forEach((mutation) => {
mutation.addedNodes.forEach(checkNodes);
});
});
function log(message, data = '') { console.log(`${_SCRIPT_SHORT_NAME}:`, message, data); }
function logError(message, data = '') { console.error(`${_SCRIPT_SHORT_NAME}:`, new Error(message), data); }
function logWarning(message, data = '') { console.warn(`${_SCRIPT_SHORT_NAME}:`, message, data); }
function logDebug(message, data = '') {
if (_DEBUG)
log(message, data);
}
function dynamicSort(property) {
let sortOrder = 1;
if (property[0] === '-') {
sortOrder = -1;
property = property.substring(1);
}
return function (a, b) {
if (sortOrder === -1)
return b[property].localeCompare(a[property]);
return a[property].localeCompare(b[property]);
};
}
function $extend(...args) {
const extended = {},
deep = Object.prototype.toString.call(args[0]) === '[object Boolean]' ? args[0] : false,
merge = function (obj) {
Object.keys(obj).forEach((prop) => {
if (Object.prototype.hasOwnProperty.call(obj, prop)) {
if (deep && Object.prototype.toString.call(obj[prop]) === '[object Object]')
extended[prop] = $extend(true, extended[prop], obj[prop]);
else if ((obj[prop] !== undefined) && (obj[prop] !== null))
extended[prop] = obj[prop];
}
});
};
for (let i = deep ? 1 : 0, { length } = args; i < length; i++) {
if (args[i])
merge(args[i]);
}
return extended;
}
function trustedHTML(htmlStr) {
if (typeof trustedTypes === 'undefined')
return htmlStr;
if (!_policyTrustedHTML)
_policyTrustedHTML = trustedTypes.createPolicy('urceTrustedHTML', { createHTML: (html) => html });
return _policyTrustedHTML.createHTML(htmlStr);
}
function untrustedHTML(htmlStr) {
if (typeof trustedTypes === 'undefined')
return htmlStr.replace(/&/g, '&').replace(/>/g, '>').replace(/</g, '<').replace(/"/g, '"e;').replace(/'/g, ''');
if (!_policyUntrustedHTML) {
_policyUntrustedHTML = trustedTypes.createPolicy('urceUntrustedHTML', {
createHTML: (html) => html.replace(/&/g, '&').replace(/>/g, '>').replace(/</g, '<').replace(/"/g, '"e;').replace(/'/g, ''')
});
}
return _policyUntrustedHTML.createHTML(htmlStr);
}
function createElem(type = '', attrs = {}, eventListener = []) {
const el = _elems[type]?.cloneNode(false) || _elems.div.cloneNode(false),
applyEventListeners = function ([evt, cb]) {
return this.addEventListener(evt, cb);
};
Object.keys(attrs).forEach((attr) => {
if ((attrs[attr] !== undefined) && (attrs[attr] !== 'undefined') && (attrs[attr] !== null) && (attrs[attr] !== 'null')) {
if ((attr === 'disabled') || (attr === 'checked') || (attr === 'selected') || (attr === 'textContent') || (attr === 'innerHTML'))
el[attr] = attrs[attr];
else
el.setAttribute(attr, attrs[attr]);
}
});
if (eventListener.length > 0) {
eventListener.forEach((obj) => {
Object.entries(obj).map(applyEventListeners.bind(el));
});
}
return el;
}
function createTextNode(str = '') {
return document.createTextNode(str);
}
function dec(s = '') {
return atob(atob(s));
}
function getRandomId() {
return Math.random().toString(36).slice(2);
}
function getOLMapExtent() {
let extent = W.map.getExtent();
if (Array.isArray(extent)) {
extent = new OpenLayers.Bounds(extent);
extent.transform('EPSG:4326', 'EPSG:3857');
}
return extent;
}
async function loadSettingsFromStorage(restoreSettings, proceedWithRestore) {
if (!restoreSettings)
logDebug('Loading settings from storage.');
const invalidRestoreSettings = [],
retainedSettings = [],
defaultSettings = {
lastSaved: 0,
lastVersion: undefined,
wmeUserId: undefined,
expandMoreInfo: false,
expandUserPreferences: true,
expandShortcuts: true,
// Comment List
commentList: 0,
commentListStyle: 'default',
commentListCollapses: {},
customSsId: '',
customTagline: '',
tagEmail: '',
autoSwitchCommentList: false,
enableAppendMode: false,
// Per Comment List Settings
perCommentListSettings: {},
// URC-E Master Settings
autoCenterOnUr: false,
autoClickOpenSolvedNi: false,
autoCloseUrPanel: (_settings.autoCloseCommentWindow),
autoSaveAfterSolvedOrNiComment: false,
autoSendReminders: false,
autoSendRemindersExceptTagged: false,
autoSetNewUrComment: false,
autoSetNewUrCommentSlur: false,
autoSetNewUrCommentWithDescription: false,
autoSetReminderUrComment: false,
placeCursorAtStart: false,
autoSwitchToUrCommentsTab: false,
autoZoomInOnNewUr: false,
autoZoomOutAfterClosePanel: false,
autoZoomOutAfterComment: false,
disableDoneNextButtons: false,
replaceNextWithDoneButton: false,
doubleClickLinkNiComments: false,
doubleClickLinkOpenComments: false,
doubleClickLinkSolvedComments: false,
hideZoomOutLinks: false,
enableUrOverflowHandling: false,
enableAutoRefresh: false,
autoScrollComments: false,
reverseCommentSort: false,
usePrepositionForSegmentNamesShortcut: false,
reminderDays: 0,
closeDays: 7,
// UR Marker Settings
enableUrPillCounts: false,
disableUrMarkerPopup: false,
urMarkerPopupDelay: 2,
urMarkerPopupTimeout: 3,
doNotShowTagNameOnPill: false,
replaceTagNameWithEditorName: false,
unstackMarkers: false,
unstackDisableAboveZoom: 15,
unstackSensitivity: 15,
customMarkersRoadworks: false,
customMarkersConstruction: false,
customMarkersClosures: false,
customMarkersEvents: false,
customMarkersNotes: false,
customMarkersWslm: false,
customMarkersBog: false,
customMarkersDifficult: false,
customMarkersNativeSl: false,
customMarkersCustom: false,
customMarkersCustomText: '',
// UR Filtering Settings
enableUrceUrFiltering: false,
invertFilters: false,
hideOutsideEditableArea: false,
doNotFilterTaggedUrs: false,
doNotHideSelectedUr: false,
disableFilteringAboveZoom: false,
disableFilteringAboveZoomLevel: 12,
disableFilteringBelowZoom: false,
disableFilteringBelowZoomLevel: 22,
// -- Lifecycle
hideWaiting: false,
hideUrsCloseNeeded: false,
hideUrsReminderNeeded: false,
// -- Hide by status
hideByStatusOpen: false,
hideByStatusClosed: false,
hideByStatusNotIdentified: false,
hideByStatusSolved: false,
hideByStatusClosedBy: false,
hideByStatusClosedByUsers: '',
// -- Hide by type
hideByTypeBlockedRoad: false,
hideByTypeGeneralError: false,
hideByTypeIncorrectAddress: false,
hideByTypeIncorrectJunction: false,
hideByTypeIncorrectRoute: false,
hideByTypeIncorrectStreetPrefixOrSuffix: false,
hideByTypeIncorrectTurn: false,
hideByTypeMissingBridgeOverpass: false,
hideByTypeMissingExit: false,
hideByTypeMissingLandmark: false,
hideByTypeMissingOrInvalidSpeedLimit: false,
hideByTypeMissingRoad: false,
hideByTypeMissingRoundabout: false,
hideByTypeMissingStreetName: false,
hideByTypeTurnNotAllowed: false,
hideByTypeUndefined: false,
hideByTypeWrongDrivingDirection: false,
// -- Hide by tag
hideByTaggedBog: false,
hideByTaggedClosure: false,
hideByTaggedConstruction: false,
hideByTaggedDifficult: false,
hideByTaggedEvent: false,
hideByTaggedNote: false,
hideByTaggedRoadworks: false,
hideByTaggedWslm: false,
// -- Hide by age of submission
hideByAgeOfSubmissionLessThan: false,
hideByAgeOfSubmissionLessThanDaysOld: '',
hideByAgeOfSubmissionMoreThan: false,
hideByAgeOfSubmissionMoreThanDaysOld: '',
// -- Hide by Descriptions / Comments / Following
hideFollowing: false,
hideNotFollowing: false,
hideWithDescription: false,
hideWithoutDescription: false,
hideWithCommentsFromMe: false,
hideWithoutCommentsFromMe: false,
hideFirstCommentByMe: false,
hideFirstCommentNotByMe: false,
hideLastCommentByMe: false,
hideLastCommentNotByMe: false,
hideLastCommentByReporter: false,
hideLastCommentNotByReporter: false,
hideByCommentCountLessThan: false,
hideByCommentCountLessThanNumber: '',
hideByCommentCountMoreThan: false,
hideByCommentCountMoreThanNumber: '',
hideByAgeOfFirstCommentLessThan: false,
hideByAgeOfFirstCommentLessThanDaysOld: '',
hideByAgeOfFirstCommentMoreThan: false,
hideByAgeOfFirstCommentMoreThanDaysOld: '',
hideByAgeOfLastCommentLessThan: false,
hideByAgeOfLastCommentLessThanDaysOld: '',
hideByAgeOfLastCommentMoreThan: false,
hideByAgeOfLastCommentMoreThanDaysOld: '',
hideByKeywordIncluding: false,
hideByKeywordIncludingKeyword: '',
hideByKeywordNotIncluding: false,
hideByKeywordNotIncludingKeyword: '',
hideByKeywordCaseInsensitive: false,
hideWithCommentBy: false,
hideWithCommentByUsers: '',
hideWithoutCommentBy: false,
hideWithoutCommentByUsers: ''
};
if (restoreSettings && (restoreSettings !== 'resetSettings') && !proceedWithRestore) {
Object.keys(restoreSettings).forEach((prop) => {
if (!defaultSettings.hasOwnProperty(prop)) {
invalidRestoreSettings.push(prop);
delete (restoreSettings[prop]);
}
else if ((restoreSettings[prop] === 'true') || (restoreSettings[prop] === true)) {
restoreSettings[prop] = true;
}
else if ((restoreSettings[prop] === 'false') || (restoreSettings[prop] === false)) {
restoreSettings[prop] = false;
}
else if ((typeof restoreSettings[prop] !== 'object')
&& !isNaN(restoreSettings[prop])
&& (restoreSettings[prop].length > 0)
&& (restoreSettings[prop] !== +restoreSettings[prop])
) {
restoreSettings[prop] = +restoreSettings[prop];
}
});
Object.keys(_settings).forEach((prop) => {
if (!restoreSettings.hasOwnProperty(prop)) {
restoreSettings[prop] = _settings[prop];
retainedSettings.push(prop);
}
});
const divElem = createElem('div');
let divElemDiv = createElem('div', { style: 'font-weight:bold;', textContent: `${I18n.t('urce.prompts.RestoreSettingsNumOfSettings')}: ` });
divElemDiv.appendChild(createElem('div', {
style: 'display:inline-block;font-weight:normal', textContent: Object.keys(restoreSettings).length - retainedSettings.length
}));
divElem.appendChild(divElemDiv);
divElemDiv = createElem('div', { style: 'font-weight:bold;', textContent: `${I18n.t('urce.prompts.RestoreSettingsRetainedSettings')}: ` });
if (retainedSettings.length > 0) {
const divElemDivDiv = createElem('div', { style: 'font-weight:normal;display:inline-block;', textContent: retainedSettings.join(', ') }),
iElem = createElem('i');
iElem.appendChild(createElem('div', { style: 'display:inline-block;padding-left:4px;', textContent: '(' }));
iElem.appendChild(createElem('div', { style: 'display:inline-block;font-weight:bold;', textContent: `${I18n.t('urce.common.Total')}: ` }));
iElem.appendChild(createElem('div', { style: 'display:inline-block;padding-left:4px;', textContent: `${retainedSettings.length})` }));
divElemDivDiv.appendChild(iElem);
divElemDiv.appendChild(divElemDivDiv);
}
else {
divElemDiv.appendChild(createElem('i', { style: 'display:inline-block;font-weight:normal;font-style:italic;', textContent: I18n.t('urce.common.None') }));
}
divElem.appendChild(divElemDiv);
divElemDiv = createElem('div', { style: 'font-weight:bold;', textContent: `${I18n.t('urce.prompts.RestoreSettingsInvalidSettings')}: ` });
if (invalidRestoreSettings.length > 0) {
const divElemDivDiv = createElem('div', { style: 'font-weight:normal;display:inline-block;', textContent: invalidRestoreSettings.join(', ') }),
iElem = createElem('i');
iElem.appendChild(createElem('div', { style: 'display:inline-block;padding-left:4px;', textContent: '(' }));
iElem.appendChild(createElem('div', { style: 'display:inline-block;font-weight:bold;', textContent: `${I18n.t('urce.common.Total')}: ` }));
iElem.appendChild(createElem('div', { style: 'display:inline-block;padding-left:4px;', textContent: `${invalidRestoreSettings.length})` }));
divElemDivDiv.appendChild(iElem);
divElemDiv.appendChild(divElemDivDiv);
}
else {
divElemDiv.appendChild(createElem('i', { style: 'display:inline-block;font-weight:normal;font-style:italic;', textContent: I18n.t('urce.common.None') }));
}
divElem.appendChild(divElemDiv);
divElem.appendChild(createElem('br'));
divElem.appendChild(createElem('br'));
divElem.appendChild(createElem('div', { style: 'font-weight:bold;', textContent: I18n.t('urce.prompts.RestoreSettingsConfirmation') }));
WazeWrap.Alerts.confirm(
_SCRIPT_SHORT_NAME,
divElem.innerHTML,
() => { loadSettingsFromStorage(restoreSettings, true); },
() => { },
I18n.t('urce.common.Yes'),
I18n.t('urce.common.No')
);
return Promise.resolve();
}
const loadedSettings = (restoreSettings === 'resetSettings') ? {} : restoreSettings || JSON.parse(localStorage.getItem(_SETTINGS_STORE_NAME));
_settings = $extend(true, {}, defaultSettings, loadedSettings);
const serverSettings = await WazeWrap.Remote.RetrieveSettings(_SETTINGS_STORE_NAME);
if (!restoreSettings && (serverSettings?.lastSaved > _settings.lastSaved)) {
_settings = $extend(true, _settings, serverSettings);
_timeouts.saveSettingsToStorage = window.setTimeout(saveSettingsToStorage, 5000);
}
if (_settings.wmeUserId !== _wmeUserId)
_settings.wmeUserId = _wmeUserId;
// Remove old settings
['autoCloseCommentWindow', 'hideClosedUrs', 'showOthersUrsPastReminderClose', 'onlyShowMyUrs', 'hideTaggedUrs', 'hideUrsWoComments', 'hideUrsWoCommentsOrDescriptions',
'hideUrsWoCommentsWithDescriptions', 'hideUrsWithUserReplies', 'disableAboveZoomLevel', 'hideByAgeOfLastCommentLessThanDaysAgo',
'hideByAgeOfLastCommentMoreThanDaysAgo', 'hideByAgeOfFirstCommentMoreThanDaysAgo', 'hideByTypeWazeAutomatic', 'sortCommentsOldestFirst', 'unfollowUrAfterSend'
].forEach((oldSetting) => {
if (_settings.hasOwnProperty(oldSetting))
delete (_settings[oldSetting]);
});
// Fix bad settings
['reminderDays', 'closeDays', 'hideByAgeOfLastCommentMoreThanDaysOld', 'hideByAgeOfLastCommentLessThanDaysOld', 'hideByAgeOfFirstCommentMoreThanDaysOld',
'hideByAgeOfFirstCommentLessThanDaysOld', 'hideByCommentCountMoreThanNumber', 'hideByCommentCountLessThanNumber', 'hideByAgeOfSubmissionMoreThanDaysOld',
'hideByAgeOfSubmissionLessThanDaysOld'].forEach((setting) => {
if ((_settings[setting] === undefined) || (_settings[setting] === null) || ((_settings[setting].length === 0) && (_settings[setting] !== '')))
_settings[setting] = '';
});
['disableFilteringAboveZoomLevel', 'disableFilteringBelowZoomLevel', 'unstackDisableAboveZoom'].forEach((setting) => {
if (_settings[setting] < 11)
_settings[setting] += 12;
});
_timeouts.saveSettingsToStorage = window.setTimeout(saveSettingsToStorage, 5000);
if (proceedWithRestore) {
await initGui(false);
await changeCommentList(_settings.commentList, false, true);
handleUrLayer('settingsToggle', undefined, getMapUrsObjArr());
saveSettingsToStorage();
WazeWrap.Alerts.success(_SCRIPT_SHORT_NAME, ((restoreSettings === 'resetSettings') ? `${I18n.t('urce.prompts.ResetSettingsComplete')}.` : `${I18n.t('urce.prompts.RestoreSettingsComplete')}.`));
}
return Promise.resolve();
}
function showScriptInfoAlert() {
if (_ALERT_UPDATE && (_SCRIPT_VERSION !== _settings.lastVersion)) {
const divElemRoot = createElem('div');
divElemRoot.appendChild(createElem('p', { textContent: 'What\'s New:' }));
const ulElem = createElem('ul');
if (_SCRIPT_VERSION_CHANGES.length > 0) {
for (let idx = 0, { length } = _SCRIPT_VERSION_CHANGES; idx < length; idx++)
ulElem.appendChild(createElem('li', { innerHTML: _SCRIPT_VERSION_CHANGES[idx] }));
}
else {
ulElem.appendChild(createElem('li', { textContent: 'Nothing major.' }));
}
divElemRoot.appendChild(ulElem);
WazeWrap.Interface.ShowScriptUpdate(_SCRIPT_SHORT_NAME, _SCRIPT_VERSION, divElemRoot.innerHTML, (_IS_BETA_VERSION ? dec(_BETA_DL_URL) : _PROD_DL_URL).replace(/code\/.*\.js/, ''), _FORUM_URL);
}
}
async function saveSettingsToStorage() {
checkTimeout({ timeout: 'saveSettingsToStorage' });
if (localStorage) {
_settings.commentListCollapses = _settings.commentListCollapses || {};
_settings.commentListCollapses[_settings.commentList] = await getCollapsedGroups();
_settings.lastVersion = _SCRIPT_VERSION;
_settings.lastSaved = Date.now();
localStorage.setItem(_SETTINGS_STORE_NAME, JSON.stringify(_settings));
WazeWrap.Remote.SaveSettings(_SETTINGS_STORE_NAME, _settings);
logDebug('Settings saved.');
}
}
function checkTimeout(obj) {
if (obj.toIndex) {
if (_timeouts[obj.timeout]?.[obj.toIndex]) {
window.clearTimeout(_timeouts[obj.timeout][obj.toIndex]);
delete (_timeouts[obj.timeout][obj.toIndex]);
}
}
else {
if (_timeouts[obj.timeout])
window.clearTimeout(_timeouts[obj.timeout]);
_timeouts[obj.timeout] = undefined;
}
}
async function checkSidebarHeight() {
const urceNode = document.querySelector('#sidepanel-urc-e .URCE-divTabs'),
userscriptsApiDocsLinkNode = document.querySelector('#user-info .userscripts-api-docs-link-container'),
wzMapOlFooter = document.querySelector('#waze-map-container .wz-map-ol-footer'),
{ top } = urceNode.getBoundingClientRect(),
{ scrollY, pageYOffset } = window,
{ scrollTop } = document.body;
if (!userscriptsApiDocsLinkNode || !wzMapOlFooter)
return;
let offset = (top + (scrollY || pageYOffset || scrollTop)) || 0;
offset += userscriptsApiDocsLinkNode.offsetHeight;
offset += wzMapOlFooter.offsetHeight;
if (offset !== +getComputedStyle(urceNode).getPropertyValue('--height-offset')) {
logDebug(`Changing --height-offset to: ${offset}`);
urceNode.style.setProperty('--height-offset', `${offset}px`);
}
}
function disableWme(htmlElem, remove = true) {
if (remove && document.getElementById('urce-disableWme')) {
document.getElementById('urce-disableWme').remove();
}
else if (!remove && !document.getElementById('urce-disableWme')) {
const divElemRoot = createElem('div', { id: 'urce-disableWme', class: 'URCE-disableWme-main' }),
messageDiv = createElem('div', { id: 'urce-disableWme-text', class: 'URCE-disableWme-text' });
messageDiv.appendChild(htmlElem);
divElemRoot.appendChild(messageDiv);
document.body.appendChild(divElemRoot);
}
}
function dismissAlertBoxInPanel(evt, idx) {
idx = idx || this?.attributes?.index?.value || -1;
if ((idx > 0) && document.getElementById(`urceAlertPanelBox-${idx}`))
document.getElementById(`urceAlertPanelBox-${idx}`).remove();
}
function alertBoxInPanel(docFrags, panelBoxTitle, panelBoxDismiss, index) {
if (document.getElementById(`urceAlertPanelBox-${index}`))
document.getElementById(`urceAlertPanelBox-${index}`).remove();
const divElemRoot = createElem('div', { id: `urceAlertPanelBox-${index}`, class: 'URCE-divWarningBox', title: panelBoxTitle || '' });
divElemRoot.appendChild(docFrags);
if (panelBoxDismiss) {
const divElemDiv = createElem('div', { id: `urceAlertPanelBox-${index}-dismiss`, class: 'URCE-divDismiss', index }, [{ click: dismissAlertBoxInPanel }]);
divElemDiv.appendChild(createElem('i', { class: 'w-icon w-icon-x', style: 'font-weight:900;' }));
divElemRoot.insertBefore(divElemDiv, divElemRoot.firstChild);
}
const panelUrceComments = document.getElementById('panel-urce-comments');
panelUrceComments.insertBefore(divElemRoot, panelUrceComments.firstChild);
}
function isPanelReady(waitForAttr = '', retryInterval = 10, maxTries = 200) {
return new Promise((resolve) => {
(function retry(waitForAttrStr, tries, toIndex, retryInt, maxNumTries) {
checkTimeout({ timeout: 'isPanelReady', toIndex });
if (tries > maxNumTries)
resolve({ error: true });
else if (!W.problemsController?.editController?.viewModel?.get(waitForAttrStr))
resolve({ error: false });
else
_timeouts.isPanelReady[toIndex] = window.setTimeout(retry, retryInt, waitForAttrStr, ++tries, toIndex, retryInt, maxNumTries);
}(waitForAttr, 1, getRandomId(), retryInterval, maxTries));
});
}
function getDomElement(element, shadowHost) {
return new Promise((resolve) => {
(function retry(elementStr, shadowHostStr, tries, toIndex, retryInt, maxNumTries, retJquery) {
checkTimeout({ timeout: 'isDomElementReady', toIndex });
if (tries > maxNumTries) {
logError(`Timed out waiting for DOM element to appear: ${elementStr}`);
resolve(undefined);
}
else {
let returnElem;
if (shadowHostStr) {
const shadowHostElem = document.querySelector(shadowHostStr);
if (shadowHostElem?.shadowRoot)
returnElem = shadowHostElem.shadowRoot.querySelector(elementStr);
}
else {
returnElem = document.querySelector(elementStr);
}
if (returnElem)
resolve(returnElem);
else
_timeouts.isDomElementReady[toIndex] = window.setTimeout(retry, retryInt, elementStr, shadowHostStr, ++tries, toIndex, retryInt, maxNumTries, retJquery);
}
}(element, shadowHost, 1, getRandomId(), 10, 200));
});
}
function handleReadyError(reopenUrPanel = false, spinnerStop = false, spinnerName = '', errorDisplay = false, errorText = '') {
if (errorText.length > 0) {
logError(errorText);
if (errorDisplay)
WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, errorText);
}
if (spinnerStop)
doSpinner(spinnerName, false);
const urId = document.querySelector('.update-requests .marker-selected')?.attributes?.['data-id']?.value;
if (reopenUrPanel && (+urId > 0))
openUrPanel(0, reopenUrPanel);
}
function getCollapsedGroups() {
return new Promise((resolve) => {
const getDivs = document.querySelectorAll('div[id$="_body_urce"]'),
rObj = {};
for (let idx = 0, { length } = getDivs; idx < length; idx++) {
if (getDivs[idx].id.includes('urceComments-for-'))
rObj[getDivs[idx].id] = getDivs[idx].classList.contains('URCE-collapsed');
}
resolve(rObj);
});
}
function doSpinner(spinnerName = '', spin = true) {
const btn = document.getElementById('urceUrMarkerProcessingSpinner');
if (!btn) {
Object.keys(_spinners).forEach((a) => { _spinners[a] = false; });
}
else if (!spin) {
_spinners[spinnerName] = false;
if (!Object.values(_spinners).some((a) => a === true)) {
btn.classList.remove('fa-spin');
btn.style.color = 'lightgray';
btn.setAttribute('title', I18n.t('urce.mouseOver.URMarkerProcessingInactive'));
}
}
else {
_spinners[spinnerName] = true;
if (!btn.classList.contains('fa-spin')) {
btn.classList.add('fa-spin');
btn.style.color = 'black';
btn.setAttribute('title', I18n.t('urce.mouseOver.URMarkerProcessingActive'));
}
}
}
function checkRestrictions(evt) {
return new Promise((resolve) => {
(function retry(tries, toIndex, event) {
checkTimeout({ timeout: 'checkRestrictions', toIndex });
// 2023.04.05.01: W.model.getTopCountry() and W.model.getTopState() return null when zoom level < 12.
if (W.map.getOLMap().getZoom() < 12) {
resolve();
return;
}
doSpinner('checkRestrictions', true);
const displayWarning = (content, remove) => {
if (remove) {
document.querySelectorAll('[id|="restrictionsEnforcedWarning"').forEach((el) => { el.style.display = 'none'; });
}
else if (((event?.[0]?.type === 'init') || (event?.[0]?.type === 'modeChange'))
|| (!document.getElementById('restrictionsEnforcedWarning').value)
) {
_restrictionsEnforcedTitle = content;
}
else {
document.querySelectorAll('[id|="restrictionsEnforcedWarning"').forEach((el) => {
el.firstChild.setAttribute('title', content);
el.style.display = '';
});
}
};
let moved = false,
state,
country;
if ((tries === 1) && (event?.[0]?.type !== 'init') && (event?.[0]?.type !== 'modeChange')) {
_timeouts.checkRestrictions[toIndex] = window.setTimeout(retry, 500, ++tries, toIndex, event);
}
else if (tries < 301) {
if ((event?.[0]?.type === 'state') && (event?.[0]?.getName() === W.model.getTopState().getName())) {
country = (event[0].getAttribute('countryID') !== 0) ? W.model.countries.getObjectById(event[0].getAttribute('countryID')).getAttribute('abbr') : W.model.getTopCountry().getAttribute('abbr');
state = event[0].getName();
}
else if ((event?.[0]?.type === 'country') && (event?.[0]?.getAttribute('abbr') === W.model.getTopCountry().getAttribute('abbr'))) {
country = event[0].getAttribute('abbr');
state = false;
}
if (!country && !W.model.getTopCountry()) {
logDebug(`Waiting on Waze model for countries to populate. Try ${tries} of 300.`);
_timeouts.checkRestrictions[toIndex] = window.setTimeout(retry, 100, ++tries, toIndex, event);
}
else {
checkTimeout({ timeout: 'checkRestrictions', toIndex });
country = country || W.model.getTopCountry().getAttribute('abbr');
state = state || (W.model.getTopState() ? W.model.getTopState().getName() : false);
if (state !== _currentArea.state) {
_currentArea.state = state;
moved = true;
}
if (country !== _currentArea.country) {
_currentArea.country = country;
moved = true;
}
if (moved) {
logDebug((((event?.[0]?.type === 'init') || (event?.[0]?.type === 'modeChange')) ? 'Setting up restrictions.' : 'Checking restrictions.'));
_restrictionsEnforce = {};
let restrictionsAlertBannerTitle = `${I18n.t('urce.prompts.RestrictionsEnforced')}\n\n${I18n.t('urce.prompts.RestrictionsEnforcedTitle')}:\n`;
if (_restrictions[country]) {
if (_restrictions[country][state]) {
const countryName = W.model.countries.getObjectArray().filter((c) => c.getAttribute('abbr') === country)?.[0]?.getName();
if (countryName)
restrictionsAlertBannerTitle += `\n${countryName} - ${state}:`;
else
restrictionsAlertBannerTitle += `\n${state}:`;
Object.keys(_restrictions[country][state]).forEach((restriction) => {
_restrictionsEnforce[restriction] = _restrictions[country][state][restriction];
restrictionsAlertBannerTitle += `\n${I18n.t(`urce.prefs.${restriction.charAt(0).toUpperCase()}${restriction.slice(1)}`)}: `;
if (_restrictionsEnforce[restriction] === true)
restrictionsAlertBannerTitle += I18n.t('urce.common.Enabled');
else if (_restrictionsEnforce[restriction] === false)
restrictionsAlertBannerTitle += I18n.t('urce.common.Disabled');
else
restrictionsAlertBannerTitle += I18n.t('common.time.days', { days: _restrictionsEnforce[restriction] });
});
}
else if (Object.keys(_restrictions[country]?.ALL || {}).length > 0) {
const countryName = W.model.countries.getObjectArray().filter((c) => c.getAttribute('abbr') === country)?.[0]?.getName();
if (countryName)
restrictionsAlertBannerTitle += `\n${countryName} - ${I18n.t('urce.common.All')}:`;
else
restrictionsAlertBannerTitle += `\n${I18n.t('urce.common.All')}:`;
Object.keys(_restrictions[country].ALL).forEach((restriction) => {
_restrictionsEnforce[restriction] = _restrictions[country].ALL[restriction];
restrictionsAlertBannerTitle += `\n${I18n.t(`urce.prefs.${restriction.charAt(0).toUpperCase()}${restriction.slice(1)}`)}: `;
if (_restrictionsEnforce[restriction] === true)
restrictionsAlertBannerTitle += I18n.t('urce.common.Enabled');
else if (_restrictionsEnforce[restriction] === false)
restrictionsAlertBannerTitle += I18n.t('urce.common.Disabled');
else
restrictionsAlertBannerTitle += I18n.t('common.time.days', { days: _restrictionsEnforce[restriction] });
});
}
if (Object.values(_restrictionsEnforce).length > 0)
displayWarning(restrictionsAlertBannerTitle, false);
else
displayWarning(false, true);
}
else {
displayWarning(false, true);
}
resolve();
}
else {
resolve();
}
}
}
else {
resolve(logError('Unable to check for restrictions'));
}
doSpinner('checkRestrictions', false);
}(1, getRandomId(), evt));
});
}
function getMapUrsObjArr(urIds = []) {
const mUrsObjArr = [],
objsArray = (urIds.length > 0) ? W.model.mapUpdateRequests.getByIds(urIds) : W.model.mapUpdateRequests.getObjectArray();
objsArray.forEach((urObj) => {
if (!mUrsObjArr.includes(urObj.getID()))
mUrsObjArr.push(urObj);
});
return mUrsObjArr;
}
function mUrsAdded(objectsArr) {
if (objectsArr?.length === 0)
return;
const zoomLevel = W.map.getOLMap().getZoom();
let filter = true;
if ((_settings.disableFilteringAboveZoom && (zoomLevel < _settings.disableFilteringAboveZoomLevel))
|| (_settings.disableFilteringBelowZoom && (zoomLevel > _settings.disableFilteringBelowZoomLevel))
)
filter = false;
handleUrLayer('mUrsAdded', filter, objectsArr.sort((a, b) => a.getID() - b.getID()));
}
function mUrsRemoved(objectsArr = []) {
if (W.model.mapUpdateRequests.getObjectArray().length === 0)
_mapUpdateRequests = {};
else
objectsArr.forEach((mUrObj) => delete (_mapUpdateRequests[mUrObj.getID()]));
}
async function getUpdateRequestSessions(urIds = []) {
let data = {};
try {
data = await W.controller.descartesClient.getUpdateRequestSessionsByIds(urIds);
if (data?.updateRequestSessions?.objects.length > 0)
// 2023.04.05.01: No need to merge the data to the W.map.mapUpdateRequests repo. Let WME control that repo.
// W.model.mergeResponse(data);
data = Object.fromEntries(data.updateRequestSessions.objects.map((o) => [o.getID(), o]));
else
data = {};
}
catch (error) {
logWarning(error);
}
return Promise.resolve(data);
}
async function handleAfterCommentMutation(domElem) {
logDebug(`Handling new comment mutation for urId: ${_selUr.urId}`);
doSpinner('handleAfterCommentMutation', true);
if (_settings.autoZoomOutAfterComment)
autoZoomOut();
if (_settings.autoCloseUrPanel || _selUr.doubleClick) {
await autoCloseUrPanel();
}
else {
await updateUrceData(getMapUrsObjArr([_selUr.urId]));
if (!domElem.querySelector('#urceDaysAgo')) {
domElem.querySelector('span.date').style.float = 'right';
domElem.shadowRoot.appendChild(createElem('style', {
textContent: '.key-with-image-wrapper, .key-wrapper { width: 100% } '
+ '.wz-list-item, .wz-list-item.with-subtitle { --wz-list-item-vertical-padding: 0px !important; margin: 2px 0px !important; }'
}));
const divElemRoot = createElem('div', { class: 'date urce' });
divElemRoot.appendChild(createElem('div', {
textContent: `(${parseDaysAgo(daysAgo(W.model.updateRequestSessions.getObjectById(_selUr.urId).getAttribute('comments')[(_mapUpdateRequests[_selUr.urId].urceData.commentCount - 1)].createdOn))})`
}));
domElem.firstChild.appendChild(divElemRoot);
}
if (_settings.reverseCommentSort) {
const commentList = await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation.section .conversation-view .comment-list'),
numComments = commentList.children.length;
domElem.remove();
commentList.insertBefore(domElem, commentList.firstChild);
autoScrollComments(numComments);
}
if (_settings.autoSaveAfterSolvedOrNiComment && ((_selUr.newStatus === 'solved') || (_selUr.newStatus === 'notidentified')))
document.querySelector('wz-button[id="save-button"]').click();
else
handleUrLayer('sendComment', undefined, getMapUrsObjArr([_selUr.urId]));
}
doSpinner('handleAfterCommentMutation', false);
}
async function handleAfterCloseUpdateContainer() {
/**
* 2024.11.26
* Commented out because we are back to needing to know if the urPanel closes and opens in the same mutation set.
* This can likely be deleted in next update if no issues are found from removing it. It is in one other place in the code.
*
_urPanelContainerObserver.disconnect();
_urPanelContainerObserver.observe(document.getElementById('panel-container'), {
childList: true, attributes: true, attributeOldValue: true, characterData: false, characterDataOldValue: false, subtree: true
});
*/
/**
* 2023.03.21
* Commented out because the marker is still selected when the panel is first closed.
* We need to go ahead and process the rest as it's important for the rest of our script. This will have a side effect of the zoom features being
* a little wonky.
* TODO: Find another way to know if we clicked a new UR marker with the current one still open OR if we truly closed the UR panel.
*
if (parseInt($('.update-requests .marker-selected').data('id')) > 0)
return;
*/
const { urId, newStatus } = _selUr;
_selUr = {
doubleClick: false,
handling: false,
newStatus: undefined,
urId: -1,
urOpen: false
};
if (_urDataStateObserver.isObserving) {
_urDataStateObserver.disconnect();
_urDataStateObserver.isObserving = false;
}
if (_urCommentsObserver.isObserving) {
_urCommentsObserver.disconnect();
_urCommentsObserver.isObserving = false;
}
if (_settings.autoZoomOutAfterClosePanel)
autoZoomOut();
if (_settings.autoSaveAfterSolvedOrNiComment && ((newStatus === 'solved') || (newStatus === 'notidentified'))) {
document.querySelector('wz-button[id="save-button"]').click();
}
else {
if (_settings.autoSwitchToUrCommentsTab)
autoSwitchToPrevTab();
handleUrLayer('close', undefined, getMapUrsObjArr([urId]));
}
}
function handleAfterSave() {
if (_settings.autoZoomOutAfterComment)
autoZoomOut();
if (_settings.autoSwitchToUrCommentsTab)
autoSwitchToPrevTab();
handleUrLayer('save', undefined, getMapUrsObjArr());
}
async function handleUpdateRequestContainer() {
// 2024.03.21: Now that the UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085), don't have time to fix marker (un)stacking. Removing for now.
// restackMarkers();
if (!_commentListLoaded)
return;
// 2024.03.21: UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085).
// const selectedUrId = document.querySelector('.update-requests .marker-selected')?.attributes?.['data-id']?.value;
const selectedUrId = W.map.getLayerByName('update_requests')?.features?.filter((feature) => feature.attributes?.wazeFeature?.isSelected)[0]?.attributes?.wazeFeature?.id || -1;
if ((+selectedUrId > 0)
&& (!(_selUr.urId > 0) || (_selUr.urId !== +selectedUrId))
) {
_selUr.urId = +selectedUrId;
logDebug(`Selected UR from handleURContainer: ${_selUr.urId}`);
}
else if (!(_selUr.urId > 0)) {
/** 2024.03.21: Let's just remove the needUrId safety net for now. This will keep the script "working" if something else breaks when looking for the URID.
*
_needUrId = true;
const docFrags = document.createDocumentFragment();
docFrags.appendChild(createElem('div', { textContent: I18n.t('urce.prompts.WaitingToGetUrId') }));
docFrags.appendChild(createElem('br'));
docFrags.appendChild(createElem('div', { textContent: I18n.t('urce.common.PleaseWait') }));
maskBoxes(docFrags, false, 'needUrId', true);
*/
return;
}
if (_settings.replaceNextWithDoneButton && W.problemsController.editController.showNext) {
openUrPanel(_selUr.urId, true);
return;
}
const clickedSendToSendComment = () => { logDebug('Clicked send to submit the comment.'); },
expandMoreInfoCallback = function () {
if (this?.parentElement?.classList?.contains('collapsed'))
_settings.expandMoreInfo = true;
else
_settings.expandMoreInfo = false;
saveSettingsToStorage();
},
expandUserPreferencesCallback = function () {
if (this?.parentElement?.classList?.contains('collapsed'))
_settings.expandUserPreferences = true;
else
_settings.expandUserPreferences = false;
saveSettingsToStorage();
};
_selUr.handling = true;
/**
* 2024.11.26
* Commented out because we are back to needing to know if the urPanel closes and opens in the same mutation set.
* This can likely be deleted in next update if no issues are found from removing it. It is in one other place in the code.
*
_urPanelContainerObserver.disconnect();
_urPanelContainerObserver.observe(document.getElementById('panel-container'), {
childList: true, attributes: false, attributeOldValue: false, characterData: false, characterDataOldValue: false, subtree: false
});
*/
doSpinner('handleUpdateRequestContainer', true);
_restoreZoom = W.map.getOLMap().getZoom();
if (_timeouts.popup)
hidePopup();
logDebug(`Handling update request container for urId: ${_selUr.urId}`);
if (_settings.autoSwitchCommentList) {
const topCountryAbbr = W.model.getTopCountry().getAttribute('abbr'),
topStateName = W.model.getTopState().getName();
if ((_autoSwitch[topCountryAbbr]?.ALL > -1) || (_autoSwitch[topCountryAbbr]?.[topStateName] > -1)) {
let commentList;
if (_autoSwitch[topCountryAbbr][topStateName] > -1)
commentList = _autoSwitch[topCountryAbbr][topStateName];
else if (_autoSwitch[topCountryAbbr].ALL)
commentList = _autoSwitch[topCountryAbbr].ALL;
else
commentList = -1;
if ((commentList > -1) && (commentList !== _currentCommentList))
await changeCommentList(commentList, (commentList !== _settings.commentList), false);
}
else if (_currentCommentList !== _settings.commentList) {
await changeCommentList(_settings.commentList, true, false);
}
}
await updateUrceData(getMapUrsObjArr([_selUr.urId]));
let domElement = await getDomElement('#panel-container div[class^="container"] .sub-title');
if (!domElement) {
handleReadyError(true, true, 'handleUpdateRequestContainer', false, 'isDomElementReady: .sub-title');
return;
}
if (!domElement.textContent.includes(_selUr.urId))
domElement.append(` (${_selUr.urId}) `);
domElement = await getDomElement('#panel-container div[class^="container"] .reported');
if (!domElement) {
handleReadyError(true, true, 'handleUpdateRequestContainer', false, '');
return;
}
const mapUrObj = W.model.mapUpdateRequests.getObjectById(_selUr.urId),
urSessionsObj = W.model.updateRequestSessions.getObjectById(_selUr.urId);
if (!domElement.textContent.endsWith(')'))
domElement.textContent += ` (${parseDaysAgo(_mapUpdateRequests[_selUr.urId].urceData.driveDaysOld)})`;
if (mapUrObj.getAttribute('description')) {
const content = await getDomElement('#panel-container div[class^="container"] .body .problem-data .description .content');
if (content?.children.length === 0) {
const divElemRoot = createElem('div', { class: 'URCE-divDesc', textContent: content.textContent });
content.textContent = '';
content.appendChild(divElemRoot);
}
}
if (_mapUpdateRequests[_selUr.urId].urceData.commentCount > 0) {
if (!(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .comment .comment-title'))) {
handleReadyError(false, false, '', false, 'isConversationLoaded: loadingConversation');
}
else {
const comments = document.querySelectorAll('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .comment .comment-title');
for (let idx = 0, { commentCount } = _mapUpdateRequests[_selUr.urId].urceData; idx < commentCount; idx++) {
const currComment = comments[idx];
if (currComment.getElementsByClassName('date urce').length === 0) {
currComment.querySelector('span.date').style.float = 'right';
currComment.parentElement.shadowRoot.appendChild(createElem('style', {
textContent: '.key-with-image-wrapper, .key-wrapper { width: 100% } '
+ '.wz-list-item, .wz-list-item.with-subtitle { --wz-list-item-vertical-padding: 0px !important; margin: 2px 0px !important; }'
}));
currComment.appendChild(createElem('div', {
class: 'date urce', textContent: `(${parseDaysAgo(daysAgo(urSessionsObj.getAttribute('comments')[idx].createdOn))})`
}));
}
}
}
if (_settings.reverseCommentSort && (_mapUpdateRequests[_selUr.urId].urceData.commentCount > 1)) {
const commentList = await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation.section .conversation-view .comment-list');
if (commentList) {
const docFrags = document.createDocumentFragment(),
reverseCommentList = [...commentList.querySelectorAll('wz-list-item.comment')].reverse();
reverseCommentList.forEach((el) => docFrags.appendChild(el));
commentList.replaceChildren(docFrags);
}
}
}
_selUr.urOpen = mapUrObj.getOpenState();
if (_settings.autoSwitchToUrCommentsTab)
autoSwitchToUrceTab();
if (_settings.disableDoneNextButtons) {
domElement = await getDomElement('#panel-container wz-card[class^="panel"].problem-edit .actions .content .navigation');
if (domElement)
domElement.style.display = 'none';
}
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .focus'))?.addEventListener('click', recenterOnUr);
domElement = await getDomElement('textarea[id^=wz-textarea-]', '#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-text');
if (!domElement) {
handleReadyError(true, true, 'handleUpdateRequestContainer', false, '');
return;
}
domElement.style.backgroundColor = _settings.enableAppendMode ? 'peachpuff' : '';
domElement.addEventListener('keyup', checkValue);
if (!document.getElementById('urceShortcuts')) {
const urceShortcutsExpand = function () {
const iElem = this.querySelector('i');
if (iElem.classList.contains('w-icon-chevron-up'))
_settings.expandShortcuts = false;
else
_settings.expandShortcuts = true;
iElem.classList.toggle('w-icon-chevron-down');
iElem.classList.toggle('w-icon-chevron-up');
if (_settings.expandShortcuts)
document.getElementById('urceShortcutsExpandDiv').style.display = '';
else
document.getElementById('urceShortcutsExpandDiv').style.display = 'none';
saveSettingsToStorage();
},
createIElem = (id = '', classStr = '', style = '', title = '') => createElem('i', {
id, class: classStr, style, title
}, [{ click: handleClickedShortcut }]),
createSeparator = (lPad = 4, rPad = 4) => {
const elem = createElem('div', { class: 'separator', textContent: '||' });
elem.style.padding = `0 ${rPad}px 0 ${lPad}px`;
return elem;
};
let divElemDiv = createElem('div', {
id: 'urceShortcutsExpand', textContent: I18n.t('urce.urPanel.Shortcuts')
}, [{ click: urceShortcutsExpand }]);
let divElemDivDiv = createElem('div', { class: 'chevron' });
divElemDivDiv.appendChild(createElem('i', { class: `w-icon ${(_settings.expandShortcuts ? 'w-icon-chevron-up' : 'w-icon-chevron-down')} URCE-chevron` }));
divElemDiv.appendChild(divElemDivDiv);
const divElemRoot = createElem('div', { id: 'urceShortcuts' });
divElemRoot.appendChild(divElemDiv);
divElemDiv = createElem('div', { id: 'urceShortcutsExpandDiv' });
if (!_settings.expandShortcuts)
divElemDiv.style.display = 'none';
divElemDivDiv = createElem('div');
divElemDivDiv.appendChild(createIElem('urceShortcuts-selSegs', 'w-icon w-icon-road', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertSelSegsTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-selSegsWithCity', 'w-icon w-icon-road', 'cursor:pointer;', I18n.t('urce.urPanel.InsertSelSegsWithCityTitle')));
divElemDivDiv.appendChild(createIElem(
'urceShortcuts-selSegsWithCity',
'w-icon w-icon-plus',
'cursor:pointer;float:left;margin:-6px 0 0 -5px;position:absolute;font-weight:600;',
I18n.t('urce.urPanel.InsertSelSegsWithCityTitle')
));
divElemDivDiv.appendChild(createSeparator(11, 4));
divElemDivDiv.appendChild(createIElem('urceShortcuts-placeName', 'w-icon w-icon-hotel', 'cursor:pointer;', I18n.t('urce.urPanel.InsertPlaceNameTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-placeAddress', 'w-icon w-icon-location', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertPlaceAddressTitle')));
divElemDivDiv.appendChild(createSeparator());
divElemDivDiv.appendChild(createIElem('urceShortcuts-description', 'w-icon w-icon-doc', 'cursor:pointer;', I18n.t('urce.urPanel.InsertDescriptionTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-wazeUsername', 'w-icon w-icon-wazer', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertWazeUsernameTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-urType', 'w-icon w-icon-info', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertUrTypeTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-customTagline', 'w-icon w-icon-tag', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertCustomTaglineTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-urPermalink', 'w-icon w-icon-link', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertUrPermalinkTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-urId', 'w-icon w-icon-node', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertUrIdTitle')));
divElemDiv.appendChild(divElemDivDiv);
divElemDivDiv = createElem('div');
divElemDivDiv.appendChild(createIElem('urceShortcuts-driveTime', 'w-icon w-icon-clock', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertTimeTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-driveDayOfWeek', 'w-icon w-icon-clock-fill', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertDayOfWeekTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-driveDate', 'w-icon w-icon-calendar', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertDateTitle')));
divElemDivDiv.appendChild(createSeparator());
divElemDivDiv.appendChild(createIElem('urceShortcuts-driveTimeCasual', 'w-icon w-icon-clock', 'cursor:pointer;', I18n.t('urce.urPanel.InsertTimeCasualTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-driveDateCasual', 'w-icon w-icon-calendar', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertDateCasualTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-driveDateTimeCasualMode', 'w-icon w-icon-available-fill', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertDateTimeCasualModeTitle')));
divElemDivDiv.appendChild(createElem('div', { class: 'driveDateText', textContent: `- ${I18n.t('urce.urPanel.DriveDate')}` }));
divElemDiv.appendChild(divElemDivDiv);
divElemDivDiv = createElem('div');
divElemDivDiv.appendChild(createIElem('urceShortcuts-currentTime', 'w-icon w-icon-clock', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertCurrentTimeTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-currentDayOfWeek', 'w-icon w-icon-clock-fill', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertCurrentDayOfWeekTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-currentDate', 'w-icon w-icon-calendar', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertCurrentDateTitle')));
divElemDivDiv.appendChild(createSeparator());
divElemDivDiv.appendChild(createIElem('urceShortcuts-currentTimeCasual', 'w-icon w-icon-clock', 'cursor:pointer;', I18n.t('urce.urPanel.InsertCurrentTimeCasualTitle')));
divElemDivDiv.appendChild(createIElem('urceShortcuts-currentDateCasual', 'w-icon w-icon-calendar', 'cursor:pointer;padding-left:4px;', I18n.t('urce.urPanel.InsertCurrentDateCasualTitle')));
divElemDivDiv.appendChild(createElem('div', { class: 'currentDateText', textContent: `- ${I18n.t('urce.urPanel.CurrentDate')}` }));
divElemDiv.appendChild(divElemDivDiv);
divElemRoot.appendChild(divElemDiv);
domElement = document.querySelector('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-form');
domElement.insertBefore(divElemRoot, domElement.firstChild);
}
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-form .send-button'))?.addEventListener('click', clickedSendToSendComment);
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-form .send-button'))?.shadowRoot.appendChild(createElem('style', {
textContent: '.wz-button.md { height: 30px !important; min-width: 60px !important; max-widt: 140px !important; padding: 0px 10px !important; overflow: hidden; }'
}));
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-form .new-comment-text'))?.shadowRoot.appendChild(createElem('style', {
textContent: '.wz-textarea textarea { font-size: 13px !important; line-height: 14px; padding: 6px !important; } '
+ '.wz-textarea .length-text { font-size: 12px !important; padding: 2px 6px 0px 0px !important; }'
}));
if (!_urDataStateObserver.isObserving) {
_urDataStateObserver.observe(document.querySelector('#panel-container wz-card[class^="panel"].problem-edit'), {
childList: false, attributes: true, attributeOldValue: true, characterData: false, characterDataOldValue: false, subtree: false
});
_urDataStateObserver.isObserving = true;
}
if (!_urCommentsObserver.isObserving) {
_urCommentsObserver.observe(document.querySelector('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation.section .conversation-view .comment-list'), {
childList: true, attributes: false, attributeOldValue: false, characterData: false, characterDataOldValue: false, subtree: false
});
_urCommentsObserver.isObserving = true;
}
if (!(await isPanelReady('loadingMoreInfo', 10, 200)).error) {
domElement = await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .more-info');
if (_settings.expandMoreInfo && W.problemsController.editController.adapter.isMoreInfoAvailable() && domElement.classList.contains('collapsed'))
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .more-info .title'))?.click();
else if (!_settings.expandMoreInfo && !domElement.classList.contains('collapsed'))
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .more-info .title'))?.click();
}
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .more-info .title'))?.addEventListener('click', expandMoreInfoCallback);
domElement = await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .reporter-preferences');
if (_settings.expandUserPreferences && domElement.classList.contains('collapsed'))
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .reporter-preferences .title'))?.click();
else if (!_settings.expandUserPreferences && !domElement.classList.contains('collapsed'))
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .reporter-preferences .title'))?.click();
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .reporter-preferences .title'))?.addEventListener('click', expandUserPreferencesCallback);
if ((await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation'))?.classList?.contains('collapsed'))
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .title'))?.click();
domElement = await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"]');
if (domElement)
domElement.scrollTop = domElement.scrollHeight;
autoScrollComments(_mapUpdateRequests[_selUr.urId].urceData.commentCount, 10, 1);
if (_mapUpdateRequests[_selUr.urId].urceData.commentCount === 0) {
if (_settings.autoZoomInOnNewUr)
autoZoomIn();
const { commentNum } = Object.values(_defaultComments).find((defaultComment) => defaultComment.urNum === mapUrObj.getAttribute('type'));
if (_selUr.urOpen && commentNum) {
if (
((!mapUrObj.getAttribute('description') || (mapUrObj.getAttribute('description').toLowerCase() === 'reported map issue'))
&& (_settings.perCommentListSettings[_currentCommentList].autoSetNewUrComment || (_restrictionsEnforce.autoSetNewUrComment === true))
&& (_restrictionsEnforce.autoSetNewUrComment !== false))
|| (mapUrObj.getAttribute('description')
&& (_settings.perCommentListSettings[_currentCommentList].autoSetNewUrCommentWithDescription || (_restrictionsEnforce.autoSetNewUrCommentWithDescription === true))
&& ((_restrictionsEnforce.autoSetNewUrCommentWithDescription !== false))
&& (mapUrObj.getAttribute('type') !== 23))
|| ((_settings.perCommentListSettings[_currentCommentList].autoSetNewUrCommentSlur || (_restrictionsEnforce.autoSetNewUrCommentSlur === true))
&& (_restrictionsEnforce.autoSetNewUrCommentSlur !== false)
&& mapUrObj.getAttribute('type') === 23)
) {
if (_settings.autoClickOpenSolvedNi)
autoClickOpenSolvedNi(commentNum);
postUrComment(_commentList[commentNum].comment, false);
}
}
}
else if (_mapUpdateRequests[_selUr.urId].urceData.commentCount === 1) {
if (_selUr.urOpen
&& (_settings.perCommentListSettings[_currentCommentList].autoSetReminderUrComment || (_restrictionsEnforce.autoSetReminderUrComment === true))
&& (_restrictionsEnforce.autoSetReminderUrComment !== false)
&& _defaultComments.dr.commentNum
&& (_mapUpdateRequests[_selUr.urId].urceData.commentCount > 0)
&& (_settings.perCommentListSettings[_currentCommentList].reminderDays !== 0)
&& (_restrictionsEnforce.reminderDays !== 0)
&& ((_mapUpdateRequests[_selUr.urId].urceData.lastCommentDaysOld > (_settings.perCommentListSettings[_currentCommentList].reminderDays - 1))
|| (_mapUpdateRequests[_selUr.urId].urceData.lastCommentDaysOld > (_restrictionsEnforce.reminderDays - 1)))
&& (_mapUpdateRequests[_selUr.urId].urceData.lastCommentBy > 0)
) {
if (_settings.autoClickOpenSolvedNi)
autoClickOpenSolvedNi(_defaultComments.dr.commentNum);
postUrComment(_commentList[_defaultComments.dr.commentNum].comment, false);
}
}
if (_settings.autoCenterOnUr)
recenterOnUr(_selUr.urId);
doSpinner('handleUpdateRequestContainer', false);
}
function checkValue() {
const varsFound = this.value.match(/\B\$\S*\$\B/gm) || this.value.match(/(\$SELSEGS|\$USERNAME|\$URD)/gm);
if (varsFound) {
let title;
if (this.value.includes('$SELSEGS'))
title = I18n.t('urce.prompts.SelSegsFound');
else if (this.value.includes('$PLACE_ADDRESS$'))
title = I18n.t('urce.prompts.PlaceAddressFound');
else if (this.value.inclues('$PLACE_NAME$'))
title = I18n.t('urce.prompts.PlaceNameFound');
else
title = I18n.t('urce.prompts.VarFound').replaceAll('$VARSFOUND$', varsFound.join(', '));
document.querySelector('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-form .send-button').setAttribute('title', title);
}
else if (document.querySelector('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-form .send-button').disabled) {
document.querySelector('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-form .send-button').disabled = false;
document.querySelector('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-form .send-button').setAttribute('title', '');
}
}
async function handleClickedComment(commentNum, doubleClick) {
_selUr.doubleClick = doubleClick;
const domElement = await getDomElement('textarea[id^=wz-textarea-]', '#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-text');
if (!domElement) {
logWarning('No comment box found after clicking a comment from the list.');
WazeWrap.Alerts.info(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.NoCommentBox'));
return;
}
if (doubleClick)
domElement.addEventListener('autoclicksendbutton', autoClickSendButton);
if (_settings.autoClickOpenSolvedNi && _selUr.urOpen)
autoClickOpenSolvedNi(commentNum);
postUrComment(_commentList[commentNum].comment, doubleClick);
}
function autoSwitchToUrceTab() {
_restoreDrawerTab = document.getElementById('drawer')?.querySelector('[selected="true"]')?.querySelector('.w-icon') || 'none-selected';
if (!_restoreDrawerTab.classList?.contains('w-icon-script'))
document.querySelector('.w-icon-script').click();
_restoreTab = _restoreTab || document.querySelector('#user-tabs .nav-tabs .active > a');
_restoreTabPosition = _restoreTabPosition || document.querySelector('#user-info .tab-content').scrollTop;
document.querySelector('img#urceIcon').closest('a[href^="#userscript"').click();
document.querySelector('a[href="#panel-urce-comments"]').click();
document.querySelector('#user-info .tab-content').scrollTop = 0;
document.querySelector('#user-info .tab-content .URCE-divTabs').scrollTop = 0;
}
function autoSwitchToPrevTab() {
if (!document.querySelector('#panel-container .show')) {
if (_restoreDrawerTab?.classList?.contains('w-icon-script')) {
if (_restoreTab) {
_restoreTab.click();
document.querySelector('#user-info .tab-content').scrollTop = _restoreTabPosition;
}
}
else if (_restoreDrawerTab && _restoreDrawerTab !== 'none-selected') {
_restoreDrawerTab.click();
}
else if (_restoreDrawerTab === 'none-selected') {
document.querySelector('.w-icon-script').click();
}
_restoreDrawerTab = undefined;
_restoreTab = undefined;
_restoreTabPosition = undefined;
}
}
async function autoCloseUrPanel() {
const areCommentsLoaded = await new Promise((resolve) => {
(function retry(tries, retryInt, maxNumTries) {
checkTimeout({ timeout: 'autoCloseUrPanel' });
if (tries > maxNumTries)
resolve({ error: true });
else if (document.querySelector('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .problem-data .more-info')?.classList?.contains('loading') === false)
resolve({ error: false });
else
_timeouts.autoCloseUrPanel = window.setTimeout(retry, ++tries, retryInt, maxNumTries);
}(1, 10, 200));
});
if (areCommentsLoaded.error) {
handleReadyError(false, false, '', false, '');
return Promise.resolve();
}
W.problemsController.editController.destroy();
return Promise.resolve();
}
async function autoClickSendButton() {
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-form .send-button'))?.click();
(await getDomElement('textarea[id^=wz-textarea-]', '#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-text'))
?.removeEventListener('autoclicksendbutton', autoClickSendButton);
}
async function autoClickOpenSolvedNi(commentNum) {
if ((_commentList[commentNum].urstatus === 'notidentified') && (_selUr.newStatus !== 'notidentified'))
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit .actions .content .controls-container input[value="not-identified"]'))?.click();
else if ((_commentList[commentNum].urstatus === 'solved') && (_selUr.newStatus !== 'solved'))
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit .actions .content .controls-container input[value="solved"]'))?.click();
else if ((_commentList[commentNum].urstatus === 'open') && ((_selUr.newStatus === 'solved') || (_selUr.newStatus === 'notidentified')))
(await getDomElement('#panel-container wz-card[class^="panel"].problem-edit .actions .content .controls-container input[value="open"]'))?.click();
}
function autoZoomIn() {
if (W.map.getOLMap().getZoom() < 17) {
/** 2024.03.22: The location object is no longer x: y: with 900913 coordinates. THis was changed in WME some time ago but never fixed in URC-E.
*
const urGeo = W.model.mapUpdateRequests.getObjectById(_selUr.urId).getLocation(),
lonlat = new OpenLayers.LonLat(urGeo.x, urGeo.y);
*/
const [x, y] = W.model.mapUpdateRequests.getObjectById(_selUr.urId).getLocation().coordinates,
lonlat = WazeWrap.Geometry.ConvertTo900913(x, y);
W.map.getOLMap().moveTo(lonlat, 17);
}
}
function autoZoomOut() {
if (_restoreZoom && !document.querySelector('#panel-container .show')) {
if (_restoreZoom !== W.map.getOLMap().getZoom()) {
W.map.getOLMap().zoomTo(_restoreZoom);
_restoreZoom = null;
}
}
}
function convertTimeOfDayToCasual(hour) {
let casualText;
if ((hour > 20) || (hour < 4))
casualText = I18n.t('urce.time.Night');
else if ((hour > 3) && (hour < 12))
casualText = I18n.t('urce.time.Morning');
else if ((hour > 11) && (hour < 18))
casualText = I18n.t('urce.time.Afternoon');
else if ((hour > 17) && (hour < 21))
casualText = I18n.t('urce.time.Evening');
else
casualText = '';
return casualText;
}
function formatText(text = '', replaceVars = false, shortcutClicked = false, urId = -1) {
if (!(urId > 0) && _selUr?.urId)
({ urId } = _selUr);
if (replaceVars && shortcutClicked && text.includes('$SELSEGS')) {
const selFeatures = W.selectionManager.getSelectedDataModelObjects(),
getStreetName = function (address) {
if (!address.isEmptyStreet())
return address.getStreetName();
if (address.getAltStreets().length > 0) {
const altStreets = address.getAltStreets();
for (let i = 0, { length } = altStreets; i < length; i++) {
if (!altStreets[i].isEmptyStreet())
return altStreets[i].getStreetName();
}
}
return I18n.t('urce.tools.UnknownRoadName');
},
getCityName = function (address) {
if (!address.getCity().isEmpty())
return { cityName: address.getCityName(), inOrNear: I18n.t('urce.common.In') };
if (address.getAltStreets().length > 0) {
const altStreets = address.getAltStreets();
for (let i = 0, end = altStreets.length; i < end; i++) {
if (!altStreets[i].getCity().isEmpty())
return { cityName: altStreets[i].getCityName(), inOrNear: I18n.t('urce.common.Near') };
}
}
return { cityName: '', inOrNear: '' };
};
let output = '';
if ((selFeatures.length > 0) && (selFeatures.length < 3)) {
let street1Name = '',
street2Name = '',
inOrNear = '',
cityName = '';
for (let idx = 0, { length } = selFeatures; idx < length; idx++) {
if (selFeatures[idx].getType() === 'segment') {
if (selFeatures.length > 1) {
const address = selFeatures[idx].getAddress();
if (idx === 0) {
street1Name = getStreetName(address);
if (text.includes('$SELSEGS_WITH_CITY$'))
({ cityName, inOrNear } = getCityName(address));
}
else {
street2Name = getStreetName(address);
if ((cityName === '') && text.includes('$SELSEGS_WITH_CITY$'))
({ cityName, inOrNear } = getCityName(address));
}
}
else {
const address = selFeatures[idx].getAddress();
street1Name = getStreetName(address);
if (text.includes('$SELSEGS_WITH_CITY$'))
({ cityName, inOrNear } = getCityName(address));
}
}
}
if ((street1Name !== '') || (street2Name !== '')) {
if (street2Name !== '') {
if (cityName !== '')
output = I18n.t('urce.tools.IntersectionOfWithCity');
else
output = I18n.t('urce.tools.IntersectionOf');
}
else if (cityName !== '') {
output = I18n.t('urce.tools.SegmentWithCity');
}
else {
output = '$SEG1NAME$';
}
output = output.replaceAll('$SEGCITY$', cityName).replaceAll('$SEG1NAME$', street1Name).replaceAll('$SEG2NAME$', street2Name).replaceAll('$IN_NEAR$', inOrNear);
if (_settings.usePrepositionForSegmentNamesShortcut)
output = `${((selFeatures.length > 1) ? I18n.t('urce.common.Near') : I18n.t('edit.on').toLowerCase())} ${output}`;
text = text.replace(/\$SELSEGS(\$|_WITH_CITY\$)?/gm, output);
}
else {
WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.SelSegsInsertError'));
}
}
else {
WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.SelSegsInsertError'));
}
}
if (replaceVars && (urId > -1)) {
const mapUrObj = W.model.mapUpdateRequests.getObjectById(urId);
if (/(\$(CURRENTDATE_DAY_OF_WEEK|CURRENTDATE_DATE|CURRENTDATE_DATE_CASUAL|CURRENTDATE_TIME|CURRENTDATE_TIME_CASUAL)\$)/gm.test(text)) {
if (text.includes('$CURRENTDATE_DAY_OF_WEEK$'))
text = text.replaceAll('$CURRENTDATE_DAY_OF_WEEK$', new Date().toLocaleDateString(I18n.currentLocale(), { weekday: 'long' }));
if (text.includes('$CURRENTDATE_DATE$'))
text = text.replaceAll('$CURRENTDATE_DATE$', new Date().toLocaleDateString(I18n.currentLocale(), { month: '2-digit', day: '2-digit', year: 'numeric' }));
if (text.includes('$CURRENTDATE_DATE_CASUAL$'))
text = text.replaceAll('$CURRENTDATE_DATE_CASUAL$', new Date().toLocaleDateString(I18n.currentLocale(), { month: 'long', day: '2-digit' }));
if (text.includes('$CURRENTDATE_TIME$'))
text = text.replaceAll('$CURRENTDATE_TIME$', new Date().toLocaleDateString(I18n.currentLocale(), { hour: '2-digit', minute: '2-digit', timeZoneName: 'short' }));
if (text.includes('$CURRENTDATE_TIME_CASUAL$'))
text = text.replaceAll('$CURRENTDATE_TIME_CASUAL$', convertTimeOfDayToCasual(new Date().getHours()));
}
if (/(\$(DRIVEDATE_DAY_OF_WEEK|DRIVEDATE_DATE|DRIVEDATE_DATE_CASUAL|DRIVEDATE_DAYS_AGO|DRIVEDATE_TIME|DRIVEDATE_TIME_CASUAL|DRIVEDATE_TIME_CASUALMODE)\$)/gm.test(text)) {
if (mapUrObj) {
if (text.includes('$DRIVEDATE_DAY_OF_WEEK$')) {
if (mapUrObj.getDriveDate() > -1) {
text = text.replaceAll(
'$DRIVEDATE_DAY_OF_WEEK$',
new Date(mapUrObj.getDriveDate()).toLocaleDateString(I18n.currentLocale(), { weekday: 'long' })
);
}
else {
text = text.replaceAll('$DRIVEDATE_DAY_OF_WEEK$', '');
}
}
if (text.includes('$DRIVEDATE_DATE$')) {
if (mapUrObj.getDriveDate() > -1) {
text = text.replaceAll(
'$DRIVEDATE_DATE$',
new Date(mapUrObj.getDriveDate()).toLocaleDateString(
I18n.currentLocale(),
{ month: '2-digit', day: '2-digit', year: 'numeric' }
)
);
}
else {
text = text.replaceAll('$DRIVEDATE_DATE$', '');
}
}
if (text.includes('$DRIVEDATE_DATE_CASUAL$')) {
if (mapUrObj.getDriveDate() > -1) {
text = text.replaceAll(
'$DRIVEDATE_DATE_CASUAL$',
new Date(mapUrObj.getDriveDate()).toLocaleDateString(I18n.currentLocale(), { month: 'long', day: '2-digit' })
);
}
else {
text = text.replaceAll('$DRIVEDATE_DATE_CASUAL$', '');
}
}
if (text.includes('$DRIVEDATE_DAYS_AGO$')) {
if (_mapUpdateRequests[urId]?.urceData?.driveDaysOld > -1)
text = text.replaceAll('$DRIVEDATE_DAYS_AGO$', parseDaysAgo(_mapUpdateRequests[urId].urceData.driveDaysOld));
else
text = text.replaceAll('$DRIVEDATE_DAYS_AGO$', '');
}
if (text.includes('$DRIVEDATE_TIME$')) {
if (mapUrObj.getDriveDate() > -1) {
text = text.replaceAll(
'$DRIVEDATE_TIME$',
new Date(mapUrObj.getDriveDate()).toLocaleTimeString(
I18n.currentLocale(),
{ hour: '2-digit', minute: '2-digit', timeZoneName: 'short' }
)
);
}
else {
text = text.replaceAll('$DRIVEDATE_TIME$', '');
}
}
if (text.includes('$DRIVEDATE_TIME_CASUAL$')) {
if (mapUrObj.getDriveDate() > -1)
text = text.replaceAll('$DRIVEDATE_TIME_CASUAL$', convertTimeOfDayToCasual(new Date(mapUrObj.getDriveDate()).getHours()));
else
text = text.replaceAll('$DRIVEDATE_TIME_CASUAL$', '');
}
if (text.includes('$DRIVEDATE_TIME_CASUALMODE$')) {
if (_mapUpdateRequests[urId]?.urceData) {
const driveDaysAgo = _mapUpdateRequests[urId].urceData.driveDaysOld,
driveHour = new Date(mapUrObj.getDriveDate()).getHours();
let dayOfWeek = new Date(mapUrObj.getDriveDate()).getDay(),
casualText;
if ((driveDaysAgo < 21) && (driveHour > -1) && (driveHour < 4))
dayOfWeek = (dayOfWeek > 0) ? dayOfWeek - 1 : 6;
if (driveDaysAgo === 0) {
if ((driveHour > 20) && (driveHour < 25))
casualText = I18n.t('urce.time.Tonight');
else if ((driveHour > -1) && (driveHour < 4))
casualText = I18n.t('urce.time.LastNight');
else
casualText = I18n.t('urce.time.ThisCasualTOD').replaceAll('$THIS_CASUAL_TOD$', convertTimeOfDayToCasual(driveHour));
}
else if (driveDaysAgo === 1) {
if ((driveHour > 20) && (driveHour < 25))
casualText = I18n.t('urce.time.LastNight');
else if ((driveHour > -1) && (driveHour < 4))
casualText = I18n.t('urce.time.ThisWeekTOD').replaceAll('$CASUAL_TOD$', convertTimeOfDayToCasual(driveHour));
else
casualText = I18n.t('urce.time.YesterdayCasualTOD').replaceAll('$YESTERDAY_CASUAL_TOD$', convertTimeOfDayToCasual(driveHour));
casualText = casualText.replaceAll('$DAY_NAME$', I18n.translations[I18n.currentLocale()].date.day_names[dayOfWeek]);
}
else if ((driveDaysAgo > 1) && (driveDaysAgo < 21)) {
if ((driveDaysAgo > 1) && (driveDaysAgo < 7)) {
if ((driveDaysAgo === 6) && (driveHour > -1) && (driveHour < 4))
casualText = I18n.t('urce.time.LastWeekTOD').replaceAll('$CASUAL_TOD$', convertTimeOfDayToCasual(driveHour));
else
casualText = I18n.t('urce.time.ThisWeekTOD').replaceAll('$CASUAL_TOD$', convertTimeOfDayToCasual(driveHour));
}
else if ((driveDaysAgo > 6) && (driveDaysAgo < 14)) {
casualText = I18n.t('urce.time.LastWeekTOD').replaceAll('$CASUAL_TOD$', convertTimeOfDayToCasual(driveHour));
}
else if ((driveDaysAgo > 13) && (driveDaysAgo < 21)) {
casualText = I18n.t('urce.time.TwoWeeksAgo');
}
casualText = casualText.replaceAll('$DAY_NAME$', I18n.translations[I18n.currentLocale()].date.day_names[dayOfWeek]);
}
else if ((driveDaysAgo > 20) && (driveDaysAgo < 28)) {
casualText = I18n.t('urce.time.ThreeWeeksAgo');
}
else if ((driveDaysAgo > 27) && (driveDaysAgo < 61)) {
casualText = I18n.t('urce.time.AFewWeeksAgo');
}
else if ((driveDaysAgo > 60) && (driveDaysAgo < 121)) {
casualText = I18n.t('urce.time.ACoupleMonthsAgo');
}
else if (driveDaysAgo > 120) {
casualText = I18n.t('urce.time.AWhileBack');
}
else {
casualText = '';
}
text = text.replaceAll('$DRIVEDATE_TIME_CASUALMODE$', casualText);
}
else {
text = text.replaceAll('$DRIVEDATE_TIME_CASUALMODE$', '');
}
}
}
}
if (text.includes('$URD')) {
if (mapUrObj.getAttribute('description'))
text = text.replace(/("?\$URD\$?"?)+/gmi, `"${mapUrObj.getAttribute('description')}"`).replace(/\n+/gmi, '');
else
text = text.replace(/("?\$URD\$?"?)+/gmi, '');
}
if (text.includes('$CUSTOMTAGLINE$')) {
if (_settings.perCommentListSettings[_currentCommentList].customTagline.length > 0)
text = text.replaceAll('$CUSTOMTAGLINE$', formatText(_settings.perCommentListSettings[_currentCommentList].customTagline, true, shortcutClicked, urId));
else
text = text.replaceAll('$CUSTOMTAGLINE$', '');
}
if (text.includes('$URTYPE$'))
text = text.replaceAll('$URTYPE$', mapUrObj.getTypeText());
if (text.includes('$PERMALINK$')) {
if (mapUrObj) {
// 2024.03.21: The location object changed in previous version of WME, but URC-E wasn't fixed for it until now.
// const lonLat = WazeWrap.Geometry.ConvertTo4326(mapUrObj.getLocation().x, mapUrObj.getLocation().y),
const [lon, lat] = mapUrObj.getLocation().coordinates,
urlParams = new URLSearchParams(window.location.search);
const urPermalink = `https://${document.location.hostname.replace(/beta/i, 'www')}${document.location.pathname}?${(urlParams.get('env') ? `env=${urlParams.get('env')}&` : '')}lon=${lon.toFixed(5)}&lat=${lat.toFixed(5)}&s=20489175039&zoomLevel=17&mapUpdateRequest=${urId}`;
text = text.replaceAll('$PERMALINK$', urPermalink);
}
else {
text = text.replaceAll('$PERMALINK$', '');
}
}
if (text.includes('$URID$'))
text = text.replaceAll('$URID$', urId);
}
if (replaceVars && text.includes('$CLOSED_NOR_EMAIL_TAG$')) {
if ((_settings.perCommentListSettings[_currentCommentList].tagEmail.length > 0) && (W.loginManager.user.getUsername().length > 0))
text = text.replaceAll('$CLOSED_NOR_EMAIL_TAG$', `Since this report is closed, please send further correspondence to ${_settings.perCommentListSettings[_currentCommentList].tagEmail} and include ${W.loginManager.user.getUsername()} in the subject line.`);
else
text = text.replaceAll('$CLOSED_NOR_EMAIL_TAG$', '');
}
if (replaceVars && text.includes('$TAG_EMAIL$')) {
if (_settings.perCommentListSettings[_currentCommentList].tagEmail.length > 0)
text = text.replaceAll('$TAG_EMAIL$', _settings.perCommentListSettings[_currentCommentList].tagEmail);
else
text = text.replaceAll('$TAG_EMAIL$', '');
}
if (replaceVars && text.includes('$USERNAME')) {
if (W.loginManager.user.getUsername())
text = text.replace(/(\$USERNAME\$?)+/gmi, W.loginManager.user.getUsername());
else
text = text.replace(/(\$USERNAME\$?)+/gmi, '');
}
if (replaceVars && text.includes('$PLACE_NAME$')) {
const placeObj = W.selectionManager.getSelectedDataModelObjects()[0];
if (placeObj?.getType() === 'venue') {
if (placeObj.isResidential() === true)
text = text.replaceAll('$PLACE_NAME$', I18n.t('objects.venue.fields.residential'));
else
text = text.replaceAll('$PLACE_NAME$', ((placeObj.getName().length > 0) ? placeObj.getName() : I18n.t('urce.tools.UnknownVenueName')));
}
else {
WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.PlaceNameInsertError'));
}
}
if (replaceVars && text.includes('$PLACE_ADDRESS$')) {
const placeObj = W.selectionManager.getSelectedDataModelObjects()[0];
if ((placeObj?.type === 'venue') && placeObj.getAddress().format())
text = text.replaceAll('$PLACE_ADDRESS$', placeObj.getAddress().format());
else
WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.PlaceAddressInsertError'));
}
if (replaceVars && (_customReplaceVars?.length > 0)) {
_customReplaceVars.forEach((customReplaceVar) => {
if (text.includes(customReplaceVar.customVar))
text = text.replace(customReplaceVar.customVar, formatText(customReplaceVar.replaceText, true, shortcutClicked, urId));
});
}
return text.replace(/\\[r|n]+/gm, '\n');
}
async function handleClickedShortcut() {
const shortcut = this.id.substring(14);
doSpinner('handleClickedShortcut', true);
let domElement = await getDomElement('textarea[id^=wz-textarea-]', '#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-text');
if (!domElement) {
handleReadyError(false, true, 'handleClickedShortcut', true, 'UR panel comment box is missing.');
return;
}
const cursorPos = domElement.selectionStart,
currVal = domElement.value;
let newVal = currVal.slice(0, cursorPos),
useCurrVal = false,
replaceText;
if (shortcut === 'selSegs') {
if (/\$SELSEGS(\$|_WITH_CITY\$)?/.test(currVal)) {
useCurrVal = true;
replaceText = currVal;
}
else {
replaceText = '$SELSEGS$';
}
}
else if (shortcut === 'selSegsWithCity') {
const checkVal = currVal.match(/(\$SELSEGS(\$|(?!_))|\$SELSEGS_WITH_CITY\$?)/);
if (checkVal?.length > 0) {
useCurrVal = true;
replaceText = currVal.replace(checkVal[0], '$SELSEGS_WITH_CITY$');
}
else {
replaceText = '$SELSEGS_WITH_CITY$';
}
}
else if (shortcut === 'driveTime') {
replaceText = '$DRIVEDATE_TIME$';
}
else if (shortcut === 'driveDayOfWeek') {
replaceText = '$DRIVEDATE_DAY_OF_WEEK$';
}
else if (shortcut === 'driveDate') {
replaceText = '$DRIVEDATE_DATE$';
}
else if (shortcut === 'description') {
replaceText = '$URD$';
}
else if (shortcut === 'wazeUsername') {
replaceText = '$USERNAME$';
}
else if (shortcut === 'driveTimeCasual') {
replaceText = '$DRIVEDATE_TIME_CASUAL$';
}
else if (shortcut === 'driveDateCasual') {
replaceText = '$DRIVEDATE_DATE_CASUAL$';
}
else if (shortcut === 'driveDateTimeCasualMode') {
replaceText = '$DRIVEDATE_TIME_CASUALMODE$';
}
else if (shortcut === 'currentDayOfWeek') {
replaceText = '$CURRENTDATE_DAY_OF_WEEK$';
}
else if (shortcut === 'currentDate') {
replaceText = '$CURRENTDATE_DATE$';
}
else if (shortcut === 'currentDateCasual') {
replaceText = '$CURRENTDATE_DATE_CASUAL$';
}
else if (shortcut === 'currentTime') {
replaceText = '$CURRENTDATE_TIME$';
}
else if (shortcut === 'currentTimeCasual') {
replaceText = '$CURRENTDATE_TIME_CASUAL$';
}
else if (shortcut === 'urType') {
replaceText = '$URTYPE$';
}
else if (shortcut === 'customTagline') {
replaceText = '$CUSTOMTAGLINE$';
}
else if (shortcut === 'placeName') {
if (currVal.includes('$PLACE_NAME$')) {
useCurrVal = true;
replaceText = currVal;
}
else {
replaceText = '$PLACE_NAME$';
}
}
else if (shortcut === 'placeAddress') {
if (currVal.includes('$PLACE_ADDRESS$')) {
useCurrVal = true;
replaceText = currVal;
}
else {
replaceText = '$PLACE_ADDRESS$';
}
}
else if (shortcut === 'urPermalink') {
replaceText = '$PERMALINK$';
}
else if (shortcut === 'urId') {
replaceText = '$URID$';
}
else {
doSpinner('handleClickedShortcut', false);
return;
}
let outputText = formatText(replaceText, true, true, -1);
if ((((shortcut === 'selSegs') || (shortcut === 'selSegsWithCity')) && /\$SELSEGS\$?/gm.test(outputText))
|| ((shortcut === 'placeName') && outputText.includes('$PLACE_NAME$'))
|| ((shortcut === 'placeAddress') && outputText.includes('$PLACE_ADDRESS$'))
) {
doSpinner('handleClickedShortcut', false);
return;
}
if (outputText && !useCurrVal) {
if ((newVal.length > 0) && !/\s/.test(newVal.slice(-1)))
outputText = ` ${outputText}`;
if (currVal.slice(cursorPos).length > 0) {
if (!/[\t\f\v ]/.test(currVal.substring(cursorPos, 1)))
outputText = `${outputText} `;
}
newVal = `${newVal}${outputText}${currVal.slice(cursorPos)}`;
if (newVal.length > 2000) {
WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.CommentTooLong'));
doSpinner('handleClickedShortcut', false);
return;
}
domElement = await getDomElement('textarea[id^=wz-textarea-]', '#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-text');
if (!domElement) {
logWarning('Timed out waiting for DOM elements before setting value of comment box after clicking a shortcut with setSelectionRange.');
}
else {
domElement.value = newVal;
domElement.dispatchEvent(new Event('input'));
domElement.dispatchEvent(new KeyboardEvent('keyup'));
domElement.setSelectionRange(cursorPos + outputText.length, cursorPos + outputText.length);
domElement.focus();
}
}
else if (useCurrVal) {
domElement = await getDomElement('textarea[id^=wz-textarea-]', '#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-text');
if (!domElement) {
logWarning('Timed out waiting for DOM elements before setting value of comment box after clicking a shortcut without setSelectionRange.');
}
else {
domElement.value = outputText;
domElement.dispatchEvent(new Event('input'));
domElement.dispatchEvent(new KeyboardEvent('keyup'));
domElement.setSelectionRange(cursorPos + outputText.length, cursorPos + outputText.length);
domElement.focus();
}
}
doSpinner('handleClickedShortcut', false);
}
async function autoPostReminderComment(urId) {
if (W.map.getOLMap().getZoom() < 10)
return Promise.resolve({ error: true, message: 'zoomIn' });
const comment = formatText(_commentList[_defaultComments.dr.commentNum].comment, true, false, urId);
try {
if (/\B\$\S*\$\B/gm.test(comment) || /(\$SELSEGS|\$USERNAME|\$URD)/gm.test(comment))
throw new Error(`Did not auto-post reminder comment for urId ${urId} because a variable was not replaced.`);
if (!W.model.updateRequestSessions.getObjectById(urId)) {
const data = await W.controller.descartesClient.getUpdateRequestSessionsByIds([urId]);
if (data.updateRequestSessions.objects.length > 0)
W.model.mergeResponse(data);
else
throw new Error(`Failed to merge updateRequestSession for urId ${urId}`);
}
W.model.updateRequestSessions.getObjectById(urId).addComment(comment);
W.model.mapUpdateRequests.getObjectById(urId).setAttribute('autoSentReminder', true);
return Promise.resolve({ error: false });
}
catch (error) {
W.model.mapUpdateRequests.getObjectById(urId).setAttribute('autoSentReminder', false);
return Promise.resolve({ error: true, message: error });
}
}
async function postUrComment(comment, doubleClick) {
doSpinner('postUrComment', true);
let commentOutput,
cursorPos,
newCursorPos,
postNls = 0;
let domElement = await getDomElement('textarea[id^=wz-textarea-]', '#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-text');
if (!domElement) {
logError('UR panel comment box is missing at beginning of postUrComment function.');
handleReadyError(false, true, 'postUrComment', true, 'UR panel comment box is missing.');
return;
}
if (_settings.enableAppendMode && (domElement.value !== '') && !doubleClick) {
cursorPos = domElement.selectionStart;
newCursorPos = cursorPos;
const currVal = domElement.value;
let newVal = currVal.slice(0, cursorPos);
if ((newVal.length > 0) && /[\n\r]/.test(newVal.slice(-1))) {
if (!/[\n\r]/.test(newVal.slice(-2, -1))) {
newVal += '\n';
newCursorPos++;
}
}
else {
newVal += '\n\n';
newCursorPos += 2;
}
newVal += formatText(comment, true, false, -1);
if (currVal.slice(cursorPos).length > 0) {
if (/[\n\r]/.test(currVal.substring(cursorPos, 1))) {
if (!/[\n\r]/.test(currVal.substring(cursorPos + 1, 1))) {
newVal += '\n';
postNls++;
}
}
else {
newVal += '\n\n';
postNls += 2;
}
newVal += currVal.slice(cursorPos);
}
commentOutput = newVal;
}
else {
commentOutput = formatText(comment, true, false, -1);
}
if (commentOutput.length > 2000) {
WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.CommentTooLong'));
logError(I18n.t('urce.prompts.CommentTooLong'));
}
else {
domElement = await getDomElement('textarea[id^=wz-textarea-]', '#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-text');
if (!domElement) {
logError('Timed out waiting on text box to set value with setSelectionRange.');
handleReadyError(false, true, 'postUrComment', true, 'UR panel comment box is missing.');
return;
}
domElement.value = commentOutput;
domElement.dispatchEvent(new Event('input'));
if (doubleClick && !/\B\$\S*\$\B/gm.test(commentOutput) && !/(\$SELSEGS|\$USERNAME|\$URD)/gm.test(commentOutput)) {
domElement.dispatchEvent(new Event('autoclicksendbutton', { bubbles: true }));
}
else {
let selectionRange;
if (newCursorPos === undefined) {
if (_settings.perCommentListSettings[_currentCommentList].placeCursorAtStart)
selectionRange = 0;
else
selectionRange = 0 + comment.replace(/\\[r|n]+/gm, ' ').length + postNls;
}
else {
selectionRange = newCursorPos + comment.replace(/\\[r|n]+/gm, ' ').length + postNls;
}
domElement.dispatchEvent(new KeyboardEvent('keyup'));
domElement.setSelectionRange(selectionRange, selectionRange);
domElement.focus();
}
}
doSpinner('postUrComment', false);
}
function getUsernameAndRank(userId) {
const userObj = W.model.users.getObjectById(userId);
let username,
rank;
if (userObj) {
username = userObj.getAttribute('userName') || userId;
rank = userObj.getRank() + 1;
}
else {
username = userId;
rank = '?';
}
return { username, rank };
}
function parsePxString(pxString) {
return +pxString.split('px')[0];
}
function parseDaysAgo(days) {
if (days === 0)
return I18n.t('date.today');
if (days === 1)
return I18n.t('date.yesterday');
return I18n.t('common.time.ago', { time: I18n.t('common.time.days', { days }) });
}
function daysAgo(timeInMs) {
const today = new Date(),
referenceDate = new Date(timeInMs),
msInDay = 24 * 60 * 60 * 1000;
referenceDate.setHours(0, 0, 0, 0);
today.setHours(0, 0, 0, 0);
const numDaysAgo = Math.round((+today - +referenceDate) / msInDay);
return numDaysAgo;
}
/** 2024.03.21: Now that the UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085), don't have time to fix marker (un)stacking. Removing for now.
*
function isIdAlreadyUnstacked(urId) {
if (_markerStackArray.length === 0)
return false;
for (let idx = 0, { length } = _markerStackArray; idx < length; idx++) {
if (_markerStackArray[idx].urId === urId)
return true;
}
return false;
}
function stackListObj(urId, x, y) {
return { urId: +urId, x: +x, y: +y };
}
function restackMarkers() {
if (_markerStackArray.length === 0)
return;
let filter = true;
if ((_settings.disableFilteringAboveZoom && (W.map.getOLMap().getZoom() < _settings.disableFilteringAboveZoomLevel))
|| (_settings.disableFilteringBelowZoom && (W.map.getOLMap().getZoom() > _settings.disableFilteringBelowZoomLevel))
)
filter = false;
const markerMapCollection = W.map.getLayerByName('update_requests').markers,
markerModelCollection = Object.fromEntries(W.model.mapUpdateRequests.getObjectArray().map((o) => [o.getID(), o]));
if (markerMapCollection) {
for (let idx = 0, { length } = markerMapCollection; idx < length; idx++) {
const markerDataId = markerMapCollection[idx].element?.attributes?.['data-id'].value;
if (markerDataId && markerModelCollection.hasOwnProperty(markerDataId)) {
const markerModelObj = markerModelCollection[markerDataId];
if (markerModelObj.getLocation().urceRealX) {
const location = markerModelObj.getLocation();
location.x = location.urceRealX;
location.y = location.urceRealY;
delete (location.urceRealX);
delete (location.urceRealY);
}
if (!(filter
&& _settings.enableUrceUrFiltering
&& _mapUpdateRequests[markerDataId]?.urceData?.hideUr
&& (!((_selUr.urId === markerModelObj.getID()) && _settings.doNotHideSelectedUr))
&& (!((_mapUpdateRequests[markerDataId]?.urceData?.tagType !== -1) && _settings.doNotFilterTaggedUrs)))
) {
if (markerMapCollection[idx].element.style.display === 'none')
markerMapCollection[idx].element.style.display = '';
}
}
}
for (let idx = 0, { length } = _markerStackArray; idx < length; idx++) {
const markerStackArrayCurUrId = _markerStackArray[idx].urId.toString(),
markerObj = markerMapCollection.find((marker) => marker.element?.attributes?.['data-id']?.value === markerStackArrayCurUrId);
if (markerObj) {
markerObj.element.style.left = `${_markerStackArray[idx].x}px`;
markerObj.element.style.top = `${_markerStackArray[idx].y}px`;
}
}
_markerStackArray = [];
_unstackedMasterId = null;
}
}
function checkMarkerStacking(urId, unstackedX, unstackedY) {
urId = +urId;
if (!_settings.unstackMarkers || (isIdAlreadyUnstacked(urId) === true))
return;
doSpinner('checkMarkerStacking', true);
const stackList = [],
markerMapCollection = W.map.getLayerByName('update_requests').markers,
markerModelCollection = Object.fromEntries(W.model.mapUpdateRequests.getObjectArray().map((o) => [o.getID(), o]));
let offset = 0.000000001;
stackList.push(urId);
if (markerMapCollection) {
const layerShowHidden = W.map.getLayerByName('update_requests').showHidden;
for (let idx = 0, { length } = markerMapCollection; idx < length; idx++) {
const markerDataId = markerMapCollection[idx].element?.attributes?.['data-id'].value;
if (markerDataId && markerModelCollection.hasOwnProperty(markerDataId)) {
const iconDiv = markerMapCollection[idx].element;
if (!(iconDiv.classList.contains('recently-closed') && (layerShowHidden === false))
&& (iconDiv.style.display !== 'none')
&& (iconDiv.style.visibility !== 'hidden')
) {
if (+markerDataId !== urId) {
const xDiff = unstackedX - parsePxString(iconDiv.style.left),
yDiff = unstackedY - parsePxString(iconDiv.style.top),
distSquared = ((xDiff * xDiff) + (yDiff * yDiff));
if (distSquared < (_settings.unstackSensitivity * _settings.unstackSensitivity))
stackList.push(+markerDataId);
}
}
}
}
}
if (stackList.length > 0) {
if (stackList.length === 1)
logDebug('Single marker highlighted. Adjusting geometry properties to prevent recentering.');
else if (W.map.getOLMap().getZoom() < _settings.unstackDisableAboveZoom)
logDebug(`Zoom level is ${W.map.getOLMap().getZoom()} which is less than setting for disable above zoom: ${_settings.unstackDisableAboveZoom}. Adjusting geometry properties to prevent recentering.`);
else
logDebug(`${stackList.length} markers are stacked!`);
if (_unstackedMasterId !== urId) {
logDebug('Unstacked ID mismatch, relocating markers.');
restackMarkers();
_unstackedMasterId = urId;
_markerStackArray = [];
_markerStackArray.push(stackListObj(urId, unstackedX, unstackedY));
for (let idx = 0, { length } = stackList; idx < length; idx++) {
const thisUrId = stackList[idx],
markerObj = markerMapCollection.find((marker) => marker.element?.attributes?.['data-id']?.value === thisUrId.toString()),
iconDiv = markerObj?.element,
x = parsePxString(iconDiv.style.left),
y = parsePxString(iconDiv.style.top);
_markerStackArray.push(stackListObj(thisUrId, x, y));
if (!((W.map.getOLMap().getZoom() < _settings.unstackDisableAboveZoom) || (stackList.length === 1))) {
if (!markerModelCollection[thisUrId].getAttribute('urceRealX')) {
const location = markerModelCollection[thisUrId].getLocation();
location.urceRealX = (location.realX) ? location.realX : location.x;
location.x = (location.realX) ? (location.realX + offset) : (location.x + offset);
location.urceRealY = (location.realY) ? location.realY : location.y;
location.y = (location.realY) ? (location.realY + offset) : (location.y + offset);
offset += 0.000000001;
}
iconDiv.style.left = `${unstackedX}px`;
iconDiv.style.top = `${unstackedY}px`;
unstackedX += 10;
unstackedY -= 30;
}
}
if (!((W.map.getOLMap().getZoom() < _settings.unstackDisableAboveZoom) || (stackList.length === 1))) {
for (let idx = 0, { length } = markerMapCollection; idx < length; idx++) {
const markerDataId = markerMapCollection[idx].element?.attributes?.['data-id'].value;
if (markerDataId && !isIdAlreadyUnstacked(+markerDataId))
markerMapCollection[idx].element.style.display = 'none';
}
}
}
}
else {
restackMarkers();
}
doSpinner('checkMarkerStacking', false);
}
*/
async function markerMouseOver() {
if (_mouseIsDown)
return;
const popupDelayTime = (Date.now() + (_settings.urMarkerPopupDelay * 100)),
wazeFeature = W.userscripts.getDataModelByFeatureElement(this);
let popupX,
popupY;
// 2024.03.21: UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085).
// if (this.className.includes('user-generated')) {
if (wazeFeature?.type === 'mapUpdateRequest') {
// 2024.03.21: UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085).
// const markerId = this.attributes?.['data-id']?.value ? +this.attributes['data-id'].value : -1;
const markerId = wazeFeature.getID();
if ((markerId > 0) && ((_mousedOverMarkerId !== markerId) || (getComputedStyle(document.getElementById('urceDiv')).visibility === 'hidden'))) {
_mousedOverMarkerId = markerId;
const targetTab = `_urceTab_${Math.round(Math.random() * 1000000)}`,
popupXOffset = parsePxString(getComputedStyle(document.getElementById('sidebar')).width) + parsePxString(getComputedStyle(document.getElementById('drawer')).width),
// 2024.03.21: UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085).
// unstackedX = parsePxString(this.style.left),
// unstackedY = parsePxString(this.style.top);
unstackedX = +this.getAttribute('x'),
unstackedY = +this.getAttribute('y');
// 2024.03.21: Now that the UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085), don't have time to fix marker (un)stacking. Removing for now.
// checkMarkerStacking(markerId, unstackedX, unstackedY);
if (!_settings.disableUrMarkerPopup) {
doSpinner('markerMouseDown', true);
if (!_mapUpdateRequests[markerId]?.urceData)
await updateUrceData(getMapUrsObjArr([markerId]));
let divElemRoot,
urLink,
x,
y;
const docFrags = document.createDocumentFragment(),
popupDelay = (Date.now() > popupDelayTime) ? -1 : (popupDelayTime - Date.now()),
ulElem = createElem('ul'),
mapUrObj = W.model.mapUpdateRequests.getObjectById(markerId);
if (mapUrObj.getLocation().urceRealX)
x = mapUrObj.getLocation().urceRealX;
else if (mapUrObj.getLocation().realX)
x = mapUrObj.getLocation().realX;
else
([x] = mapUrObj.getLocation().coordinates);
if (mapUrObj.getLocation().urceRealY)
y = mapUrObj.getLocation().urceRealY;
else if (mapUrObj.getLocation().realY)
y = mapUrObj.getLocation().realY;
else
([, y] = mapUrObj.getLocation().coordinates);
// 2024.03.21: The location object changed in previous version of WME, but URC-E wasn't fixed for it until now.
// const urPos = WazeWrap.Geometry.ConvertTo4326(x, y);
urLink = document.location.href;
urLink = `${urLink.substring(0, urLink.indexOf('?zoom'))}?zoomLevel=17&lat=${y}&lon=${x}&mapUpdateRequest=${markerId}`;
popupX = unstackedX - parsePxString(W.map.getSegmentLayer().div.style.left) + popupXOffset + 20;
popupY = unstackedY - parsePxString(W.map.getSegmentLayer().div.style.top) + 20;
docFrags.appendChild(createElem('div', {
style: 'font-weight:bold;', textContent: `${I18n.t('problems.panel.titles.map_update_request')} (${markerId}): ${I18n.t(`update_requests.types.${mapUrObj.getAttribute('type')}`)}`
}));
if (!mapUrObj.getAttribute('description')) {
divElemRoot = createElem('div', { style: 'font-style:italic;', textContent: I18n.t('urce.mouseOver.NoDescription') });
}
else {
divElemRoot = createElem('div', {
textContent: mapUrObj.getAttribute('description')
});
}
docFrags.appendChild(divElemRoot);
if (_mapUpdateRequests[markerId].urceData.driveDaysOld > -1) {
divElemRoot = createElem('div', {
style: 'font-style:italic;', textContent: `${I18n.t('mte.edit.submitted')} ${parseDaysAgo(_mapUpdateRequests[markerId].urceData.driveDaysOld)} `
});
if (mapUrObj.getDriveDate() > -1) {
divElemRoot.textContent += `(${new Date(mapUrObj.getDriveDate()).toLocaleDateString('en-us')}`
+ ` ${new Date(mapUrObj.getDriveDate()).toLocaleTimeString('en-us')}) `;
}
if (mapUrObj.getAttribute('guestUserName')) {
divElemRoot.textContent += I18n.t('urce.mouseOver.ViaLivemap');
if (mapUrObj.getAttribute('guestUserName') !== '')
divElemRoot.textContent += ` by ${mapUrObj.getAttribute('guestUserName').replace(/<\/?[^>]+(>|$)/g, '')}`;
}
docFrags.appendChild(divElemRoot);
}
if ((mapUrObj.getAttribute('resolvedOn')) && (_mapUpdateRequests[markerId].urceData.resolveDaysAgo > -1)) {
docFrags.appendChild(createElem('div', {
style: 'font-style:italic;',
textContent: `${I18n.t('urce.urStatus.Closed')} ${parseDaysAgo(_mapUpdateRequests[markerId].urceData.resolveDaysAgo)} `
+ `(${new Date(mapUrObj.getAttribute('resolvedOn')).toLocaleDateString('en-us')}`
+ ` ${new Date(mapUrObj.getAttribute('resolvedOn')).toLocaleTimeString('en-us')})`
}));
divElemRoot = createElem('div', { style: 'font-style:italic;' });
divElemRoot.appendChild(createTextNode(`${I18n.t('urce.mouseOver.MarkedAs')} `));
if (mapUrObj.getResolutionState() === 0)
divElemRoot.appendChild(createTextNode(I18n.t('venues.update_requests.panel.solved')));
else if (mapUrObj.getResolutionState() === 1)
divElemRoot.appendChild(createTextNode(I18n.t('urce.urStatus.NotIdentified')));
else
divElemRoot.appendChild(createTextNode(I18n.t('segment.direction.0')));
if (mapUrObj.getAttribute('resolvedBy')) {
divElemRoot.appendChild(createTextNode(` ${I18n.t('element_history.changed_by')} `));
divElemRoot.appendChild(createElem('a', {
href: W.Config.user_profile.url + getUsernameAndRank(mapUrObj.getAttribute('resolvedBy')).username,
textContent: getUsernameAndRank(mapUrObj.getAttribute('resolvedBy')).username
}));
divElemRoot.appendChild(createTextNode(` (${getUsernameAndRank(mapUrObj.getAttribute('resolvedBy')).rank})`));
}
docFrags.appendChild(divElemRoot);
}
docFrags.appendChild(createElem('br'));
divElemRoot = createElem('div', { textContent: `${_mapUpdateRequests[markerId].urceData.commentCount} ${I18n.t('urce.tabs.Comments')}` });
if (!_mapUpdateRequests[markerId].urceData.commentsByMe && (_mapUpdateRequests[markerId].urceData.commentCount > 0))
divElemRoot.textContent += ` (${I18n.t('urce.mouseOver.NoneByMe')})`;
if ((_mapUpdateRequests[markerId].urceData.commentCount > 0) && (_mapUpdateRequests[markerId].urceData.lastCommentDaysOld > -1))
divElemRoot.textContent += `, ${I18n.t('element_history.actions.default.UPDATE')} ${parseDaysAgo(_mapUpdateRequests[markerId].urceData.lastCommentDaysOld)}`;
docFrags.appendChild(divElemRoot);
if (_mapUpdateRequests[markerId].urceData.commentCount > 0) {
divElemRoot = createElem('div');
divElemRoot.appendChild(createTextNode(`${I18n.t('urce.mouseOver.FirstComment')}: ${parseDaysAgo(_mapUpdateRequests[markerId].urceData.firstCommentDaysOld)} ${I18n.t('element_history.changed_by')} `));
if (_mapUpdateRequests[markerId].urceData.firstCommentBy === -1) {
divElemRoot.appendChild(createTextNode(I18n.t('conversation.reporter')));
}
else {
divElemRoot.appendChild(createElem('a', {
href: W.Config.user_profile.url + getUsernameAndRank(_mapUpdateRequests[markerId].urceData.firstCommentBy).username,
textContent: getUsernameAndRank(_mapUpdateRequests[markerId].urceData.firstCommentBy).username
}));
divElemRoot.appendChild(createTextNode(` (${getUsernameAndRank(_mapUpdateRequests[markerId].urceData.firstCommentBy).rank})`));
}
docFrags.appendChild(divElemRoot);
if (_mapUpdateRequests[markerId].urceData.commentCount > 1) {
divElemRoot = createElem('div');
divElemRoot.appendChild(createTextNode(`${I18n.t('urce.mouseOver.LastComment')}: ${parseDaysAgo(_mapUpdateRequests[markerId].urceData.lastCommentDaysOld)} ${I18n.t('element_history.changed_by')} `));
if (_mapUpdateRequests[markerId].urceData.lastCommentBy === -1) {
divElemRoot.appendChild(createTextNode(I18n.t('conversation.reporter')));
}
else {
divElemRoot.appendChild(createElem('a', {
href: W.Config.user_profile.url + getUsernameAndRank(_mapUpdateRequests[markerId].urceData.lastCommentBy).username,
textContent: getUsernameAndRank(_mapUpdateRequests[markerId].urceData.lastCommentBy).username
}));
divElemRoot.appendChild(createTextNode(` (${getUsernameAndRank(_mapUpdateRequests[markerId].urceData.lastCommentBy).rank})`));
}
docFrags.appendChild(divElemRoot);
}
divElemRoot = createElem('div', { textContent: `${I18n.t('urce.mouseOver.ReporterHasCommented')}: ` });
divElemRoot.appendChild(createElem('div', {
style: 'font-style:italic;display:inline-block;', textContent: _mapUpdateRequests[markerId].urceData.reporterHasCommented ? I18n.t('urce.common.Yes') : I18n.t('urce.common.No')
}));
docFrags.appendChild(divElemRoot);
}
docFrags.appendChild(createElem('hr'));
let liElem = createElem('li');
liElem.appendChild(createElem('a', {
id: '_urceOpenInNewTab', href: urLink, target: targetTab, textContent: I18n.t('urce.mouseOver.OpenInNewTab')
}, [{ mouseup: saveSettingsToStorage }]));
liElem.appendChild(createTextNode(' - '));
liElem.appendChild(createElem('a', {
id: '_urceRecenterSession', href: '#', 'data-id': markerId, textContent: I18n.t('urce.mouseOver.CenterInCurrentTab')
}, [{ click: recenterOnUr }]));
ulElem.appendChild(liElem);
const lmLink = document.getElementById('livemap-link')?.href || document.querySelector('.livemap-link')?.href || undefined;
if (lmLink) {
liElem = createElem('li');
liElem.appendChild(createElem('a', {
href: `${(lmLink.includes('?') ? lmLink.substring(0, lmLink.indexOf('?')) : lmLink)}?zoomLevel=17&lat=${y}&lon=${x}&layers=BTTTT`,
target: `${targetTab}_lmTab`,
textContent: I18n.t('urce.mouseOver.OpenInNewLivemapTab')
}));
ulElem.appendChild(liElem);
}
docFrags.appendChild(ulElem);
if (popupDelay < 0) {
handlePopup({
docFrags, popupX, popupY, markerId
});
}
else {
checkTimeout({ timeout: 'popupDelay' });
_timeouts.popupDelay = window.setTimeout(handlePopup, popupDelay, {
docFrags, popupX, popupY, markerId
});
}
doSpinner('markerMouseDown', false);
}
}
}
}
function markerMouseOut(evt) {
const newUrId = evt.relatedTarget?.attributes?.['data-id']?.value;
// 2024.03.21: Now that the UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085), don't have time to fix marker (un)stacking. Removing for now.
// if (((+newUrId > 0) && isIdAlreadyUnstacked(+newUrId)) ||
if (((evt.relatedTarget?.id === 'urceDiv') || evt?.relatedTarget?.id?.includes('urceCounts') || evt?.relatedTarget?.parentNode?.id?.includes('urce'))
)
return;
if (!+newUrId)
_mousedOverMarkerId = null;
hidePopup();
// 2024.03.21: Now that the UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085), don't have time to fix marker (un)stacking. Removing for now.
// restackMarkers();
}
/** 2023.05.16.01: Removed markerClick(evt) as it is no longer needed with the the _urMarkerObserver now watching for 'marker-selected' class mutation.
* This change could have taken place back when the MO was updated, but wasn't really discovered as unncessary until troubleshooting pillClick()
* (now gone) being a problem.
*
function markerClick(evt) {
const evtUrId = evt.target.attributes?.['data-id']?.value;
if (!_selUr.handling && _commentListLoaded && (!(_selUr.urId > 0) || (_selUr.urId !== +evtUrId))) {
_selUr = {
doubleClick: false,
handling: false,
newStatus: undefined,
urId: +evtUrId,
urOpen: false
};
logDebug(`Clicked UR marker for urId: ${_selUr.urId}`);
}
}
*/
function checkPopupTimeouts() {
checkTimeout({ timeout: 'popup' });
checkTimeout({ timeout: 'popupDelay' });
}
function dispatchPopupDoubleClick() {
hidePopup({ doubleClick: true });
}
function handlePopup(popupObj) {
if (_mousedOverMarkerId !== popupObj.markerId)
return;
const urceDiv = document.getElementById('urceDiv'),
urceDivContent = urceDiv.querySelector('.urceDivContent');
urceDivContent.replaceChildren(popupObj.docFrags);
Object.assign(urceDiv.style, { height: 'auto', width: 'auto' });
let rw = +urceDiv.clientWidth;
if (rw > (window.innerWidth * 0.45)) {
rw = (window.innerWidth * 0.45);
urceDiv.style.width = `${rw}px`;
}
const rh = +urceDiv.clientHeight;
if ((popupObj.popupX + rw) > window.innerWidth)
popupObj.popupX -= (rw + 20);
if ((popupObj.popupY + rh) > window.innerHeight)
popupObj.popupY -= (((popupObj.popupY + rh) - window.innerHeight) + 30);
popupObj.popupX = (popupObj.popupX < 0) ? 0 : popupObj.popupX;
popupObj.popupY = (popupObj.popupY < 0) ? 0 : popupObj.popupY;
Object.assign(urceDiv.style, { top: `${popupObj.popupY}px`, left: `${popupObj.popupX}px`, visibility: 'visible' });
checkTimeout({ timeout: 'popup' });
checkTimeout({ timeout: 'popupDelay' });
if (_settings.urMarkerPopupTimeout > 0)
_timeouts.popup = window.setTimeout(hidePopup, (_settings.urMarkerPopupTimeout * 1000));
}
function hidePopup(evt) {
const newUrId = evt?.relatedTarget?.attributes?.['data-id']?.value;
checkTimeout({ timeout: 'popup' });
if (getComputedStyle(document.getElementById('urceDiv')).visibility !== 'hidden')
document.getElementById('urceDiv').style.visibility = 'hidden';
// 2024.03.21: Now that the UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085), don't have time to fix marker (un)stacking. Removing for now.
// if (((+newUrId > 0) && isIdAlreadyUnstacked(+newUrId)) ||
if (((evt?.relatedTarget?.id === 'urceDiv') || evt?.relatedTarget?.id?.includes('urceCounts') || evt?.relatedTarget?.parentNode?.id?.includes('urce'))
) {
if (!evt?.doubleClick)
return;
}
if (!+newUrId && (evt?.type === 'mouseleave') && ((evt?.target?.id === 'urceDiv') || (evt?.target?.offsetParent?.id === 'urceDiv')))
_mousedOverMarkerId = null;
// 2024.03.21: Now that the UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085), don't have time to fix marker (un)stacking. Removing for now.
// if (!_mousedOverMarkerId)
// restackMarkers();
}
async function openUrPanel(urId = -1, closeUrPanel = false) {
if (!(urId > 0) && _selUr?.urId)
({ urId } = _selUr);
if (!(urId > 0))
return;
if (closeUrPanel)
await autoCloseUrPanel();
const t = (_settings.replaceNextWithDoneButton)
? { showNext: false, nextButtonString: I18n.t('problems.panel.done') }
: { showNext: true, nextButtonString: I18n.t('problems.panel.next') };
if (urId !== _selUr.urId) {
_selUr = {
doubleClick: false,
handling: false,
newStatus: undefined,
urId,
urOpen: false
};
}
W.reqres.request('problems:browse', $extend(t, { problem: W.model.mapUpdateRequests.getObjectById(urId) }));
}
function recenterOnUr(urId = -1) {
if (this?.classList?.contains('focus'))
urId = _selUr.urId;
if (this?.attributes?.['data-id']?.value)
urId = +this.attributes['data-id'].value;
if (urId < 0)
return;
if (this?.id === '_urceRecenterSession')
openUrPanel(urId, true);
hidePopup();
const urGeo = W.model.mapUpdateRequests.getObjectById(urId).getLocation();
W.map.setCenter({ lon: urGeo.x, lat: urGeo.y });
}
function addCustomMarker(urId, urOpen, customType, node) {
let useCustomMarker = false;
if (customType === 0)
useCustomMarker = _settings.customMarkersRoadworks;
if (customType === 1)
useCustomMarker = _settings.customMarkersConstruction;
if (customType === 2)
useCustomMarker = _settings.customMarkersClosures;
if (customType === 3)
useCustomMarker = _settings.customMarkersEvents;
if (customType === 4)
useCustomMarker = _settings.customMarkersNotes;
if (customType === 5)
useCustomMarker = _settings.customMarkersWslm;
if (customType === 6)
useCustomMarker = _settings.customMarkersBog;
if (customType === 7)
useCustomMarker = _settings.customMarkersDifficult;
if (customType === 98)
useCustomMarker = _settings.customMarkersNativeSl;
if (customType === 99)
useCustomMarker = _settings.customMarkersCustom;
if (useCustomMarker)
return renderCustomMarker(urId, urOpen, customType, node);
return removeCustomMarker(urId);
}
function removeCustomMarker(urId) {
if (document.getElementById(`urceCustomMarker_${urId}`)) {
document.getElementById(`urceCustomMarker_${urId}`).remove();
return 'removed';
}
return false;
}
function customMarkersEnabledCheck() {
if (_settings.customMarkersRoadworks || _settings.customMarkersConstruction || _settings.customMarkersClosures || _settings.customMarkersEvents || _settings.customMarkersNotes
|| _settings.customMarkersWslm || _settings.customMarkersBog || _settings.customMarkersDifficult || _settings.customMarkersNativeSl || _settings.customMarkersCustom
)
return true;
return false;
}
function renderCustomMarker(urId, urOpen, customType, node) {
// URO+ Alt Markers, courtesy of URO+. Thank you!
const uroAltMarkers = [
// each altMarker has 4 variants: 0 = normal open, 1 = selected open, 2 = normal closed, 3 = selected closed
// 0: closure UR
[
'',
'',
'',
''
],
// 1: roadworks UR
[
'',
'',
'',
''
],
// 2: custom keyword UR
[
'',
'',
'',
''
],
// 3: note UR
[
'',
'',
'',
''
],
// 4: event UR
[
'',
'',
'',
''
],
// 5: WMSL/SLUR UR
[
'',
'',
'',
''
],
// 6: Elgin MP
[
'',
'',
'',
''
],
// 7: TrafficCast MP
[
'',
'',
'',
''
],
// 8: TrafficMaster MP
[
'',
'',
'',
''
],
// 9: CalTrans
[
'',
'',
'',
''
],
// 10: TfL
[
'',
'',
'',
''
],
// 11: BOG
[
'',
'',
'',
''
],
// 12: Difficult turn
[
'',
'',
'',
''
]
];
const docFrags = document.createDocumentFragment(),
spanElem = createElem('span', { id: `urceCustomMarker_${urId}`, style: 'position:absolute;pointer-events:none;top:-3px;left:-2px;' }),
status = document.getElementById(`urceCustomMarker_${urId}`) ? 'updated' : 'added';
spanElem.appendChild(createElem('img', { src: uroAltMarkers[getCustomMarkerIdx(customType)][(!urOpen) ? 2 : 0] }));
docFrags.appendChild(spanElem);
node.appendChild(docFrags);
return status;
}
function getCustomMarkerIdx(customType) {
if (customType === 0) // ROADWORKS
return 1;
if (customType === 1) // CONSTRUCTION
return 1;
if (customType === 2) // CLOSURE
return 0;
if (customType === 3) // EVENT
return 4;
if (customType === 4) // NOTE
return 3;
if (customType === 5) // WSLM
return 5;
if (customType === 6) // BOG
return 11;
if (customType === 7) // DIFFICULT
return 12;
if (customType === 98) // Native speed limit URs
return 5;
if (customType === 99) // custom text
return 2;
return -1;
}
function convertTagToCustomType(tag) {
if (tag === 'ROADWORKS')
return 0;
if (tag === 'CONSTRUCTION')
return 1;
if (tag === 'CLOSURE')
return 2;
if (tag === 'EVENT')
return 3;
if (tag === 'NOTE')
return 4;
if (tag === 'WSLM')
return 5;
if ((tag === 'BOG') || (tag === 'BOTG'))
return 6;
if (tag === 'DIFFICULT')
return 7;
return -1;
}
function updateUrMapMarkers(mUrsObjArr, filter) {
const closeDays = _restrictionsEnforce.closeDays || _settings.perCommentListSettings[_currentCommentList].closeDays || 7,
markerChanges = {
markers: {
hidden: [],
unhidden: [],
missing: []
},
pills: {
added: [],
updated: [],
removed: []
},
customMarkers: {
added: [],
updated: [],
removed: []
}
};
mUrsObjArr.forEach((mUrObj) => {
const mUrObjUrId = mUrObj.getID(),
// 2024.03.21: UR markers layer is no longer a marker layer and is a feature layer in latest WME (v2.217-13-g1d79af085).
// marker = document.querySelector(`.map-problem.user-generated[data-id="${mUrObjUrId}"]`),
marker = W.userscripts.getFeatureElementByDataModel(mUrObj),
mUrObjUrceData = _mapUpdateRequests[mUrObjUrId];
if (marker && mUrObjUrceData?.urceData) {
if (filter
&& _settings.enableUrceUrFiltering
&& (mUrObjUrceData.urceData.hideUr || (_settings.hideOutsideEditableArea && !mUrObj.canEdit()))
&& (!((_selUr.urId === mUrObjUrId) && _settings.doNotHideSelectedUr))
&& (!((mUrObjUrceData.urceData.tagType !== -1) && _settings.doNotFilterTaggedUrs))
) {
markerChanges.markers.hidden.push(mUrObjUrId);
marker.style.display = 'none';
}
else {
if (!marker.dataset.urceHasListeners) {
marker.addEventListener('mouseover', markerMouseOver);
marker.addEventListener('mouseout', markerMouseOut);
marker.dataset.urceHasListeners = true;
}
if (marker.style.display === 'none') {
markerChanges.markers.unhidden.push(mUrObjUrId);
marker.style.display = '';
}
if (_settings.enableUrPillCounts || customMarkersEnabledCheck()) {
if (_settings.enableUrPillCounts) {
let tagContent = '',
tagOffset,
urCountBackground;
if (mUrObjUrceData.urceData.tagType !== -1)
urCountBackground = '#CCCCCC';
else if ((mUrObjUrceData.urceData.commentCount > 0) && (mUrObjUrceData.urceData.lastCommentDaysOld > (closeDays - 1)))
urCountBackground = '#FF8B8B';
else if (!mUrObjUrceData.urceData.commentsByMe && (mUrObjUrceData.urceData.lastCommentBy < 0) && (mUrObjUrceData.urceData.lastCommentDaysOld < closeDays))
urCountBackground = '#FFCC99';
else if (!mUrObjUrceData.urceData.commentsByMe && (mUrObjUrceData.urceData.lastCommentBy > -1) && (mUrObjUrceData.urceData.lastCommentDaysOld < closeDays))
urCountBackground = '#FFFF99';
else if (mUrObjUrceData.urceData.commentsByMe && (mUrObjUrceData.urceData.lastCommentBy < 0))
urCountBackground = '#79B5C7';
else if (_wmeUserId === mUrObjUrceData.urceData.lastCommentBy)
urCountBackground = '#FFFFFF';
else
urCountBackground = '#84FFFF';
if ((mUrObjUrceData.urceData.tagType !== -1) && _settings.doNotShowTagNameOnPill) {
if (mUrObjUrceData.urceData.commentCount > 0)
tagContent += `${mUrObjUrceData.urceData.commentCount}c `;
tagContent += `${((mUrObjUrceData.urceData.lastCommentDaysOld !== -1) ? mUrObjUrceData.urceData.lastCommentDaysOld : mUrObjUrceData.urceData.driveDaysOld)}d`;
tagOffset = (tagContent.length < 3) ? 0 : Math.round(tagContent.length * 2.28);
}
else if (mUrObjUrceData.urceData.tagType !== -1) {
tagContent += mUrObjUrceData.urceData.tagType;
if (mUrObjUrceData.urceData.commentCount > 0)
tagContent += ` ${mUrObjUrceData.urceData.commentCount}c`;
tagOffset = (tagContent.length < 10) ? Math.round(tagContent.length * 2.86) : Math.round(tagContent.length * 3.33);
}
else {
if (mUrObjUrceData.urceData.commentCount > 0)
tagContent += `${mUrObjUrceData.urceData.commentCount}c `;
tagContent += `${((mUrObjUrceData.urceData.lastCommentDaysOld !== -1) ? mUrObjUrceData.urceData.lastCommentDaysOld : mUrObjUrceData.urceData.driveDaysOld)}d`;
tagOffset = (tagContent.length < 3) ? 0 : Math.round(tagContent.length * 2.28);
}
tagOffset = `-${tagOffset}px`;
if (document.getElementById(`urceCounters-${mUrObjUrId}`)) {
markerChanges.pills.updated.push(mUrObjUrId);
document.getElementById(`urceCounters-${mUrObjUrId}`).remove();
}
else {
markerChanges.pills.added.push(mUrObjUrId);
}
const divContainer = createElem('div', {
id: `urceCounters-${mUrObjUrId}`, 'data-id': mUrObjUrId, style: 'clear:both; margin-bottom:10px;'
});
divContainer.dataset.urceHasListeners = true;
divContainer.appendChild(createElem('div', {
id: `urceCounters-${mUrObjUrId}-text`,
'data-id': mUrObjUrId,
class: 'urceCountersPill urceCounts',
style: `background-color:${urCountBackground}; right:${tagOffset};`,
textContent: tagContent
}));
marker.appendChild(divContainer);
}
else if (document.getElementById(`urceCounters-${mUrObjUrId}`)) {
markerChanges.pills.removed.push(mUrObjUrId);
document.getElementById(`urceCounters-${mUrObjUrId}`).remove();
}
let status;
if (customMarkersEnabledCheck()) {
if (mUrObjUrceData.urceData.customType > -1)
status = addCustomMarker(mUrObjUrId, mUrObj.getOpenState(), mUrObjUrceData.urceData.customType, marker);
else
status = removeCustomMarker(mUrObjUrId);
}
else {
status = removeCustomMarker(mUrObjUrId);
}
if (status)
markerChanges.customMarkers[status].push(mUrObjUrId);
}
else {
if (document.getElementById(`urceCounters-${mUrObjUrId}`)) {
markerChanges.pills.removed.push(mUrObjUrId);
document.getElementById(`urceCounters-${mUrObjUrId}`).remove();
}
removeCustomMarker(mUrObjUrId);
}
}
}
});
if (markerChanges.markers.hidden.length > 0)
logDebug(`Hid UR markers for UR(s): ${markerChanges.markers.hidden.join(', ')} (Total: ${markerChanges.markers.hidden.length})`);
if (markerChanges.markers.unhidden.length > 0)
logDebug(`Unhid UR markers for UR(s): ${markerChanges.markers.unhidden.join(', ')} (Total: ${markerChanges.markers.unhidden.length})`);
if (markerChanges.markers.missing.length > 0)
logDebug(`Missing UR markers for UR(s): ${markerChanges.markers.missing.join(', ')} (Total: ${markerChanges.markers.missing.length})`);
if (markerChanges.pills.added.length > 0)
logDebug(`Added marker pills for UR(s): ${markerChanges.pills.added.join(', ')} (Total: ${markerChanges.pills.added.length})`);
if (markerChanges.pills.updated.length > 0)
logDebug(`Updated marker pills for UR(s): ${markerChanges.pills.updated.join(', ')} (Total: ${markerChanges.pills.updated.length})`);
if (markerChanges.pills.removed.length > 0)
logDebug(`Updated marker pills for UR(s): ${markerChanges.pills.removed.join(', ')} (Total: ${markerChanges.pills.removed.length})`);
if (markerChanges.customMarkers.added.length > 0)
logDebug(`Added custom markers for UR(s): ${markerChanges.customMarkers.added.join(', ')} (Total: ${markerChanges.customMarkers.added.length})`);
if (markerChanges.customMarkers.updated.length > 0)
logDebug(`Updated custom markers for UR(s): ${markerChanges.customMarkers.updated.join(', ')} (Total: ${markerChanges.customMarkers.updated.length})`);
if (markerChanges.customMarkers.removed.length > 0)
logDebug(`Removed custom markers for UR(s): ${markerChanges.customMarkers.removed.join(', ')} (Total: ${markerChanges.customMarkers.removed.length})`);
}
async function updateUrceData(mUrsObjArr) {
const updateMarkersArr = [];
let reopenPanel = false;
if (!mUrsObjArr)
return Promise.resolve();
const eg = getOLMapExtent(),
processMUrObjs = [...mUrsObjArr],
reminderDays = _restrictionsEnforce.reminderDays || _settings.perCommentListSettings[_currentCommentList].reminderDays || 0,
closeDays = _restrictionsEnforce.closeDays || _settings.perCommentListSettings[_currentCommentList].closeDays || 7,
tagRegex = /^.*?\[(ROADWORKS|CONSTRUCTION|CLOSURE|EVENT|NOTE|WSLM|BOG|BOTG|DIFFICULT)\].*$/gim,
tagUsernameRegex = new RegExp(` ${W.loginManager.user.getUsername()} `, 'gi'),
getUserIdsToCheck = (usersToCheck) => {
if (usersToCheck.length > 0) {
// eslint-disable-next-line array-callback-return, consistent-return
const userIdsArr = usersToCheck.split(',').map((username) => {
const userArr = W.model.users.getObjectArray().filter((o) => o.getAttribute('userName') === username);
if (userArr.length > 0)
return userArr[0].getID();
});
return userIdsArr;
}
return [];
},
checkCommentedUsers = (userIdsToCheck, userIds) => {
let retVal = false;
userIdsToCheck.forEach((userId) => {
if (userIds.includes(+userId))
retVal = true;
});
return retVal;
};
while (processMUrObjs.length > 0) {
const autoSentRemindersFor = [],
// 2023.04.05.01: Lowered to 400 due to request header size limits experienced by ojlaw
chunk = processMUrObjs.splice(0, 400),
urIds = chunk.map((a) => a.getID());
let includingKeyword,
keywordIncludingRegex,
notIncludingKeyword,
keywordNotIncludingRegex;
logDebug(`Updating urceData for urIds: ${urIds.join(', ')} (Total Count: ${urIds.length})`);
if (_settings.hideByKeywordIncludingKeyword.length > 0) {
includingKeyword = _settings.hideByKeywordIncludingKeyword.trim(); // Make regex compat ... .replace(/[-[\]{}()*+?.,\\^$|#\s]/g, '\\$&');
if (_settings.hideByKeywordCaseInsensitive)
keywordIncludingRegex = new RegExp(includingKeyword, 'gim');
else
keywordIncludingRegex = new RegExp(includingKeyword, 'gm');
}
if (_settings.hideByKeywordNotIncludingKeyword.length > 0) {
notIncludingKeyword = _settings.hideByKeywordNotIncludingKeyword.trim(); // Make regex compat ... .replace(/[-[\]{}()*+?.,\\^$|#\s]/g, '\\$&');
if (_settings.hideByKeywordCaseInsensitive)
keywordNotIncludingRegex = new RegExp(notIncludingKeyword, 'gim');
else
keywordNotIncludingRegex = new RegExp(notIncludingKeyword, 'gm');
}
// eslint-disable-next-line no-await-in-loop
const urSessions = await getUpdateRequestSessions(urIds);
for (let idx = 0, { length } = chunk; idx < length; idx++) {
const urId = chunk[idx].getAttribute('id'),
urSessionsObj = urSessions[urId];
if (urSessionsObj) {
const comments = urSessionsObj.getAttribute('comments');
let autoSendReminder = false;
let urceData = {
commentCount: comments.length,
commentsByMe: false,
commentUserIds: [],
containsSquareBrackets: false,
customType: -1,
driveDaysOld: (chunk[idx].getDriveDate()) ? daysAgo(chunk[idx].getDriveDate()) : -1,
firstCommentBy: -2,
firstCommentDaysOld: -1,
fullText: '',
hideByStatusClosedBy: false,
hideUr: false,
hideWithCommentBy: false,
hideWithoutCommentBy: false,
inMapExtent: eg.intersectsBounds(chunk[idx].getOLGeometry().getBounds()),
keywordIncluding: false,
keywordNotIncluding: false,
lastCommentBy: -2,
lastCommentDaysOld: -1,
needsClosed: false,
needsReminder: false,
reporterHasCommented: false,
resolveDaysAgo: (chunk[idx].getAttribute('resolvedOn')) ? daysAgo(chunk[idx].getAttribute('resolvedOn')) : undefined,
tagType: -1,
waiting: false
};
if (urceData.commentCount > 0) {
urceData.firstCommentDaysOld = daysAgo(comments[0].createdOn);
urceData.firstCommentBy = +comments[0].userID;
urceData.lastCommentDaysOld = daysAgo(comments[(urceData.commentCount - 1)].createdOn);
urceData.lastCommentBy = +comments[(urceData.commentCount - 1)].userID;
urceData.fullText += (chunk[idx].getAttribute('description')) ? `${chunk[idx].getAttribute('description')} ` : '';
for (let commentIdx = 0, { commentCount } = urceData; commentIdx < commentCount; commentIdx++) {
urceData.fullText += `${comments[commentIdx].text} `;
urceData.commentUserIds.push(comments[commentIdx].userID);
}
if (/\[\s*\S+[\s\S]*\]/m.test(urceData.fullText))
urceData.containsSquareBrackets = true;
if (urceData.commentUserIds.includes(_wmeUserId))
urceData.commentsByMe = true;
if (urceData.commentUserIds.includes(-1))
urceData.reporterHasCommented = true;
if (chunk[idx].getOpenState() && urceData.commentCount === 1) {
if ((reminderDays !== 0) && (_restrictionsEnforce.reminderDays !== 0) && (urceData.lastCommentDaysOld > (reminderDays - 1))) {
if ((_settings.perCommentListSettings[_currentCommentList].autoSendReminders || (_restrictionsEnforce.autoSendReminders === true))
&& (_restrictionsEnforce.autoSendReminders !== false)
&& _defaultComments.dr.commentNum
&& (urceData.lastCommentBy > 1)
&& (_wmeUserId === urceData.lastCommentBy)
&& urceData.inMapExtent
&& !urceData.containsSquareBrackets
&& ((+W.loginManager.user.getRank() > 2) || ((+W.loginManager.user.getRank() === 2) && W.loginManager.user.getAttribute('isAreaManager')))
&& !chunk[idx].getAttribute('autoSentReminder')
)
autoSendReminder = true;
else
urceData.needsReminder = true;
}
else if (((reminderDays === 0) || (_restrictionsEnforce.reminderDays === 0)) && (urceData.lastCommentDaysOld > (closeDays - 1))) {
urceData.needsClosed = true;
}
else {
urceData.waiting = true;
}
}
if (chunk[idx].getOpenState() && (urceData.commentCount > 1)) {
if (urceData.lastCommentBy > 1) {
if ((closeDays > 0) && (urceData.lastCommentDaysOld > (closeDays - 1))) {
if (_wmeUserId === urceData.lastCommentBy)
urceData.needsClosed = true;
else if (urceData.lastCommentDaysOld < (reminderDays + closeDays))
urceData.waiting = true;
else if (urceData.lastCommentDaysOld > (reminderDays + closeDays - 1))
urceData.needsClosed = true;
}
else {
urceData.waiting = true;
}
}
}
}
else {
urceData.fullText += (chunk[idx].getAttribute('description')) ? chunk[idx].getAttribute('description') : '';
}
if (!chunk[idx].getOpenState() && _settings.hideByStatusClosedBy && (_settings.hideByStatusClosedByUsers.length > 0)) {
const userIdsToCheck = getUserIdsToCheck(_settings.hideByStatusClosedByUsers);
if ((userIdsToCheck.length > 0) && userIdsToCheck.includes(chunk[idx].getAttribute('resolvedBy')))
urceData.hideByStatusClosedBy = true;
}
if (_settings.hideWithCommentBy && (_settings.hideWithCommentByUsers.length > 0)) {
const userIdsToCheck = getUserIdsToCheck(_settings.hideWithCommentByUsers);
if ((userIdsToCheck.length > 0) && (urceData.commentUserIds.length > 0)) {
if (checkCommentedUsers(userIdsToCheck, urceData.commentUserIds) === true)
urceData.hideWithCommentBy = true;
}
}
if (_settings.hideWithoutCommentBy && (_settings.hideWithoutCommentByUsers.length > 0)) {
const userIdsToCheck = getUserIdsToCheck(_settings.hideWithoutCommentByUsers);
if ((userIdsToCheck.length > 0) && (urceData.commentUserIds.length > 0)) {
if (checkCommentedUsers(userIdsToCheck, urceData.commentUserIds) !== true)
urceData.hideWithoutCommentBy = true;
}
else if (urceData.commentUserIds.length === 0) {
urceData.hideWithoutCommentBy = true;
}
}
// eslint-disable-next-line no-control-regex
urceData.fullText = urceData.fullText.replace(/[\r\n\x0B\x0C\u0085\u2028\u2029]+/g, ' ');
urceData.tagType = tagRegex.test(urceData.fullText) ? urceData.fullText.replace(tagRegex, '$1') : -1;
if (urceData.tagType !== -1) {
urceData.customType = convertTagToCustomType(urceData.tagType);
}
else if (chunk[idx].getAttribute('type') === 23) {
urceData.customType = 98;
}
else if (_settings.customMarkersCustom && (_settings.customMarkersCustomText.length > 0) && new RegExp(`\\[${_settings.customMarkersCustomText.replace(/[[\]]+/gim, '')}\\]`).test(urceData.fullText)) {
urceData.tagType = 99;
urceData.customType = 99;
}
else {
urceData.customType = -1;
}
if (keywordIncludingRegex && keywordIncludingRegex.test(urceData.fullText))
urceData.keywordIncluding = true;
if (keywordNotIncludingRegex && !keywordNotIncludingRegex.test(urceData.fullText))
urceData.keywordNotIncluding = true;
if ((urceData.tagType === -1) && (chunk[idx].getAttribute('type') === 23))
urceData.customType = 98;
else if (urceData.tagType === -1)
urceData.customType = -1;
if ((urceData.tagType === -1) || _settings.replaceTagNameWithEditorName) {
if (_settings.replaceTagNameWithEditorName && tagUsernameRegex.test(urceData.fullText))
urceData.tagType = W.loginManager.user.getUsername();
else if (!urceData.tagType)
urceData.tagType = -1;
}
if (autoSendReminder) {
if (((urceData.customType > -1) && !_settings.perCommentListSettings[_currentCommentList].autoSendRemindersExceptTagged)
|| (urceData.customType === -1)
) {
try {
// eslint-disable-next-line no-await-in-loop
const autoPostReminderCommentResult = await autoPostReminderComment(urId);
if (autoPostReminderCommentResult.error) {
if (autoPostReminderCommentResult.message === 'zoomIn')
logDebug(`Did not auto post reminder comment due to zoom being less than 10 (Zoom: ${W.map.getOLMap().getZoom()}) for urId ${urId}.`);
else
logError(autoPostReminderCommentResult.message);
}
else {
autoSentRemindersFor.push(urId);
urceData = $extend({}, urceData, {
commentCount: urceData.commentCount + 1,
commentsByMe: true,
lastCommentDaysOld: 0,
lastCommentBy: _wmeUserId,
needsReminder: false,
waiting: true
});
}
const selectedMarkerUrId = document.querySelector('.update-requests .marker-selected')?.attributes?.['data-id']?.value;
if (+selectedMarkerUrId === _selUr.urId)
reopenPanel = true;
updateMarkersArr.push(chunk[idx]);
}
catch (error) {
logWarning(error);
urceData.needsReminder = true;
}
}
else {
urceData.needsReminder = true;
}
}
if ((_settings.hideWaiting && urceData.waiting)
|| (_settings.hideUrsCloseNeeded && urceData.needsClosed)
|| (_settings.hideUrsReminderNeeded && urceData.needsReminder)
|| (_settings.hideByStatusOpen && chunk[idx].getOpenState())
|| (_settings.hideByStatusClosed && !chunk[idx].getOpenState())
|| (_settings.hideByStatusNotIdentified && (chunk[idx].getResolutionState() === 1))
|| (_settings.hideByStatusSolved && (chunk[idx].getResolutionState() === 0))
|| (_settings.hideByStatusClosedBy && urceData.hideByStatusClosedBy)
// Types
|| (_settings.hideByTypeBlockedRoad && (chunk[idx].getAttribute('type') === 19))
|| (_settings.hideByTypeGeneralError && (chunk[idx].getAttribute('type') === 10))
|| (_settings.hideByTypeIncorrectAddress && (chunk[idx].getAttribute('type') === 7))
|| (_settings.hideByTypeIncorrectJunction && (chunk[idx].getAttribute('type') === 12))
|| (_settings.hideByTypeIncorrectRoute && (chunk[idx].getAttribute('type') === 8))
|| (_settings.hideByTypeIncorrectStreetPrefixOrSuffix && (chunk[idx].getAttribute('type') === 22))
|| (_settings.hideByTypeIncorrectTurn && (chunk[idx].getAttribute('type') === 6))
|| (_settings.hideByTypeMissingBridgeOverpass && (chunk[idx].getAttribute('type') === 13))
|| (_settings.hideByTypeMissingExit && (chunk[idx].getAttribute('type') === 15))
|| (_settings.hideByTypeMissingLandmark && (chunk[idx].getAttribute('type') === 18))
|| (_settings.hideByTypeMissingOrInvalidSpeedLimit && (chunk[idx].getAttribute('type') === 23))
|| (_settings.hideByTypeMissingRoad && (chunk[idx].getAttribute('type') === 16))
|| (_settings.hideByTypeMissingRoundabout && (chunk[idx].getAttribute('type') === 9))
|| (_settings.hideByTypeMissingStreetName && (chunk[idx].getAttribute('type') === 21))
|| (_settings.hideByTypeTurnNotAllowed && (chunk[idx].getAttribute('type') === 11))
|| (_settings.hideByTypeUndefined
&& (!chunk[idx].getAttribute('type')
|| (chunk[idx].getAttribute('type') > 23)
|| (chunk[idx].getAttribute('type') < 6)
|| (chunk[idx].getAttribute('type') === 17)
|| (chunk[idx].getAttribute('type') === 20)))
|| (_settings.hideByTypeWrongDrivingDirection && (chunk[idx].getAttribute('type') === 14))
// Tags
|| (_settings.hideByTaggedBog && (urceData.customType === 6))
|| (_settings.hideByTaggedClosure && (urceData.customType === 2))
|| (_settings.hideByTaggedConstruction && (urceData.customType === 1))
|| (_settings.hideByTaggedDifficult && (urceData.customType === 7))
|| (_settings.hideByTaggedEvent && (urceData.customType === 3))
|| (_settings.hideByTaggedNote && (urceData.customType === 4))
|| (_settings.hideByTaggedRoadworks && (urceData.customType === 0))
|| (_settings.hideByTaggedWslm && (urceData.customType === 5))
// Age of submission
|| (_settings.hideByAgeOfSubmissionLessThan && (urceData.driveDaysOld < _settings.hideByAgeOfSubmissionLessThanDaysOld))
|| (_settings.hideByAgeOfSubmissionMoreThan && (urceData.driveDaysOld > _settings.hideByAgeOfSubmissionMoreThanDaysOld))
// Following, description, comments
|| (_settings.hideFollowing && urSessionsObj.getAttribute('isFollowing'))
|| (_settings.hideNotFollowing && !urSessionsObj.getAttribute('isFollowing'))
|| (_settings.hideWithDescription
&& (chunk[idx]?.getAttribute('description')?.length > 0) && (chunk[idx].getAttribute('description') !== ''))
|| (_settings.hideWithoutDescription
&& (!chunk[idx]?.getAttribute('description') || (chunk[idx]?.getAttribute('description')?.length === 0) || (chunk[idx]?.getAttribute('description') === '')))
|| (_settings.hideWithCommentsFromMe && (urceData.commentsByMe))
|| (_settings.hideWithoutCommentsFromMe && (!urceData.commentsByMe))
|| (_settings.hideLastCommentByMe && (urceData.lastCommentBy === _wmeUserId))
|| (_settings.hideLastCommentNotByMe && (urceData.lastCommentBy !== _wmeUserId))
|| (_settings.hideLastCommentByReporter && (urceData.lastCommentBy === -1))
|| (_settings.hideLastCommentNotByReporter && (urceData.lastCommentBy > 0))
|| (_settings.hideByCommentCountLessThan && (urceData.commentCount < _settings.hideByCommentCountLessThanNumber))
|| (_settings.hideByCommentCountMoreThan && (urceData.commentCount > _settings.hideByCommentCountMoreThanNumber))
|| (_settings.hideByAgeOfFirstCommentLessThan && (urceData.commentCount > 0) && (urceData.firstCommentDaysOld < _settings.hideByAgeOfFirstCommentLessThanDaysOld))
|| (_settings.hideByAgeOfFirstCommentMoreThan && (urceData.commentCount > 0) && (urceData.firstCommentDaysOld > _settings.hideByAgeOfFirstCommentMoreThanDaysOld))
|| (_settings.hideByAgeOfLastCommentLessThan && (urceData.commentCount > 0) && (urceData.lastCommentDaysOld < _settings.hideByAgeOfLastCommentLessThanDaysOld))
|| (_settings.hideByAgeOfLastCommentMoreThan && (urceData.commentCount > 0) && (urceData.lastCommentDaysOld > _settings.hideByAgeOfLastCommentMoreThanDaysOld))
|| (_settings.hideByKeywordIncluding && urceData.keywordIncluding)
|| (_settings.hideByKeywordNotIncluding && urceData.keywordNotIncluding)
|| (_settings.hideWithCommentBy && urceData.hideWithCommentBy)
|| (_settings.hideWithoutCommentBy && urceData.hideWithoutCommentBy)
)
urceData.hideUr = true;
if (_settings.invertFilters) {
if (urceData.hideUr)
urceData.hideUr = false;
else
urceData.hideUr = true;
}
_mapUpdateRequests[urId] = { urceData };
}
else {
logDebug(`Could not load urSessionsObj for urID: ${urId}`);
}
}
if (autoSentRemindersFor.length > 0) {
logDebug(`Automatically sent reminder comments to urId(s): ${autoSentRemindersFor.join(', ')} (${autoSentRemindersFor.length})`);
WazeWrap.Alerts.info(_SCRIPT_SHORT_NAME, `${I18n.t('urce.prompts.ReminderMessageAuto')}: ${autoSentRemindersFor.join(', ')} (${autoSentRemindersFor.length})`);
}
}
if (updateMarkersArr.length > 0) {
const zoomLevel = W.map.getOLMap().getZoom(),
filter = !(((_settings.disableFilteringAboveZoom && (zoomLevel < _settings.disableFilteringAboveZoomLevel))
|| (_settings.disableFilteringBelowZoom && (zoomLevel > _settings.disableFilteringBelowZoomLevel))));
updateUrMapMarkers(updateMarkersArr, filter);
}
if (reopenPanel)
openUrPanel(0, true);
return Promise.resolve();
}
async function handleUrLayer(phase, filter, mUrsObjArr) {
if (!mUrsObjArr)
return Promise.resolve();
doSpinner('handleUrLayer', true);
const zoomLevel = W.map.getOLMap().getZoom();
filter = (filter !== undefined) ? filter : !(((_settings.disableFilteringAboveZoom && (zoomLevel < _settings.disableFilteringAboveZoomLevel))
|| (_settings.disableFilteringBelowZoom && (zoomLevel > _settings.disableFilteringBelowZoomLevel))));
if (phase === 'init')
logDebug('Checking for UR markers already present before URC-E completed initialization.');
else if (phase === 'init_end')
logDebug('Updating UR markers that appeared after initialization completed.');
else if (phase === 'close')
logDebug('Updating UR markers after closing UR panel.');
else if (phase === 'settingsToggle')
logDebug('Updating UR markers after a setting toggle.');
else if (phase === 'sendComment')
logDebug('Updating UR markers after sending a comment.');
else if (phase === 'overflow')
logDebug('Updating UR markers from being added through UR overflow handling.');
else
logDebug(`Updating UR markers that appeared after a ${phase} event.`);
mUrsObjArr.sort((a, b) => a.getID() - b.getID());
if (phase === 'init')
_markerCountOnInit = mUrsObjArr.length;
if ((phase === 'overflow') && _initialUrLayerScan) {
if (_markerCountOnInit > -1)
_markerCountOnInit += mUrsObjArr.length;
else
_markerCountOnInit = mUrsObjArr.length;
}
await updateUrceData(mUrsObjArr);
updateUrMapMarkers(mUrsObjArr, filter);
// 2023.04.05.01: Used to check for conditions and either run handleUrOverflow or panel alert box or remove it. But now using W.app.on('change:loadingIssueTrackerMapData') event listener.
if (phase !== 'overflow')
_filtersAppliedOnZoom = filter;
doSpinner('handleUrLayer', false);
return Promise.resolve();
}
function getOverflowUrsFromUrl(urlStr) {
return new Promise((resolve) => {
(async function retry(url, tries, toIndex) {
logDebug(`Getting URs from: ${urlStr} (Try: ${tries})`);
checkTimeout({ timeout: 'getOverflowUrsFromUrl', toIndex });
const errorObj = { error: false, url };
GM_xmlhttpRequest({
url,
method: 'GET',
onload(res) {
if (res.status < 400) {
const data = JSON.parse(res.responseText);
if (!data) {
if (tries > 10) {
resolve({ error: { reason: 'Too many retries.' }, url });
}
else {
log('Rate limited by Waze server. Retrying overflow request.');
_timeouts.getOverflowUrsFromUrl[toIndex] = window.setTimeout(retry, 100, url, ++tries, toIndex);
}
}
else {
data.url = url;
checkTimeout({ timeout: 'getOverflowUrsFromUrl', toIndex });
resolve(data);
}
}
else if (tries > 10) {
resolve({ error: { reason: 'Too many retries.' }, url });
}
else if (res.status === 429) {
log('Rate limited by Waze server. Retrying overflow request.');
_timeouts.getOverflowUrsFromUrl[toIndex] = window.setTimeout(retry, 100, url, ++tries, toIndex);
}
else {
logWarning(`HTTP request error: ${JSON.stringify(res)}`);
_timeouts.getOverflowUrsFromUrl[toIndex] = window.setTimeout(retry, 100, url, ++tries, toIndex);
}
},
onerror(res) {
if (tries > 10) {
resolve({ error: { reason: 'Too many retries.' }, url });
}
else if (res.status === 429) {
log('Rate limited by Waze server. Retrying overflow request.');
_timeouts.getOverflowUrsFromUrl[toIndex] = window.setTimeout(retry, 100, url, ++tries, toIndex);
}
else if (!errorObj.error) {
errorObj.error = (typeof res === 'object') ? res : { res };
resolve(errorObj);
}
}
});
}(urlStr, 1, getRandomId()));
});
}
function handleUrOverflow(evt) {
if (evt?.changed?.loadingIssueTrackerMapData || evt?.attributes?.loadingIssueTrackerMapData)
return;
if (W.map.getOLMap().getZoom() < 10) {
logDebug('UR overflow handling does not work with zoom levels < 10.');
return;
}
if (W.model.mapUpdateRequests.getObjectArray().length < 500) {
dismissAlertBoxInPanel(undefined, 9999);
return;
}
if (!_settings.enableUrOverflowHandling && (W.model.mapUpdateRequests.getObjectArray().length > 499)) {
const docFrags = document.createDocumentFragment();
docFrags.appendChild(createTextNode(I18n.t('urce.prompts.UrOverflowErrorWithoutOverflowEnabled')));
alertBoxInPanel(docFrags, undefined, true, 9999);
return;
}
doSpinner('handleUrOverflow', true);
const baseUrl = `https://${document.location.host}${W.Config.paths.features}?language=en&mapUpdateRequestFilter=`
+ `${((W.issueTrackerController.getOption('app').get('issueTrackerFilter').get('mapUpdateRequestsFilter').get('status') === 'BOTH') ? '3' : '1')}%2C0&bbox=`,
vpBounds = getOLMapExtent().transform(W.map.getProjectionObject(), new OpenLayers.Projection('EPSG:4326')),
vpBoundsFrom = { lon: vpBounds.left, lat: vpBounds.bottom },
vpBoundsTo = { lon: vpBounds.right, lat: vpBounds.top },
vpCenter = getOLMapExtent().getCenter().transform(W.map.getProjectionObject(), new OpenLayers.Projection('EPSG:4326')),
processData = (data) => {
if (data.error) {
logWarning(data.error);
}
else if (data.mapUpdateRequests?.objects?.length > 499) {
logDebug('More than 499 objects returned in overflow request, queueing sub quadrants for further checking.');
const bbox = data.url.split('bbox=')[1].split(','),
bboxFrom = WazeWrap.Geometry.ConvertTo900913(bbox[0], bbox[1]),
bboxTo = WazeWrap.Geometry.ConvertTo900913(bbox[2], bbox[3]),
subQuadCenter = WazeWrap.Geometry.ConvertTo4326((bboxFrom.lon - ((bboxFrom.lon - bboxTo.lon) / 2)), (bboxTo.lat - ((bboxTo.lat - bboxFrom.lat) / 2)));
[`${baseUrl}${subQuadCenter.lon.toFixed(6)},${subQuadCenter.lat.toFixed(6)},${bbox[2]},${bbox[3]}`,
`${baseUrl}${bbox[0]},${subQuadCenter.lat.toFixed(6)},${subQuadCenter.lon.toFixed(6)},${bbox[3]}`,
`${baseUrl}${bbox[0]},${bbox[1]},${subQuadCenter.lon.toFixed(6)},${subQuadCenter.lat.toFixed(6)}`,
`${baseUrl}${subQuadCenter.lon.toFixed(6)},${bbox[1]},${bbox[2]},${subQuadCenter.lat.toFixed(6)}`
].forEach((url) => {
_overflowUrsUrls.push(url);
getOverflowUrsFromUrl(url).then(processData);
});
}
else if (data.mapUpdateRequests?.objects?.length > 0) {
const overflowUrsToPut = [];
data.mapUpdateRequests.objects.forEach((mapUr) => {
if (!W.model.mapUpdateRequests.getObjectById(mapUr.id)) {
const NewUr = require('Waze/Feature/Vector/UpdateRequest'),
toPutUr = new NewUr(mapUr),
toPutPoint = new OpenLayers.Geometry.Point(mapUr.geometry.coordinates[0], mapUr.geometry.coordinates[1])
.transform(new OpenLayers.Projection('EPSG:4326'), W.map.getProjectionObject());
toPutUr.geometry = toPutPoint;
const toPutReqBounds = new OpenLayers.Geometry.Polygon(),
toPutBounds = new OpenLayers.Bounds(toPutPoint.x, toPutPoint.y, toPutPoint.x, toPutPoint.y);
toPutReqBounds.bounds = toPutBounds;
toPutUr.requestBounds = toPutReqBounds;
overflowUrsToPut.push(toPutUr);
}
});
if (overflowUrsToPut.length > 0) {
logDebug(`${overflowUrsToPut.length} URs added from overflow.`);
W.model.mapUpdateRequests.put(overflowUrsToPut);
}
else {
logDebug('All URs submitted for overflow processing already exist on map.');
}
}
_overflowUrsUrls.splice(_overflowUrsUrls.indexOf(data.url), 1);
if (_overflowUrsUrls.length === 0)
doSpinner('handleUrOverflow', false);
};
[`${baseUrl}${vpCenter.lon.toFixed(6)},${vpCenter.lat.toFixed(6)},${vpBoundsTo.lon.toFixed(6)},${vpBoundsTo.lat.toFixed(6)}`,
`${baseUrl}${vpBoundsFrom.lon.toFixed(6)},${vpCenter.lat.toFixed(6)},${vpCenter.lon.toFixed(6)},${vpBoundsTo.lat.toFixed(6)}`,
`${baseUrl}${vpBoundsFrom.lon.toFixed(6)},${vpBoundsFrom.lat.toFixed(6)},${vpCenter.lon.toFixed(6)},${vpCenter.lat.toFixed(6)}`,
`${baseUrl}${vpCenter.lon.toFixed(6)},${vpBoundsFrom.lat.toFixed(6)},${vpBoundsTo.lon.toFixed(6)},${vpCenter.lat.toFixed(6)}`
].forEach((url) => {
_overflowUrsUrls.push(url);
getOverflowUrsFromUrl(url).then(processData);
});
}
function mouseDown() {
_mouseIsDown = true;
}
function mouseUp() {
_mouseIsDown = false;
}
/**
* 2023.04.05.01: With the removal of the handleUrOverflow call from this function, it and the event listener are no longer needed.
function invokeMoveEnd(/* evt *-/) {
/**
* Enable Auto Refresh: Disabled 2023.03.29
* Due to W.controller.reloadData() causing issues with new Issue Tracker.
* Specifically if user had more than 500 URs loaded in tracker, it would reset them back to 500 due to reloadData
* wiping and reloading the model data. For now, this setting is disabled and the functionality has been replaced
* with the already working handleOverflow function (if the user has it enabled).
const zoomLevel = evt.object.zoom || W.map.getOLMap().getZoom();
if (_settings.enableAutoRefresh
&& (zoomLevel > 14)
&& (W.model.mapUpdateRequests.getObjectArray().length > 499)
&& (W.model.actionManager.getActionsNum() === 0)
)
W.controller.reloadData();
else if (!_settings.enableUrOverflowHandling && (W.model.mapUpdateRequests.getObjectArray().length > 499))
alertBoxInPanel(I18n.t('urce.prompts.UrOverflowErrorWithoutOverflowEnabled'), undefined, true, 9999);
else
dismissAlertBoxInPanel(undefined, 9999);
*-/
// 2023.04.05.01: Used to check for conditions and either run handleUrOverflow or panel alert box or remove it. But now using W.app.on('change:loadingIssueTrackerMapData') event listener.
}
*/
function invokeZoomEnd(evt) {
const zoomLevel = evt?.object?.zoom || W.map.getOLMap().getZoom();
/**
* Enable Auto Refresh: Disabled 2023.03.29
* Due to W.controller.reloadData() causing issues with new Issue Tracker.
* Specifically if user had more than 500 URs loaded in tracker, it would reset them back to 500 due to reloadData
* wiping and reloading the model data. For now, this setting is disabled and the functionality has been replaced
* with the already working handleOverflow function (if the user has it enabled).
*
if (_settings.enableAutoRefresh && (zoomLevel > 14) && (W.model.mapUpdateRequests.getObjectArray().length > 499) && (W.model.actionManager.getActionsNum() === 0)) {
W.controller.reloadData();
return;
}
*/
let filter;
if (_settings.disableFilteringAboveZoom || _settings.disableFilteringBelowZoom) {
if (!_filtersAppliedOnZoom && _settings.disableFilteringAboveZoom && (zoomLevel > (_settings.disableFilteringAboveZoomLevel - 1)))
filter = true;
if (!_filtersAppliedOnZoom && _settings.disableFilteringBelowZoom && (zoomLevel < (_settings.disableFilteringBelowZoomLevel + 1)))
filter = true;
if (_filtersAppliedOnZoom && _settings.disableFilteringAboveZoom && (zoomLevel < _settings.disableFilteringAboveZoomLevel))
filter = false;
if (_filtersAppliedOnZoom && _settings.disableFilteringBelowZoom && (zoomLevel > _settings.disableFilteringBelowZoomLevel))
filter = false;
}
// 2023.04.05.01: Used to check for conditions and either run handleUrOverflow or panel alert box or remove it. But now using W.app.on('change:loadingIssueTrackerMapData') event listener.
if (filter !== undefined)
handleUrLayer('zoomEnd', filter, getMapUrsObjArr());
}
async function invokeModeChange(evt) {
if (evt?.changed?.mode === 1)
await initBackgroundTasks('disable', 'invokeModeChange');
else if (evt?.changed?.mode === 0)
await initBackgroundTasks('enable', 'invokeModeChange');
if ((evt?.changed?.mode === 0) || (evt?.changed?.isImperial === true) || (evt?.changed?.isImperial === false)) {
await changeCommentList(_currentCommentList, false, true);
await checkRestrictions([{ type: 'modeChange' }]);
handleUrLayer('modeChange', undefined, getMapUrsObjArr());
}
}
async function maskBoxes(docFrags, unmask, phase, maskUrPanel) {
const zIndex = (phase === 'init') ? 19999 : 10000;
if (unmask) {
document.getElementById(`urceTabLightbox-${phase}`)?.remove();
document.getElementById(`urPanelLightbox-${phase}`)?.remove();
if ((document.querySelectorAll('[id^="urceTabLightbox-"').length === 0) && document.getElementById('sidepanel-urc-e'))
document.getElementById('sidepanel-urc-e').style.position = '';
}
else if (!unmask) {
if (!document.getElementById(`urceTabLightbox-${phase}`)) {
const divElemRootDiv = createElem('div', { class: 'urceMaskLightbox text' });
divElemRootDiv.appendChild(docFrags.cloneNode());
const divElemRoot = createElem('div', { id: `urceTabLightbox-${phase}`, class: 'urceMaskLightbox', style: `z-index:${zIndex};` });
divElemRoot.appendChild(divElemRootDiv);
const domElement = document.getElementById('sidepanel-urc-e');
domElement.style.position = 'relative';
domElement.insertBefore(divElemRoot, domElement.firstChild);
}
if (maskUrPanel && !document.getElementById(`urPanelLightbox-${phase}`)) {
const domElement = await getDomElement('#panel-container wz-card[class^="panel"].problem-edit');
if (!domElement) {
handleReadyError(false, false, '', false, 'Timed out trying to add mask to UR panel.');
}
else {
domElement.firstChild.style.position = 'relative';
const divElemRootDiv = createElem('div', { class: 'urceMaskLightbox text' });
divElemRootDiv.appendChild(docFrags.cloneNode());
const divElemRoot = createElem('div', { id: `urPanelLightbox-${phase}`, class: 'urceMaskLightbox', style: `z-index:${zIndex};` });
divElemRoot.appendChild(divElemRootDiv);
domElement.firstChild.insertBefore(divElemRoot, domElement.firstChild.firstChild);
}
}
}
docFrags = undefined;
}
function autoScrollComments(commentCount = 0, retryInterval = 10, maxTries = 200) {
(async function retry(commentCountInt, tries, retryInt, maxNumTries) {
checkTimeout({ timeout: 'autoScrollComments' });
if (tries > maxNumTries) {
logWarning(`Timed out trying to scroll to the bottom. commentCount: ${commentCountInt}, tries: ${tries}, retryInt: ${retryInt}, maxTries: ${maxTries}`);
return;
}
const commentList = await getDomElement('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation.section .conversation-view .comment-list');
if (commentList && (commentCountInt > 0)
&& (commentCountInt === commentList.children.length)
&& (!_settings.autoScrollComments
|| ((commentList.scrollHeight - commentList.scrollTop - commentList.offsetHeight) > 1)
)
)
commentList.scrollTop = _settings.autoScrollComments ? commentList.scrollHeight : 0;
else if ((commentCountInt !== 0) && W.problemsController.editController.viewModel.get('loadingConversation'))
_timeouts.autoScrollComments = window.setTimeout(retry, retryInt, commentCountInt, ++tries, retryInt, maxNumTries);
}(commentCount, 1, retryInterval, maxTries));
}
function changeCommentListStyle(settingVal) {
if (_settings.commentListStyle !== settingVal) {
if (settingVal === 'default') {
document.querySelectorAll('fieldset[id^="urceComments-for-"], legend[id^="urceComments-for-"], div[id^="urceComments-for-"], fieldset[id^="urce-prefs-fieldset"], legend[id^="urce-prefs-legend"], '
+ '#URCE-expandCollapseAllComments').forEach((el) => { el.classList.remove('urStyle'); });
}
else if (settingVal === 'urStyle') {
document.querySelectorAll('fieldset[id^="urceComments-for-"], legend[id^="urceComments-for-"], div[id^="urceComments-for-"], fieldset[id^="urce-prefs-fieldset"], legend[id^="urce-prefs-legend"], '
+ '#URCE-expandCollapseAllComments').forEach((el) => { el.classList.add('urStyle'); });
}
_settings.commentListStyle = settingVal;
}
saveSettingsToStorage();
}
function changeCommentList(commentListIdx, autoSwitch, refresh) {
return new Promise((resolve) => {
doSpinner('changeCommentList', true);
refresh = (refresh === true);
commentListIdx = (isNaN(commentListIdx)) ? _settings.commentList : commentListIdx;
if (refresh || (!autoSwitch && ((commentListIdx !== _settings.commentList) || (commentListIdx !== _currentCommentList))) || (autoSwitch && (commentListIdx !== _currentCommentList))) {
let debugMsg;
if (autoSwitch)
debugMsg = 'Automatically switching comment lists';
else if (refresh)
debugMsg = 'Refreshing comment list';
else
debugMsg = 'Switching comment lists';
if (+_currentCommentList > -1)
debugMsg += ` from ${getCommentListInfo(_currentCommentList).name}`;
logDebug(`${debugMsg} to ${getCommentListInfo(commentListIdx).name}.`);
if (!autoSwitch && !refresh)
_settings.commentList = commentListIdx;
buildCommentList(commentListIdx, 'changeList', autoSwitch).catch((error) => {
error.maskUrPanel = (_selUr.urId > 0);
error.phase = 'changeList';
error.staticList = (getCommentListInfo(commentListIdx).type === 'static');
error.commentList = commentListIdx;
handleError(error);
})
.finally(() => {
if (!autoSwitch && !refresh)
saveSettingsToStorage();
doSpinner('changeCommentList', false);
resolve();
});
}
else {
doSpinner('changeCommentList', false);
resolve();
}
});
}
function getCommentListInfo(commentListIdx) {
commentListIdx = (isNaN(commentListIdx)) ? _settings.commentList : commentListIdx;
return _commentLists.find((cList) => cList.idx === commentListIdx);
}
function createCommentListCSV(section) {
section = section || 'comments';
const date = new Date(),
dateTs = `${date.getFullYear()}${(date.getMonth() < 9) ? `0${(date.getMonth() + 1)}` : (date.getMonth() + 1)
}${date.getDate() < 9 ? `0${(date.getDate() + 1)}` : (date.getDate() + 1)
}${date.getHours() < 10 ? `0${date.getHours()}` : date.getHours()
}${date.getMinutes() < 10 ? `0${date.getMinutes()}` : date.getMinutes()
}${date.getSeconds() < 10 ? `0${date.getSeconds()}` : date.getSeconds()
}`;
if (section === 'defaults') {
let idx,
defaultsArr = {};
Object.keys(_defaultComments).forEach((key) => {
if (key === 'dr')
idx = 0;
else if (key === 'dc')
idx = 1;
else
idx = _defaultComments[key].urNum;
defaultsArr[idx] = (_defaultComments[key].commentNum !== null) ? [_commentList[_defaultComments[key].commentNum].title] : [''];
});
defaultsArr = Object.values(defaultsArr);
const a = createElem('a', { href: URL.createObjectURL(new Blob([defaultsArr.join('\n')], { type: 'text/csv' })), download: `default_responses_${dateTs}.csv` });
a.click();
a.remove();
}
else if (section === 'comments') {
const commentsArr = _commentList.map((entry) => {
let urstatus;
if (entry.urstatus === 'open')
urstatus = 'Open';
else if (entry.urstatus === 'solved')
urstatus = 'Solved';
else if (entry.urstatus === 'notidentified')
urstatus = 'NotIdentified';
else if (entry.urstatus !== '')
urstatus = entry.urstatus.toUpperCase();
else
urstatus = '';
return [`"${entry.title.replace(/"/gmi, '\\"')}"|"${entry.comment.replace(/"/gmi, '\\"')}"|${urstatus}`];
});
const a = createElem('a', { href: URL.createObjectURL(new Blob([commentsArr.join('\n')], { type: 'text/csv' })), download: `comment_list_${dateTs}.csv` });
a.click();
a.remove();
}
}
function invokeCreateCommentListCsvDefaults() {
createCommentListCSV('defaults');
}
function invokeCreateCommentListCsvComments() {
createCommentListCSV('comments');
}
function createStaticToGoogleSheet(convert) {
if (convert && (getCommentListInfo(_currentCommentList).type !== 'static')) {
WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.ConversionLoadAddonFirst'));
return;
}
const createSteps = Object.keys(I18n.translations[I18n.currentLocale()].urce.tools).filter((k) => /^CreateStep[0-9]+/.test(k)),
finalSteps = Object.keys(I18n.translations[I18n.currentLocale()].urce.tools).filter((k) => /^FinalStep[0-9]+/.test(k));
let spreadsheetStep,
downloadDefaultsStep,
downloadCommentsStep;
const divElemRoot = createElem('div');
divElemRoot.appendChild(createElem('div', {
class: 'URCE-disableWme-text-header', textContent: convert ? I18n.t('urce.tools.ConvertCreateConvertProcess') : I18n.t('urce.tools.ConvertCreateCreateProcess')
}));
let divElemDiv = createElem('div', { class: 'URCE-disableWme-text-body' });
divElemDiv.appendChild(createElem('div', { style: 'font-size:14px;font-weight:600;', textContent: I18n.t('urce.tools.ConvertCreateSteps') }));
const olElem = createElem('ol');
for (let i = 0, { length } = createSteps; i < length; i++) {
if (I18n.t(`urce.tools.${createSteps[i]}`).includes('$SPREADSHEET_STEP$'))
spreadsheetStep = i + 1;
const liElem = createElem('li');
liElem.appendChild(createElem('p', {
innerHTML: trustedHTML(
I18n.t(`urce.tools.${createSteps[i]}`)
.replaceAll('$TEMPLATE_LINK$', `<a id="_openTemplate" href="https://bit.ly/urc-e_cc-template" target="_blank">${I18n.t('urce.common.Link')}</a>`)
.replaceAll('$SPREADSHEET_STEP$', '')
)
}));
olElem.appendChild(liElem);
}
if (convert) {
const convertSteps = Object.keys(I18n.translations[I18n.currentLocale()].urce.tools).filter((k) => /^ConvertStep[0-9]+/.test(k));
for (let i = 0, { length } = convertSteps; i < length; i++) {
if (I18n.t(`urce.tools.${convertSteps[i]}`).includes('$DOWNLOAD_DEFAULTS_LINK$'))
downloadDefaultsStep = createSteps.length + i + 1;
if (I18n.t(`urce.tools.${convertSteps[i]}`).includes('$DOWNLOAD_COMMENTS_LINK$'))
downloadCommentsStep = createSteps.length + i + 1;
const liElem = createElem('li');
liElem.appendChild(
createElem('p', {
innerHTML: trustedHTML(I18n.t(`urce.tools.${convertSteps[i]}`)
.replaceAll('$DOWNLOAD_DEFAULTS_LINK$', `<a id="_downloadDefaults">${I18n.t('urce.common.Link')}</a>`)
.replaceAll('$DOWNLOAD_COMMENTS_LINK$', `<a id="_downloadComments">${I18n.t('urce.common.Link')}</a>`)
.replaceAll('$DOWNLOAD_DEFAULTS_STEP$', downloadDefaultsStep)
.replaceAll('$DOWNLOAD_COMMENTS_STEP$', downloadCommentsStep))
})
);
if ((downloadDefaultsStep > 0) && (i === (downloadDefaultsStep - createSteps.length - 1)))
liElem.querySelector('#_downloadDefaults').addEventListener('click', invokeCreateCommentListCsvDefaults);
if ((downloadCommentsStep > 0) && (i === (downloadCommentsStep - createSteps.length - 1)))
liElem.querySelector('#_downloadComments').addEventListener('click', invokeCreateCommentListCsvComments);
olElem.appendChild(liElem);
}
}
for (let i = 0, { length } = finalSteps; i < length; i++) {
const liElem = createElem('li');
liElem.appendChild(createElem('p', { innerHTML: trustedHTML(I18n.t(`urce.tools.${finalSteps[i]}`).replaceAll('$SPREADSHEET_STEP$', spreadsheetStep)) }));
olElem.appendChild(liElem);
}
divElemDiv.appendChild(olElem);
divElemRoot.appendChild(divElemDiv);
divElemDiv = createElem('div', { class: 'URCE-disableWme-text-footer' });
divElemDiv.appendChild(createElem('button', {
id: '_dismissSteps', urceprefs: 'tools', class: 'urceToolsButton', textContent: I18n.t('urce.common.Finish')
}, [{ click: disableWme }]));
divElemRoot.appendChild(divElemDiv);
disableWme(divElemRoot, false);
}
function checkForStaticListArray(oldVarNameStr) {
return new Promise((resolve, reject) => {
(function retry(oldVarName, tries) {
checkTimeout({ timeout: 'checkForStaticListArray' });
if (tries > 100)
reject(new Error(I18n.t('urce.prompts.TimedOutWaitingStatic')));
else if (!(typeof unsafeWindow !== 'undefined' ? unsafeWindow : window)[`Urcomments${oldVarName}Array2`])
_timeouts.checkForStaticListArray = window.setTimeout(retry, 100, oldVarName, ++tries);
else
resolve();
}(oldVarNameStr, 1));
});
}
function convertCommentListStatic(commentListIdx) {
return new Promise((resolve, reject) => {
commentListIdx = (isNaN(commentListIdx)) ? _settings.commentList : commentListIdx;
const { oldVarName } = getCommentListInfo(commentListIdx);
checkForStaticListArray(oldVarName).then(() => {
try {
const oldUrcArr = (typeof unsafeWindow !== 'undefined' ? unsafeWindow : window)[`Urcomments${oldVarName}Array2`],
defaultReminderIdx = +(typeof unsafeWindow !== 'undefined' ? unsafeWindow : window)[`Urcomments${oldVarName}ReminderPosistion`],
closedNiIdx = +(typeof unsafeWindow !== 'undefined' ? unsafeWindow : window)[`Urcomments${oldVarName}CloseNotIdentifiedPosistion`],
data = [];
let entryIdx;
logDebug(`Converting static comment list to URC-E format for comment list: ${oldVarName}`);
data[0] = ['URCE'];
data[1] = ['2019.03.01.01'];
data[2] = ['TITLE|COMMENT|URSTATUS|DR|DC|IT|IA|IR|MRA|GE|TNA|IJ|MBO|WDD|ME|MR|ML|BR|MSN|ISPS|SL'];
if (!/<br>/gi.test(oldUrcArr[0]) && (oldUrcArr[2] !== '')) {
data[3] = ['||GROUP TITLE||||||||||||||||||'];
entryIdx = 4;
}
else {
entryIdx = 3;
}
for (let oldUrcArrIdx = 0, { length } = oldUrcArr; oldUrcArrIdx < length; oldUrcArrIdx += 3) {
if (/<br>/gi.test(oldUrcArr[oldUrcArrIdx]) || (oldUrcArr[oldUrcArrIdx + 2] === '')) {
const divElemRoot = createElem('div', { textContent: untrustedHTML(oldUrcArr[oldUrcArrIdx].replace(/<br>/gi, '')) });
oldUrcArr[oldUrcArrIdx + 2] = 'GROUP TITLE';
oldUrcArr[oldUrcArrIdx] = divElemRoot.innerText;
}
let temp = `${oldUrcArr[oldUrcArrIdx]}|${oldUrcArr[oldUrcArrIdx + 1]}|`
+ `${((oldUrcArr[oldUrcArrIdx + 2] !== '') ? oldUrcArr[oldUrcArrIdx + 2].toLowerCase() : '')}`
+ `${((oldUrcArrIdx === defaultReminderIdx) ? '|default_is_true' : '|')}`
+ `${((oldUrcArrIdx === closedNiIdx) ? '|default_is_true' : '|')}`;
for (let i = 6; i < 24; i++) {
if ((i !== 17) && (i !== 20))
temp += (((typeof unsafeWindow !== 'undefined' ? unsafeWindow : window)[`Urcomments${oldVarName}def_names`][i]) && ((typeof unsafeWindow !== 'undefined' ? unsafeWindow : window)[`Urcomments${oldVarName}def_names`][i].toLowerCase() === oldUrcArr[oldUrcArrIdx].toLowerCase())) ? '|default_is_true' : '|';
}
data[entryIdx] = [temp];
entryIdx++;
}
resolve(data);
}
catch (error) {
reject(error);
}
})
.catch((error) => {
reject(error);
});
});
}
function processCommentList(data) {
return new Promise((resolve, reject) => {
logDebug('Processing comment list data.');
if (data) {
let ssFieldNames;
const EXPECTED_FIELD_NAMES = ['TITLE', 'COMMENT', 'URSTATUS', 'DR', 'DC', 'IT', 'IA', 'IR', 'MRA', 'GE', 'TNA', 'IJ', 'MBO', 'WDD', 'ME', 'MR', 'ML', 'BR', 'MSN', 'ISPS', 'SL'],
outputItems = [],
rowObj = {},
findObjIndex = (array, fldName, value) => array.map((a) => a[fldName]).indexOf(value),
checkFieldNames = function (fldName) { return this.includes(fldName); },
createRowObj = (entry, i) => {
i = i || 0;
if (ssFieldNames[i].trim().toLowerCase() === 'comment')
rowObj[ssFieldNames[i].trim().toLowerCase()] = entry;
else if (ssFieldNames[i].trim().toLowerCase() === 'title')
rowObj[ssFieldNames[i].trim().toLowerCase()] = entry.trim();
else
rowObj[ssFieldNames[i].trim().toLowerCase()] = entry.trim().toLowerCase();
i++;
},
legendClickToggle = function () {
this.firstChild.classList.toggle('w-icon-chevron-up');
this.firstChild.classList.toggle('w-icon-chevron-down');
this.nextSibling.classList.toggle('URCE-collapsed');
saveSettingsToStorage();
},
cidSingleClick = function () {
handleClickedComment(+this.id.substring(9), false);
},
cidDoubleClick = function () {
handleClickedComment(+this.id.substring(13), true);
};
let groupDivId,
commentId = 0,
blankGroup = 0;
for (let entryIdx = 0, { length } = data; entryIdx < length; entryIdx++) {
if (entryIdx === 0) {
if (data[entryIdx][0] !== 'URCE')
return reject(new Error('Incorrect format in spreadsheet data received.'));
}
else if (entryIdx === 1) {
if (_SCRIPT_VERSION < data[entryIdx][0])
return reject(new Error(`updateRequired|${data[entryIdx][0]}`));
if (data[entryIdx][0] < _MIN_VERSION_COMMENTS)
return reject(new Error(`spreadsheetUpdateRequired|${_MIN_VERSION_COMMENTS}`));
}
else if (entryIdx === 2) {
ssFieldNames = data[entryIdx][0].split('|').map((fldName) => fldName.trim());
if (ssFieldNames.length !== EXPECTED_FIELD_NAMES.length)
return reject(new Error(`Expected ${EXPECTED_FIELD_NAMES.length} columns in comment definition data. Spreadsheet returned ${ssFieldNames.length}.`));
if (!EXPECTED_FIELD_NAMES.every(checkFieldNames.bind(ssFieldNames)))
return reject(new Error(`Script expected to see the following column names in the comment definition spreadsheet:\n${EXPECTED_FIELD_NAMES.join(', ')}\nHowever, the spreadsheet returned these:\n${ssFieldNames.join(', ')}`));
}
else if (data[entryIdx][0]) {
data[entryIdx][0].split('|').forEach((entry, i) => createRowObj(entry, i));
if (rowObj.title !== 'URCE_REMOVED_SO_SKIP') {
if (rowObj.title === 'URCE_ERROR')
return reject(new Error(`There is an unknown error in the spreadsheet output. Please contact the list owner: ${getCommentListInfo(_settings.commentList).listOwner}`));
if (rowObj.urstatus === 'custom var') {
_customReplaceVars.push({ customVar: `$${rowObj.title}$`, replaceText: rowObj.comment });
}
else if ((entryIdx === 3) && (rowObj.urstatus !== 'group title')) {
return reject(new Error(`Row 25 on the spreadsheet must be set to "GROUP TITLE" for the URSTATUS column. Please contact the list owner: ${getCommentListInfo(_settings.commentList).listOwner}`));
}
else if (rowObj.urstatus === 'group title') { // Group title row. Nothing to set in the arrays, but build html.
groupDivId = 'urceComments-for-';
if (rowObj.title !== '') {
groupDivId += rowObj.title.replace(/[^\w]+/gi, '').toLowerCase();
if (rowObj.title === rowObj.title.toUpperCase()) {
if (rowObj.title.length > 30) {
rowObj.titleMouseOver = rowObj.title;
rowObj.title = `${rowObj.title.substring(0, 30)}...`;
}
}
else if (rowObj.title.length > 35) {
rowObj.titleMouseOver = rowObj.title;
rowObj.title = `${rowObj.title.substring(0, 35)}...`;
}
}
else {
groupDivId += `blankGroup${(++blankGroup)}`;
}
const collapsed = (
_settings.commentListCollapses.hasOwnProperty(_settings.commentList)
&& _settings.commentListCollapses[_settings.commentList].hasOwnProperty(`${groupDivId}_body_urce`)
&& (_settings.commentListCollapses[_settings.commentList][`${groupDivId}_body_urce`] === true)
)
? 'URCE-collapsed'
: '',
chevron = (collapsed === 'URCE-collapsed') ? 'w-icon-chevron-down' : 'w-icon-chevron-up';
outputItems.push({
chevron,
collapsed,
groupDivId,
items: [],
name: rowObj.title,
title: rowObj.titleMouseOver
});
_commentList[commentId] = { title: rowObj.title, comment: '', urstatus: 'GROUP TITLE' };
commentId++;
}
else { // SHOULD be a normal comments row, push values to arrays and build html.
if ((rowObj.urstatus !== 'solved') && (rowObj.urstatus !== 'notidentified') && (rowObj.urstatus !== 'open') && (rowObj.urstatus !== 'blank line'))
return reject(new Error(`Your current selected list does not have a status set for ${rowObj.title}. Please contact list owner: ${getCommentListInfo(_settings.commentList).listOwner}`));
_commentList[commentId] = { title: rowObj.title, comment: rowObj.comment, urstatus: rowObj.urstatus };
if (Object.values(rowObj).includes('default_is_true')) {
const drIdx = ssFieldNames.indexOf('DR'),
splitRowDefaultCommentsBoolean = Object.values(rowObj).slice(drIdx);
for (let boolIdx = 0, len = splitRowDefaultCommentsBoolean.length; boolIdx < len; boolIdx++) {
if (splitRowDefaultCommentsBoolean[boolIdx].toLowerCase() === 'default_is_true')
_defaultComments[ssFieldNames[(boolIdx + drIdx)].toLowerCase()].commentNum = commentId;
}
}
let linkClass;
let divDoubleClickId;
let divDoubleClickStyle = 'display:initial;';
if (rowObj.urstatus === 'solved') {
linkClass = 'URCE-solvedLink';
divDoubleClickId = 'URCE-divDoubleClickSolved';
if (!_settings.doubleClickLinkSolvedComments)
divDoubleClickStyle = 'display:none;';
}
else if (rowObj.urstatus === 'notidentified') {
linkClass = 'URCE-niLink';
divDoubleClickId = 'URCE-divDoubleClickNi';
if (!_settings.doubleClickLinkNiComments)
divDoubleClickStyle = 'display:none;';
}
else {
linkClass = (rowObj.urstatus === 'blank line') ? 'URCE-blankLine' : 'URCE-openLink';
divDoubleClickId = (rowObj.title !== '') ? 'URCE-divDoubleClickOpen' : 'URCE-divDoubleClickOpen-Hidden';
if (!_settings.doubleClickLinkOpenComments || (rowObj.urstatus === 'blank line'))
divDoubleClickStyle = 'display:none;';
}
const idx = findObjIndex(outputItems, 'groupDivId', groupDivId);
outputItems[idx].items.push({
linkClass,
commentId,
title: formatText(rowObj.comment, false, false, -1),
name: rowObj.title,
divDoubleClickId,
divDoubleClickStyle,
divDoubleClickTitle: `${I18n.t('urce.common.DoubleClickTitle')}:\n${formatText(rowObj.comment, false, false, -1)}`
});
commentId++;
}
}
}
}
if (outputItems.length > 0) {
const urStyle = (_settings.commentListStyle === 'urStyle') ? ' urStyle' : '',
DOUBLE_CLICK_ICON = '';
const divElemCommentList = createElem('div');
outputItems.forEach((item) => {
const legendElem = createElem('legend', { id: `${item.groupDivId}_legend`, class: `URCE-legend ${urStyle}` }, [{ click: legendClickToggle }]);
legendElem.appendChild(createElem('i', { class: `w-icon ${item.chevron} URCE-chevron` }));
legendElem.appendChild(createElem('span', { class: 'URCE-span', title: item.title, textContent: item.name }));
const fieldsetElem = createElem('fieldset', { id: item.groupDivId, class: `URCE-field ${urStyle}` });
fieldsetElem.appendChild(legendElem);
if (item.items.length > 0) {
const divElemGroup = createElem('div', { id: `${item.groupDivId}_body_urce`, class: `${item.collapsed} URCE-group_body ${urStyle}` });
for (let idx = 0, { length } = item.items; idx < length; idx++) {
const curItem = item.items[idx],
divElemItemName = createElem('div', { style: 'width:225px;display:inline-flex;' });
divElemItemName.appendChild(createElem('a', {
id: `urce-cid-${curItem.commentId}`, class: `URCE-Comments ${curItem.linkClass} URCE-Comments`, title: curItem.title.replace(/"/g, '\''), textContent: item.items[idx].name
}, [{ click: cidSingleClick }]));
const divElemItemParent = createElem('div', { class: `URCE-divComment hover expand ${curItem.linkClass}`, style: 'position:relative' });
divElemItemParent.appendChild(divElemItemName);
const divElemDoubleClick = createElem('div', {
id: curItem.divDoubleClickId, class: 'URCE-divDoubleClick', style: curItem.divDoubleClickStyle, title: curItem.divDoubleClickTitle.replace(/"/g, '\'')
});
divElemDoubleClick.appendChild(createElem('img', {
id: `urce-img-cid-${item.items[idx].commentId}`, src: DOUBLE_CLICK_ICON, class: 'URCE-doubleClickIcon'
}, [{ dblclick: cidDoubleClick }]));
divElemItemParent.appendChild(divElemDoubleClick);
divElemGroup.appendChild(divElemItemParent);
}
fieldsetElem.appendChild(divElemGroup);
}
divElemCommentList.appendChild(fieldsetElem);
});
document.getElementById('_commentList').appendChild(divElemCommentList);
}
else {
return reject(new Error(`There is an error in the output. Please contact the list owner: ${getCommentListInfo(_settings.commentList).listOwner}`));
}
}
else {
return reject(new Error('No data passed to the JSON processing function.'));
}
return resolve();
});
}
function commentListAsync(commentListIdx) {
return new Promise((resolve, reject) => {
commentListIdx = (isNaN(commentListIdx)) ? _settings.commentList : commentListIdx;
const commentListInfo = getCommentListInfo(commentListIdx);
logDebug(`Beginning comment list async for comment list: ${commentListInfo.name}`);
GM_xmlhttpRequest({
url: `https://sheets.googleapis.com/v4/spreadsheets/${((commentListIdx === 1001) ? _settings.customSsId : dec(_URCE_SPREADSHEET_ID))}/values/${commentListInfo.gSheetRange}?key=${dec(_URCE_API_KEY)}`,
headers: { 'Content-Type': 'application/json', Referer: 'https://www.waze.com' },
method: 'GET',
onload(res) {
if (res.status < 400) {
const data = JSON.parse(res.responseText);
if (data.values?.length > 0)
resolve(data.values);
else if (data.values?.length === 0)
reject(new Error('No comments found in the spreadsheet sheet.'));
else if (data.error)
reject(new Error(`Comment list call failed with: ${data.error.message}`));
else
reject(new Error('Comment list call failed with unknown error.'));
}
else {
reject(new Error(`HTTP request error: ${JSON.stringify(res)}`));
}
},
onerror(res) {
reject(new Error(`xmlhttpRequest error: ${JSON.stringify(res)}`));
}
});
});
}
function buildCommentList(commentListIdx, phase, autoSwitch) {
return new Promise((resolve, reject) => {
doSpinner('buildCommentList', true);
commentListIdx = (isNaN(commentListIdx)) ? _settings.commentList : commentListIdx;
const docFrags = document.createDocumentFragment(),
commentListInfo = getCommentListInfo(commentListIdx),
selectionChange = function () {
if ((+this.value === 1001) && (!_settings.customSsId || (_settings.customSsId.length < 1))) {
this.value = _currentCommentList;
WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.SetCustomSsIdFirst'));
}
else {
changeCommentList(+this.value, false, true);
}
},
appendModeToggle = function () {
_settings[this.id.substring(3)] = this.checked;
if (W.problemsController.editController.adapter.problem) {
document.querySelector('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .body .conversation .new-comment-text')
.shadowRoot.querySelector('textarea[id^=wz-textarea-]').style.backgroundColor = this.checked ? 'peachpuff' : '';
}
saveSettingsToStorage();
},
expandCollapseAll = function () {
const legends = document.querySelectorAll('legend[id^="urceComments-for"]');
for (let idx = 0, { length } = legends; idx < length; idx++) {
if (this.id === 'URCE-expandAllComments') {
if (legends[idx].nextElementSibling.classList.contains('URCE-collapsed'))
legends[idx].click();
}
else if (this.id === 'URCE-collapseAllComments') {
if (!legends[idx].nextElementSibling.classList.contains('URCE-collapsed'))
legends[idx].click();
}
}
};
Object.values(_defaultComments).forEach((a) => { a.commentNum = null; });
logDebug(`Building comment list for: ${commentListInfo.name}`);
if (phase !== 'init') {
const docFrags2 = document.createDocumentFragment();
docFrags2.appendChild(createElem('div', { textContent: I18n.t('urce.prompts.SwitchingCommentLists') }));
docFrags2.appendChild(createElem('br'));
docFrags2.appendChild(createElem('div', { textContent: I18n.t('urce.common.PleaseWait') }));
maskBoxes(docFrags2, false, phase, (_selUr.urId > 0));
}
const selectElem = createElem('select', { id: '_selcurrentCommentList', title: I18n.t('urce.common.CurrentCommentListTitle') }, [{ change: selectionChange }]);
_commentLists.forEach((cList) => {
if (cList.status !== 'disabled') {
selectElem.appendChild(createElem('option', {
value: cList.idx, textContent: (autoSwitch && (cList.idx === commentListIdx)) ? `${cList.name} (${I18n.t('urce.common.AutoSwitched')})` : cList.name, selected: (cList.idx === commentListIdx)
}));
}
});
let divElemRoot = createElem('div', { class: 'URCE-commentListName', textContent: `${I18n.t('urce.common.CommentList')}: ` });
divElemRoot.appendChild(selectElem);
const divElemRestrictionEnforcedParent = createElem('div', {
id: 'restrictionsEnforcedWarning', style: `float:right;padding-top:6px;color:red;font-size:22px;${(_restrictionsEnforcedTitle ? '' : 'display:none;')}`
});
divElemRestrictionEnforcedParent.appendChild(createElem('i', { class: 'w-icon w-icon-alert-danger', title: _restrictionsEnforcedTitle || '' }));
divElemRoot.appendChild(divElemRestrictionEnforcedParent);
docFrags.appendChild(divElemRoot);
divElemRoot = createElem('div', { class: 'URCE-commentListName URCE-controls URCE-divCC' });
divElemRoot.appendChild(createElem('input', {
type: 'checkbox', id: '_cbenableAppendMode', class: 'urceSettingsCheckbox2', title: I18n.t('urce.prefs.EnableAppendModeTitle'), checked: _settings.enableAppendMode
}, [{ change: appendModeToggle }]));
divElemRoot.appendChild(createElem('label', {
for: '_cbenableAppendMode', title: I18n.t('urce.prefs.EnableAppendModeTitle'), class: 'URCE-label', textContent: I18n.t('urce.prefs.EnableAppendMode')
}));
divElemRoot.appendChild(createElem('br'));
docFrags.appendChild(divElemRoot);
divElemRoot = createElem('div', { id: 'URCE-expandCollapseAllComments', class: `URCE-expandCollapseAll${((_settings.commentListStyle === 'urStyle') ? ' urStyle' : '')}` });
divElemRoot.appendChild(createElem('div', {
id: 'URCE-expandAllComments', class: 'URCE-expandCollapseAllItem', textContent: I18n.t('urce.common.ExpandAll')
}, [{ click: expandCollapseAll }]));
divElemRoot.appendChild(createElem('div', { style: 'display:inline;', textContent: ' : ' }));
divElemRoot.appendChild(createElem('div', {
id: 'URCE-collapseAllComments', class: 'URCE-expandCollapseAllItem', textContent: I18n.t('urce.common.CollapseAll')
}, [{ click: expandCollapseAll }]));
docFrags.appendChild(divElemRoot);
document.getElementById('_commentList').replaceChildren(docFrags);
_restrictionsEnforcedTitle = undefined;
_commentList = [];
_customReplaceVars = [];
_currentCommentList = commentListIdx;
processPerCommentListSettings(commentListIdx);
if (commentListInfo.type === 'static') {
convertCommentListStatic(commentListIdx).then((data) => {
processCommentList(data).then(() => {
_commentListLoaded = true;
doSpinner('buildCommentList', false);
if (phase !== 'init')
maskBoxes(undefined, true, phase, (_selUr.urId > 0));
resolve();
})
.catch((error) => {
doSpinner('buildCommentList', false);
error.phase = phase;
error.maskUrPanel = (_selUr.urId > 0);
error.staticList = (commentListInfo.type === 'static');
error.commentList = commentListIdx;
reject(error);
});
}).catch((error) => {
doSpinner('buildCommentList', false);
error.phase = phase;
error.maskUrPanel = (_selUr.urId > 0);
error.staticList = (commentListInfo.type === 'static');
error.commentList = commentListIdx;
reject(error);
});
}
else {
commentListAsync(commentListIdx).then((data) => {
processCommentList(data).then(() => {
_commentListLoaded = true;
doSpinner('buildCommentList', false);
if (phase !== 'init')
maskBoxes(undefined, true, phase, (_selUr.urId > 0));
resolve();
})
.catch((error) => {
doSpinner('buildCommentList', false);
error.phase = phase;
error.maskUrPanel = (_selUr.urId > 0);
error.staticList = (commentListInfo.type === 'static');
error.commentList = commentListIdx;
reject(error);
});
}).catch((error) => {
doSpinner('buildCommentList', false);
error.phase = phase;
error.maskUrPanel = (_selUr.urId > 0);
error.staticList = (commentListInfo.type === 'static');
error.commentList = commentListIdx;
reject(error);
});
}
});
}
function processPerCommentListSettings(commentListIdx) {
const docFrags = document.createDocumentFragment(),
defaultPerCommentListSettings = {
autoSendReminders: _settings.autoSendReminders,
autoSendReminders_useDefault: true,
autoSendRemindersExceptTagged: _settings.autoSendRemindersExceptTagged,
autoSendRemindersExceptTagged_useDefault: true,
autoSetNewUrComment: _settings.autoSetNewUrComment,
autoSetNewUrComment_useDefault: true,
autoSetNewUrCommentSlur: _settings.autoSetNewUrCommentSlur,
autoSetNewUrCommentSlur_useDefault: true,
autoSetNewUrCommentWithDescription: _settings.autoSetNewUrCommentWithDescription,
autoSetNewUrCommentWithDescription_useDefault: true,
autoSetReminderUrComment: _settings.autoSetReminderUrComment,
autoSetReminderUrComment_useDefault: true,
placeCursorAtStart: _settings.placeCursorAtStart,
placeCursorAtStart_useDefault: true,
customTagline: _settings.customTagline,
customTagline_useDefault: true,
tagEmail: _settings.tagEmail,
tagEmail_useDefault: true,
reminderDays: _settings.reminderDays,
reminderDays_useDefault: true,
closeDays: _settings.closeDays,
closeDays_useDefault: true
};
let perCommentListSettingsChanged = false;
if (!_settings.perCommentListSettings[commentListIdx]) {
_settings.perCommentListSettings[commentListIdx] = defaultPerCommentListSettings;
perCommentListSettingsChanged = true;
}
else {
Object.keys(defaultPerCommentListSettings).forEach((prop) => {
if (!_settings.perCommentListSettings[commentListIdx].hasOwnProperty(prop)) {
_settings.perCommentListSettings[commentListIdx][prop] = defaultPerCommentListSettings[prop];
perCommentListSettingsChanged = true;
}
});
}
if (perCommentListSettingsChanged)
saveSettingsToStorage();
const perCListSettings = _settings.perCommentListSettings[commentListIdx],
settingsChange = function () {
const settingName = this.id.replace(/(_.+perCommentList_)/gmi, ''),
toggleDisabled = (elem) => {
elem.classList?.toggle('urceDisabled');
elem.toggleAttribute('disabled');
for (let i = 0, { length } = elem.children; i < length; i++)
toggleDisabled(elem.children[i]);
};
if (this.type === 'checkbox') {
_settings.perCommentListSettings[_currentCommentList][settingName] = this.checked;
saveSettingsToStorage();
}
if (this.type === 'text') {
const newVal = this.value.trim();
if ((newVal !== this.value) || (_settings.perCommentListSettings[_currentCommentList][settingName] !== newVal)) {
if (newVal !== this.value)
this.value = newVal;
_settings.perCommentListSettings[_currentCommentList][settingName] = newVal;
saveSettingsToStorage();
if (settingName === 'tagEmail')
changeCommentList(_currentCommentList, false, true);
}
}
if (this.type === 'number') {
const val = Math.min(9999, Math.max(0, Math.abs((+this.value || 0))));
if ((val !== this.value) || (_settings.perCommentListSettings[_currentCommentList][settingName] !== val)) {
if (val !== +this.value)
this.value = val;
_settings.perCommentListSettings[_currentCommentList][settingName] = val;
saveSettingsToStorage();
handleUrLayer('settingsToggle', undefined, getMapUrsObjArr());
}
}
if (settingName.includes('_useDefault')) {
for (let i = 0, { length } = this.parentElement.children; i < length; i++) {
if (this !== this.parentElement.children[i])
toggleDisabled(this.parentElement.children[i]);
}
const parentSettingName = settingName.substring(0, (settingName.length - 11));
if (_settings.perCommentListSettings[_currentCommentList][parentSettingName] !== _settings[parentSettingName]) {
_settings.perCommentListSettings[_currentCommentList][parentSettingName] = _settings[parentSettingName];
const parentSettingElem = document.querySelector(`input[id$="${parentSettingName}"]:not([urceprefs="perCommentList"])`);
if (parentSettingElem?.type === 'checkbox') {
if (_settings[parentSettingName])
document.getElementById(`_cbperCommentList_${parentSettingName}`).checked = true;
else
document.getElementById(`_cbperCommentList_${parentSettingName}`).checked = false;
document.getElementById(`_cbperCommentList_${parentSettingName}`).checked = _settings[parentSettingName];
}
else if (parentSettingElem?.type === 'number') {
document.getElementById(`_numperCommentList_${parentSettingName}`).value = _settings[parentSettingName];
}
else if (parentSettingElem?.type === 'text') {
document.getElementById(`_textperCommentList_${parentSettingName}`).value = _settings[parentSettingName];
}
else if (document.querySelector(`textarea[id$="${parentSettingName}"]:not([urceprefs="perCommentList"])`)) {
document.getElementById(`_textperCommentList_${parentSettingName}`).value = _settings[parentSettingName];
}
}
saveSettingsToStorage();
}
},
createDivSettingCb = (settingId = '', warningSetting = false) => {
const title = I18n.t(`urce.prefs.${settingId.charAt(0).toUpperCase() + settingId.slice(1)}Title`),
divElemDiv = createElem('div', {
style: `width:calc(100% - 18px);display:inline-block;${(settingId === 'autoSendRemindersExceptTagged') ? 'padding-left:15px;font-style:italic;' : ''}`
});
divElemDiv.appendChild(createElem('input', {
type: 'checkbox',
id: `_cbperCommentList_${settingId}`,
urceprefs: 'perCommentList',
class: `urceSettingsCheckbox${((perCListSettings[`${settingId}_useDefault`]) ? ' urceDisabled' : '')}`,
title,
checked: perCListSettings[settingId],
disabled: perCListSettings[`${settingId}_useDefault`]
}, [{ change: settingsChange }]));
divElemDiv.appendChild(createElem('label', {
for: `_cbperCommentList_${settingId}`,
title,
urceprefs: 'perCommentList',
class: `URCE-label${((perCListSettings[`${settingId}_useDefault`]) ? ' urceDisabled' : '')}`,
textContent: I18n.t(`urce.prefs.${settingId.charAt(0).toUpperCase() + settingId.slice(1)}`)
}));
if (warningSetting) {
const divElemWarning = createElem('div', { class: 'URCE-divWarning URCE-divWarningPre', textContent: '(' });
divElemWarning.appendChild(createElem('div', {
class: 'URCE-divWarning URCE-divWarningTitle',
title: I18n.t(`urce.prefs.${settingId.charAt(0).toUpperCase() + settingId.slice(1)}WarningTitle`),
textContent: I18n.t(`urce.prefs.${settingId.charAt(0).toUpperCase() + settingId.slice(1)}Warning`)
}));
divElemWarning.appendChild(createElem('div', { class: 'URCE-divWarning', textContent: ')' }));
divElemDiv.appendChild(divElemWarning);
}
const divElemRoot = createElem('div');
divElemRoot.appendChild(divElemDiv);
divElemRoot.appendChild(createElem('input', {
type: 'checkbox',
style: 'float:right;',
id: `_cbperCommentList_${settingId}_useDefault`,
urceprefs: 'perCommentList',
class: 'urceSettingsCheckbox',
title: I18n.t('urce.prefs.UseDefault'),
checked: perCListSettings[`${settingId}_useDefault`]
}, [{ change: settingsChange }]));
return divElemRoot;
},
createDivSettingInput = (settingId) => {
const title = I18n.t(`urce.prefs.${settingId.charAt(0).toUpperCase() + settingId.slice(1)}Title`),
// eslint-disable-next-line no-nested-ternary
type = (settingId === 'tagEmail') ? 'text' : (settingId !== 'customTagline') ? 'number' : '',
divElemDiv = createElem('div', { style: 'width:calc(100% - 18px);display:inline-block;' });
divElemDiv.appendChild(createElem('div', {
style: 'display:inline;',
title,
class: `URCE-label${((perCListSettings[`${settingId}_useDefault`]) ? ' urceDisabled' : '')}`,
urceprefs: 'perCommentList',
textContent: `${I18n.t(`urce.prefs.${settingId.charAt(0).toUpperCase() + settingId.slice(1)}`)}:`
}));
let divElemDivInput;
if (settingId === 'customTagline') {
divElemDivInput = createElem('textarea', {
id: '_textperCommentList_customTagline',
class: `URCE-textInput urceSettingsTextBox${((perCListSettings[`${settingId}_useDefault`]) ? ' urceDisabled' : '')}`,
urceprefs: 'perCommentList',
title,
style: 'resize:none;width:230px;height:50px',
textContent: perCListSettings[settingId],
disabled: perCListSettings[`${settingId}_useDefault`]
}, [{ change: settingsChange }]);
}
else {
const extraAttributes = (type === 'number') ? { min: 0, max: 9999, step: 1 } : {};
divElemDivInput = createElem('input', {
type,
id: `_${(type === 'text') ? type : 'num'}perCommentList_${settingId}`,
class: `URCE-${(type === 'text') ? 'textInput' : 'daysInput'} urceSettings${type.charAt(0).toUpperCase() + type.slice(1)}Box${((perCListSettings[`${settingId}_useDefault`]) ? ' urceDisabled' : '')}`,
urceprefs: 'perCommentList',
value: perCListSettings[settingId],
title,
disabled: perCListSettings[`${settingId}_useDefault`],
...extraAttributes
}, [{ change: settingsChange }]);
}
if (type === 'number') {
divElemDivInput.appendChild(createElem('div', {
style: 'display:inline;',
class: `URCE-label${((perCListSettings[`${settingId}_useDefault`]) ? ' urceDisabled' : '')}`,
urceprefs: 'perCommentList',
textContent: I18n.t('common.time.days', { days: 0 }).replaceAll('0 ', '')
}));
}
divElemDiv.appendChild(divElemDivInput);
const divElemRoot = createElem('div');
divElemRoot.appendChild(divElemDiv);
divElemRoot.appendChild(createElem('input', {
type: 'checkbox', style: 'float:right;', id: `_cbperCommentList_${settingId}_useDefault`, urceprefs: 'perCommentList', class: 'urceSettingsCheckbox', title: I18n.t('urce.prefs.UseDefault'), checked: perCListSettings[`${settingId}_useDefault`]
}, [{ change: settingsChange }]));
return divElemRoot;
};
docFrags.appendChild(createElem('div', { textContent: `${I18n.t('urce.prefs.SettingsFor')}: ${getCommentListInfo(commentListIdx).name}` }));
let divElemRoot = createElem('div', { class: 'URCE-controls' });
['autoSendReminders', 'autoSendRemindersExceptTagged', 'autoSetNewUrComment', 'autoSetNewUrCommentSlur', 'autoSetNewUrCommentWithDescription', 'autoSetReminderUrComment', 'placeCursorAtStart'
].forEach((settingId) => {
divElemRoot.appendChild(createDivSettingCb(settingId, (settingId === 'autoSendReminders')));
});
docFrags.appendChild(divElemRoot);
divElemRoot = createElem('div', { class: 'URCE-controls URCE-textFirst' });
['tagEmail', 'reminderDays', 'closeDays', 'customTagline'].forEach((settingId) => {
divElemRoot.appendChild(createDivSettingInput(settingId));
});
docFrags.appendChild(divElemRoot);
document.getElementById('URCE-divPerCommentListSettings').replaceChildren(docFrags);
}
function handleError(err) {
const divElemRoot = createElem('div', { class: 'URCE-divLoading' });
if (err.message?.includes('|') > -1) {
const [reason, version] = err.message.split('|');
if ((reason === 'updateRequired') || (reason === 'spreadsheetUpdateRequired')) {
const scriptLink = createElem('a', {
href: _IS_BETA_VERSION ? dec(_BETA_DL_URL) : _PROD_DL_URL, target: '_blank', textContent: _IS_BETA_VERSION ? dec(_BETA_DL_URL) : _PROD_DL_URL
}),
ssUpdateInstructionsLink = createElem('a', {
href: 'https://bit.ly/urc-e_ss-update-instructions', target: '_blank', textContent: 'https://bit.ly/urc-e_ss-update-instructions'
});
const [firstText, secondText] = I18n.t(`urce.prompts.${reason.charAt(0).toUpperCase()}${reason.slice(1)}`).split('$VERSION$');
divElemRoot.appendChild(createElem('div', { textContent: `${firstText.trim()} ${version}` }));
divElemRoot.appendChild(createElem('br'));
const divElemRootDiv = createElem('div');
divElemRootDiv.appendChild(createTextNode(`${secondText.trim().replace('$LINK$', '')} `));
divElemRootDiv.appendChild((reason === 'updateRequired') ? scriptLink : ssUpdateInstructionsLink);
divElemRoot.appendChild(divElemRootDiv);
}
}
else {
divElemRoot.appendChild(createElem('div', { textContent: (err.commentList === 1001) ? I18n.t('urce.prompts.CustomGSheetLoadError') : I18n.t('urce.common.ErrorGeneric') }));
}
logError(divElemRoot.textContent);
_commentListLoaded = false;
if (err.phase === 'changeList') {
if (err.staticList) {
divElemRoot.appendChild(createElem('br'));
divElemRoot.appendChild(createElem('div', { textContent: `${I18n.t('urce.common.Type')}: ${I18n.t('urce.common.Static')}` }));
}
else if (err.commentList) {
const commentListInfo = getCommentListInfo(err.commentList);
divElemRoot.appendChild(createElem('br'));
divElemRoot.appendChild(createElem('div', { textContent: `${I18n.t('urce.common.CommentList')}: ${commentListInfo.name}` }));
divElemRoot.appendChild(createElem('div', { textContent: `${I18n.t('urce.common.ListOwner')}: ${commentListInfo.listOwner}` }));
}
}
document.getElementById('_commentList')?.replaceChildren(divElemRoot);
WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, divElemRoot.outerHTML);
maskBoxes(undefined, true, err.phase, err.maskUrPanel);
return { divElemRoot };
}
async function initBackgroundTasks(status, phase) {
logDebug(`${((status === 'enable') ? 'Initializing' : 'Uninitializing')} background tasks.`);
if (status === 'enable') {
logDebug('Setting event listeners for UR markers.');
if (phase === 'init') {
_initialUrLayerScan = true;
await handleUrLayer('init', undefined, getMapUrsObjArr());
_initialUrLayerScan = false;
}
if (!_saveButtonObserver.isObserving
|| !_urPanelContainerObserver.isObserving
|| !_urMarkerObserver.isObserving
|| !_urceSidepanelContentObserver.isObserving
|| !_userInfoTabContentObserver.isObserving
) {
logDebug('Enabling MOs.');
if (!_saveButtonObserver.isObserving) {
_saveButtonObserver.observe(document.getElementById('save-button'), {
childList: false, attributes: true, attributeOldValue: true, characterData: false, characterDataOldValue: false, subtree: false
});
_saveButtonObserver.isObserving = true;
}
if (!_urPanelContainerObserver.isObserving) {
_urPanelContainerObserver.observe(document.getElementById('panel-container'), {
childList: true, attributes: true, attributeOldValue: true, characterData: false, characterDataOldValue: false, subtree: true
});
_urPanelContainerObserver.isObserving = true;
}
if (!_urMarkerObserver.isObserving) {
_urMarkerObserver.observe(W.map.getLayerByName('update_requests').div, {
childList: true, attributes: true, attributeOldValue: true, characterData: false, characterDataOldValue: false, subtree: true
});
_urMarkerObserver.isObserving = true;
}
if (!_urceSidepanelContentObserver.isObserving) {
_urceSidepanelContentObserver.observe(document.getElementById('sidepanel-urc-e').parentElement, {
childList: false, attributes: true, attributeOldValue: true, characterData: false, characterOldValue: false, subtree: false
});
_urceSidepanelContentObserver.isObserving = true;
}
if (!_userInfoTabContentObserver.isObserving) {
_userInfoTabContentObserver.observe(document.querySelector('#user-info .tab-content'), {
childList: true, attributes: false, attributeOldValue: false, subtree: false
});
_userInfoTabContentObserver.isObserving = true;
}
}
logDebug('Registering event hooks.');
W.map.events.registerPriority('mousedown', null, mouseDown);
W.map.events.register('zoomend', undefined, invokeZoomEnd);
W.map.events.register('mouseup', undefined, mouseUp);
W.prefs.on('change:isImperial', invokeModeChange);
W.model.mapUpdateRequests.on('objectsadded', mUrsAdded);
W.model.mapUpdateRequests.on('objectsremoved', mUrsRemoved);
W.app.on('change:loadingIssueTrackerMapData', handleUrOverflow);
W.model.states.on('objectsadded', checkRestrictions);
W.model.countries.on('objectsadded', checkRestrictions);
}
else if (status === 'disable') {
if (_saveButtonObserver.isObserving || _urPanelContainerObserver.isObserving) {
logDebug('Disabling MOs.');
if (_saveButtonObserver.isObserving) {
_saveButtonObserver.disconnect();
_saveButtonObserver.isObserving = false;
}
if (_urPanelContainerObserver.isObserving) {
_urPanelContainerObserver.disconnect();
_urPanelContainerObserver.isObserving = false;
}
}
logDebug('Disabling event listeners for UR markers.');
const markerMapCollection = W.map.getLayerByName('update_requests').markers;
if (markerMapCollection) {
for (let idx = 0, { length } = markerMapCollection; idx < length; idx++) {
if (markerMapCollection[idx].element?.attributes?.['data-id'].value) {
const iconDiv = markerMapCollection[idx].element;
iconDiv.removeEventListener('mouseover', markerMouseOver);
iconDiv.removeEventListener('mouseout', markerMouseOut);
iconDiv.dataset.urceHasListeners = false;
}
}
}
logDebug('Unregistering map.events event hook.');
W.map.events.unregister('mousedown', undefined, mouseDown);
W.map.events.unregister('zoomend', undefined, invokeZoomEnd);
W.map.events.unregister('mouseup', undefined, mouseUp);
W.model.mapUpdateRequests.off('objectsadded', mUrsAdded);
W.model.mapUpdateRequests.off('objectsremoved', mUrsRemoved);
W.app.off('change:loadingIssueTrackerMapData', handleUrOverflow);
W.model.states.off('objectsadded', checkRestrictions);
W.model.countries.off('objectsadded', checkRestrictions);
}
return Promise.resolve();
}
function injectCss() {
logDebug('Injecting CSS.');
document.head.appendChild(createElem('style', {
type: 'text/css',
textContent: ''
// Comments tab
+ '#sidepanel-urc-e #panel-urce-comments .URCE-Comments { text-decoration:none; cursor:pointer; color: #000000; font-size:11px; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-commentListName { padding-top:5px; font-size:10px; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divLoading { text-align:left; color:red; font-size:12px; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divCCLinks { text-align:center; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divIcon { height:0px; position:relative; top:-3px; left:-100px; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-icon { cursor:default; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment { padding:0px 4px 0px 4px; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment:before, #panel-urce-comments .URCE-divComment.hover:after { '
+ ' content:""; position:absolute; bottom:-2px; width:0px; height:2px; transition:all 0.2s ease-in-out; transition-duration:0.5s; opacity:0;'
+ '}'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.expand.URCE-openLink:before { left:calc(50%); background-color:#000000; margin-right:5px; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.expand.URCE-openLink:after { right:calc(50%); background-color:#000000; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.expand.URCE-solvedLink:before { left:calc(50%); background-color:#008F00; margin-right:5px; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.expand.URCE-solvedLink:after { right:calc(50%); background-color:#008F00;}'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.expand.URCE-niLink:before { left:calc(50%); background-color:#E68A00; margin-right:5px; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.expand.URCE-niLink:after { right:calc(50%); background-color:#E68A00; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover:hover { cursor:pointer; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.URCE-blankLine:hover { cursor:default; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment:hover:after { width:100%; opacity:1; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment:hover:before { width:100%; opacity:1; margin-right:5px; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover:hover.expand:after { width:50%; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover:hover.expand:before { width:50%; margin-right:5px; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-solvedLink { color:#008F00; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-niLink { color:#E68A00; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-openLink { color:#000000; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-doubleClickIcon { padding-top:4px; height:14px; float:right; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-divDoubleClick { display:inline; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-span { cursor:pointer; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-group_body { line-height: 15px; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-group_body.urStyle { padding-left:23px !important; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-controls input[type="checkbox"] { margin:2px; vertical-align:middle; cursor:pointer; }'
+ '#sidepanel-urc-e #panel-urce-comments .URCE-controls label { font-weight:normal; cursor:pointer; display:inline-block; position:relative; padding-left:16px; }'
// Settings tab
+ '#sidepanel-urc-e #panel-urce-settings .URCE-divDefaultSettings {'
+ ' background-color:lightgray; border-top:1px solid gray; border-bottom:1px solid gray; margin-top:8px; text-align:center; font-size:10px; font-weight:600;'
+ ' text-transform:uppercase;'
+ '}'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-divWarningPre { margin-left:3px; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-divWarning { display:inline; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-divWarningTitle { color:red; text-decoration:underline; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-daysInput { width:38px; height:20px; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-textInput { width:175px; height:20px; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-textInput2 { width:75px; height:20px; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-span { font-size:12px; text-transform:uppercase; cursor:pointer; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-controls { padding:5px 0 5px 0; font-size:10px;}'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-controls .URCE-subHeading { font-weight:600; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-controls .URCE-textFirst, .URCE-controls.URCE-textFirst { padding:0 0 0 16px !important; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-controls .URCE-textFirst.urceDisabled, .URCE-controls.URCE-textFirst.urceDisabled, div.URCE-label.urceDisabled { color:#808080; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-controls .URCE-divDaysInline { display:inline; padding-left:3px; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-controls .URCE-divDaysInline.urceDisabled { display:inline; padding-left:2px; cursor:default; color:#808080; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-controls input[type="checkbox"] { margin:2px; vertical-align:middle; cursor:pointer; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-controls input[type="checkbox"][disabled] { margin:2px; vertical-align:middle; cursor:default; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-controls select { height:22px; vertical-align:middle; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-controls label { font-weight:normal; cursor:pointer; display:inline-block; position:relative; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-controls label.urceDisabled { font-weight:normal; cursor:default; color:#808080; display:inline-block; position:relative; }'
+ '#sidepanel-urc-e #panel-urce-settings .URCE-spreadsheetLink { font-size:11px; text-align:right; }'
// Tools tab
+ '#sidepanel-urc-e #panel-urce-tools .URCE-divCC { text-align:center; }'
+ '#sidepanel-urc-e #panel-urce-tools .URCE-span { font-size:12px; text-transform:uppercase; cursor:pointer; }'
+ '#sidepanel-urc-e #panel-urce-tools .urceToolsButtonFile {'
+ ' font-size:11px; background-color:lightgray; border:1px solid gray; cursor:default; height:22px; margin-top:6px; border-radius:4px;'
+ '}'
+ '#sidepanel-urc-e #panel-urce-tools .URCE-divRestoreFileError { font-weight:600; margin:6px; font-size:11px; text-align:left; }'
+ '#sidepanel-urc-e #panel-urce-tools #urceGSheetCreateConvert { margin-top:5px; }'
// disable WME
+ '#urce-disableWme.URCE-disableWme-main { position:absolute; top:0; left:0; width:100%; height:100%; background:rgba(0,0,0,0.75); z-index:2000; }'
+ '#urce-disableWme #urce-disableWme-text.URCE-disableWme-text {'
+ ' position:absolute; padding:8px; top:50%; left:50%; transform:translate(-50%,-50%); border:2px solid black; border-radius:5px;box-shadow:5px 5px 10px black;'
+ ' background-color:#2F96B4; color:white; z-index:20001;'
+ '}'
+ '#urce-disableWme #urce-disableWme-text .URCE-disableWme-text-header { border-bottom:1px solid;padding: 0 5px 5px 5px;font-size: 18px;text-align:center;font-weight: 600; }'
+ '#urce-disableWme #urce-disableWme-text .URCE-disableWme-text-body { border-bottom: 1px solid;padding: 5px 0 0 0; }'
+ '#urce-disableWme #urce-disableWme-text .URCE-disableWme-text-footer { padding-top:5px; text-align:center; }'
+ '#urce-disableWme #urce-disableWme-text li { font-weight:600; font-size:13px; }'
+ '#urce-disableWme #urce-disableWme-text p { font-weight:normal; margin:0; font-size:13px; }'
+ '#urce-disableWme #urce-disableWme-text a { cursor:pointer; }'
// Common
+ '.urceToolsButton {'
+ ' font-size:11px; margin-left:10px; background-color:lightgray; border:none !important; cursor:default; height:22px; border-radius:4px; border:1px solid gray;'
+ '}'
+ '.urceToolsButton.active, .urceToolsButtonFile:hover, .urceToolsButton:hover { background-color:gray !important; }'
+ '#sidepanel-urc-e .URCE-divDismiss {'
+ ' display:inline-block; float:right; width:20px; height:20px; margin-top:-12px; border-radius:50%; border:1px solid black; background-color:white; text-align:center; '
+ ' cursor:pointer; font-size:18px;'
+ '}'
+ '#sidepanel-urc-e .URCE-divWarningBox { background-color:indianred; border:1px solid silver; margin:6px 0 6px 0; font-size:12px; border-radius:4px; padding:5px; font-weight:600; }'
+ '#sidepanel-urc-e .URCE-collapsed { display:none; }'
+ '#sidepanel-urc-e .URCE-expandCollapseAll { font-size:9px; margin-bottom:-10px; text-align:right; }'
+ '#sidepanel-urc-e .URCE-expandCollapseAll.urStyle { margin-bottom:unset !important; }'
+ '#sidepanel-urc-e .URCE-expandCollapseAllItem { display:inline; cursor:pointer; }'
+ '#sidepanel-urc-e .URCE-chevron { cursor:pointer; font-size:22px; margin-right:3px; float:left; }'
+ '#sidepanel-urc-e .URCE-field { border:1px solid silver; padding:5px; border-radius:4px; -webkit-padding-before:0; }'
+ '#sidepanel-urc-e .URCE-field.urStyle { border:unset !important; padding:unset !important; border-radius:unset !important; }'
+ '#sidepanel-urc-e .URCE-legend { margin-bottom:0px; border-bottom-style:none; width:auto; }'
+ '#sidepanel-urc-e .URCE-legend.urStyle {'
+ ' border-bottom-style:unset !important; margin-bottom:2px !important; width:100% !important; background-color:#F6F7F7 !important; line-height:20px !important;'
+ ' padding:0 2px 0 2px !important; border-top:1px solid #C0C0C0 !important; border-bottom:1px solid #C0C0C0 !important;'
+ '}'
+ '#sidepanel-urc-e .URCE-divCC { /* padding-top:2px !important; */ }'
+ '#sidepanel-urc-e .URCE-label { white-space:pre-line; margin:0 0 0 0; }'
+ '#sidepanel-urc-e .URCE-span { font-size:13px; font-weight:600; }'
+ '#sidepanel-urc-e .URCE-spanTitle { font-size:14px; font-weight:600; }'
+ '#sidepanel-urc-e .URCE-spanVersion { font-size:11px; margin-left:11px; color:#aaa; }'
+ '#sidepanel-urc-e .URCE-divTabs { padding-right:5px; height:calc(100vh - var(--height-offset)); }'
+ '#sidepanel-urc-e .URCE-navTabs { padding:0 0 6px; }'
+ '#sidepanel-urc-e .URCE-navTabs li { flex-grow:1 !important; }' // Compatibility with FUME "Compress/enhance side panel contents" setting
+ '#panel-urce-comments { padding: 0px !important; width:100% !important; }'
+ '#panel-urce-settings { padding: 0px !important; width:100% !important; }'
+ '#panel-urce-tools { padding: 0px !important; width:100% !important; }'
// Main Tabs
+ '.URCE-tabIcon { margin-bottom:3px; width:20px; }'
+ '.URCE-urFilteringToggleBtn { margin-left:4px; cursor:pointer; font-size:18px; margin-top:2px; float:right; }'
+ '.URCE-spinner { margin-left:2px; line-height:2; font-size:12px; color:lightgray; }'
// urceDiv
+ '#urceDiv { position:absolute; visibility:hidden; top:0; left:0; z-index:15000; background-color:aliceBlue; border-width:3px; border-style:solid; border-radius:10px;'
+ ' box-shadow:5px 5px 10px silver;'
+ '}'
+ '#urceDiv hr { border-top:1px solid #000000; }'
+ '#urceDiv .urceDivCloseButton { float:right; cursor:pointer; padding:2px 6px; margin-top:-15px; margin-right:-15px; background-color:aliceblue; border-width:3px; border-style:solid;'
+ ' border-radius:10px; box-shadow: 5px 5px 10px silver; font-weight:600;'
+ '}'
+ '#urceDiv .urceDivContent { padding:5px 15px 0px 5px; }'
+ '#urceDiv .urceDivDisablePopups { float:right; cursor:pointer; margin-top:-12px; margin-right:10px; font-size:10px; text-decoration:underline; }'
// UR panel Manipulation
+ '#urceShortcuts { text-align:left; padding-bottom:8px; font-size:14px; width:100%; }'
+ '#urceShortcuts .chevron { display:inline; float:right; }'
+ '#urceShortcuts .currentDateText { display:inline-block; padding-left:26px; font-size:12px; vertical-align:top; }'
+ '#urceShortcuts .driveDateText { display:inline-block; padding-left:8px; font-size:12px; vertical-align:top; }'
+ '#urceShortcuts .separator { display:inline-block; font-size:12px; vertical-align:text-top; }'
+ '#urceShortcuts i.URCE-chevron { font-weight:900; }'
+ '#urceShortcutsExpand { padding-bottom:4px; font-size:13px; cursor:pointer; border-bottom:1px solid darkgray; }'
+ '#urceShortcutsExpandDiv { border-bottom:1px solid darkgray; padding: 5px 0 5px 0; }'
+ '#panel-container wz-card[class^="panel"].problem-edit { width:380px; max-height:87vh; }'
+ '#panel-container wz-card[class^="panel"].problem-edit.problem-edit { --wz-card-width: 100%; }'
+ '#panel-container wz-card[class^="panel"].problem-edit>* { max-height:87vh; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .conversation-view .comment-list { padding: 0px 6px; margin-bottom: 6px; max-height: 26vh; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .conversation-view .new-comment-form .new-comment-text { margin-bottom: 0px; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .conversation-view .comment .comment-title .date.urce { display: flex; justify-content: flex-end; margin-top: -4px; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .issue-panel-header { padding-top: 5px; padding-bottom: 5px; font-size: 12px; line-height: 14px; padding-right: 0px; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .issue-panel-header .main-title { font-size: 14px; line-height: 14px; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .issue-panel-header .dot { top: 6px; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .section .content { padding: 5px 12px; font-size: 12px; line-height: 14px; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .section .content .URCE-divDesc { max-height: 82px; overflow-y: auto; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .section .title { padding: 0 6px 0 6px; font-size: 13px; line-height: 13px; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .actions .controls-container { margin-top: -2px; margin-bottom: -2px; text-align: center; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .more-info .more-info-checkbox label { font-size: 12px; line-height: 14px; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .actions .controls-container label[for|="state"] { height: 22px; width: unset; min-width: 162px; line-height: 26px; margin: 2px; }'
+ '#panel-container wz-card[class^="panel"].problem-edit[data-state="open"] .actions .controls-container label[for="state-solved"] { display: inline-block; }'
+ '#panel-container wz-card[class^="panel"].problem-edit[data-state="open"] .actions .controls-container label[for|="state-not-identified"] { display: inline-block; }'
+ '#panel-container wz-card[class^="panel"].problem-edit[data-state="solved"] .actions .controls-container label[for|="state-open"], '
+ ' #panel-container wz-card[class^="panel"].problem-edit[data-state="not-identified"] .actions .controls-container label[for|="state-open"] { display: inline-block !important; }'
+ '#panel-container wz-card[class^="panel"].problem-edit[data-state="not-identified"] .actions .controls-container label[for|="state-not-identified"], '
+ ' #panel-container wz-card[class^="panel"].problem-edit[data-state="not-identified"] .actions .controls-container label[for|="state-solved"] { display: none !important; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .actions .navigation .waze-plain-btn { height: 30px; line-height: 18px; font-size: 13px; }'
+ '#panel-container wz-card[class^="panel"].problem-edit .actions .no-permissions-alert { margin-bottom: 8px; margin-top: 2px; padding: 4px; }'
// Map content
+ '.urceCountersPill { color:black; position:absolute; top:30px; display:block; width:auto; white-space:nowrap; padding-left:5px; padding-right:5px; border:1px solid; border-radius:25px; }'
// URCE Lightbox
+ '.urceMaskLightbox { position:absolute; width:100%; height:100%; background:rgba(0,0,0,.75); color:white; display:flex; align-items:center; justify-content:center; }'
+ '.urceMaskLightbox .text { font-weight:800; }'
}));
}
function initCommentsTab() {
logDebug('Initializing Comments tab.');
const zoomOutLinkClicked = function () {
if (document.querySelector('#panel-container wz-card[class^="panel"].problem-edit div[class^="container"] .close-panel'))
autoCloseUrPanel();
W.map.getOLMap().zoomTo(+this.getAttribute('zoomTo'));
};
const imgDiv = createElem('div', { id: 'urceIcon', class: 'URCE-divIcon' }),
contentHeaderDiv = createElem('div', { id: '_divZoomOutLinks', class: 'URCE-divCCLinks', style: _settings.hideZoomOutLinks ? 'display:none;' : '' });
imgDiv.replaceChildren(createElem('img', { src: GM_info.script.icon, class: 'URCE-icon' }));
contentHeaderDiv.appendChild(imgDiv);
contentHeaderDiv.appendChild(createElem('a', {
id: 'zoomOutLink1', class: 'URCE-Comments', zoomTo: '12', title: I18n.t('urce.commentsTab.ZoomOutLink1Title'), textContent: I18n.t('urce.commentsTab.ZoomOutLink1')
}, [{ click: zoomOutLinkClicked }]));
contentHeaderDiv.appendChild(createElem('br'));
contentHeaderDiv.appendChild(createElem('a', {
id: 'zoomOutLink2', class: 'URCE-Comments', zoomTo: '14', title: I18n.t('urce.commentsTab.ZoomOutLink2Title'), textContent: I18n.t('urce.commentsTab.ZoomOutLink2')
}, [{ click: zoomOutLinkClicked }]));
contentHeaderDiv.appendChild(createElem('br'));
contentHeaderDiv.appendChild(createElem('a', {
id: 'zoomOutLink3', class: 'URCE-Comments', zoomTo: '15', title: I18n.t('urce.commentsTab.ZoomOutLink3Title'), textContent: I18n.t('urce.commentsTab.ZoomOutLink3')
}, [{ click: zoomOutLinkClicked }]));
contentHeaderDiv.appendChild(createElem('br'));
let docFrags = document.createDocumentFragment();
docFrags.appendChild(contentHeaderDiv);
docFrags.appendChild(createElem('div', { id: '_commentList', class: 'URCE-divCC' }));
document.getElementById('panel-urce-comments').replaceChildren(docFrags);
if (_needTranslation) {
docFrags = document.createDocumentFragment();
let divElemRoot = createElem('div');
divElemRoot.appendChild(createTextNode('URC-E does not currently have a translation for your WME Language Setting ('));
divElemRoot.appendChild(createElem('div', { style: 'display:inline-block;font-style:italic;', textContent: I18n.currentLocale() }));
divElemRoot.appendChild(createTextNode('). Translations are setup on a Google Sheet, so they are simple to do.'));
docFrags.appendChild(divElemRoot);
docFrags.appendChild(createElem('br'));
divElemRoot = createElem('div');
divElemRoot.appendChild(createTextNode('If you would like to provide a translation for your WME Language Setting ('));
divElemRoot.appendChild(createElem('div', { style: 'display:inline-block;font-style:italic;', textContent: I18n.currentLocale() }));
divElemRoot.appendChild(createTextNode(`), please contact ${_SCRIPT_AUTHOR} via forum PM or Discord, or click reply on the `));
divElemRoot.appendChild(createElem('a', { href: 'https://www.waze.com/forum/viewtopic.php?f=819&t=275608#p1920278', target: '_blank', textContent: 'forum thread' }));
divElemRoot.appendChild(createTextNode('.'));
docFrags.appendChild(divElemRoot);
alertBoxInPanel(docFrags, undefined, true, 9998);
}
}
function initToolsTab() {
logDebug('Initializing Tools tab.');
const docFrags = document.createDocumentFragment(),
urStyle = (_settings.commentListStyle === 'urStyle') ? ' urStyle' : '',
expandCollapseAll = function () {
const legends = document.querySelectorAll('legend[id^="urce-tools-legend"]');
for (let idx = 0, { length } = legends; idx < length; idx++) {
if (this.id === 'URCE-expandAllTools') {
if (legends[idx].nextElementSibling.classList.contains('URCE-collapsed'))
legends[idx].click();
}
else if (this.id === 'URCE-collapseAllTools') {
if (!legends[idx].nextElementSibling.classList.contains('URCE-collapsed'))
legends[idx].click();
}
}
},
backupSettingsClick = () => {
saveSettingsToStorage();
const a = createElem('a', {
href: URL.createObjectURL(new Blob([JSON.stringify({ URCE: _settings })], { type: 'application/json' })), download: `urce-settings-v${_SCRIPT_VERSION}.json`
});
a.click();
a.remove();
},
restoreSettingsClick = function () {
if (!this.classList.contains('active')) {
this.classList.add('active');
document.getElementById('urceRestoreSettingsFile').style.display = '';
}
else {
this.classList.remove('active');
document.getElementById('_filerestoreSettings').value = '';
document.getElementById('urceRestoreSettingsFile').style.display = 'none';
document.getElementById('urceRestoreSettingsFile').classList.remove('error');
document.getElementById('urceRestoreSettingsFileError').replaceChildren();
}
},
resetSettingsClick = () => {
WazeWrap.Alerts.confirm(
_SCRIPT_SHORT_NAME,
I18n.t('urce.prompts.ResetSettingsConfirmation'),
() => { loadSettingsFromStorage('resetSettings', true); },
() => { },
I18n.t('urce.common.Yes'),
I18n.t('urce.common.No')
);
},
inputFileChange = function () {
const file = this.files[0];
if (((file.type === 'application/json') || (file.type === 'text/json')) && (file.size < 102400)) {
const reader = new FileReader();
reader.onload = function () {
let restoreSettings;
try {
restoreSettings = JSON.parse(this.result);
}
catch (error) {
logError('Unable to parse the input file.', error);
document.getElementById('_filerestoreSettings').value = '';
return;
}
if (!restoreSettings.hasOwnProperty('URCE')) {
logWarning('Invalid URCE settings JSON file.');
document.getElementById('urceRestoreSettingsFileError').style.display = '';
document.getElementById('urceRestoreSettingsFile').classList.add('error');
}
else {
loadSettingsFromStorage(restoreSettings.URCE, false);
}
};
reader.readAsText(file);
}
else {
logWarning('Invalid URCE settings JSON file.');
document.getElementById('urceRestoreSettingsFileError').style.display = '';
document.getElementById('urceRestoreSettingsFile').classList.add('error');
}
},
legendClickToggle = function () {
this.firstChild.classList.toggle('w-icon-chevron-up');
this.firstChild.classList.toggle('w-icon-chevron-down');
this.nextSibling.classList.toggle('URCE-collapsed');
saveSettingsToStorage();
},
buildFieldsetSection = (idTag, textContent, divAttrs = { class: 'URCE-controls URCE-divCC' }) => {
const legendElem = createElem('legend', { id: `urce-tools-legend-${idTag}`, class: `URCE-legend${urStyle}` }, [{ click: legendClickToggle }]);
legendElem.appendChild(createElem('i', { class: 'w-icon w-icon-chevron-up URCE-chevron' }));
legendElem.appendChild(createElem('span', { class: 'URCE-span', textContent }));
const fieldsetElem = createElem('fieldset', { id: `urce-tools-fieldset-${idTag}`, class: `URCE-field${urStyle}` });
fieldsetElem.appendChild(legendElem);
const divElemRoot = createElem('div', divAttrs);
fieldsetElem.appendChild(divElemRoot);
return { fieldsetElem, contentDiv: divElemRoot };
},
convertCurrentCustomClick = () => createStaticToGoogleSheet(true),
createNewCustomClick = () => createStaticToGoogleSheet(false);
const contentHeaderDiv = createElem('div', { id: 'expandCollapseAllTools', class: `URCE-expandCollapseAll${urStyle}` });
contentHeaderDiv.appendChild(createElem('div', {
id: 'URCE-expandAllTools', class: 'URCE-expandCollapseAllItem', textContent: I18n.t('urce.common.ExpandAll')
}, [{ click: expandCollapseAll }]));
contentHeaderDiv.appendChild(createElem('div', { style: 'display:inline;', textContent: ' : ' }));
contentHeaderDiv.appendChild(createElem('div', {
id: 'URCE-collapseAllTools', class: 'URCE-expandCollapseAllItem', textContent: I18n.t('urce.common.CollapseAll')
}, [{ click: expandCollapseAll }]));
docFrags.appendChild(contentHeaderDiv);
let { fieldsetElem, contentDiv } = buildFieldsetSection('settings', I18n.t('urce.tabs.Settings'));
contentDiv.appendChild(createElem('button', {
id: '_butbackupSettings', urceprefs: 'tools', class: 'urceToolsButton', title: I18n.t('urce.tools.BackupSettingsTitle'), textContent: I18n.t('urce.common.Backup')
}, [{ click: backupSettingsClick }]));
contentDiv.appendChild(createElem('button', {
id: '_butrestoreSettings', urceprefs: 'tools', class: 'urceToolsButton', title: I18n.t('urce.tools.RestoreSettingsTitle'), textContent: I18n.t('urce.common.Restore')
}, [{ click: restoreSettingsClick }]));
contentDiv.appendChild(createElem('button', {
id: '_butresetSettings', urceprefs: 'tools', class: 'urceToolsButton', title: I18n.t('urce.tools.ResetSettingsTitle'), textContent: I18n.t('urce.common.Reset')
}, [{ click: resetSettingsClick }]));
const divElemRoot = createElem('div', { id: 'urceRestoreSettingsFile', style: 'display:none;' });
divElemRoot.appendChild(createElem('input', {
type: 'file', id: '_filerestoreSettings', urceprefs: 'tools', class: 'urceToolsButtonFile', title: I18n.t('urce.tools.RestoreSettingsSelectFileTitle')
}, [{ change: inputFileChange }]));
divElemRoot.appendChild(createElem('div', {
id: 'urceRestoreSettingsFileError', class: 'URCE-divRestoreFileError error', style: 'display:none;', textContent: `* ${I18n.t('urce.tools.RestoreSettingsFileError')}`
}));
contentDiv.appendChild(divElemRoot);
docFrags.appendChild(fieldsetElem);
({ fieldsetElem, contentDiv } = buildFieldsetSection('customGoogleSpreadsheet', I18n.t('urce.tabs.Settings'), { id: 'urceGSheetCreateConvert', urceprefs: 'tools' }));
contentDiv.appendChild(createElem('button', {
id: '_butconvertCurrentCustom', urceprefs: 'tools', class: 'urceToolsButton', title: I18n.t('urce.tools.ConvertCurrentCustomTitle'), textContent: I18n.t('urce.tools.ConvertCurrentCustom')
}, [{ click: convertCurrentCustomClick }]));
contentDiv.appendChild(createElem('button', {
id: '_butcreateNewCustom', urceprefs: 'tools', class: 'urceToolsButton', title: I18n.t('urce.tools.CreateNewCustomTitle'), textContent: I18n.t('urce.tools.CreateNewCustom')
}, [{ click: createNewCustomClick }]));
docFrags.appendChild(fieldsetElem);
document.getElementById('panel-urce-tools').replaceChildren(docFrags);
}
function initSettingsTab() {
logDebug('Initializing Settings tab.');
const docFrags = document.createDocumentFragment(),
urStyle = (_settings.commentListStyle === 'urStyle') ? ' urStyle' : '',
expandCollapseAll = function () {
const legends = document.querySelectorAll('legend[id^="urce-prefs-legend"');
for (let idx = 0, { length } = legends; idx < length; idx++) {
if (this.id === 'URCE-expandAllSettings') {
if (legends[idx].nextElementSibling.classList.contains('URCE-collapsed'))
legends[idx].click();
}
else if (this.id === 'URCE-collapseAllSettings') {
if (!legends[idx].nextElementSibling.classList.contains('URCE-collapsed'))
legends[idx].click();
}
}
},
legendClickToggle = function () {
this.firstChild.classList.toggle('w-icon-chevron-up');
this.firstChild.classList.toggle('w-icon-chevron-down');
this.nextSibling.classList.toggle('URCE-collapsed');
},
commentListSelectionChange = function () {
if ((+this.value === 1001) && (!_settings.customSsId || (_settings.customSsId.length < 1))) {
this.value = _currentCommentList;
WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.SetCustomSsIdFirst'));
}
else {
changeCommentList(+this.value, false, false);
}
},
commentListStyleSelectionChange = function () {
changeCommentListStyle(this.value);
},
spreadsheetLinkClick = function (e) {
e.preventDefault();
e.stopPropagation();
window.open(this.href, '_blank');
},
urceSettingsCheckboxChange = function () {
let otherSettingName;
const settingName = this.id.substring(3),
urcePrefs = this.attributes.urceprefs.value;
if (settingName === 'hideFollowing')
otherSettingName = 'hideNotFollowing';
if (settingName === 'hideNotFollowing')
otherSettingName = 'hideFollowing';
if (settingName === 'hideWithDescription')
otherSettingName = 'hideWithoutDescription';
if (settingName === 'hideWithoutDescription')
otherSettingName = 'hideWithDescription';
if (settingName === 'hideWithCommentsFromMe')
otherSettingName = 'hideWithoutCommentsFromMe';
if (settingName === 'hideWithoutCommentsFromMe')
otherSettingName = 'hideWithCommentsFromMe';
if (settingName === 'hideFirstCommentByMe')
otherSettingName = 'hideFirstCommentNotByMe';
if (settingName === 'hideFirstCommentNotByMe')
otherSettingName = 'hideFirstCommentByMe';
if (settingName === 'hideLastCommentByMe')
otherSettingName = 'hideLastCommentNotByMe';
if (settingName === 'hideLastCommentNotByMe')
otherSettingName = 'hideLastCommentByMe';
if (settingName === 'hideLastCommentByReporter')
otherSettingName = 'hideLastCommentNotByReporter';
if (settingName === 'hideLastCommentNotByReporter')
otherSettingName = 'hideLastCommentByReporter';
if (settingName === 'replaceNextWithDoneButton')
otherSettingName = 'disableDoneNextButtons';
if (settingName === 'disableDoneNextButtons')
otherSettingName = 'replaceNextWithDoneButton';
if (otherSettingName) {
if (this.checked && document.getElementById(`_cb${otherSettingName}`)?.checked) {
_settings[otherSettingName] = false;
document.getElementById(`_cb${otherSettingName}`).checked = false;
}
}
if (settingName === 'disableDoneNextButtons') {
if (this.checked)
document.querySelector('#panel-container .content .navigation').style.display = 'none';
else
document.querySelector('#panel-container .content .navigation').style.display = '';
}
if (settingName === 'hideZoomOutLinks') {
if (this.checked)
document.getElementById('_divZoomOutLinks').style.display = 'none';
else
document.getElementById('_divZoomOutLinks').style.display = '';
}
if (settingName === 'unstackMarkers') {
if (this.checked) {
document.querySelectorAll('[urceprefs$="-unstack"]').forEach((el) => {
el.disabled = false;
el.classList.remove('urceDisabled');
});
}
else {
document.querySelectorAll('[urceprefs$="-unstack"]').forEach((el) => {
el.disabled = true;
el.classList.add('urceDisabled');
});
}
}
if (settingName === 'autoClickOpenSolvedNi') {
if (document.getElementById('_cbdoubleClickLinkNiComments').checked) {
_settings.doubleClickLinkNiComments = false;
document.getElementById('_cbdoubleClickLinkNiComments').checked = false;
document.getElementById('URCE-divDoubleClickNi').style.display = 'none';
}
if (document.getElementById('_cbdoubleClickLinkOpenComments').checked) {
_settings.doubleClickLinkOpenComments = false;
document.getElementById('_cbdoubleClickLinkOpenComments').checked = false;
document.getElementById('URCE-divDoubleClickOpen').style.display = 'none';
}
if (document.getElementById('_cbdoubleClickLinkSolvedComments').checked) {
_settings.doubleClickLinkSolvedComments = false;
document.getElementById('_cbdoubleClickLinkSolvedComments').checked = false;
document.getElementById('URCE-divDoubleClickSolved').style.display = 'none';
}
}
if (settingName === 'doubleClickLinkOpenComments') {
if (!this.checked) {
document.getElementById('URCE-divDoubleClickOpen').style.display = 'none';
}
else {
if (!document.getElementById('_cbautoClickOpenSolvedNi').checked) {
_settings.autoClickOpenSolvedNi = true;
document.getElementById('_cbautoClickOpenSolvedNi').checked = true;
}
document.getElementById('URCE-divDoubleClickOpen').style.display = '';
}
}
if (settingName === 'doubleClickLinkNiComments') {
if (!this.checked) {
document.getElementById('URCE-divDoubleClickNi').style.display = 'none';
}
else {
if (!document.getElementById('_cbautoClickOpenSolvedNi').checked) {
_settings.autoClickOpenSolvedNi = true;
document.getElementById('_cbautoClickOpenSolvedNi').checked = true;
}
document.getElementById('URCE-divDoubleClickNi').style.display = '';
}
}
if (settingName === 'doubleClickLinkSolvedComments') {
if (!this.checked) {
document.getElementById('URCE-divDoubleClickSolved').style.display = 'none';
}
else {
if (!document.getElementById('_cbautoClickOpenSolvedNi').checked) {
_settings.autoClickOpenSolvedNi = true;
document.getElementById('_cbautoClickOpenSolvedNi').checked = true;
}
document.getElementById('URCE-divDoubleClickSolved').style.display = '';
}
}
if (urcePrefs === 'markerMaster') {
if (!this.checked) {
document.querySelectorAll('[urceprefs=marker]').forEach((el) => {
el.disabled = true;
el.classList.add('urceDisabled');
});
}
else {
document.querySelectorAll('[urceprefs=marker]').forEach((el) => {
el.disabled = false;
el.classList.remove('urceDisabled');
});
}
}
if (urcePrefs === 'filteringMaster') {
if (!this.checked) {
document.querySelectorAll('[urceprefs=filtering]').forEach((el) => {
el.disabled = true;
el.classList.add('urceDisabled');
});
document.getElementById('urceUrFilteringToggleBtn').style.color = '#ccc';
}
else {
document.querySelectorAll('[urceprefs=filtering]').forEach((el) => {
el.disabled = false;
el.classList.remove('urceDisabled');
});
document.getElementById('urceUrFilteringToggleBtn').style.color = '#00bd00';
}
}
_settings[settingName] = this.checked;
if (document.getElementById(`_cbperCommentList_${settingName}_useDefault`)?.checked
&& (document.getElementById(`_cbperCommentList_${settingName}`)?.checked !== this.checked)
)
document.getElementById(`_cbperCommentList_${settingName}`).checked = this.checked;
Object.values(_settings.perCommentListSettings).forEach((arr) => {
if (arr[`${settingName}_useDefault`]) {
if (arr[settingName] !== this.checked)
arr[settingName] = this.checked;
}
});
saveSettingsToStorage();
if (settingName === 'invertFilters')
initSettingsTab();
if (((urcePrefs.includes('marker') || urcePrefs.includes('filtering')) && !settingName.includes('unstack'))
|| (settingName === 'enableUrOverflowHandling')
)
handleUrLayer('settingsToggle', undefined, getMapUrsObjArr());
},
urceSettingsNumberBoxChange = function () {
const settingName = this.id.substring(4),
maxVal = ['disableFilteringAboveZoomLevel', 'disableFilteringBelowZoomLevel', 'unstackDisableAboveZoom'].includes(settingName) ? 22 : 9999,
minVal = ['disableFilteringAboveZoomLevel', 'disableFilteringBelowZoomLevel', 'unstackDisableAboveZoom'].includes(settingName) ? 12 : 0,
val = Math.min(maxVal, Math.max(minVal, +Math.abs(+this.value || minVal)));
if ((val !== +this.value) || (_settings[settingName] !== val)) {
if (val !== +this.value)
this.value = val;
_settings[settingName] = val;
if (document.getElementById(`_cbperCommentList_${settingName}_useDefault`)?.checked
&& (document.getElementById(`_numperCommentList_${settingName}`)?.value !== val)
)
document.getElementById(`_numperCommentList_${settingName}`).value = val;
Object.values(_settings.perCommentListSettings).forEach((arr) => {
if (arr[`${settingName}_useDefault`]) {
if (arr[settingName] !== val)
arr[settingName] = val;
}
});
saveSettingsToStorage();
if (!settingName.includes('unstack'))
handleUrLayer('settingsToggle', undefined, getMapUrsObjArr());
}
},
urceSettingsTextBoxChange = function () {
const settingName = this.id.substring(5),
val = this.value.trim();
if ((val !== this.value) || (_settings[settingName] !== val)) {
if (val !== this.value)
this.value = val;
_settings[settingName] = val;
if (document.getElementById(`_cbperCommentList_${settingName}_useDefault`)?.checked && (document.getElementById(`_numperCommentList_${settingName}`).value !== val))
document.getElementById(`_textperCommentList_${settingName}`).value = val;
Object.values(_settings.perCommentListSettings).forEach((arr) => {
if (arr[`${settingName}_useDefault`]) {
if (arr[settingName] !== val)
arr[settingName] = val;
}
});
saveSettingsToStorage();
if (settingName === 'customSsId') {
if (val && (val.length > 0)) {
const customSsIdLinkClick = function (e) {
e.preventDefault();
e.stopPropagation();
window.open(this.href, '_blank');
},
customSsIdLink = createElem('a', {
class: 'URCE-Controls URCE-spreadsheetLink',
id: 'urce-customSpreadsheet-link',
title: I18n.t('urce.prefs.CustomSpreadsheetLinkTitle'),
href: `https://docs.google.com/spreadsheets/d/${val}/edit`,
textContent: I18n.t('urce.prefs.CustomSpreadsheetLink')
}, [{ click: customSsIdLinkClick }]);
if (!document.getElementById('urceCustomSpreadsheetLinkDiv')) {
const divElemRoot = createElem('div', { id: 'urceCustomSpreadsheetLinkDiv', class: 'URCE-spreadsheetLink' });
divElemRoot.appendChild(customSsIdLink);
document.getElementById('urceSpreadsheetLinks').appendChild(divElemRoot);
}
else {
const urceCustomSpreadsheetLinkDiv = document.getElementById('urceCustomSpreadsheetLinkDiv');
urceCustomSpreadsheetLinkDiv.replaceChildren(customSsIdLink);
}
}
else if (document.getElementById('urceCustomSpreadsheetLinkDiv')) {
document.getElementById('urceCustomSpreadsheetLinkDiv').remove();
}
}
if ((settingName !== 'tagEmail') && (settingName !== 'customSsId') && (settingName !== 'customTagline'))
handleUrLayer('settingsToggle', undefined, getMapUrsObjArr());
else if ((settingName === 'tagEmail') || (settingName === 'customTagline') || ((settingName === 'customSsId') && (_currentCommentList === 1001)))
changeCommentList(_settings.commentList, false, true);
}
},
buildFieldsetSection = (idTag, textContent, divAttrs = { class: 'URCE-controls URCE-divCC' }) => {
const legendElem = createElem('legend', { id: `urce-prefs-legend-${idTag}`, class: `URCE-legend${urStyle}` }, [{ click: legendClickToggle }]);
legendElem.appendChild(createElem('i', { class: 'w-icon w-icon-chevron-up URCE-chevron' }));
legendElem.appendChild(createElem('span', { class: 'URCE-span', textContent }));
const fieldsetElem = createElem('fieldset', { id: `urce-prefs-fieldset-${idTag}`, class: `URCE-field${urStyle}` });
fieldsetElem.appendChild(legendElem);
const divElemRoot = createElem('div', divAttrs);
fieldsetElem.appendChild(divElemRoot);
return { fieldsetElem, contentDiv: divElemRoot };
},
buildStandardCbSetting = (setting, urceprefs) => {
let translationName,
title,
disabled = false,
urceDisabled = '';
if (setting === 'disableFilteringBelowZoom') {
translationName = I18n.t('urce.prefs.DisableFilteringBelowZoomLevel');
title = I18n.t('urce.prefs.DisableFilteringBelowZoomLevelTitle');
}
else if (setting === 'disableFilteringAboveZoom') {
translationName = I18n.t('urce.prefs.DisableFilteringAboveZoomLevel');
title = I18n.t('urce.prefs.DisableFilteringAboveZoomLevelTitle');
}
else {
if (setting === 'hideByStatusOpen')
translationName = I18n.t('venues.update_requests.panel.action.open');
else if (setting === 'hideByStatusClosed')
translationName = I18n.t('urce.urStatus.Closed');
else if (setting === 'hideByStatusNotIdentified')
translationName = I18n.t('urce.urStatus.NotIdentified');
else if (setting === 'hideByStatusSolved')
translationName = I18n.t('venues.update_requests.panel.solved');
else if (setting === 'hideByStatusClosedBy')
translationName = `${I18n.t('urce.common.ClosedBy')}: `;
else if (setting === 'hideByTypeIncorrectTurn')
translationName = I18n.t('update_requests.types.6');
else if (setting === 'hideByTypeIncorrectAddress')
translationName = I18n.t('update_requests.types.7');
else if (setting === 'hideByTypeIncorrectRoute')
translationName = I18n.t('update_requests.types.8');
else if (setting === 'hideByTypeMissingRoundabout')
translationName = I18n.t('update_requests.types.9');
else if (setting === 'hideByTypeGeneralError')
translationName = I18n.t('update_requests.types.10');
else if (setting === 'hideByTypeTurnNotAllowed')
translationName = I18n.t('update_requests.types.11');
else if (setting === 'hideByTypeIncorrectJunction')
translationName = I18n.t('update_requests.types.12');
else if (setting === 'hideByTypeMissingBridgeOverpass')
translationName = I18n.t('update_requests.types.13');
else if (setting === 'hideByTypeWrongDrivingDirection')
translationName = I18n.t('update_requests.types.14');
else if (setting === 'hideByTypeMissingExit')
translationName = I18n.t('update_requests.types.15');
else if (setting === 'hideByTypeMissingRoad')
translationName = I18n.t('update_requests.types.16');
else if (setting === 'hideByTypeMissingLandmark')
translationName = I18n.t('update_requests.types.18');
else if (setting === 'hideByTypeBlockedRoad')
translationName = I18n.t('update_requests.types.19');
else if (setting === 'hideByTypeMissingStreetName')
translationName = I18n.t('update_requests.types.21');
else if (setting === 'hideByTypeIncorrectStreetPrefixOrSuffix')
translationName = I18n.t('update_requests.types.22');
else if (setting === 'hideByTypeMissingOrInvalidSpeedLimit')
translationName = I18n.t('update_requests.types.23');
else if (setting === 'hideByTypeUndefined')
translationName = I18n.t('urce.urTypes.Undefined');
else
translationName = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}`);
title = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}Title`);
}
if (((urceprefs === 'filtering') && !_settings.enableUrceUrFiltering)
|| ((urceprefs === 'marker') && !_settings.enableUrPillCounts)
) {
disabled = true;
urceDisabled = ' urceDisabled';
}
const divElemRoot = createElem('div');
divElemRoot.appendChild(
createElem('input', {
type: 'checkbox', id: `_cb${setting}`, urceprefs, class: `urceSettingsCheckbox${urceDisabled}`, title, disabled, checked: (_settings[setting] === true)
}, [{ change: urceSettingsCheckboxChange }])
);
divElemRoot.appendChild(createElem('label', {
for: `_cb${setting}`, urceprefs, title, class: `URCE-label${urceDisabled}`, disabled, textContent: translationName
}));
if (setting === 'autoSendReminders') {
const divElemWarning = createElem('div', { class: 'URCE-divWarning URCE-divWarningPre', textContent: '(' });
divElemWarning.appendChild(
createElem('div', {
class: 'URCE-divWarning URCE-divWarningTitle', title: I18n.t('urce.prefs.AutoSendRemindersWarningTitle'), textContent: I18n.t('urce.prefs.AutoSendRemindersWarning')
})
);
divElemWarning.appendChild(createElem('div', { class: 'URCE-divWarning', textContent: ')' }));
divElemRoot.appendChild(divElemWarning);
const divElemDiv = createElem('div', { style: 'padding-left:15px; font-style:italic;' });
divElemDiv.appendChild(
createElem('input', {
type: 'checkbox',
id: '_cbautoSendRemindersExceptTagged',
urceprefs,
class: 'urceSettingsCheckbox',
title: I18n.t('urce.prefs.AutoSendRemindersExceptTaggedTitle'),
checked: (_settings.autoSendRemindersExceptTagged === true)
}, [{ change: urceSettingsCheckboxChange }])
);
divElemDiv.appendChild(
createElem('label', {
for: '_cbautoSendRemindersExceptTagged',
urceprefs,
title: I18n.t('urce.prefs.AutoSendRemindersExceptTaggedTitle'),
class: 'URCE-label',
textContent: I18n.t('urce.prefs.AutoSendRemindersExceptTagged')
})
);
divElemRoot.appendChild(divElemDiv);
}
if ((setting === 'disableFilteringAboveZoom') || (setting === 'disableFilteringBelowZoom')) {
const divElemDiv = createElem('div', { class: `URCE-divDaysInline${urceDisabled}`, urceprefs, disabled });
divElemDiv.appendChild(createElem('input', {
type: 'number',
id: `_num${setting}Level`,
class: `URCE-daysInput urceSettingsNumberBox${urceDisabled}`,
urceprefs: 'filtering',
min: '12',
max: '22',
step: '1',
value: _settings[`${setting}Level`],
title,
disabled
}, [{ change: urceSettingsNumberBoxChange }]));
divElemRoot.appendChild(divElemDiv);
}
if (setting === 'hideByStatusClosedBy') {
const divElemDiv = createElem('div', { class: `URCE-divDaysInline${urceDisabled}`, urceprefs, disabled });
divElemDiv.appendChild(createElem('input', {
type: 'text',
id: '_texthideByStatusClosedByUsers',
class: `urceSettingsTextBox${urceDisabled}`,
style: 'width:150px;height:20px;',
urceprefs: 'filtering',
value: _settings.hideByStatusClosedByUsers,
title,
disabled
}, [{ change: urceSettingsTextBoxChange }]));
divElemRoot.appendChild(divElemDiv);
}
return divElemRoot;
},
buildTagCbSetting = (setting, urceprefs, translationName, title) => {
let disabled = false,
urceDisabled = '';
if ((urceprefs === 'filtering') && !_settings.enableUrceUrFiltering) {
disabled = true;
urceDisabled = ' urceDisabled';
}
const divElemRoot = createElem('div');
divElemRoot.appendChild(createElem('input', {
type: 'checkbox', id: `_cb${setting}`, urceprefs, class: `urceSettingsCheckbox${urceDisabled}`, title, disabled, checked: (_settings[setting] === true)
}, [{ change: urceSettingsCheckboxChange }]));
divElemRoot.appendChild(createElem('label', {
for: `_cb${setting}`, urceprefs, title, class: `URCE-label${urceDisabled}`, disabled, textContent: translationName
}));
return divElemRoot;
},
buildTextFirstTextSetting = (setting, urceprefs) => {
let disabled = false,
urceDisabled = '';
if ((urceprefs === 'filtering') && !_settings.enableUrceUrFiltering) {
disabled = true;
urceDisabled = ' urceDisabled';
}
const title = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}Title`),
divElemRoot = createElem('div', {
title, class: `URCE-label${urceDisabled}`, urceprefs, disabled, textContent: `${I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}`)}: `
});
divElemRoot.appendChild(createElem('input', {
type: 'text', id: `_text${setting}`, class: `URCE-textInput urceSettingsTextBox${urceDisabled}`, urceprefs, value: _settings[setting], title, disabled
}, [{ change: urceSettingsTextBoxChange }]));
return divElemRoot;
},
buildTextFirstNumSetting = (setting, urceprefs, min, max, step, postText) => {
let disabled = false,
urceDisabled = '';
if (((urceprefs === 'filtering') && !_settings.enableUrceUrFiltering) || ((urceprefs === 'marker-nodisable-unstack') && !_settings.unstackMarkers)) {
disabled = true;
urceDisabled = ' urceDisabled';
}
const translationName = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}`),
title = formatText(I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}Title`), false, false, -1),
divElemRoot = createElem('div', {
title, class: `URCE-label${urceDisabled}`, urceprefs, disabled, textContent: `${translationName} `
});
divElemRoot.appendChild(createElem('input', {
type: 'number', id: `_num${setting}`, class: `URCE-daysInput urceSettingsNumberBox${urceDisabled}`, urceprefs, value: _settings[setting], title, min, max, step, disabled
}, [{ change: urceSettingsNumberBoxChange }]));
if (postText) {
divElemRoot.appendChild(createElem('div', {
title, class: `URCE-divDaysInline${urceDisabled}`, urceprefs, disabled, textContent: postText
}));
}
return divElemRoot;
},
buildTextFirstDoubleCbSetting = (setting1, setting2, textBefore, urceprefs) => {
let translationName1,
translationName2,
disabled = false,
urceDisabled = '';
if ((urceprefs === 'filtering') && !_settings.enableUrceUrFiltering) {
disabled = true;
urceDisabled = ' urceDisabled';
}
if (setting1 === 'hideFollowing') {
translationName1 = I18n.t('urce.common.Following');
translationName2 = I18n.t('urce.common.NotFollowing');
}
else if ((setting1 === 'hideWithDescription') || (setting1 === 'hideWithCommentsFromMe')) {
translationName1 = I18n.t('urce.common.With');
translationName2 = I18n.t('urce.common.Without');
}
else if ((setting1 === 'hideFirstCommentByMe') || (setting1 === 'hideLastCommentByMe') || (setting1 === 'hideLastCommentByReporter')) {
translationName1 = I18n.t('urce.common.Yes');
translationName2 = I18n.t('urce.common.No');
}
const translationTitle1 = I18n.t(`urce.prefs.${setting1.charAt(0).toUpperCase()}${setting1.slice(1)}Title`),
translationTitle2 = I18n.t(`urce.prefs.${setting2.charAt(0).toUpperCase()}${setting2.slice(1)}Title`),
divElemRoot = createElem('div', { style: 'display:inline-flex;', textContent: `${textBefore}: ` }),
divElemDiv = createElem('div', { style: 'display:inline;' });
divElemDiv.appendChild(createElem('input', {
type: 'checkbox', id: `_cb${setting1}`, urceprefs, class: `urceSettingsCheckbox${urceDisabled}`, title: translationTitle1, disabled, checked: (_settings[setting1] === true)
}, [{ change: urceSettingsCheckboxChange }]));
divElemDiv.appendChild(createElem('label', {
for: `_cb${setting1}`, urceprefs, class: `URCE-label${urceDisabled}`, title: translationTitle1, disabled, textContent: translationName1
}));
divElemDiv.appendChild(createElem('input', {
type: 'checkbox', id: `_cb${setting2}`, urceprefs, class: `urceSettingsCheckbox${urceDisabled}`, title: translationTitle2, disabled, checked: (_settings[setting2] === true)
}, [{ change: urceSettingsCheckboxChange }]));
divElemDiv.appendChild(createElem('label', {
for: `_cb${setting2}`, urceprefs, class: `URCE-label${urceDisabled}`, title: translationTitle2, disabled, textContent: translationName2
}));
divElemRoot.appendChild(divElemDiv);
return divElemRoot;
},
buildCbandNumSetting = (setting, numSetting, urceprefs, min, max, step, postText) => {
let translationName,
disabled = false,
urceDisabled = '';
if ((urceprefs === 'filtering') && !_settings.enableUrceUrFiltering) {
disabled = true;
urceDisabled = ' urceDisabled';
}
if ((setting === 'hideByAgeOfSubmissionLessThan') || (setting === 'hideByCommentCountLessThan'))
translationName = I18n.t('urce.common.LessThan');
else if ((setting === 'hideByAgeOfSubmissionMoreThan') || (setting === 'hideByCommentCountMoreThan'))
translationName = I18n.t('urce.common.MoreThan');
else
translationName = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}`);
const title = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}Title`),
divElemRoot = createElem('div');
divElemRoot.appendChild(createElem('input', {
type: 'checkbox', id: `_cb${setting}`, urceprefs, class: `urceSettingsCheckbox${urceDisabled}`, title, disabled, checked: (_settings[setting] === true)
}, [{ change: urceSettingsCheckboxChange }]));
divElemRoot.appendChild(createElem('label', {
for: `_cb${setting}`, urceprefs, class: `URCE-label${urceDisabled}`, title, disabled, textContent: `${translationName}: `
}));
const divElemDiv = createElem('div', { class: `URCE-divDaysInline${urceDisabled}`, urceprefs, disabled });
divElemDiv.appendChild(createElem('input', {
type: 'number', id: `_num${numSetting}`, urceprefs, class: `URCE-daysInput urceSettingsNumberBox${urceDisabled}`, value: _settings[numSetting], title, min, max, step, disabled
}, [{ change: urceSettingsNumberBoxChange }]));
divElemRoot.appendChild(divElemDiv);
if (postText) {
divElemRoot.appendChild(createElem('div', {
class: `URCE-divDaysInline${urceDisabled}`, urceprefs, disabled, textContent: postText
}));
}
return divElemRoot;
},
buildCbandTextSetting = (setting, textSetting, urceprefs, postText) => {
let disabled = false,
urceDisabled = '';
if ((urceprefs === 'filtering') && !_settings.enableUrceUrFiltering) {
disabled = true;
urceDisabled = ' urceDisabled';
}
const translationName = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}`),
title = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}Title`),
divElemRooot = createElem('div', { style: 'display:inline-flex;' });
divElemRooot.appendChild(createElem('input', {
type: 'checkbox', id: `_cb${setting}`, urceprefs, class: `urceSettingsCheckbox${urceDisabled}`, title, disabled, checked: (_settings[setting] === true)
}, [{ change: urceSettingsCheckboxChange }]));
divElemRooot.appendChild(createElem('label', {
for: `_cb${setting}`, urceprefs, class: `URCE-label${urceDisabled}`, title, disabled, textContent: `${translationName}: `
}));
const divElemDiv = createElem('div', { class: `URCE-divDaysInline${urceDisabled}`, urceprefs, disabled });
divElemDiv.appendChild(createElem('input', {
type: 'text', id: `_text${textSetting}`, urceprefs, class: `URCE-textInput2 urceSettingsTextBox${urceDisabled}`, value: _settings[textSetting], title, disabled
}, [{ change: urceSettingsTextBoxChange }]));
divElemRooot.appendChild(divElemDiv);
if (postText) {
divElemRooot.appendChild(createElem('div', {
class: `URCE-divDaysInline${urceDisabled}`, urceprefs, disabled, textContent: postText
}));
}
return divElemRooot;
};
const contentHeaderDiv = createElem('div', { id: 'expandCollapseAllSettings', class: `URCE-expandCollapseAll${urStyle}` });
contentHeaderDiv.appendChild(createElem('div', {
id: 'URCE-expandAllSettings', class: 'URCE-expandCollapseAllItem', textContent: I18n.t('urce.common.ExpandAll')
}, [{ click: expandCollapseAll }]));
contentHeaderDiv.appendChild(createElem('div', { style: 'display:inline;', textContent: ' : ' }));
contentHeaderDiv.appendChild(createElem('div', {
id: 'URCE-collapseAllSettings', class: 'URCE-expandCollapseAllItem', textContent: I18n.t('urce.common.CollapseAll')
}, [{ click: expandCollapseAll }]));
docFrags.appendChild(contentHeaderDiv);
// Comment List
let { fieldsetElem, contentDiv } = buildFieldsetSection('commentList', I18n.t('urce.common.CommentList')),
contentDivDiv = createElem('div', { textContent: `${I18n.t('urce.prefs.DefaultList')}: ` }),
selectElem = createElem('select', { id: '_selCommentList', title: I18n.t('urce.prefs.DefaultListTitle'), urceprefs: 'commentList' }, [{ change: commentListSelectionChange }]);
_commentLists.forEach((cList) => {
if (cList.status !== 'disabled')
selectElem.appendChild(createElem('option', { value: cList.idx, textContent: cList.name, selected: (cList.idx === _settings.commentList) }));
});
contentDivDiv.appendChild(selectElem);
let contentDivDivDiv = createElem('div', { id: 'restrictionsEnforcedWarning-Settings', style: `float:right;padding-right:4px;color:red;font-size:20px;${(_restrictionsEnforcedTitle ? '' : 'display:none;')}` });
contentDivDivDiv.appendChild(createElem('i', { class: 'w-icon w-icon-alert-danger', 'aria-hidden': true, title: _restrictionsEnforcedTitle || '' }));
contentDivDiv.appendChild(contentDivDivDiv);
contentDiv.appendChild(contentDivDiv);
contentDivDiv = createElem('div', { textContent: `${I18n.t('urce.prefs.CustomSsId')}: ` });
contentDivDiv.appendChild(createElem('input', {
type: 'text',
id: '_textcustomSsId',
class: 'URCE-textInput urceSettingsTextBox',
urceprefs: 'commentList',
value: _settings.customSsId,
title: I18n.t('urce.prefs.CustomSsIdTitle'),
style: 'width:100px;margin-left:5px;'
}, [{ change: urceSettingsTextBoxChange }]));
contentDiv.appendChild(contentDivDiv);
contentDivDiv = createElem('div', { textContent: `${I18n.t('urce.common.Style')}: ` });
selectElem = createElem('select', {
id: '_selCommentListStyle', title: I18n.t('urce.prefs.CommentListStyleTitle'), urceprefs: 'commentList'
}, [{ change: commentListStyleSelectionChange }]);
selectElem.appendChild(createElem('option', { value: 'default', textContent: I18n.t('urce.prefs.StyleDefault'), selected: (_settings.commentListStyle === 'default') }));
selectElem.appendChild(createElem('option', { value: 'urStyle', textContent: I18n.t('urce.prefs.StyleUrStyle'), selected: (_settings.commentListStyle === 'urStyle') }));
contentDivDiv.appendChild(selectElem);
contentDiv.appendChild(contentDivDiv);
contentDiv.appendChild(createElem('input', {
type: 'checkbox', id: '_cbautoSwitchCommentList', class: 'urceSettingsCheckbox', urceprefs: 'commentList', title: I18n.t('urce.prefs.AutoSwitchCommentListTitle'), checked: _settings.autoSwitchCommentList
}, [{ change: urceSettingsCheckboxChange }]));
contentDiv.appendChild(createElem('label', {
for: '_cbautoSwitchCommentList', urceprefs: 'commentList', title: I18n.t('urce.prefs.AutoSwitchCommentListTitle'), class: 'URCE-label', textContent: I18n.t('urce.prefs.AutoSwitchCommentList')
}));
contentDivDiv = createElem('div', { id: 'urceSpreadsheetLinks' });
contentDivDivDiv = createElem('div', { id: 'urceSpreadsheetLinkDiv', class: 'URCE-spreadsheetLink' });
contentDivDivDiv.appendChild(createElem('a', {
id: 'urce-spreadsheet-link', class: 'URCE-Controls URCE-spreadsheetLink', title: I18n.t('urce.prefs.SpreadsheetLinkTitle'), href: 'http://bit.ly/urc-e_ss', textContent: 'URC-E Master Spreadsheet'
}, [{ click: spreadsheetLinkClick }]));
contentDivDiv.appendChild(contentDivDivDiv);
if (_settings.customSsId?.length > 0) {
contentDivDivDiv = createElem('div', { id: 'urceCustomSpreadsheetLinkDiv', class: 'URCE-spreadsheetLink' });
contentDivDivDiv.appendChild(createElem('a', {
id: 'urce-customSpreadsheet-link',
class: 'URCE-Controls URCE-spreadsheetLink',
title: I18n.t('urce.prefs.CustomSpreadsheetLinkTitle'),
href: `https://docs.google.com/spreadsheets/d/${_settings.customSsId}/edit`,
textContent: I18n.t('urce.prefs.CustomSpreadsheetLink')
}, [{ click: spreadsheetLinkClick }]));
contentDivDiv.appendChild(contentDivDivDiv);
}
contentDiv.appendChild(contentDivDiv);
docFrags.appendChild(fieldsetElem);
// Per Comment List Settings Settings
({ fieldsetElem } = buildFieldsetSection('perCommentListSettings', I18n.t('urce.prefs.PerCommentListSettings'), { id: 'URCE-divPerCommentListSettings' }));
docFrags.appendChild(fieldsetElem);
// URC-E Master Settings
({ fieldsetElem, contentDiv } = buildFieldsetSection('urc-e-prefs', I18n.t('urce.prefs.UrcePrefs')));
['autoCenterOnUr', 'autoClickOpenSolvedNi', 'autoCloseUrPanel', 'autoSaveAfterSolvedOrNiComment', 'autoSendReminders', 'autoSetNewUrComment', 'autoSetNewUrCommentSlur',
'autoSetNewUrCommentWithDescription', 'autoSetReminderUrComment', 'placeCursorAtStart', 'autoSwitchToUrCommentsTab', 'autoZoomInOnNewUr', 'autoZoomOutAfterClosePanel',
'autoZoomOutAfterComment', 'disableDoneNextButtons', 'replaceNextWithDoneButton', 'doubleClickLinkNiComments', 'doubleClickLinkOpenComments', 'doubleClickLinkSolvedComments',
'hideZoomOutLinks', 'enableUrOverflowHandling', 'enableAutoRefresh', 'expandMoreInfo', 'expandUserPreferences', 'expandShortcuts', 'autoScrollComments', 'reverseCommentSort',
'usePrepositionForSegmentNamesShortcut'
].forEach((setting) => { contentDiv.appendChild(buildStandardCbSetting(setting, 'urce')); });
contentDivDiv = createElem('div', { class: 'URCE-controls URCE-textFirst' });
contentDivDiv.appendChild(buildTextFirstTextSetting('tagEmail', 'commentList'));
contentDivDiv.appendChild(buildTextFirstNumSetting('reminderDays', 'urce', '0', '9999', '1', I18n.t('common.time.days', { days: 0 }).replaceAll('0 ', '')));
contentDivDiv.appendChild(buildTextFirstNumSetting('closeDays', 'urce', '1', '9999', '1', I18n.t('common.time.days', { days: 0 }).replaceAll('0 ', '')));
contentDivDivDiv = createElem('div', {
class: 'URCE-label', urceprefs: 'commentList', title: I18n.t('urce.prefs.CustomTaglineTitle'), textContent: `${I18n.t('urce.prefs.CustomTagline')}: `
});
contentDivDivDiv.appendChild(createElem('textarea', {
id: '_textcustomTagline',
class: 'URCE-textInput urceSettingsTextBox',
style: 'width:230px;height:50px;resize:none;',
urceprefs: 'commentList',
title: I18n.t('urce.prefs.CustomTaglineTitle'),
textContent: _settings.customTagline
}, [{ change: urceSettingsTextBoxChange }]));
contentDivDiv.appendChild(contentDivDivDiv);
contentDiv.appendChild(contentDivDiv);
docFrags.appendChild(fieldsetElem);
// UR Marker Settings
({ fieldsetElem, contentDiv } = buildFieldsetSection('ur-marker-prefs', I18n.t('urce.prefs.UrMarkerPrefs')));
contentDiv.appendChild(buildStandardCbSetting('enableUrPillCounts', 'markerMaster'));
contentDiv.appendChild(buildStandardCbSetting('disableUrMarkerPopup', 'marker-nodisable'));
contentDivDiv = createElem('div', { class: 'URCE-textFirst', urceprefs: 'marker-nodisable' });
contentDivDiv.appendChild(buildTextFirstNumSetting('urMarkerPopupDelay', 'marker-nodisable', '1', '99', '1', ' * 100ms'));
contentDivDiv.appendChild(buildTextFirstNumSetting('urMarkerPopupTimeout', 'marker-nodisable', '1', '99', '1', I18n.t('datetime.distance_in_words.x_seconds.other', { count: 30 }).replaceAll('30', '')));
contentDiv.appendChild(contentDivDiv);
['doNotShowTagNameOnPill', 'replaceTagNameWithEditorName'].forEach((setting) => { contentDiv.appendChild(buildStandardCbSetting(setting, 'marker')); });
contentDiv.appendChild(buildStandardCbSetting('unstackMarkers', 'marker-nodisable'));
contentDivDiv = createElem('div', { class: 'URCE-textFirst', urceprefs: 'marker-nodisable-stack' });
contentDivDiv.appendChild(buildTextFirstNumSetting('unstackSensitivity', 'marker-nodisable-unstack', '1', '99', '1', undefined));
contentDivDiv.appendChild(buildTextFirstNumSetting('unstackDisableAboveZoom', 'marker-nodisable-unstack', '12', '22', '1', undefined));
contentDiv.appendChild(contentDivDiv);
// -- Custom markers
contentDivDiv = createElem('div');
contentDivDiv.appendChild(createElem('div', { class: 'URCE-subHeading', style: 'font-weight:600;', textContent: `${I18n.t('urce.prefs.UseCustomMarkersFor')}:` }));
['Bog', 'Closures', 'Construction', 'Difficult', 'Events', 'Notes', 'Roadworks', 'Wslm', 'NativeSl', 'Custom'].forEach((setting) => {
let translationName,
translationTitle;
if ((setting === 'Closures') || (setting === 'Events') || (setting === 'Notes')) {
translationName = I18n.t(`urce.tags.${setting.slice(0, -1)}`);
translationTitle = I18n.t(`urce.prefs.${setting.slice(0, -1)}Title`);
}
else if (setting === 'NativeSl') {
translationName = I18n.t('urce.prefs.NativeSpeedLimits');
translationTitle = I18n.t('urce.prefs.NativeSpeedLimitsTitle');
}
else if (setting === 'Custom') {
translationName = `${I18n.t('urce.common.Custom')}: `;
translationTitle = I18n.t('urce.prefs.CustomTitle');
}
else {
translationName = I18n.t(`urce.tags.${setting}`);
translationTitle = I18n.t(`urce.prefs.${setting}Title`);
}
contentDivDiv.appendChild(buildTagCbSetting(`customMarkers${setting}`, 'marker-nodisable', translationName, translationTitle));
});
contentDivDivDiv = createElem('div', { class: 'URCE-divDaysInline' });
contentDivDivDiv.appendChild(createElem('input', {
type: 'text', id: '_textcustomMarkersCustomText', class: 'urceSettingsTextBox', style: 'width:100px;height:20px;', value: _settings.customMarkersCustomText, title: I18n.t('urce.prefs.CustomTitle')
}, [{ change: urceSettingsTextBoxChange }]));
contentDivDiv.appendChild(contentDivDivDiv);
contentDiv.appendChild(contentDivDiv);
docFrags.appendChild(fieldsetElem);
// UR Filtering Settings
({ fieldsetElem, contentDiv } = buildFieldsetSection('ur-filtering-prefs', I18n.t('urce.prefs.UrFilteringPrefs')));
contentDiv.appendChild(buildStandardCbSetting('enableUrceUrFiltering', 'filteringMaster'));
['invertFilters', 'hideOutsideEditableArea', 'doNotFilterTaggedUrs', 'doNotHideSelectedUr', 'disableFilteringAboveZoom', 'disableFilteringBelowZoom'].forEach((setting) => {
contentDiv.appendChild(buildStandardCbSetting(setting, 'filtering'));
});
// -- Lifecycle
contentDiv.appendChild(createElem('div', {
class: 'URCE-subHeading', style: 'font-weight:600;', textContent: `${I18n.t(`urce.prefs.LifeCycleStatus${((_settings.invertFilters) ? 'Inverted' : '')}`)}:`
}));
['hideWaiting', 'hideUrsCloseNeeded', 'hideUrsReminderNeeded'].forEach((setting) => { contentDiv.appendChild(buildStandardCbSetting(setting, 'filtering')); });
// -- Hide by status
contentDiv.appendChild(createElem('div', {
class: 'URCE-subHeading', style: 'font-weight:600;', textContent: `${I18n.t(`urce.prefs.HideByStatus${((_settings.invertFilters) ? 'Inverted' : '')}`)}:`
}));
['hideByStatusOpen', 'hideByStatusClosed', 'hideByStatusNotIdentified', 'hideByStatusSolved', 'hideByStatusClosedBy'].forEach((setting) => {
contentDiv.appendChild(buildStandardCbSetting(setting, 'filtering'));
});
// -- Hide by type
contentDiv.appendChild(createElem('div', {
class: 'URCE-subHeading', style: 'font-weight:600;', textContent: `${I18n.t(`urce.prefs.HideByType${((_settings.invertFilters) ? 'Inverted' : '')}`)}:`
}));
['hideByTypeBlockedRoad', 'hideByTypeGeneralError', 'hideByTypeIncorrectAddress', 'hideByTypeIncorrectJunction', 'hideByTypeIncorrectRoute', 'hideByTypeIncorrectStreetPrefixOrSuffix',
'hideByTypeIncorrectTurn', 'hideByTypeMissingBridgeOverpass', 'hideByTypeMissingExit', 'hideByTypeMissingLandmark', 'hideByTypeMissingOrInvalidSpeedLimit', 'hideByTypeMissingRoad',
'hideByTypeMissingRoundabout', 'hideByTypeMissingStreetName', 'hideByTypeTurnNotAllowed', 'hideByTypeUndefined', 'hideByTypeWrongDrivingDirection'
].forEach((setting) => { contentDiv.appendChild(buildStandardCbSetting(setting, 'filtering')); });
// -- Hide by tagged
contentDiv.appendChild(createElem('div', {
class: 'URCE-subHeading', style: 'font-weight:600;', textContent: `${I18n.t(`urce.prefs.HideByTagged${((_settings.invertFilters) ? 'Inverted' : '')}`)}:`
}));
['Bog', 'Closure', 'Construction', 'Difficult', 'Event', 'Note', 'Roadworks', 'Wslm'].forEach((setting) => {
contentDiv.appendChild(buildTagCbSetting(`hideByTagged${setting}`, 'filtering', I18n.t(`urce.tags.${setting}`), I18n.t(`urce.prefs.HideByTagged${setting}Title`), true));
});
// -- Hide by age of submission
contentDiv.appendChild(createElem('div', {
class: 'URCE-subHeading', style: 'font-weight:600;', textContent: `${I18n.t(`urce.prefs.HideByAgeOfSubmission${((_settings.invertFilters) ? 'Inverted' : '')}`)}:`
}));
contentDiv.appendChild(
buildCbandNumSetting('hideByAgeOfSubmissionLessThan', 'hideByAgeOfSubmissionLessThanDaysOld', 'filtering', '0', '9999', '1', I18n.t('common.time.days', { days: 0 }).replaceAll('0 ', ''))
);
contentDiv.appendChild(
buildCbandNumSetting('hideByAgeOfSubmissionMoreThan', 'hideByAgeOfSubmissionMoreThanDaysOld', 'filtering', '0', '9999', '1', I18n.t('common.time.days', { days: 0 }).replaceAll('0 ', ''))
);
// -- Hide by description, comments, following
contentDiv.appendChild(createElem('div', {
class: 'URCE-subHeading', style: 'font-weight:600;', textContent: `${I18n.t(`urce.prefs.DescriptionCommentsFollowing${((_settings.invertFilters) ? 'Inverted' : '')}`)}:`
}));
contentDivDiv = createElem('div', { class: 'URCE-textFirst', urceprefs: 'filtering' });
contentDivDiv.appendChild(buildTextFirstDoubleCbSetting('hideFollowing', 'hideNotFollowing', I18n.t('urce.common.Following'), 'filtering'));
contentDivDiv.appendChild(buildTextFirstDoubleCbSetting('hideWithDescription', 'hideWithoutDescription', I18n.t('objects.venue.fields.description'), 'filtering'));
contentDivDiv.appendChild(buildTextFirstDoubleCbSetting('hideWithCommentsFromMe', 'hideWithoutCommentsFromMe', I18n.t('urce.prefs.HideCommentsFromMe'), 'filtering'));
contentDivDiv.appendChild(buildTextFirstDoubleCbSetting('hideFirstCommentByMe', 'hideFirstCommentNotByMe', I18n.t('urce.prefs.HideFirstCommentByMe'), 'filtering'));
contentDivDiv.appendChild(buildTextFirstDoubleCbSetting('hideLastCommentByMe', 'hideLastCommentNotByMe', I18n.t('urce.prefs.HideLastCommentByMe'), 'filtering'));
contentDivDiv.appendChild(buildTextFirstDoubleCbSetting('hideLastCommentByReporter', 'hideLastCommentNotByReporter', I18n.t('urce.prefs.HideLastCommentByReporter'), 'filtering'));
contentDiv.appendChild(contentDivDiv);
contentDiv.appendChild(buildCbandNumSetting('hideByCommentCountLessThan', 'hideByCommentCountLessThanNumber', 'filtering', '0', '9999', '1', I18n.t('urce.tabs.Comments')));
contentDiv.appendChild(buildCbandNumSetting('hideByCommentCountMoreThan', 'hideByCommentCountMoreThanNumber', 'filtering', '0', '9999', '1', I18n.t('urce.tabs.Comments')));
contentDiv.appendChild(
buildCbandNumSetting('hideByAgeOfFirstCommentLessThan', 'hideByAgeOfFirstCommentLessThanDaysOld', 'filtering', '0', '9999', '1', I18n.t('common.time.days', { days: 0 }).replaceAll('0 ', ''))
);
contentDiv.appendChild(
buildCbandNumSetting('hideByAgeOfFirstCommentMoreThan', 'hideByAgeOfFirstCommentMoreThanDaysOld', 'filtering', '0', '9999', '1', I18n.t('common.time.days', { days: 0 }).replaceAll('0 ', ''))
);
contentDiv.appendChild(
buildCbandNumSetting('hideByAgeOfLastCommentLessThan', 'hideByAgeOfLastCommentLessThanDaysOld', 'filtering', '0', '9999', '1', I18n.t('common.time.days', { days: 0 }).replaceAll('0 ', ''))
);
contentDiv.appendChild(
buildCbandNumSetting('hideByAgeOfLastCommentMoreThan', 'hideByAgeOfLastCommentMoreThanDaysOld', 'filtering', '0', '9999', '1', I18n.t('common.time.days', { days: 0 }).replaceAll('0 ', ''))
);
contentDiv.appendChild(buildCbandTextSetting('hideByKeywordIncluding', 'hideByKeywordIncludingKeyword', 'filtering', undefined));
contentDiv.appendChild(buildCbandTextSetting('hideByKeywordNotIncluding', 'hideByKeywordNotIncludingKeyword', 'filtering', undefined));
contentDivDiv = createElem('div', { style: 'padding-left:15px;font-style:italic;' });
contentDivDiv.appendChild(buildStandardCbSetting('hideByKeywordCaseInsensitive', 'filtering'));
contentDiv.appendChild(contentDivDiv);
contentDiv.appendChild(buildCbandTextSetting('hideWithCommentBy', 'hideWithCommentByUsers', 'filtering', undefined));
contentDiv.appendChild(buildCbandTextSetting('hideWithoutCommentBy', 'hideWithoutCommentByUsers', 'filtering', undefined));
docFrags.appendChild(fieldsetElem);
document.getElementById('panel-urce-settings').appendChild(docFrags);
/**
* Disable settings for compatibility
* Enable Auto Refresh: Disabled 2023.03.29
* Due to W.controller.reloadData() causing issues with new Issue Tracker.
* Specifically if user had more than 500 URs loaded in tracker, it would reset them back to 500 due to reloadData
* wiping and reloading the model data. For now, this setting is disabled and the functionality has been replaced
* with the already working handleOverflow function (if the user has it enabled).
*/
document.querySelectorAll('#_cbenableAutoRefresh, label[for="_cbenableAutoRefresh"').forEach((el) => {
el.disabled = true;
el.classList.add('urceDisabled');
el.setAttribute('title', `${I18n.t('urce.prefs.DisabledUnusedSettingTitle')}\r\n\r\n${I18n.t('urce.prefs.EnableAutoRefreshTitle')}`);
});
}
async function initGui(firstCall = true, structureOnly = false) {
if (_initError && !structureOnly)
return Promise.resolve();
logDebug('Initializing GUI.');
injectCss();
if (!document.getElementById('urceDiv')) {
const disableUrMarkerPopups = () => {
hidePopup({ doubleClick: true });
document.getElementById('_cbdisableUrMarkerPopup').click();
},
divElemRoot = createElem('div', { id: 'urceDiv' }, [{ mouseleave: hidePopup }, { mouseenter: checkPopupTimeouts }, { dblclick: dispatchPopupDoubleClick }]);
divElemRoot.appendChild(createElem('span', { class: 'urceDivCloseButton', textContent: 'X' }, [{ click: hidePopup }]));
divElemRoot.appendChild(createElem('div', { class: 'urceDivContent' }));
divElemRoot.appendChild(createElem('span', { class: 'urceDivDisablePopups', textContent: I18n.t('urce.prefs.DisableUrMarkerPopup') }, [{ click: disableUrMarkerPopups }]));
document.body.appendChild(divElemRoot);
}
if (firstCall) {
const docFrags = document.createDocumentFragment(),
{ tabLabel, tabPane } = W.userscripts.registerSidebarTab('URC-E'),
urceUrFilteringToggle = (evt) => {
evt.stopPropagation();
document.getElementById('_cbenableUrceUrFiltering').click();
};
// Tab Label
docFrags.appendChild(createElem('img', { id: 'urceIcon', class: 'URCE-tabIcon', src: GM_info.script.icon }));
docFrags.appendChild(
createElem('span', { id: 'urceUrMarkerProcessingSpinner', class: 'fa fa-spinner URCE-spinner', title: I18n.t('urce.mouseOver.URMarkerProcessingInactive') })
);
docFrags.appendChild(
createElem('span', {
id: 'urceUrFilteringToggleBtn', class: 'w-icon w-icon-filter URCE-urFilteringToggleBtn', style: `color:${(_settings.enableUrceUrFiltering ? '#00bd00' : '#ccc')};`, title: I18n.t('urce.mouseOver.ToggleUrceURFiltering')
}, [{ click: urceUrFilteringToggle }])
);
tabLabel.appendChild(docFrags);
tabLabel.title = 'URC-E';
tabLabel.style.display = 'flex';
tabLabel.style.justifyContent = 'center';
// Tab Pane
docFrags.appendChild(createElem('span', { class: 'URCE-spanTitle', textContent: _SCRIPT_SHORT_NAME }));
docFrags.appendChild(createElem('span', { class: 'URCE-spanVersion', textContent: _SCRIPT_VERSION }));
let liElem = createElem('li', { class: 'active' });
liElem.appendChild(createElem('a', {
'data-toggle': 'tab', href: '#panel-urce-comments', 'aria-expanded': true, textContent: I18n.t('urce.tabs.Comments')
}));
const ulElem = createElem('ul', { class: 'nav nav-tabs' });
ulElem.appendChild(liElem);
liElem = createElem('li');
liElem.appendChild(createElem('a', {
'data-toggle': 'tab', href: '#panel-urce-settings', 'aria-expanded': true, textContent: I18n.t('urce.tabs.Settings')
}));
ulElem.appendChild(liElem);
liElem = createElem('li');
liElem.appendChild(createElem('a', {
'data-toggle': 'tab', href: '#panel-urce-tools', 'aria-expanded': true, textContent: I18n.t('urce.tabs.Tools')
}));
ulElem.appendChild(liElem);
let divElemRoot = createElem('div', { class: 'URCE-navTabs' });
divElemRoot.appendChild(ulElem);
docFrags.appendChild(divElemRoot);
divElemRoot = createElem('div', { class: 'tab-content URCE-divTabs', style: '--height-offset:0px;' });
divElemRoot.appendChild(createElem('div', { id: 'panel-urce-comments', class: 'tab-pane active' }));
divElemRoot.appendChild(createElem('div', { id: 'panel-urce-settings', class: 'tab-pane' }));
divElemRoot.appendChild(createElem('div', { id: 'panel-urce-tools', class: 'tab-pane' }));
docFrags.appendChild(divElemRoot);
tabPane.appendChild(docFrags);
Object.assign(tabPane.parentElement.style, { width: 'auto', padding: '0 15px' });
tabPane.id = 'sidepanel-urc-e';
await W.userscripts.waitForElementConnected(tabPane);
showScriptInfoAlert();
}
if (!structureOnly) {
initSettingsTab();
initCommentsTab();
initToolsTab();
}
return Promise.resolve();
}
function initCommentLists() {
return new Promise((resolve, reject) => {
logDebug('Initializing available comment lists.');
const postProcess = function (errorText) {
if (errorText && _STATIC_ONLY_USERS.includes(W.loginManager.user.getUsername())) {
_commentLists.push({
idx: 1, name: 'Custom', status: 'enabled', type: 'static', oldVarName: 'Custom', listOwner: 'Custom', gSheetRange: ''
});
_commentLists.push({
idx: 3, name: 'USA - SER', status: 'enabled', type: 'static', oldVarName: 'USA_Southeast', listOwner: 'itzwolf', gSheetRange: ''
});
resolve();
}
else if (errorText) {
reject(new Error(errorText));
}
else {
resolve();
}
};
GM_xmlhttpRequest({
url: `https://sheets.googleapis.com/v4/spreadsheets/${dec(_URCE_SPREADSHEET_ID)}/values/CommentLists!A3:G?key=${dec(_URCE_API_KEY)}`,
headers: { 'Content-Type': 'application/json', Referer: 'https://www.waze.com' },
method: 'GET',
onload(res) {
let errorText;
if (res.status < 400) {
const data = JSON.parse(res.responseText);
if (data.values?.length > 0) {
let ssFieldNames;
const EXPECTED_FIELD_NAMES = ['idx', 'name', 'status', 'type', 'oldVarName', 'Prefix', 'listOwner'],
checkFieldNames = function (fldName) { return this.includes(fldName); };
for (let entryIdx = 0, { length } = data.values; entryIdx < length; entryIdx++) {
if (entryIdx === 0) {
if (_SCRIPT_VERSION < data.values[entryIdx][0]) {
errorText = `updateRequired|${data.values[entryIdx][0]}`;
break;
}
if (data.values[entryIdx][0] < _MIN_VERSION_COMMENTLISTS) {
errorText = `spreadsheetUpdateRequired|${_MIN_VERSION_COMMENTLISTS}`;
break;
}
}
else if (entryIdx === 1) {
ssFieldNames = data.values[entryIdx].map((fldName) => fldName.trim());
if (ssFieldNames.length !== EXPECTED_FIELD_NAMES.length)
errorText = `Expected ${EXPECTED_FIELD_NAMES.length} columns in comment lists data. Spreadsheet returned ${ssFieldNames.length}.`;
else if (!EXPECTED_FIELD_NAMES.every(checkFieldNames.bind(ssFieldNames)))
errorText = `Script expected to see the following column names in the comment definition spreadsheet:\n${EXPECTED_FIELD_NAMES.join(', ')}\nHowever, the spreadsheet returned these:\n${ssFieldNames.join(', ')}`;
if (errorText)
break;
}
else {
const output = Object.create({});
for (let valIdx = 0, len = data.values[entryIdx].length; valIdx < len; valIdx++) {
if (ssFieldNames[valIdx] === 'idx')
output[ssFieldNames[valIdx]] = +data.values[entryIdx][valIdx];
else if (ssFieldNames[valIdx] === 'Prefix')
output.gSheetRange = `${data.values[entryIdx][valIdx]}_Output_(do_not_edit)!A1:A`;
else
output[ssFieldNames[valIdx]] = data.values[entryIdx][valIdx];
}
_commentLists.push(output);
}
}
if (!errorText)
_commentLists.sort(dynamicSort('name'));
}
else {
errorText = errorText || 'No lists available.';
}
}
else {
errorText = errorText || res;
}
postProcess(errorText);
},
onerror(res) {
postProcess(`xmlhttpRequest error: ${JSON.stringify(res)}`);
}
});
});
}
function initAutoSwitchArrays() {
return new Promise((resolve, reject) => {
logDebug('Initializing auto switch setup.');
const postProcess = function (errorText) {
if (!errorText || (errorText && _STATIC_ONLY_USERS.includes(W.loginManager.user.getUsername())))
resolve();
else
reject(new Error(errorText));
};
GM_xmlhttpRequest({
url: `https://sheets.googleapis.com/v4/spreadsheets/${dec(_URCE_SPREADSHEET_ID)}/values/CommentLists_AutoSwitch!A3:ZZ?majorDimension=COLUMNS&key=${dec(_URCE_API_KEY)}`,
headers: { 'Content-Type': 'application/json', Referer: 'https://www.waze.com' },
method: 'GET',
onload(res) {
let errorText;
if (res.status < 400) {
const data = JSON.parse(res.responseText);
if (data.values?.length > 0) {
for (let entryIdx = 0, { length } = data.values; entryIdx < length; entryIdx++) {
if (entryIdx === 0) {
if (_SCRIPT_VERSION < data.values[entryIdx][0]) {
errorText = `updateRequired|${data.values[entryIdx][0]}`;
break;
}
if (data.values[entryIdx][0] < _MIN_VERSION_AUTOSWITCH) {
errorText = `spreadsheetUpdateRequired|${_MIN_VERSION_AUTOSWITCH}`;
break;
}
}
if (data.values[entryIdx].length > 3) {
const commentListNum = +data.values[entryIdx][2];
for (let idx = 3, len = data.values[entryIdx].length; idx < len; idx++) {
const values = data.values[entryIdx][idx].split('.');
if ((values[0] === 'state') && !_autoSwitch[values[1]])
_autoSwitch[values[1]] = {};
if (values[0] === 'state')
_autoSwitch[values[1]][values[2]] = commentListNum;
if ((values[0] === 'country') && !_autoSwitch[values[1]])
_autoSwitch[values[1]] = { ALL: commentListNum };
}
}
}
}
else {
errorText = errorText || 'No autoswitch data available';
}
}
else {
errorText = errorText || res;
}
postProcess(errorText);
},
onerror(res) {
postProcess(`xmlhttpRequest error: ${JSON.stringify(res)}`);
}
});
});
}
function initRestrictions() {
return new Promise((resolve, reject) => {
logDebug('Initializing restrictions.');
const postProcess = function (errorText) {
if (!errorText || (errorText && _STATIC_ONLY_USERS.includes(W.loginManager.user.getUsername())))
resolve();
else
reject(new Error(errorText));
};
GM_xmlhttpRequest({
url: `https://sheets.googleapis.com/v4/spreadsheets/${dec(_URCE_SPREADSHEET_ID)}/values/Restrictions!A3:ZZ?majorDimension=COLUMNS&key=${dec(_URCE_API_KEY)}`,
headers: { 'Content-Type': 'application/json', Referer: 'https://www.waze.com' },
method: 'GET',
onload(res) {
let errorText;
if (res.status < 400) {
const data = JSON.parse(res.responseText);
if (data.values?.length > 0) {
for (let entryIdx = 0, { length } = data.values; entryIdx < length; entryIdx++) {
if (entryIdx === 0) {
if (_SCRIPT_VERSION < data.values[entryIdx][0]) {
errorText = `updateRequired|${data.values[entryIdx][0]}`;
break;
}
if (data.values[entryIdx][0] < _MIN_VERSION_RESTRICTIONS) {
errorText = `spreadsheetUpdateRequired|${_MIN_VERSION_RESTRICTIONS}`;
break;
}
}
if (data.values[entryIdx].length > 1) {
const restriction = data.values[entryIdx][1];
for (let idx = 2, len = data.values[entryIdx].length; idx < len; idx++) {
const values = data.values[entryIdx][idx].split('.');
if (!_restrictions[values[1]])
_restrictions[values[1]] = {};
if (values[0] === 'state') {
if (!_restrictions[values[1]][values[2]])
_restrictions[values[1]][values[2]] = {};
_restrictions[values[1]][values[2]][restriction] = ((restriction !== 'reminderDays') && (restriction !== 'closeDays'))
? (values[3] === 'true')
: +values[3];
}
if (values[0] === 'country') {
if (!_restrictions[values[1]][values[2]])
_restrictions[values[1]][values[2]] = {};
_restrictions[values[1]][values[2]][restriction] = ((restriction !== 'reminderDays') && (restriction !== 'closeDays'))
? (values[3] === 'true')
: +values[3];
}
}
}
}
}
else {
errorText = errorText || 'No restrictions data available';
}
}
else {
errorText = errorText || res;
}
postProcess(errorText);
},
onerror(res) {
postProcess(`xmlhttpRequest error: ${JSON.stringify(res)}`);
}
});
});
}
async function initFinish(urId, urPanelMissing) {
checkTimeout({ timeout: 'initUrIdInUrl' });
if (_initUrIdInUrlObserver?.isObserving) {
_initUrIdInUrlObserver.isObserving = false;
_initUrIdInUrlObserver.disconnect();
}
await initBackgroundTasks('enable', 'initFinish');
if (W.model.mapUpdateRequests.getObjectArray().length > _markerCountOnInit)
await handleUrLayer('init_end', undefined, getMapUrsObjArr());
_markerCountOnInit = -1;
maskBoxes(undefined, true, 'init', (urId > 0));
if (urPanelMissing && W.model.mapUpdateRequests.getObjectById(urId)) {
logDebug(`UR ${urId} marker in URL. UR panel did not appear after 15 seconds, attempting to activate marker.`);
openUrPanel(urId, false);
}
else if (urPanelMissing) {
log(`UR ${urId} marker in URL. UR panel did not appear after 15 seconds. Marker not found.`);
}
else {
logDebug(`UR ${urId} marker in URL. Re-opening.`);
openUrPanel(urId, true);
}
}
function initCheckForUrPanel(urId, tries) {
checkTimeout({ timeout: 'initUrIdInUrl' });
if (tries < 150) {
if (document.getElementById('panel-container')?.children.length === 0) {
if (document.querySelectorAll(`.map-problem.user-generated[data-id="${urId}"]`)?.length > 0)
openUrPanel(urId, true);
else
_timeouts.initUrIdInUrl = window.setTimeout(initCheckForUrPanel, 100, urId, ++tries);
}
}
else {
logWarning(`UR ${urId} found in URL, but UR panel failed to open after 15 seconds.`);
initFinish(urId, true);
}
}
function checkUrceVersion() {
if (_IS_ALPHA_VERSION)
return;
let updateMonitor;
try {
updateMonitor = new WazeWrap.Alerts.ScriptUpdateMonitor(_SCRIPT_LONG_NAME, _SCRIPT_VERSION, (_IS_BETA_VERSION ? dec(_BETA_DL_URL) : _PROD_DL_URL), GM_xmlhttpRequest);
updateMonitor.start();
}
catch (err) {
logError('Upgrade version check:', err);
}
}
async function onWazeWrapReady() {
log('Initializing.');
checkUrceVersion();
_wmeUserId = W.loginManager.user.getID();
await loadSettingsFromStorage(false, false);
const urIdInUrl = +window.location.search.split('mapUpdateRequest=')[1],
afterInitGuiError = (divElem) => document.getElementById('panel-urce-comments')?.replaceChildren(divElem),
afterInitGui = () => {
if (_initError)
return;
const afterBuildCommentList = () => {
window.addEventListener('beforeunload', saveSettingsToStorage, false);
log(`Fully initialized in ${Math.round(performance.now() - _LOAD_BEGIN_TIME)} ms.`);
if (!urIdInUrl || !(urIdInUrl > 0)) {
initBackgroundTasks('enable', 'init').then(() => {
if (W.model.mapUpdateRequests.getObjectArray().length > _markerCountOnInit)
handleUrLayer('init_end', undefined, getMapUrsObjArr());
_markerCountOnInit = -1;
if (W.model.mapUpdateRequests.getObjectArray().length > 499)
handleUrOverflow({ changed: { loadingIssueTrackerMapData: false } });
maskBoxes(undefined, true, 'init', (urIdInUrl > 0));
});
}
else if (document.getElementById('panel-container')?.children.length === 0) {
logDebug(`urId ${urIdInUrl} found in URL, but the UR panel has not shown up yet. Waiting up to 15 seconds.`);
_initUrIdInUrlObserver = new MutationObserver((mutations) => {
mutations.forEach((mutation) => {
if ((mutation.type === 'attributes')
&& mutation.target.parentNode.matches('#panel-container')
&& mutation.target.classList.contains('show')
&& !mutation.oldValue.includes('show')
)
initFinish(urIdInUrl, false);
});
});
_initUrIdInUrlObserver.observe(document.getElementById('panel-container'), {
childList: true, attributes: true, attributeOldValue: true, characterData: true, characterDataOldValue: true, subtree: true
});
_initUrIdInUrlObserver.isObserving = true;
_timeouts.initUrIdInUrl = window.setTimeout(initCheckForUrPanel, 100, urIdInUrl, 1);
}
else {
initFinish(urIdInUrl, false);
}
doSpinner('init', false);
},
afterCheckRestrictions = () => {
buildCommentList(undefined, 'init', false)
.catch((error) => { handleError(error); })
.finally(afterBuildCommentList);
};
doSpinner('init', true);
const docFrags = document.createDocumentFragment();
docFrags.appendChild(createElem('div', { textContent: I18n.t('urce.prompts.WaitingOnInit') }));
docFrags.appendChild(createElem('br'));
docFrags.appendChild(createElem('div', { textContent: I18n.t('urce.common.PleaseWait') }));
maskBoxes(docFrags, false, 'init', (urIdInUrl > 0));
checkRestrictions([{ type: 'init' }]).then(afterCheckRestrictions);
};
Promise.all([loadTranslations(), initCommentLists(), initAutoSwitchArrays(), initRestrictions()]).catch(async (error) => {
const areTranslationsReady = () => new Promise((resolve) => {
(function retry(tries) {
checkTimeout({ timeout: 'areTranslationsReady' });
if (tries > 100)
resolve();
else if (!I18n.translations[I18n.currentLocale()].urce)
_timeouts.areTranslationsReady = window.setTimeout(retry, 100, ++tries);
else
resolve();
}(1));
});
_initError = true;
error.staticList = false;
error.phase = 'init';
error.maskUrPanel = (urIdInUrl > 0);
error.commentList = false;
await areTranslationsReady();
const { divElemRoot } = handleError(error);
await initGui(true, true);
afterInitGuiError(divElemRoot);
})
.then(initGui)
.then(afterInitGui);
}
function onWmeReady(tries = 1) {
if (typeof tries === 'object')
tries = 1;
checkTimeout({ timeout: 'onWmeReady' });
if (WazeWrap?.Ready) {
logDebug('WazeWrap is ready. Proceeding with initialization.');
onWazeWrapReady();
}
else if (tries < 1000) {
logDebug(`WazeWrap is not in Ready state. Retrying ${tries} of 1000.`);
_timeouts.onWmeReady = window.setTimeout(onWmeReady, 200, ++tries);
}
else {
logError('onWmeReady timed out waiting for WazeWrap Ready state.');
}
}
function onWmeInitialized() {
if (W.userscripts?.state?.isReady) {
logDebug('W is ready and already in "wme-ready" state. Proceeding with initialization.');
onWmeReady(1);
}
else {
logDebug('W is ready, but state is not "wme-ready". Adding event listener.');
document.addEventListener('wme-ready', onWmeReady, { once: true });
}
}
function bootstrap() {
if (!W) {
logDebug('W is not available. Adding event listener.');
document.addEventListener('wme-initialized', onWmeInitialized, { once: true });
}
else {
onWmeInitialized();
}
}
bootstrap();
function loadTranslations() {
return new Promise((resolve, reject) => {
logDebug('Loading translations.');
const postProcess = function (errorText) {
if (errorText && !_STATIC_ONLY_USERS.includes(W.loginManager.user.getUsername()))
reject(new Error(errorText));
else
resolve();
};
/* FIX FOR WME MISSING DAY NAMES 2021.07.13 - snhroc */
if (!I18n.translations[I18n.locale].date.day_names) {
I18n.translations[I18n.locale].date.day_names = [];
Object.entries(I18n.translations[I18n.locale].date).forEach(([k, v]) => {
if (k.indexOf('day_names_') === 0)
I18n.translations[I18n.locale].date.day_names.push(v);
});
}
/* END FIX */
GM_xmlhttpRequest({
url: `https://sheets.googleapis.com/v4/spreadsheets/${dec(_URCE_SPREADSHEET_ID)}/values/Script_Translations!A3:AA?key=${dec(_URCE_API_KEY)}`,
headers: { 'Content-Type': 'application/json', Referer: 'https://www.waze.com' },
method: 'GET',
onload(res) {
let errorText;
if (res.status < 400) {
const data = JSON.parse(res.responseText),
translationLocales = [];
let translations = {};
if (data.values?.length > 0) {
for (let entryIdx = 0, { length } = data.values; entryIdx < length; entryIdx++) {
if (entryIdx === 0) {
if (_SCRIPT_VERSION < data.values[entryIdx][0]) {
errorText = `updateRequired|${data.values[entryIdx][0]}`;
break;
}
if (data.values[entryIdx][0] < _MIN_VERSION_TRANSLATIONS) {
errorText = `spreadsheetUpdateRequired|${_MIN_VERSION_TRANSLATIONS}`;
break;
}
}
else if (entryIdx === 1) {
for (let idx = 1, len = data.values[entryIdx].length; idx < len; idx++) {
translationLocales.push(data.values[entryIdx][idx].trim());
translations[data.values[entryIdx][idx].trim()] = {};
}
}
else {
let translationDefinition = [];
for (let valIdx = 0, len = data.values[entryIdx].length; valIdx < len; valIdx++) {
if (valIdx === 0) {
translationDefinition = data.values[entryIdx][valIdx].split('.');
}
else {
const translationLocale = translationLocales[(valIdx - 1)],
translationDef0 = translationDefinition[0],
translationDef1 = translationDefinition[1];
if (typeof translations[translationLocale][translationDef0] === 'undefined')
translations[translationLocale][translationDef0] = {};
translations[translationLocale][translationDef0][translationDef1] = data.values[entryIdx][valIdx]
.replaceAll('$SCRIPT_AUTHOR$', _SCRIPT_AUTHOR).replace(/\\[r|n]+/gm, '\n');
}
}
}
}
}
else {
errorText = errorText || 'No translations available.';
}
translations = (!errorText && (data?.values.length > 0)) ? translations : {
en: {
commentsTab: {
ZoomOutLink1: 'Zoom out 12 & close UR',
ZoomOutLink1Title: 'Zooms all the way out and closes the UR panel.',
ZoomOutLink2: 'Zoom out 14 & close UR',
ZoomOutLink2Title: 'Zooms out to level 14 and closes the UR panel.',
ZoomOutLink3: 'Zoom out 15 & close UR',
ZoomOutLink3Title: 'Zooms out to level 15 and closes the UR panel.'
},
common: {
All: 'All',
AutoSwitched: 'auto switched',
Backup: 'Backup',
ClosedBy: 'Closed by',
CollapseAll: 'Collapse all',
CommentList: 'Comment List',
CurrentCommentListTitle: 'You can change the currently loaded comment list using this drop down.\nChanging this drop down will not be saved as a setting and '
+ 'will not change your default list (located on the settings tab).\nThis is only to allow you to quickly switch between lists.',
Custom: 'Custom',
Description: 'Description',
Disabled: 'Disabled',
DoubleClickTitle: 'Double click here to send this comment',
Enabled: 'Enabled',
ErrorGeneric: 'An error has occurred within URC-E. Please contact a WazeDev team member.',
ErrorHeader: 'URC-E Error',
ExpandAll: 'Expand all',
Failed: 'Failed',
Finish: 'Finish',
Following: 'Following',
In: 'in',
LessThan: 'Less than',
Link: 'Link',
List: 'List',
ListOwner: 'List owner',
Loading: 'Loading',
MoreThan: 'More than',
Near: 'near',
NeedsChecked: 'Needs checked',
No: 'No',
None: 'None',
NotApplicable: 'Not applicable',
NotFollowing: 'Not following',
PleaseWait: 'Please wait',
Reset: 'Reset',
Restore: 'Restore',
Static: 'Static',
Style: 'Style',
Success: 'Success',
Total: 'Total',
Type: 'Type',
With: 'With',
Without: 'Without',
Yes: 'Yes'
},
mouseOver: {
CenterInCurrentTab: 'Center in current tab',
FirstComment: 'First comment',
LastComment: 'Last comment',
MarkedAs: 'Marked as',
NoDescription: 'No description',
NoneByMe: 'none by me',
OpenInNewLivemapTab: 'Open in new Livemap tab',
OpenInNewTab: 'Open in new tab',
ReporterHasCommented: 'Reporter has commented',
ToggleUrceURFiltering: 'Toggle URC-E UR filtering.',
URMarkerProcessingActive: 'URC-E is currently processing UR markers... Please wait.',
URMarkerProcessingInactive: 'URC-E is not currently processing any UR markers.',
ViaLivemap: 'via Livemap'
},
prefs: {
DisabledUnusedSettingTitle: 'This setting is currently disabled and unused due to WME compatibility issues.',
DefaultList: 'Default list',
DefaultListTitle: 'Select the custom list you would like to use. CommentTeam is the default. If you would like your comment list built into this script or have '
+ 'suggestions on the CommentTeam list, please contact a WazeDev team member.',
CustomSsId: 'Custom Google Spreadsheet ID',
CustomSsIdTitle: 'Enter the Google Spreadsheet ID for the Custom comment list you would like to load when you select \'Custom G Sheet\' comment list.',
CommentListStyleTitle: 'Select the style you would like the URC-E panel to be displayed in. This only affects the look of the tab, no functionality is changed.',
SpreadsheetLink: 'URC-E Master Spreadsheet',
SpreadsheetLinkTitle: 'Click here to view the URC-E master spreadsheet.',
CustomSpreadsheetLink: 'Custom Google Spreadsheet',
CustomSpreadsheetLinkTitle: 'Click here to view your custom google spreadsheet.',
StyleDefault: 'Default',
StyleUrStyle: 'UR Style',
TagEmail: 'Tag email',
TagEmailTitle: 'Some comment lists have specific comments that use a replacement tag.\nThe replacement tag is used to specify an email address to send correspondence '
+ 'to.\nIf you are setup to use one of these email addresses, please specify it here. If not, leave it blank.',
CustomTagline: 'Custom tagline',
CustomTaglineTitle: 'Some comments use a custom tagline variable.\nSpecify the text, if any, you would like to have put in place of this custom tagline variable in the comment.',
AutoSwitchCommentList: 'Automatically switch comment lists',
AutoSwitchCommentListTitle: 'Automatically switch to the comment list designated for the area the UR is in, if there is a list associated with the area.\nOpening a UR '
+ 'in an area that does not have a list associated will use the \'Comment List\' you have selected above.',
PerCommentListSettings: 'Per comment list settings',
SettingsFor: 'Settings for',
UseDefault: 'Use \'URC-E Master Settings\' setting',
UrcePrefs: 'URC-E Master Settings',
AutoCenterOnUr: 'Auto center on UR',
AutoCenterOnUrTitle: 'Auto center the map to the selected UR at the current map zoom level.',
AutoClickOpenSolvedNi: 'Auto click open, solved or not identified',
AutoClickOpenSolvedNiTitle: 'Suppress the message about recent pending questions to the reporter and then, depending on the choice set for that comment, automatically '
+ 'select Open, Solved or Not Identified.',
AutoCloseUrPanel: 'Auto close UR panel',
AutoCloseUrPanelTitle: 'Automatically close the UR panel after you click send on a comment.',
AutoSaveAfterSolvedOrNiComment: 'Auto save after solved or NI',
AutoSaveAfterSolvedOrNiCommentTitle: 'This will automatically click the save button after you close the UR panel in with you set the UR status to Solved or Not Identified.',
AutoSendReminders: 'Auto send reminders',
AutoSendRemindersTitle: 'Automatically send the reminder comment to the URs in the map window (as you pan around) you were the last to comment on and it has reached the '
+ 'days specified in \'Reminder Days\'. Restricted to editor rank 4+.',
AutoSendRemindersWarning: 'WARNING',
AutoSendRemindersWarningTitle: 'AUTOMATICALLY SEND REMINDERS at the reminder days setting.\nThis only happens when they are visible on your screen.\n\nNOTE: When using '
+ 'this feature you should not leave URs open unless you asked a question\nthat needs a response from the reporter, as this script will send reminders to '
+ 'all open URs\nafter \'Reminder days\'.',
AutoSendRemindersExceptTagged: 'Except tagged',
AutoSendRemindersExceptTaggedTitle: 'Enabling this setting will prevent the \'Auto send reminders\' setting (if enabled as well) from automatically sending a reminder '
+ 'comment to a tagged UR.',
AutoSetNewUrComment: 'Auto set new UR comment (w/o description)',
AutoSetNewUrCommentTitle: 'Automatically set the default UR comment for the UR type on new (do not already have comments) URs that do not have a description.',
AutoSetNewUrCommentSlur: 'Auto set new UR comment (SLURs)',
AutoSetNewUrCommentSlurTitle: 'Automatically set the default UR comment for new (do not already have comments) SLURs.',
AutoSetNewUrCommentWithDescription: 'Auto set new UR comment (with description)',
AutoSetNewUrCommentWithDescriptionTitle: 'Automatically set the default UR comment for the UR type on new (do not already have comments) URs that have a description.',
AutoSetReminderUrComment: 'Auto set reminder UR comment',
AutoSetReminderUrCommentTitle: 'Automatically set the UR reminder comment for URs that are older than the \'Reminder days\' setting and have only one comment.',
PlaceCursorAtStart: 'Place cursor at start',
PlaceCursorAtStartTitle: 'Place the cursor at the start of new or reminder comment when automatically set via auto set options above.',
AutoSwitchToUrCommentsTab: 'Auto switch to the URC-E tab',
AutoSwitchToUrCommentsTabTitle: 'Automatically switch to the URComments-Enhanced tab when opening a UR. When the UR panel is closed you will be switched back to your previous '
+ 'tab.',
AutoZoomInOnNewUr: 'Auto zoom in on new UR',
AutoZoomInOnNewUrTitle: 'Automatically zoom in when opening new (no comments) URs.',
AutoZoomOutAfterClosePanel: 'Auto zoom out after close panel',
AutoZoomOutAfterClosePanelTitle: 'Automatically zoom the map back to the previous zoom after closing the UR panel.',
AutoZoomOutAfterComment: 'Auto zoom out after comment',
AutoZoomOutAfterCommentTitle: 'Automatically zoom the map back to the previous zoom after clicking send on a UR comment.',
DisableDoneNextButtons: 'Disable done / next buttons',
DisableDoneNextButtonsTitle: 'Disable the done / next buttons at the bottom of the UR panel.',
ReplaceNextWithDoneButton: 'Replace Next UR button with Done button',
ReplaceNextWithDoneButtonTitle: 'Replace the Next Update Request button with a Done button.',
DoubleClickLinkNiComments: 'Double click link - NI comments',
DoubleClickLinkNiCommentsTitle: 'Add an image (extra link) to the \'not identified\' comments. When double clicked it will automatically set and send the UR comment of the '
+ 'one you double clicked, and then launch all of the other options that are enabled.',
DoubleClickLinkOpenComments: 'Double click link - Open comments',
DoubleClickLinkOpenCommentsTitle: 'Add an image (extra link) to the \'open\' comments. When double clicked it will automatically set and send the UR comment of the one you '
+ 'double clicked, and then launch all of the other options that are enabled.',
DoubleClickLinkSolvedComments: 'Double click link - Solved comments',
DoubleClickLinkSolvedCommentsTitle: 'Add an image (extra link) to the \'solved\' comments. When double clicked it will automatically set and send the UR comment of the one '
+ 'you double clicked, and then launch all of the other options that are enabled.',
HideZoomOutLinks: 'Hide zoom out links',
HideZoomOutLinksTitle: 'Hide the zoom out links on the comments tab.',
EnableUrOverflowHandling: 'Enable UR overflow handling',
EnableUrOverflowHandlingTitle: 'If this setting is enabled and there are more than 499 URs on the screen, URC-E will attempt to gather more URs and add them to the map, '
+ 'if they do not already exist.\nWME does not display more than 500 URs on a single screen on its own.',
EnableAutoRefresh: 'Enable auto refresh on zoom / pan',
EnableAutoRefreshTitle: 'Reloads the map data when zooming or panning to show URs that may have been missed due to WME\'s 500 UR limit. Will only reload if the zoom '
+ 'level is between 15 and 22, there are not pending edits, and there are more than 499 URs loaded.',
ExpandMoreInfo: 'Auto expand more info section',
ExpandMoreInfoTitle: 'Automatically expand the more info section of the UR Panel.',
ExpandUserPreferences: 'Auto expand user preferences section',
ExpandUserPreferencesTitle: 'Automatically expand the user preferences section of the UR Panel.',
ExpandShortcuts: 'Auto expand shortcuts section',
ExpandShortcutsTitle: 'Automatically expand the shortcuts section of the UR Panel.',
AutoScrollComments: 'Auto scroll comments in UR panel',
AutoScrollCommentsTitle: 'Automatically scroll the comments in the UR panel to the bottom if enabled, or top if disabled.',
ReverseCommentSort: 'Sort Comments in reverse order',
ReverseCommentSortTitle: 'Sort comments in the UR Panel in the reverse order.',
UsePrepositionForSegmentNamesShortcut: 'Use preposition phrase for segment names shortcut',
UsePrepositionForSegmentNamesShortcutTitle: 'Use preposition phrase (on/near) for segment names shortcut.',
UrMarkerPrefs: 'UR Marker Settings',
EnableUrPillCounts: 'Enable UR pill counts',
EnableUrPillCountsTitle: 'Enable or disable the pill with UR counts on the map marker.',
DisableUrMarkerPopup: 'Disable UR marker popups',
DisableUrMarkerPopupTitle: 'Do not show the UR popup tooltip when you mouse over a UR marker.',
UrMarkerPopupDelay: 'UR marker popup delay',
UrMarkerPopupDelayTitle: 'The number of milliseconds (* 100) to delay before the UR marker tooltip will be displayed.',
UrMarkerPopupTimeout: 'UR marker popup timeout',
UrMarkerPopupTimeoutTitle: 'Specify the number of seconds to leave the UR marker tooltip displayed, while hovering over the marker.\nLeaving the marker, unless to the '
+ 'tooltip itself, will cause the tooltip to close.\nEntering the tooltip will cancel the timer and leaving the tooltip will close the tooltip.\nDouble '
+ 'click to quickly close.',
DoNotShowTagNameOnPill: 'Don\'t show tag name on pill',
DoNotShowTagNameOnPillTitle: 'Do not show the tag name on the pill where there is a tag. Example: [NOTE]',
ReplaceTagNameWithEditorName: 'Replace tag name with editor name',
ReplaceTagNameWithEditorNameTitle: 'When a UR has the logged in editors name in the description or any of the comments of the UR (not the name Waze automatically adds '
+ 'when commenting), replace the tag type with the editors name.',
UnstackMarkers: 'Unstack markers',
UnstackMarkersTitle: 'Attempt to unstack markers by offsetting them. Similar to how URO+ unstacks markers.',
UnstackSensitivity: 'Unstack sensitivity',
UnstackSensitivityTitle: 'Specify the sensitivity for which markers are considered stacked.\nDefault: 15',
UnstackDisableAboveZoom: 'Unstack disable when zoom level <',
UnstackDisableAboveZoomTitle: 'When you zoom out wider than the specified zoom level, marker unstacking will be disabled.\nDefault: 15',
UseCustomMarkersFor: 'Use Custom Markers for',
BogTitle: 'Replace default UR marker with custom marker for the URs with \'[BOG]\' (boots on ground) / \'[BOTG]\' (boots on the ground) in the description or comments.',
ClosureTitle: 'Replace default UR marker with custom marker for the URs with \'[CLOSURE]\' in the description or comments.',
ConstructionTitle: 'Replace default UR marker with custom marker for the URs with \'[CONSTRUCTION]\' in the description or comments.',
DifficultTitle: 'Replace default UR marker with custom marker for the URs with \'[DIFFICULT]\' in the description or comments.',
EventTitle: 'Replace default UR marker with custom marker for the URs with \'[EVENT]\' in the description or comments.',
NoteTitle: 'Replace default UR marker with custom marker for the URs with \'[NOTE]\' in the description or comments.',
RoadworksTitle: 'Replace default UR marker with custom marker for the URs with \'[ROADWORKS]\' in the description or comments. Used in the UK.',
WslmTitle: 'Waze Speed Limit Marker',
NativeSpeedLimits: 'Native speed limits',
NativeSpeedLimitsTitle: 'Replace default UR marker with custom marker for the URs with \'speed limit\' type.',
CustomTitle: 'Replace default UR marker with custom marker for the URs with the text in the box to the right in the description or comments.',
UrFilteringPrefs: 'UR Filtering Settings',
EnableUrceUrFiltering: 'Enable URC-E UR filtering',
EnableUrceUrFilteringTitle: 'Enable or disable URComments-Enhanced built-in UR filtering.',
InvertFilters: 'Invert filters',
InvertFiltersTitle: 'This will invert the filters you select / do not select.\nExample: If a filter were to match a selected / enabled setting, it would normally '
+ 'be hidden.\nBut, if you enable \'invert filters\', the marker would be shown and all others that do not match will be hidden.',
HideOutsideEditableArea: 'Hide outside editable area',
HideOutsideEditableAreaTitle: 'Hide URs outside your editable area.',
DoNotFilterTaggedUrs: 'Do not filter tagged URs',
DoNotFilterTaggedUrsTitle: 'Do not filter URs that are tagged with a [] tag. Example: [NOTE]',
DoNotHideSelectedUr: 'Do not hide selected UR',
DoNotHideSelectedUrTitle: 'Do not hide a UR if it is currently being selected.',
DisableFilteringAboveZoomLevel: 'Disable filtering when zoom level <',
DisableFilteringAboveZoomLevelTitle: 'Disable UR filtering when zoomed out wider than the specified zoom level. Set to \'12\' to enable all filtering.',
DisableFilteringBelowZoomLevel: 'Disable filtering when zoom level >',
DisableFilteringBelowZoomLevelTitle: 'Disable UR filtering when zoomed in tighter than the specified zoom level. Set to \'22\' to enable all filtering.',
LifeCycleStatus: 'Hide by lifecycle status',
LifeCycleStatusInverted: 'Show by lifecycle status',
HideWaiting: 'Waiting',
HideWaitingTitle: 'Hide/show URs that do not currently need work.',
HideUrsCloseNeeded: 'Close needed',
HideUrsCloseNeededTitle: 'Hide/show URs that need closing.',
HideUrsReminderNeeded: 'Reminders needed',
HideUrsReminderNeededTitle: 'Hide/show URs where reminders are needed.',
HideByStatus: 'Hide by status',
HideByStatusInverted: 'Show by status',
HideByStatusOpenTitle: 'Hide/show all open URs.',
HideByStatusClosedTitle: 'Hide/show all closed (solved and not identified) URs.',
HideByStatusNotIdentifiedTitle: 'Hide/show all closed as not identified URs.',
HideByStatusSolvedTitle: 'Hide/show all closed as solved URs.',
HideByStatusClosedByTitle: 'Hide/show closed URs that were closed by any of the specified users.\nInput username(s) in text box. Separate multiple entries with a comma.',
HideByType: 'Hide by type',
HideByTypeInverted: 'Show by type',
HideByTypeBlockedRoadTitle: 'Hide/show all blocked road URs.',
HideByTypeGeneralErrorTitle: 'Hide/show all general error URs.',
HideByTypeIncorrectAddressTitle: 'Hide/show all incorrect address URs.',
HideByTypeIncorrectJunctionTitle: 'Hide/show all incorrect junction URs.',
HideByTypeIncorrectRouteTitle: 'Hide/show all incorrect route URs.',
HideByTypeIncorrectStreetPrefixOrSuffixTitle: 'Hide/show all incorrect street prefix or suffix URs.',
HideByTypeIncorrectTurnTitle: 'Hide/show all incorrect turn URs.',
HideByTypeMissingBridgeOverpassTitle: 'Hide/show all missing bridge overpass URs.',
HideByTypeMissingExitTitle: 'Hide/show all missing exit URs.',
HideByTypeMissingLandmarkTitle: 'Hide/show all missing landmark URs.',
HideByTypeMissingOrInvalidSpeedLimitTitle: 'Hide/show all missing or invalid speed limit URs.',
HideByTypeMissingRoadTitle: 'Hide/show all missing road URs.',
HideByTypeMissingRoundaboutTitle: 'Hide/show all missing roundabout URs.',
HideByTypeMissingStreetNameTitle: 'Hide/show all missing street name URs.',
HideByTypeTurnNotAllowedTitle: 'Hide/show all turn not allowed URs.',
HideByTypeUndefinedTitle: 'Hide/show all undefined URs.',
HideByTypeWrongDrivingDirectionTitle: 'Hide/show all wrong driving direction URs.',
HideByTagged: 'Hide by tagged',
HideByTaggedInverted: 'Show by tagged',
HideByTaggedBogTitle: 'Hide/show all URs with [BOG] (boots on ground) / [BOTG] (boots on the ground) in description or comments.',
HideByTaggedClosureTitle: 'Hide/show all URs with [CLOSURE] in description or comments.',
HideByTaggedConstructionTitle: 'Hide/show all URs with [CONSTRUCTION] in description or comments.',
HideByTaggedDifficultTitle: 'Hide/show all URs with [DIFFICULT] in description or comments.',
HideByTaggedEventTitle: 'Hide/show all URs with [EVENT] in description or comments.',
HideByTaggedNoteTitle: 'Hide/show all URs with [NOTE] in description or comments.',
HideByTaggedRoadworksTitle: 'Hide/show all URs with [ROADWORKS] in description or comments. Used in the UK.',
HideByTaggedWslmTitle: 'Hide/show all URs with [WSLM] in description or comments.',
HideByAgeOfSubmission: 'Hide by age of submission',
HideByAgeOfSubmissionInverted: 'Show by age of submission',
HideByAgeOfSubmissionLessThanTitle: 'Hide/show URs that were originally created less than specified number of days ago.',
HideByAgeOfSubmissionMoreThanTitle: 'Hide/show URs that were originally created more than specified number of days ago.',
DescriptionCommentsFollowing: 'Hide by description, comment, following',
DescriptionCommentsFollowingInverted: 'Show by description, comment, following',
HideFollowingTitle: 'Hide/show URs you are following.',
HideNotFollowingTitle: 'Hide/show URs you are not following.',
HideWithDescriptionTitle: 'Hide/show URs that have a description.',
HideWithoutDescriptionTitle: 'Hide/show URs that do not have a description.',
HideCommentsFromMe: 'Comments from me',
HideWithCommentsFromMeTitle: 'Hide/show URs you have commented on.',
HideWithoutCommentsFromMeTitle: 'Hide/show URs you have not commented on.',
HideFirstCommentByMe: 'First comment by me',
HideFirstCommentByMeTitle: 'Hide/show URs where you were the first person to comment.',
HideFirstCommentNotByMeTitle: 'Hide/show URs where someone else was the first person to comment.',
HideLastCommentByMe: 'Last comment by me',
HideLastCommentByMeTitle: 'Hide/show URs where you are the last person to comment.',
HideLastCommentNotByMeTitle: 'Hide/show URs where someone else is the last person to comment.',
HideLastCommentByReporter: 'Last comment by reporter',
HideLastCommentByReporterTitle: 'Hide/show URs where the reporter is the last person to comment.',
HideLastCommentNotByReporterTitle: 'Hide/show URs where the reporter is not the last person to comment.',
HideByCommentCountLessThanTitle: 'Hide/show URs that contain less comments than the number specified.',
HideByCommentCountMoreThanTitle: 'Hide/show URs that contain more comments than the number specified.',
HideByAgeOfFirstCommentLessThan: 'First comment less than',
HideByAgeOfFirstCommentLessThanTitle: 'Hide/show URs where the first comment is less than the days specified ago.',
HideByAgeOfFirstCommentMoreThan: 'First comment more than',
HideByAgeOfFirstCommentMoreThanTitle: 'Hide/show URs where the first comment is more than the days specified ago.',
HideByAgeOfLastCommentLessThan: 'Last comment less than',
HideByAgeOfLastCommentLessThanTitle: 'Hide/show URs where the last comment is less than the days specified ago.',
HideByAgeOfLastCommentMoreThan: 'Last comment more than',
HideByAgeOfLastCommentMoreThanTitle: 'Hide/show URs where the last comment is more than the days specified ago.',
HideByKeywordIncluding: 'Including keyword',
HideByKeywordIncludingTitle: 'Hide/show URs that include the custom word / text specified. Regex compatible.',
HideByKeywordNotIncluding: 'Not including keyword',
HideByKeywordNotIncludingTitle: 'Hide/show URs that do not include the custom word / text specified. Regex compatible.',
HideByKeywordCaseInsensitive: 'Case-insensitive keyword matches',
HideByKeywordCaseInsensitiveTitle: 'If enabled, searching for the above including or not including keywords will be done using case insensitive searching.',
HideWithCommentBy: 'With comment by',
HideWithCommentByTitle: 'Hide/show URs that have been commented on by the specified user(s).\nUse a comma (,) to separate usernames.',
HideWithoutCommentBy: 'Without comment by',
HideWithoutCommentByTitle: 'Hide/show URs that have not been commented on by the specified user(s).\nUse a comma (,) to separate usernames.',
ReminderDays: 'Reminder days',
ReminderDaysTitle: 'Number of days to use when calculating UR filtering and when setting and/or sending the reminder comment.\nThis is the number of days since the first '
+ 'comment.\nSet to 0 if you do not use reminders.',
CloseDays: 'Close days',
CloseDaysTitle: 'Number of days to use when calculating UR filtering.\nThis is the number of days since the last comment.\nExample: If you close 4 days after the last '
+ 'comment, set to 4.\nAnything less than this time will be considered \'waiting\' as long as there is at least one comment already.',
EnableAppendMode: 'Enable append comment mode',
EnableAppendModeTitle: 'Enabling append comment mode will allow you to append a comment to the existing text in the new-comment box.\nThe comment is appended with a blank '
+ 'line between the existing text and the new text.\nThe status of the UR is set to the status of the new comment you clicked to append.\nIf the comment would end '
+ 'up being longer than 2000 characters, append mode will give a warning and not alter the text in the comment box, but the status would have been changed.'
},
tabs: {
Comments: 'Comments',
Settings: 'Settings',
Tools: 'Tools'
},
tags: {
Bog: '[BOG] / [BOTG]',
Closure: '[CLOSURE]',
Construction: '[CONSTRUCTION]',
Difficult: '[DIFFICULT]',
Event: '[EVENT]',
Note: '[NOTE]',
Roadworks: '[ROADWORKS]',
Wslm: '[WSLM]'
},
tools: {
BackupSettingsTitle: 'Download a backup copy of your URC-E settings in JSON format.\nThis backup can be used to restore your settings to another computer, or in the event '
+ 'you lose your settings.\n\nNote: Please do not modify the JSON file in any way. The format is crucial to proper restoral of settings.',
CreateStep1: 'Open the template spreadsheet: $TEMPLATE_LINK$',
CreateStep2: 'Click <i>File</i> then <i>Make a copy</i>.',
CreateStep3: 'Give your file a name and specify the folder you want to save it in.',
CreateStep4: 'Click <i>Make a copy</i>.',
CreateStep5: 'Your sheet will open in a new tab. Look at the URL to get the <i>Spreadsheet ID</i>.<br>It is between <i>https://docs.google.com/spreadsheets/d/</i> and <i>/edit#...</i>.\n<br><b>Example:</b> <i>1JVtw4xwjKmPX_H1Xo1uwyYwxguM-oSi0LotD4lEwmK4</i> $SPREADSHEET_STEP$',
CreateStep6: 'Click <i>Share</i> at the top right.',
CreateStep7: 'Under <i>General access</i>, click the down arrow next to <i>Restricted</i>.',
CreateStep8: 'Select <i>Anyone with the link</i>.',
CreateStep9: 'Ensure <i>Viewer</i> is listed as to right and <i>Anyone on the internet with the link can view</i> below.',
CreateStep10: 'Click <i>Done</i>.',
ConvertStep1: 'Download your converted default UR type responses: $DOWNLOAD_DEFAULTS_LINK$',
ConvertStep2: 'Download your converted comment list: $DOWNLOAD_COMMENTS_LINK$',
ConvertStep3: 'In your custom Google Sheet, select the <b>CustomComments</b> tab, then click cell <b>B5</b>.',
ConvertStep4: 'Click <i>File</i> then <i>Import</i>.',
ConvertStep5: 'Click <i>Upload</i> at the top.',
ConvertStep6: 'Either click <i>Browse</i> and select your <i>default UR type responses</i> file that was downloaded in step $DOWNLOAD_DEFAULTS_STEP$ '
+ '(<b>default_responses_DATETIME.csv</b>),<br>or drag and drop your <i>default UR type responses</i> file to the box.',
ConvertStep7: 'For <i>Import location</i> select <b>Replace data at selected cell</b>.',
ConvertStep8: 'For <i>Separator type</i> select <i>Custom</i> and put <b>|</b> in the box.',
ConvertStep9: 'Click <i>Import data</i>.',
ConvertStep10: 'Click cell <b>A25</b>.',
ConvertStep11: 'Click <i>File</i> then <i>Import</i>.',
ConvertStep12: 'Click <i>Upload</i> at the top.',
ConvertStep13: 'Either click <i>Browse</i> and select your <i>comment list</i> file that was downloaded in step $DOWNLOAD_COMMENTS_STEP$ '
+ '(<b>comment_list_DATETIME.csv</b>),<br>or drag and drop your <i>comment list</i> file to the box.',
ConvertStep14: 'For <i>Import location</i> select <b>Replace data at selected cell</b>.',
ConvertStep15: 'For <i>Separator type</i> select <i>Custom</i> and put <b>|</b> in the box.',
ConvertStep16: 'Click <i>Import data</i>.',
FinalStep1: 'Put your <b>Spreadsheet ID</b> (step $SPREADSHEET_STEP$) in the Custom <i>Google Spreadsheet ID</i> settings box in URC-E settings.',
FinalStep2: 'You can now use the <i>Custom G Sheet</i> comment list.',
ConvertCreateConvertProcess: 'Convert Process',
ConvertCreateCreateProcess: 'Create Process',
ConvertCreateSteps: 'Steps',
ConvertCurrentCustom: 'Convert current custom',
ConvertCurrentCustomTitle: 'Convert your current custom comment list addon to URC-E style Google Sheet for easy maintenance, sharing, etc.',
CreateNewCustom: 'Create new custom',
CreateNewCustomTitle: 'Create your own URC-E custom comment list Google sheet for easy maintenance, sharing, etc.',
CustomGoogleSpreadsheet: 'Custom Google Spreadsheet',
IntersectionOf: 'the intersection of $SEG1NAME$ and $SEG2NAME$',
IntersectionOfWithCity: 'the intersection of $SEG1NAME$ and $SEG2NAME$ in $SEGCITY$',
ResetSettingsTitle: 'Reset all URC-E settings back to their default values.\n\nNote: Almost all settings default to disabled.',
RestoreSettingsFileError: 'Invalid URC-E settings JSON file.',
RestoreSettingsSelectFileTitle: 'Select the JSON file created by the URC-E \'Backup\' settings button.',
RestoreSettingsTitle: 'Restore a backup copy of your URC-E settings from the JSON backup file created with the backup settings button.\n\nNote: Please do not modify the '
+ 'JSON file in any way. The format is crucial to proper restoral of settings.',
SegmentWithCity: '$SEG1NAME$ in $SEGCITY$',
UnknownRoadName: 'unnamed road',
UnknownVenueName: 'unnamed venue'
},
urPanel: {
CurrentDate: 'Current date',
DriveDate: 'Drive date',
InsertCurrentDateCasualTitle: 'Shortcut - Current date (casual): Click this icon to insert the current date into the new comment box at the cursor position (full month '
+ 'name, 2-digit day in locale format).',
InsertCurrentDateTitle: 'Shortcut - Current date: Click this icon to insert the current date into the new comment box at the cursor position (2-digit month, 2-digit day, '
+ '4-digit year in locale format).',
InsertCurrentDayOfWeekTitle: 'Shortcut - Current day of week: Click this icon to insert the current day of the week into the new comment box at the cursor position (full '
+ 'day of week name in locale language).',
InsertCurrentTimeCasualTitle: 'Shortcut - Current time of day (casual): Click this icon to insert the current time of day into the new comment box at the cursor position '
+ '(in locale language).\n\n04:00am-11:59am: morning\n12:00pm-05:59pm: afternoon\n06:00pm-08:59pm: evening\n09:00pm-03:59am: night',
InsertCurrentTimeTitle: 'Shortcut - Current time of day: Click this icon to insert the current time of day into the new comment box at the cursor position (2-digit hour, '
+ '2-digit minute in locale format).',
InsertCustomTaglineTitle: 'Shortcut - Custom tagline: Click this icon to insert your custom tagline into the new comment box at the cursor position.',
InsertDateTimeCasualModeTitle: 'Shortcut - Drive day and time (fully casual): Click this icon to insert the drive date into the new comment box at the cursor position.\n\n'
+ '0 days: this morning, this afternoon, this evening, tonight\n'
+ '1 days: yesterday morning, yesterday afternoon, yesterrday evening, last night\n'
+ '2-6 days: Day_of_Week morning/afternoon/evening/night\n'
+ '7-13 days: last Day_of_Week morning/afternoon/evening/night\n'
+ '14-20 days: Day_of_Week before last\n'
+ '21-27 days: three weeks ago\n'
+ '28-60 days: a few weeks ago\n'
+ '61-120 days: a couple months back\n'
+ '121+ days: a while ago',
InsertDateCasualTitle: 'Shortcut - Drive date (casual): Click this icon to insert the drive date into the new comment box at the cursor position (full month name, 2-digit '
+ 'day in locale format).',
InsertDateTitle: 'Shortcut - Drive date: Click this icon to insert the drive date into the new comment box at the cursor position (2-digit month, 2-digit day, 4-digit '
+ 'year in locale format).',
InsertDayOfWeekTitle: 'Shortcut - Drive day of week: Click this icon to insert the drive date day of the week into the new comment box at the cursor position (full day of '
+ 'week name in locale language).',
InsertDescriptionTitle: 'Shortcut - Description: Click this icon to insert the UR description into the new comment box at the cursor position.',
InsertPlaceAddressTitle: 'Shortcut - Place address: Click this icon to either replace \'$PLACE_ADDRESS$\' with the address of the currently selected place, or it will '
+ 'insert the address of the currently selected place into the comment box at the current cursor position.',
InsertPlaceNameTitle: 'Shortcut - Place name: Click this icon to either replace \'$PLACE_NAME$\' with the name of the currently selected place, or it will insert the name '
+ 'of the currently selected place into the comment box at the current cursor position.',
InsertSelSegsTitle: 'Shortcut - Segment name(s): Click this icon to either replace \'$SELSEGS$\' (or \'$SELSEGS\') with the name of the currently selected segment(s), or\nit '
+ 'will insert the name of the currently selected segment(s) into the comment box at the cursor position.',
InsertSelSegsWithCityTitle: 'Shortcut - Segment name(s) with city: Click this icon to either replace \'$SELSEGS_WITH_CITY$\' (or \'$SELSEGS$\' or \'$SELSEGS\') with the '
+ 'name and city of the currently selected segment(s), or \nit will insert the name and city of the currently selected segment(s) into the comment box at the '
+ 'cursor position.',
InsertTimeCasualTitle: 'Shortcut - Drive time of day (casual): Click this icon to insert the drive date time of day into the new comment box at the cursor position (in '
+ 'locale language).\n\n04:00am-11:59am: morning\n12:00pm-05:59pm: afternoon\n06:00pm-08:59pm: evening\n09:00pm-03:59am: night',
InsertTimeTitle: 'Shortcut - Drive time of day: Click this icon to insert the drive date time of day into the new comment box at the cursor position (2-digit hour, '
+ '2-digit minute in locale format).',
InsertUrIdTitle: 'Shortcut - Insert UR ID number.',
InsertUrPermalinkTitle: 'Shortcut - Insert permalink to this UR. URL will include your locale, your \'env\', the latitude and longitude of this UR, zoom level of 17, '
+ 'mapUpdateRequest of this UR ID number, and s=20489175039 (which ensures the UR layer is turned on).',
InsertUrTypeTitle: 'Shortcut - UR Type: Click this icon to insert the UR type into the new comment box at the cursor position.',
InsertWazeUsernameTitle: 'Shortcut - Waze username: Click this icon to insert your Waze username into the new comment box at the cursor position.',
Shortcuts: 'Shortcuts'
},
urStatus: {
Closed: 'Closed',
NotIdentified: 'Not identified'
},
urTypes: {
Undefined: 'Undefined'
},
prompts: {
CommentInsertTimedOut: 'URC-E timed out waiting for the comment text box to become available.',
CommentTooLong: 'Appending another comment to the current text will cause the comment to be longer than 2000 characters, which is too long. URC-E did not append the '
+ 'selected comment and left the new-comment box with the same text it had.',
ConversionLoadAddonFirst: 'In order to convert a custom comments addon script to the new URC-E style Google Sheet, you must first ensure your custom comments addon is '
+ 'enabled in TamperMonkey and selected in URC-E\'s Comment List settings box.',
CustomGSheetLoadError: 'An error has occurred loading your URC-E custom comment Google sheet.<br><br>Please make sure you have done the following:\n<ul><li>Set the correct '
+ '<b><i>Custom Google Spreadsheet ID</i></b> (should be the ID of the copy created in the previous step) set on the settings tab.\n<li>Ensured your spreadsheet '
+ 'is shared to everyone with a link can view.\n</ul>If all these have been done and you are still having issues, please contact a WazeDev team member.',
NoCommentBox: 'URC-E: Unable to find the comment box! In order for this script to work, you need to have a UR open.',
PlaceAddressFound: 'The selected comment contains \'$PLACE_ADDRESS$\'.\n\nIn order to replace this text with the place address, please\nselect a place and click the place '
+ 'address shortcut button in the UR panel.',
PlaceAddressInsertError: 'In order to use the <i class="w-icon w-icon-location" aria-hidden="true"></i> button in the UR Panel, you must first select a place.',
PlaceNameFound: 'The selected comment contains \'$PLACE_NAME$\'.\n\nIn order to replace this text with the place name, please\nselect a place and click the place name '
+ 'shortcut button in the UR panel.',
PlaceNameInsertError: 'In order to use the <i class="w-icon w-icon-location"></i> button in the UR Panel, you must first select a place.',
ReminderMessageAuto: 'Automatically sent reminder message to UR(s)',
ResetSettings: 'Reset Settings',
ResetSettingsComplete: 'Settings reset complete',
ResetSettingsConfirmation: 'Are you sure you want to reset URC-E settings back to their default values?',
RestoreSettings: 'Restore Settings',
RestoreSettingsComplete: 'Settings restore complete',
RestoreSettingsConfirmation: 'Would you like to proceed with restoring your URC-E settings?',
RestoreSettingsInvalidSettings: 'Invalid settings removed from JSON file',
RestoreSettingsNumOfSettings: 'Number of settings to restore',
RestoreSettingsRetainedSettings: 'Retained current settings',
RestrictionsEnforced: 'Restrictions enforced!',
RestrictionsEnforcedTitle: 'The following restrictions have been enforced',
SelSegsFound: 'The selected comment contains \'$SELSEGS$\'.\n\nIn order to replace this text with the road name(s), please\nselect one or two segments and click the road '
+ 'button in the\nUR panel.',
SelSegsInsertError: 'In order to use the <i class="w-icon w-icon-road"></i> button in the UR Panel, you must first select one or two segments.',
SetCustomSsIdFirst: 'Before you can select <b><i>Custom G Sheet</i></b> as your comment list, you must first set the <b><i>Custom Google Spreadsheet ID</i></b> setting on '
+ 'the URC-E settings tab.',
SpreadsheetUpdateRequired: 'You are using an older version of the URC-E spreadsheet, which has caused an error. Please update your spreadsheet to at least version: $VERSION$ '
+ 'See instructions available in this forum post: $LINK$',
SwitchingCommentLists: 'Switching comment lists',
TimedOutWaitingStatic: 'Timed out waiting for the static list to become available. Is it enabled?',
UpdateRequired: 'You are using an older version of URC-E, which has caused an error. Please update to at least version: $VERSION$ GreasyFork Link: $LINK$',
UrOverflowErrorWithoutOverflowEnabled: 'WME will not load more than 500 URs per screen. Some URs may be missing. You can try to enable overflow handling, or zoom in and refresh.',
VarFound: 'The selected comment contains variables: $VARSFOUND$.\n\nUntil these are replaced, the send button is disabled.',
WaitingOnInit: 'Waiting for URC-E to fully initialize',
WaitingToGetUrId: 'Waiting to get urId'
},
time: {
ACoupleMonthsAgo: 'a couple months ago',
AFewWeeksAgo: 'a few weeks ago',
Afternoon: 'afternoon',
AWhileBack: 'a while ago',
Evening: 'evening',
Morning: 'morning',
LastNight: 'last night',
LastWeekTOD: 'last $DAY_NAME$ $CASUAL_TOD$',
Night: 'night',
ThisCasualTOD: 'this $THIS_CASUAL_TOD$',
ThisWeekTOD: '$DAY_NAME$ $CASUAL_TOD$',
Tonight: 'tonight',
TwoWeeksAgo: '$DAY_NAME$ before last',
ThreeWeeksAgo: 'three weeks ago',
YesterdayCasualTOD: 'yesterday $YESTERDAY_CASUAL_TOD$'
}
}
};
translations['en-US'] = { ...translations.en };
const locale = I18n.currentLocale();
if (translations[locale]) {
Object.keys(translations[locale]).forEach((obj1) => {
if (typeof translations[locale][obj1] === 'object') {
Object.keys(translations[locale][obj1]).forEach((obj2) => {
if (translations[locale][obj1][obj2].length === 0)
delete (translations[locale][obj1][obj2]);
});
}
else if ((typeof translations[locale][obj1] === 'string') && (translations[locale][obj1].length === 0)) {
delete (translations[locale][obj1]);
}
});
}
I18n.translations[locale].urce = $extend(true, {}, translations.en, translations[locale]);
if (!Object.keys(translations).includes(locale))
_needTranslation = true;
}
else {
errorText = errorText || res;
}
postProcess(errorText);
},
onerror(res) {
postProcess(`xmlhttpRequest error: ${JSON.stringify(res)}`);
}
});
});
}
}
)();