WME E40 Geometry

Setup POI geometry properties in one click

Fra 20.08.2022. Se den seneste versjonen.

// ==UserScript==
// @name         WME E40 Geometry
// @version      0.4.0
// @description  Setup POI geometry properties in one click
// @author       Anton Shevchuk
// @license      MIT License
// @match        https://www.waze.com/editor*
// @match        https://www.waze.com/*/editor*
// @match        https://beta.waze.com/editor*
// @match        https://beta.waze.com/*/editor*
// @exclude      https://www.waze.com/user/editor*
// @exclude      https://beta.waze.com/user/editor*
// @grant        none
// @icon         
// @require      https://greasyfork.dpdns.org/scripts/389117-apihelper/code/APIHelper.js?version=1082818
// @require      https://greasyfork.dpdns.org/scripts/389577-apihelperui/code/APIHelperUI.js?version=1082967
// @supportURL   https://github.com/AntonShevchuk/wme-e40/issues
// @namespace    https://greasyfork.dpdns.org/users/227648
// ==/UserScript==

/* jshint esversion: 6 */
/* global require */
/* global $ */
/* global W */
/* global OpenLayers */
/* global I18n */
/* global APIHelper */
/* global APIHelperUI */

(function ($) {
  'use strict'

  let helper
  let panel
  let tab

  let OL = OpenLayers

  // Script name, uses as unique index
  const NAME = 'E40'

  // Translations
  const TRANSLATION = {
    'en': {
      title: 'Geometry',
      description: 'Change geometry in the current view area',
      orthogonalize: 'Orthogonalize',
      simplify: 'Simplify',
      scale: 'Scale',
      copy: 'Copy',
    },
    'uk': {
      title: 'Геометрія',
      description: 'Змінити геометрію об’єктів у поточному розташуванні',
      orthogonalize: 'Вирівняти',
      simplify: 'Спростити',
      scale: 'Масштабувати',
      copy: 'Копіювати',
    },
    'ru': {
      title: 'Геометрия',
      description: 'Изменить геометрию объектов в текущем расположении',
      orthogonalize: 'Выровнять',
      simplify: 'Упростить',
      scale: 'Масштабировать',
      copy: 'Копировать',
    }
  }

  APIHelper.bootstrap()
  APIHelper.addTranslation(NAME, TRANSLATION)
  APIHelper.addStyle(
    'button.waze-btn.e40 { margin: 0 4px 4px 0; padding: 2px; width: 42px; } ' +
    'button.waze-btn.e40:hover { box-shadow: 0 2px 8px 0 rgba(0, 0, 0, 0.1), inset 0 0 100px 100px rgba(255, 255, 255, 0.3); } '
  )

  const panelButtons = {
    A: {
      title: '🔲',
      description: I18n.t(NAME).orthogonalize,
      shortcut: 'S+49',
      callback: () => orthogonalize()
    },
    B: {
      title: '<i class="fa fa-magic"></i>',
      description: I18n.t(NAME).simplify,
      shortcut: 'S+50',
      callback: () => simplify()
    },
    C: {
      title: '500m²',
      description: I18n.t(NAME).scale,
      shortcut: 'S+51',
      callback: () => scaleSelected(500)
    },
    D: {
      title: '650m²',
      description: I18n.t(NAME).scale,
      shortcut: 'S+52',
      callback: () => scaleSelected(650)
    },
    E: {
      title: '>650',
      description: I18n.t(NAME).scale,
      shortcut: 'S+53',
      callback: () => scaleSelected(650, true)
    },
    F: {
      title: '<i class="fa fa-clone" aria-hidden="true"></i>',
      description: I18n.t(NAME).copy,
      shortcut: 'S+53',
      callback: () => copyPlace()
    }
  }

  const tabButtons = {
    A: {
      title: '🔲',
      description: I18n.t(NAME).orthogonalize,
      shortcut: null,
      callback: () => orthogonalizeAll()
    },
    B: {
      title: '<i class="fa fa-magic"></i>',
      description: I18n.t(NAME).simplify,
      shortcut: null,
      callback: () => simplifyAll()
    },
    C: {
      title: '>500',
      description: I18n.t(NAME).scale,
      shortcut: null,
      callback: () => scaleAll(500, true)
    }
  }

  let WazeActionUpdateFeatureGeometry
  let WazeActionUpdateFeatureAddress
  let WazeFeatureVectorLandmark
  let WazeActionAddLandmark

  /**
   * Get selected Area POI
   * @return {Array}
   */
  function getSelectedPlaces () {
    let selected
    selected = APIHelper.getSelectedVenues()
    selected = selected.filter((el) => !el.isPoint())
    return selected
  }

  // Scale selected place(s) to X m²
  function scaleSelected (x, orMore = false) {
    scaleArray(getSelectedPlaces(), x, orMore)
    return false
  }

  // Scale all places in the editor area to X m²
  function scaleAll (x = 650, orMore = true) {
    scaleArray(APIHelper.getVenues(), x, orMore)
    return false
  }

  function scaleArray (elements, x, orMore = false) {
    for (let i = 0; i < elements.length; i++) {
      let selected = elements[i]
      try {
        let oldGeometry = selected.geometry.clone()
        let newGeometry = selected.geometry.clone()

        let scale = Math.sqrt((x + 5) / oldGeometry.getGeodesicArea(W.map.getProjectionObject()))
        if (scale < 1 && orMore) {
          continue
        }
        newGeometry.resize(scale, newGeometry.getCentroid())

        let action = new WazeActionUpdateFeatureGeometry(selected, W.model.venues, oldGeometry, newGeometry)
        W.model.actionManager.add(action)
      } catch (e) {
        log('skipped')
      }
    }
  }

  // Orthogonalize selected place(s)
  function orthogonalize () {
    orthogonalizeArray(getSelectedPlaces())
    return false
  }

  // Orthogonalize all places in the editor area
  function orthogonalizeAll () {
    // skip parking, natural and outdoors
    // TODO: make options for filters
    orthogonalizeArray(APIHelper.getVenues(['OUTDOORS', 'PARKING_LOT', 'NATURAL_FEATURES']))
    return false
  }

  function orthogonalizeArray (elements) {
    for (let i = 0; i < elements.length; i++) {
      let selected = elements[i]
      try {
        let oldGeometry = selected.geometry.clone()
        let newGeometry = orthogonalizeGeometry(selected.geometry.clone().components[0].components)

        if (!compare(oldGeometry.components[0].components, newGeometry)) {
          selected.geometry.components[0].components = [].concat(newGeometry)
          selected.geometry.components[0].clearBounds()

          let action = new WazeActionUpdateFeatureGeometry(selected, W.model.venues, oldGeometry, selected.geometry)
          W.model.actionManager.add(action)
        }
      } catch (e) {
        log('skipped')
        console.log(e)
      }
    }
    return false
  }

  function orthogonalizeGeometry (geometry, threshold = 12) {
    let nomthreshold = threshold, // degrees within right or straight to alter
      lowerThreshold = Math.cos((90 - nomthreshold) * Math.PI / 180),
      upperThreshold = Math.cos(nomthreshold * Math.PI / 180)

    function Orthogonalize () {
      let nodes = geometry,
        points = nodes.slice(0, -1).map(function (n) {
          let p = n.clone().transform(new OpenLayers.Projection('EPSG:900913'), new OpenLayers.Projection('EPSG:4326'))
          p.y = lat2latp(p.y)
          return p
        }),
        corner = { i: 0, dotp: 1 },
        epsilon = 1e-4,
        i, j, score, motions

      // Triangle
      if (nodes.length === 4) {
        for (i = 0; i < 1000; i++) {
          motions = points.map(calcMotion)

          let tmp = addPoints(points[corner.i], motions[corner.i])
          points[corner.i].x = tmp.x
          points[corner.i].y = tmp.y

          score = corner.dotp
          if (score < epsilon)
            break
        }

        let n = points[corner.i]
        n.y = latp2lat(n.y)
        let pp = n.transform(new OpenLayers.Projection('EPSG:4326'), new OpenLayers.Projection('EPSG:900913'))

        let id = nodes[corner.i].id
        for (i = 0; i < nodes.length; i++) {
          if (nodes[i].id != id)
            continue

          nodes[i].x = pp.x
          nodes[i].y = pp.y
        }

        return nodes
      } else {
        let best,
          originalPoints = nodes.slice(0, -1).map(function (n) {
            let p = n.clone().transform(new OpenLayers.Projection('EPSG:900913'), new OpenLayers.Projection('EPSG:4326'))
            p.y = lat2latp(p.y)
            return p
          })
        score = Infinity

        for (i = 0; i < 1000; i++) {
          motions = points.map(calcMotion)
          for (j = 0; j < motions.length; j++) {
            let tmp = addPoints(points[j], motions[j])
            points[j].x = tmp.x
            points[j].y = tmp.y
          }
          var newScore = squareness(points)
          if (newScore < score) {
            best = [].concat(points)
            score = newScore
          }
          if (score < epsilon)
            break
        }

        points = best

        for (i = 0; i < points.length; i++) {
          // only move the points that actually moved
          if (originalPoints[i].x !== points[i].x || originalPoints[i].y !== points[i].y) {
            let n = points[i]
            n.y = latp2lat(n.y)
            let pp = n.transform(new OpenLayers.Projection('EPSG:4326'), new OpenLayers.Projection('EPSG:900913'))

            let id = nodes[i].id
            for (j = 0; j < nodes.length; j++) {
              if (nodes[j].id != id)
                continue

              nodes[j].x = pp.x
              nodes[j].y = pp.y
            }
          }
        }

        // remove empty nodes on straight sections
        for (i = 0; i < points.length; i++) {
          let dotp = normalizedDotProduct(i, points)
          if (dotp < -1 + epsilon) {
            let id = nodes[i].id
            for (j = 0; j < nodes.length; j++) {
              if (nodes[j].id != id)
                continue

              nodes[j] = false
            }
          }
        }

        return nodes.filter(item => item !== false)
      }

      function calcMotion (b, i, array) {
        let a = array[(i - 1 + array.length) % array.length],
          c = array[(i + 1) % array.length],
          p = subtractPoints(a, b),
          q = subtractPoints(c, b),
          scale, dotp

        scale = 2 * Math.min(euclideanDistance(p, { x: 0, y: 0 }), euclideanDistance(q, { x: 0, y: 0 }))
        p = normalizePoint(p, 1.0)
        q = normalizePoint(q, 1.0)

        dotp = filterDotProduct(p.x * q.x + p.y * q.y)

        // nasty hack to deal with almost-straight segments (angle is closer to 180 than to 90/270).
        if (array.length > 3) {
          if (dotp < -0.707106781186547)
            dotp += 1.0
        } else if (dotp && Math.abs(dotp) < corner.dotp) {
          corner.i = i
          corner.dotp = Math.abs(dotp)
        }

        return normalizePoint(addPoints(p, q), 0.1 * dotp * scale)
      }
    }

    function lat2latp (lat) {
      return 180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * (Math.PI / 180) / 2))
    }

    function latp2lat (a) {
      return 180 / Math.PI * (2 * Math.atan(Math.exp(a * Math.PI / 180)) - Math.PI / 2)
    }

    function squareness (points) {
      return points.reduce(function (sum, val, i, array) {
        let dotp = normalizedDotProduct(i, array)

        dotp = filterDotProduct(dotp)
        return sum + 2.0 * Math.min(Math.abs(dotp - 1.0), Math.min(Math.abs(dotp), Math.abs(dotp + 1)))
      }, 0)
    }

    function normalizedDotProduct (i, points) {
      let a = points[(i - 1 + points.length) % points.length],
        b = points[i],
        c = points[(i + 1) % points.length],
        p = subtractPoints(a, b),
        q = subtractPoints(c, b)

      p = normalizePoint(p, 1.0)
      q = normalizePoint(q, 1.0)

      return p.x * q.x + p.y * q.y
    }

    function subtractPoints (a, b) {
      return { x: a.x - b.x, y: a.y - b.y }
    }

    function addPoints (a, b) {
      return { x: a.x + b.x, y: a.y + b.y }
    }

    function euclideanDistance (a, b) {
      let x = a.x - b.x, y = a.y - b.y
      return Math.sqrt((x * x) + (y * y))
    }

    function normalizePoint (point, scale) {
      let vector = { x: 0, y: 0 }
      let length = Math.sqrt(point.x * point.x + point.y * point.y)
      if (length !== 0) {
        vector.x = point.x / length
        vector.y = point.y / length
      }

      vector.x *= scale
      vector.y *= scale

      return vector
    }

    function filterDotProduct (dotp) {
      if (lowerThreshold > Math.abs(dotp) || Math.abs(dotp) > upperThreshold)
        return dotp

      return 0
    }

    function isDisabled (nodes) {
      let points = nodes.slice(0, -1).map(function (n) {
        let p = n.toLonLat().transform(new OpenLayers.Projection('EPSG:900913'), new OpenLayers.Projection('EPSG:4326'))
        return { x: p.lat, y: p.lon }
      })

      return squareness(points)
    }

    return Orthogonalize()
  }

  // Simplify selected place(s)
  function simplify (factor = 8) {
    simplifyArray(getSelectedPlaces(), factor)
    return false
  }

  // Simplify all places in the editor area
  function simplifyAll () {
    // skip parking, natural and outdoors
    // TODO: make options for filters
    simplifyArray(APIHelper.getVenues(['OUTDOORS', 'PARKING_LOT', 'NATURAL_FEATURES']))
    return false
  }

  function simplifyArray (elements, factor = 8) {
    for (let i = 0; i < elements.length; i++) {
      let selected = elements[i]
      try {
        let oldGeometry = selected.geometry.clone()
        let ls = new OL.Geometry.LineString(oldGeometry.components[0].components)
        ls = ls.simplify(factor)
        let newGeometry = new OL.Geometry.Polygon(new OL.Geometry.LinearRing(ls.components))

        if (newGeometry.components[0].components.length < oldGeometry.components[0].components.length) {
          W.model.actionManager.add(new WazeActionUpdateFeatureGeometry(selected, W.model.venues, oldGeometry, newGeometry))
        }
      } catch (e) {
        log('skipped')
      }
    }
    return false
  }

  // Compare two polygons point-by-point
  function compare (geo1, geo2) {
    if (geo1.length !== geo2.length) {
      return false
    }
    for (let i = 0; i < geo1.length; i++) {
      if (Math.abs(geo1[i].x - geo2[i].x) > .1
        || Math.abs(geo1[i].y - geo2[i].y) > .1) {
        return false
      }
    }
    return true
  }

  function copyPlace(){
    let venues = APIHelper.getSelectedVenues()

    if (venues.length > 0) {
      let oldPlace = venues[0];
      let newPlace = new WazeFeatureVectorLandmark;
      newPlace.attributes.name = oldPlace.attributes.name + ' (copy)';
      newPlace.attributes.phone = oldPlace.attributes.phone;
      newPlace.attributes.url = oldPlace.attributes.url;
      newPlace.attributes.categories = [].concat(oldPlace.attributes.categories);
      newPlace.attributes.aliases = [].concat(oldPlace.attributes.aliases);
      newPlace.attributes.description = oldPlace.attributes.description;
      newPlace.attributes.houseNumber = oldPlace.attributes.houseNumber;
      newPlace.attributes.lockRank = oldPlace.attributes.lockRank;
      newPlace.attributes.geometry = oldPlace.attributes.geometry.clone();

      if (oldPlace.attributes.geometry.toString().match(/^POLYGON/)) {
        for (let i = 0; i < newPlace.attributes.geometry.components[0].components.length - 1; i++) {
          newPlace.attributes.geometry.components[0].components[i].x += 5
          newPlace.attributes.geometry.components[0].components[i].y += 5
        }
      } else {
        // Geometry not used for points
        // But who knows?
        newPlace.attributes.geometry.x += 5
        newPlace.attributes.geometry.y += 5
      }

      newPlace.attributes.services = [].concat(oldPlace.attributes.services);
      newPlace.attributes.openingHours = [].concat(oldPlace.attributes.openingHours);
      newPlace.attributes.streetID = oldPlace.attributes.streetID;

      if (oldPlace.attributes.categories.includes('GAS_STATION')) {
        newPlace.attributes.brand = oldPlace.attributes.brand
      }

      if (oldPlace.attributes.categories.includes('PARKING_LOT')) {
        newPlace.attributes.categoryAttributes.PARKING_LOT = {}

        let attributes = oldPlace.attributes.categoryAttributes.PARKING_LOT
        if ((attributes.lotType != null))
          newPlace.attributes.categoryAttributes.PARKING_LOT.lotType = [].concat(oldPlace.attributes.categoryAttributes.PARKING_LOT.lotType)
        if ((attributes.canExitWhileClosed != null))
          newPlace.attributes.categoryAttributes.PARKING_LOT.canExitWhileClosed = oldPlace.attributes.categoryAttributes.PARKING_LOT.canExitWhileClosed
        if ((attributes.costType != null))
          newPlace.attributes.categoryAttributes.PARKING_LOT.costType = oldPlace.attributes.categoryAttributes.PARKING_LOT.costType
        if ((attributes.estimatedNumberOfSpots != null))
          newPlace.attributes.categoryAttributes.PARKING_LOT.estimatedNumberOfSpots = oldPlace.attributes.categoryAttributes.PARKING_LOT.estimatedNumberOfSpots
        if ((attributes.hasTBR != null))
          newPlace.attributes.categoryAttributes.PARKING_LOT.hasTBR = oldPlace.attributes.categoryAttributes.PARKING_LOT.hasTBR
        if ((attributes.lotType != null))
          newPlace.attributes.categoryAttributes.PARKING_LOT.lotType = [].concat(oldPlace.attributes.categoryAttributes.PARKING_LOT.lotType)
        if ((attributes.parkingType != null))
          newPlace.attributes.categoryAttributes.PARKING_LOT.parkingType = oldPlace.attributes.categoryAttributes.PARKING_LOT.parkingType
        if ((attributes.paymentType != null))
          newPlace.attributes.categoryAttributes.PARKING_LOT.paymentType = [].concat(oldPlace.attributes.categoryAttributes.PARKING_LOT.paymentType)
      }

      W.model.actionManager.add(new WazeActionAddLandmark(newPlace));
      W.selectionManager.setSelectedModels(newPlace);
    }
  }

  // Simple console.log wrapper
  function log (message) {
    console.log(NAME + ': ' + message)
  }

  $(document)
    .on('init.apihelper', ready)
    .on('landmark.apihelper', createPanel)
    .on('landmark-collection.apihelper', createPanel)

  function ready () {
    // Require Waze components
    WazeActionUpdateFeatureGeometry = require('Waze/Action/UpdateFeatureGeometry')
    WazeActionUpdateFeatureAddress = require('Waze/Action/UpdateFeatureAddress')
    WazeFeatureVectorLandmark = require('Waze/Feature/Vector/Landmark')
    WazeActionAddLandmark = require('Waze/Action/AddLandmark')


    helper = new APIHelperUI(NAME)

    panel = helper.createPanel(I18n.t(NAME).title)
    panel.addButtons(panelButtons)

    if (W.loginManager.user.getRank() > 2) {
      tab = helper.createTab(
        I18n.t(NAME).title,
        I18n.t(NAME).description,
        '<i class="w-icon panel-header-component-icon w-icon-polygon"></i>'
      )
      tab.addButtons(tabButtons)
      tab.inject()
    }

    W.model.actionManager.events.register('afterundoaction', null, updateLabel)
    W.model.actionManager.events.register('afterclearactions', null, updateLabel)
    W.model.actionManager.events.register('afteraction', null, updateLabel)
  }

  function createPanel (event, element) {
    if (element.querySelector('div.form-group.e40')) {
      return
    }
    let places = getSelectedPlaces()
    if (places.length === 0) {
      return
    }

    element.prepend(panel.html())
    updateLabel()
  }

  function updateLabel () {
    let places = getSelectedPlaces()
    if (places.length === 0) {
      return
    }
    let info = []
    for (let i = 0; i < places.length; i++) {
      let selected = places[i]
      info.push(Math.round(selected.geometry.getGeodesicArea(W.map.getProjectionObject())) + 'm²')
    }
    let label = I18n.t(NAME).title
    if (info.length) {
      label += ' (' + info.join(', ') + ')'
    }
    panel.html().querySelector('label').innerText = label
  }

  // external API
  window.E40 = {
    scale: function (x) {
      scaleSelected(x)
    }
  }
})(window.jQuery)
长期地址
遇到问题?请前往 GitHub 提 Issues,或加Q群1031348184

赞助商

Fishcpy

广告

Rainyun

注册一下就行

Rainyun

一年攒够 12 元