WME Closure Details

Provide access to all closure details within the closures list

目前为 2018-02-22 提交的版本。查看 最新版本

// ==UserScript==
// @name        WME Closure Details
// @namespace   http://www.tomputtemans.com/
// @description Provide access to all closure details within the closures list
// @include     /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor.*$/
// @icon        
// @version     1.0.3
// @grant       none
// ==/UserScript==

function init(e) {
  if (e && e.user === null) {
    return;
  }
  if (document.getElementById('user-info') === null) {
    setTimeout(init, 500);
    return;
  }
  if (typeof W.loginManager === 'undefined' || W.app.modeController === 'undefined' || W.selectionManager === 'undefined') {
    setTimeout(init, 300);
    return;
  }
  if (!W.loginManager.user) {
    W.loginManager.events.register('login', null, init);
    W.loginManager.events.register('loginStatus', null, init);
    // Double check as event might have triggered already
    if (!W.loginManager.user) {
      return;
    }
  }

  // Deal with events mode
  if (W.app.modeController) {
    W.app.modeController.model.bind('change:mode', function(model, modeId) {
      if (modeId === 0) {
        observeContentsPane();
      }
    });
  }
  observeContentsPane();
}
init();

function observeContentsPane() {
  function handleMutations(mutations) {
    mutations.forEach(function(mutation) {
      var closureBlocks = mutation.target.querySelectorAll('.closure-item');
      var selectedIDs = W.selectionManager.selectedItems.filter(function(obj) {
        return obj.model.type == 'segment';
      }).map(function(obj) {
        return obj.model.attributes.id;
      });
      var selectedClosures = W.model.roadClosures.getObjectArray().filter(function(closure) {
        return selectedIDs.indexOf(closure.segID) != -1;
      });
      for (var i = 0; i < closureBlocks.length; i++) {
        var closureBlock = closureBlocks[i];
        var buttons = closureBlock.querySelectorAll('a');
        for (var j = 0; j < buttons.length; j++) {
          buttons[j].addEventListener('click', function() {
            var tooltip = document.querySelector('body .tooltip');
            if (tooltip) {
              tooltip.parentNode.removeChild(tooltip);
            }
          });
        }
        var matchedClosure = selectedClosures.find(function(closure) {
          return getLocalizedTime(closure.startDate) == getTimeFromBlock(closureBlock.querySelector('.start-date')) &&
            getLocalizedTime(closure.endDate) == getTimeFromBlock(closureBlock.querySelector('.end-date'));
        });
        if (matchedClosure) {
          var description = '<div>';
          if (matchedClosure.reason !== '') {
            description = '<strong style="font-size:120%">' + matchedClosure.reason + '</strong>';
          } else {
            description = '<em style="font-size:120%">No description set</em>';
          }
          description += '<table style="border-width:0;text-align:left"><tbody><tr><th style="text-align:right;padding-right:0.4em">Created by</th><td>' + getUsername(matchedClosure.createdBy) + '</td></tr><tr><th style="text-align:right;padding-right:0.4em">Created on</th><td>' + I18n.l('time.formats.long', matchedClosure.createdOn) + '</td></tr>';
          if (matchedClosure.updatedBy) {
            description += '<tr><th style="text-align:right;padding-right:0.4em">Updated by</th><td>' + getUsername(matchedClosure.updatedBy) + '</td></tr>';
          }
          if (matchedClosure.updatedOn) {
            description += '<tr><th style="text-align:right;padding-right:0.4em">Updated on</th><td>' + I18n.l('time.formats.long', matchedClosure.updatedOn) + '</td></tr>';
          }
          description += '<tr><td colspan="2" style="text-align:center"><em>' + (matchedClosure.permanent ? 'Ignores traffic' : 'Listens to traffic') + '</em></td></tr></tbody></table></div>';
          $(closureBlock).tooltip({
            placement: 'right',
            container: 'body',
            trigger: 'hover',
            html: true,
            title: description
          });
        }
      }
    });
    if (document.querySelector('.contents .main')) {
      (new MutationObserver(handleMutations)).observe(document.querySelector('.contents .main'), {
        childList: true
      });
    }
  }
  (new MutationObserver(handleMutations)).observe(document.querySelector('.contents'), {
    childList: true
  });
}

function getTimeFromBlock(node) {
  return node.querySelector('.date').textContent + ' ' + node.querySelector('.time').textContent;
}

function getLocalizedTime(date) {
  var splitDate = date.split(' ');
  return I18n.l('date.formats.default', splitDate[0]) + ' ' + splitDate[1];
}

function getUsername(id) {
  var user = W.model.users.get(id);
  if (user) {
    return user.userName;
  } else {
    return id + '(user not loaded)';
  }
}