- // ==UserScript==
- // @name WME E85 Simplify Street Geometry
- // @name:uk WME 🇺🇦 E85 Simplify Street Geometry
- // @version 0.2.4
- // @description Simplify Street Geometry, looks like fork
- // @description:uk Спрощуємо та вирівнюємо геометрію вулиць
- // @license MIT License
- // @author Anton Shevchuk
- // @namespace https://greasyfork.org/users/227648-anton-shevchuk
- // @supportURL https://github.com/AntonShevchuk/wme-e85/issues
- // @match https://*.waze.com/editor*
- // @match https://*.waze.com/*/editor*
- // @exclude https://*.waze.com/user/editor*
- // @icon 
- // @grant none
- // @require https://update.greasyfork.org/scripts/389765/1090053/CommonUtils.js
- // @require https://update.greasyfork.org/scripts/450160/1218867/WME-Bootstrap.js
- // @require https://update.greasyfork.org/scripts/452563/1218878/WME.js
- // @require https://update.greasyfork.org/scripts/450221/1137043/WME-Base.js
- // @require https://update.greasyfork.org/scripts/450320/1555446/WME-UI.js
- // ==/UserScript==
-
- /* jshint esversion: 8 */
-
- /* global require */
- /* global $, jQuery */
- /* global W */
- /* global I18n */
- /* global OpenLayers */
- /* global WME, WMEBase */
- /* global WMEUI, WMEUIHelper, WMEUIHelperPanel, WMEUIHelperModal, WMEUIHelperTab, WMEUIShortcut */
- /* global Container, Settings, SimpleCache, Tools */
-
- (function () {
- 'use strict'
-
- // Script name, uses as unique index
- const NAME = 'E85'
-
- // Translations
- const TRANSLATION = {
- 'en': {
- title: 'Street Geometry',
- description: 'Simplify and straighten up streets',
- buttons: {
- A: 'Simplify',
- B: 'Straighten',
- C: '∡90°',
- },
- settings: {
- title: 'Settings',
- description: 'Settings for simplifying segments',
- simplifyShort: 'Remove a fragment shorter than',
- simplifyAngle: 'If the angle is bigger than',
- simplifyTwoShort: 'and fragments shorter than',
- },
- },
- 'uk': {
- title: 'Геометрія вулиць',
- description: 'Спрощуйте та вирівнюйте вулиці',
- buttons: {
- A: 'Спростити',
- B: 'Вирівняти',
- C: '∡90°',
- },
- settings: {
- title: 'Налаштування',
- description: 'Для спрощення сегментів будуть враховані наступні параметри',
- simplifyShort: 'Видаляти фрагменти менші ніж',
- simplifyAngle: 'Або якщо кут більше ніж',
- simplifyTwoShort: 'та фрагменти меньші ніж',
- },
- },
- 'ru': {
- title: 'Геометрия улиц',
- description: 'Упрощайте и выравнивайте геометрию улиц',
- buttons: {
- A: 'Упростить',
- B: 'Выровнять',
- C: '∡90°',
- },
- settings: {
- title: 'Настройки',
- description: 'Параметры для упрощения геометрии сегмента',
- simplifyShort: 'Если фрагмент короче, чем',
- simplifyAngle: 'Или угол больше чем',
- simplifyTwoShort: 'и фрагменты меньше, чем',
- },
- }
- }
-
- const STYLE =
- 'button.e85.e85-A { background-color: #0f9; margin-right: 2px }' +
- 'button.e85.e85-B { background-color: #09f; color: #fff }' +
- 'button.e85.e85-C { background-color: #f99; margin-left: 2px }' +
- 'button.e85.e85-A:disabled, button.e85.e85-B:disabled { background-color: #ccc }' +
- '.e85 legend { cursor:pointer; font-size: 12px; font-weight: bold; width: auto; text-align: right; border: 0; margin: 0; padding: 0 8px; }' +
- '.e85 fieldset { border: 1px solid #ddd; padding: 8px; }' +
- '.e85 fieldset.e85 div.controls label { white-space: normal; font-weight: normal; line-height: 32px; font-size: 13px; }' +
- '.e85 fieldset.e85 div.controls input[type="number"] { float:right; wight: 32px }' +
- 'p.e85-info { border-top: 1px solid #ccc; color: #777; font-size: x-small; margin-top: 15px; padding-top: 10px; text-align: center; }'
-
- WMEUI.addTranslation(NAME, TRANSLATION)
- WMEUI.addStyle(STYLE)
-
- const BUTTONS = {
- A: {
- title: I18n.t(NAME).buttons.A,
- description: I18n.t(NAME).buttons.A,
- shortcut: '',
- },
- B: {
- title: I18n.t(NAME).buttons.B,
- description: I18n.t(NAME).buttons.B,
- shortcut: '',
- },
- C: {
- title: I18n.t(NAME).buttons.C,
- description: I18n.t(NAME).buttons.C,
- shortcut: '',
- },
- }
-
- // Default settings
- const SETTINGS = {
- simplifyShort: 5,
- simplifyAngle: 176,
- simplifyTwoShort: 50,
- }
-
- let WazeActionAddNode
- let WazeActionMoveNode
- let WazeActionMultiAction
- let WazeActionUpdateSegmentGeometry
-
- class E85 extends WMEBase {
- /**
- * Initial UI elements
- * @param {Object} buttons
- */
- init (buttons) {
- /** @type {WMEUIHelper} */
- this.helper = new WMEUIHelper(this.name)
-
- /** @type {WMEUIHelperTab} */
- this.tab = this.helper.createTab(
- I18n.t(this.name).title,
- {
- image: GM_info.script.icon
- }
- )
-
- // Setup options for script
- let fieldset = this.helper.createFieldset(I18n.t(NAME).settings.title)
- fieldset.addText('description', I18n.t(NAME).settings.description)
- let settings = this.settings.get()
- for (let item in settings) {
- if (settings.hasOwnProperty(item)) {
- fieldset.addNumber(
- 'settings-' + item,
- I18n.t(NAME).settings[item],
- event => this.settings.set([item], event.target.value),
- this.settings.get(item),
- (item === 'simplifyAngle') ? 150 : 0,
- (item === 'simplifyAngle') ? 180 : 200,
- 1
- )
- }
- }
- this.tab.addElement(fieldset)
- this.tab.addText(
- 'info',
- '<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version
- )
-
- // Inject custom HTML to container in the WME interface
- this.tab.inject()
- }
-
- /**
- * Handler for `segment.wme` event
- * @param {jQuery.Event} event
- * @param {HTMLElement} element
- * @param {W.model} model
- * @return {void}
- */
- onSegment (event, element, model) {
- // Skip for blocked roads
- if (model.isLockedByHigherRank() || !model.isGeometryEditable()) {
- return
- }
-
- let panel = this.helper.createPanel(I18n.t(this.name).title)
- let simplifyButton = panel.addButton(
- 'A',
- BUTTONS.A.title,
- BUTTONS.A.description,
- () => this.simplifySegmentGeometry(model),
- BUTTONS.A.shortcut
- )
-
- let straightenButton = panel.addButton(
- 'B',
- BUTTONS.B.title,
- BUTTONS.B.description,
- () => this.straightenSegmentGeometry(model),
- BUTTONS.B.shortcut
- )
- if (model.getGeometry().coordinates.length < 3) {
- simplifyButton.html().disabled = true
- straightenButton.html().disabled = true
- }
-
- const existingFormGroup = element.querySelector('div.form-group.e85');
- if (existingFormGroup) {
- existingFormGroup.replaceWith(panel.html());
- } else {
- element.prepend(panel.html());
- }
- }
-
- /**
- * Handler for `segments.wme` event
- * @param {jQuery.Event} event
- * @param {HTMLElement} element
- * @param {Array} models
- * @return {void}
- */
- onSegments (event, element, models) {
- // Skip for locked roads
- if (models.filter((model) => model.isLockedByHigherRank() || !model.isGeometryEditable()).length > 0) {
- element.querySelector('div.form-group.e85')?.remove()
- return
- }
-
- let panel = this.helper.createPanel(I18n.t(this.name).title)
- let simplifyButton = panel.addButton(
- 'A',
- BUTTONS.A.title,
- BUTTONS.A.description,
- () => this.simplifyStreetGeometry(models),
- BUTTONS.A.shortcut
- )
-
- // Don't straighten multiple components
- let straightenButton = panel.addButton(
- 'B',
- BUTTONS.B.title,
- BUTTONS.B.description,
- () => this.straightenStreetGeometry(models),
- BUTTONS.B.shortcut
- )
-
- let modelWithComponents = models.filter(model => model.getGeometry().coordinates.length > 2)
-
- if (modelWithComponents.length === 0) {
- simplifyButton.html().disabled = true
- }
-
- if (W.selectionManager.getSegmentSelection().multipleConnectedComponents) {
- straightenButton.html().disabled = true
- }
-
- if (!W.selectionManager.getSegmentSelection().multipleConnectedComponents
- && models.length === 2) {
- panel.addButton(
- 'C',
- BUTTONS.C.title,
- BUTTONS.C.description,
- () => this.orthogonalizeStreetGeometry(models[0], models[1]),
- BUTTONS.C.shortcut
- )
- }
-
- const existingFormGroup = element.querySelector('div.form-group.e85');
- if (existingFormGroup) {
- existingFormGroup.replaceWith(panel.html());
- } else {
- element.prepend(panel.html());
- }
- }
-
- /**
- * Remove geometry nodes on the target segment
- * @param {Object} model
- * @return {void}
- */
- simplifySegmentGeometry (model) {
- if (model.getGeometry().coordinates.length < 3) {
- return
- }
-
- this.group('simplify segment geometry')
- this.log('check geometry of the segment with ID ' + model.getID())
- let nodes = []
-
- // calculate angles for every inside point
- for (let i = 0; i < model.getGeometry().coordinates.length - 2; i++) {
- let nodeStart = model.getGeometry().coordinates[i],
- nodeCenter = model.getGeometry().coordinates[i + 1],
- nodeEnd = model.getGeometry().coordinates[i + 2]
-
- nodes[i] = {
- angle: Math.round(this.findAngle(nodeStart, nodeCenter, nodeEnd)),
- start: Math.round(this.findLength(nodeStart, nodeCenter)),
- end: Math.round(this.findLength(nodeCenter, nodeEnd)),
- }
- this.log('point ' + (i+1) + ' : ' + nodes[i].angle + '°, ' + nodes[i].start + 'm, ' + nodes[i].end + 'm')
- }
-
- let removeNodes = []
-
- for (let i = 0; i < nodes.length; i++) {
- let node = nodes[i]
-
- // mark to remove a node with short START segment
- if (node.start < this.settings.get('simplifyShort')) {
- this.log('found too short segment: ' + node.start + 'm')
- removeNodes.push(i+1)
- continue // skip next rule
- }
- // mark to remove a node with short END segment and big ANGLE
- if (node.angle >= this.settings.get('simplifyAngle')
- && node.end < this.settings.get('simplifyShort')) {
- this.log('found too short fragment: ' + node.end + 'm')
- removeNodes.push(i+1)
- i++ // skip next node
- continue // skip next rule
- }
- // mark to remove a node with big angle and short segments
- if (node.angle >= this.settings.get('simplifyAngle')
- && node.start + node.end < this.settings.get('simplifyTwoShort')) {
- this.log(
- 'found point with short fragment: ' + node.start + ' + ' + node.end + ' = ' +
- (node.start + node.end) + 'm and angle equal to ' + node.angle + '°'
- )
- removeNodes.push(i+1)
- // continue // skip next rule
- }
- }
-
- // remove nodes from geometry
- if (removeNodes.length) {
- let newGeometry = { ... model.getGeometry() }
- let coordinates = []
- for (let i = 0; i < newGeometry.coordinates.length; i++) {
- if (removeNodes.indexOf(i) === -1) {
- coordinates.push(newGeometry.coordinates[i])
- }
- }
- newGeometry.coordinates = coordinates
- W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(model, model.getGeometry(), newGeometry))
- }
- this.groupEnd()
- }
-
- /**
- * Calculates the angle (in radians) between two vectors pointing outward from one center
- *
- * @param {Object} start first point
- * @param {Object} center second point
- * @param {Object} end third point
- */
- findAngle (start, center, end) {
- let b = Math.pow(center[0] - start[0], 2) + Math.pow(center[1] - start[1], 2),
- a = Math.pow(center[0] - end[0], 2) + Math.pow(center[1] - end[1], 2),
- c = Math.pow(end[0] - start[0], 2) + Math.pow(end[1] - start[1], 2)
- return Math.acos((a + b - c) / Math.sqrt(4 * a * b)) * (180 / Math.PI)
- }
-
- /**
- * Get the length of the line by point coordinates
- * @param {Array<number,number>} start point
- * @param {Array<number,number>} end point
- * @return {Number} length in meters
- */
- findLength (start, end) {
- return distance(start[0], start[1], end[0], end[1])
- }
-
- /**
- * Remove geometry nodes on the target segments
- * @param {Array} models
- * @return {void}
- */
- simplifyStreetGeometry (models) {
- this.group('simplify street geometry')
- for (let i = 0; i < models.length; i++) {
- this.simplifySegmentGeometry(models[i])
- }
- this.groupEnd()
- }
-
- /**
- * Aligns the segments into a straight line by moving the intermediate
- * nodes to the intersection points of the perpendiculars with
- * the calculated line passing through the start and end nodes of the selection.
- *
- * A, B, and C are the parameters of the calculated line equation:
- * Ax + By + C = 0
- *
- * @param {Array} models
- * @return {void}
- */
- straightenStreetGeometry (models) {
- this.group('straighten street geometry')
- this.log('calculating the formula for the straight line')
-
- let segmentSelection = W.selectionManager.getSegmentSelection()
-
- if (segmentSelection.multipleConnectedComponents) {
- this.log('don\'t try to straighten multiple segments without connection')
- }
-
- let
- allNodeIds = [], // all nodes for selected segments
- dupNodeIds = [], // only nodes inside connections
- virtualNodes = [] // virtual nodes of segments
-
- models.forEach(segment => {
- this.log('straighten segment #' + segment.getID())
-
- // simplify segment to straight
- this.straightenSegmentGeometry(segment)
-
- // collect the nodes
- allNodeIds.push(segment.getFromNode().getID())
- allNodeIds.push(segment.getToNode().getID())
- virtualNodes = virtualNodes.concat(segment.getVirtualNodes())
- })
-
- if (virtualNodes.length ) {
- this.log('⚠️ virtual nodes are present, please disconnect all trails and rails from the segments and try again')
-
- // doesn't work, but why? what is wrong with this code?
- // virtualNodes.forEach(node => {
- // let element = document.getElementById(node.getOLGeometry.id)
- // element.setAttribute("fill","#dd7700")
- //
- // element.addEventListener("click", () => {
- // element.setAttribute("fill","#00ece3")
- // });
- // })
-
- return
- }
-
- allNodeIds.forEach((nodeId, idx) => {
- if (allNodeIds.indexOf(nodeId, idx + 1) > -1) {
- if (!dupNodeIds.includes(nodeId))
- dupNodeIds.push(nodeId);
- }
- });
-
- let distinctNodeIds = [...new Set(allNodeIds)];
- let endPointNodeIds = distinctNodeIds.filter((nodeId) => !dupNodeIds.includes(nodeId));
- let endPointNodes = W.model.nodes.getByIds(endPointNodeIds),
- endPointNode1Geo = endPointNodes[0].getGeometry().coordinates,
- endPointNode2Geo = endPointNodes[1].getGeometry().coordinates
-
- const a = endPointNode2Geo[1] - endPointNode1Geo[1],
- b = endPointNode1Geo[0] - endPointNode2Geo[0],
- c = endPointNode2Geo[0] * endPointNode1Geo[1] - endPointNode1Geo[0] * endPointNode2Geo[1];
-
- dupNodeIds.forEach((nodeId) => {
- const node = W.model.nodes.getObjectById(nodeId),
- nodeCoordinates = node.getGeometry().coordinates;
- const d = nodeCoordinates[1] * a - nodeCoordinates[0] * b,
- newCoordinates = getIntersectCoordinates(a, b, c, d);
-
- this.log('move node #' + nodeId + ' to [' + newCoordinates[0] + ';' + newCoordinates[1] + ']')
- this.moveNode(node, newCoordinates)
- });
-
- // I don't understand why doesn't it work, in the WME all looks good, but it fails when try to save changes
- // virtualNodes.forEach((node) => {
- // const nodeCoordinates = node.getGeometry().coordinates;
- // const d = nodeCoordinates[1] * a - nodeCoordinates[0] * b,
- // newCoordinates = getIntersectCoordinates(a, b, c, d);
- //
- // this.log('move node #' + node.getID() + ' to [' + newCoordinates[0] + ';' + newCoordinates[1] + ']')
- // this.moveNode(node, newCoordinates)
- // });
-
- this.groupEnd()
- }
-
- /**
- * Orthogonalize two segments
- * This method move the node to new point
- *
- * @param {Object} segment1
- * @param {Object} segment2
- * @return {void}
- */
- orthogonalizeStreetGeometry (segment1, segment2) {
- this.log('orthogonalize street geometry')
-
- if (segment1.getType() !== 'segment'
- || segment2.getType() !== 'segment') {
- this.log('only segments must be selected')
- return
- }
-
- /**
- * Extract coordinates from components
- * @param {Object} segment
- * @param {'first'|'second'|'last-but-one'|'last'} position
- * @return {*[]}
- */
- function getCoordinatesFromComponent(segment, position) {
- let pos = 0
- switch (position) {
- case 'first':
- pos = 0
- break
- case 'second':
- pos = 1
- break
- case 'last-but-one':
- pos = segment.getOLGeometry().components.length - 2
- break
- case 'last':
- pos = segment.getOLGeometry().components.length - 1
- break
- }
- return [
- segment.getOLGeometry().components[pos].x,
- segment.getOLGeometry().components[pos].y,
- ]
- }
-
- let A, B, C, commonNode
-
- if (segment1.getToNode().getID() === segment2.getFromNode().getID()) {
- // A → B → C
- commonNode = segment1.getToNode()
- A = getCoordinatesFromComponent(segment1, 'last-but-one')
- B = getCoordinatesFromComponent(segment1, 'last')
- C = getCoordinatesFromComponent(segment2, 'second')
- } else if (segment1.getFromNode().getID() === segment2.getFromNode().getID()) {
- // B ← A → C
- commonNode = segment1.getFromNode()
- A = getCoordinatesFromComponent(segment1, 'second')
- B = getCoordinatesFromComponent(segment1, 'first')
- C = getCoordinatesFromComponent(segment2, 'second')
- } else if (segment1.getToNode().getID() === segment2.getToNode().getID()) {
- // A → B ← C
- commonNode = segment1.getToNode()
- A = getCoordinatesFromComponent(segment1, 'last-but-one')
- B = getCoordinatesFromComponent(segment1, 'last')
- C = getCoordinatesFromComponent(segment2, 'last-but-one')
- } else if (segment1.getFromNode().getID() === segment2.getToNode().getID()) {
- // B ← A ← C
- commonNode = segment1.getFromNode()
- A = getCoordinatesFromComponent(segment1, 'second')
- B = getCoordinatesFromComponent(segment1, 'first')
- C = getCoordinatesFromComponent(segment2, 'last-but-one')
- }
-
- if (!commonNode) {
- this.log('segments does not have common node')
- return
- }
-
- this.log('common node coords [' + B[0] + ';' + B[1] + ']')
-
- // Coordinates of points A, B and C
- // First selected segment uses it as line for calculation
- let intersection = findIntersection(A, B, C)
-
- // Uses OpenLayers with convertation, because
- intersection = W.userscripts.toGeoJSONGeometry(new OpenLayers.Geometry.Point( ...intersection ))
-
- this.log('point of the intersection is [' + intersection[0] + ', ' + intersection[1] +']')
-
- this.moveNode(commonNode, intersection.coordinates)
- }
-
- /**
- * Straighten up segment, remove all geometry nodes except first and last
- * @param {Object} segment
- */
- straightenSegmentGeometry (segment) {
- this.group('straighten segment geometry')
- if (segment.getGeometry().coordinates.length > 2) {
- let multiAction = new WazeActionMultiAction()
- let newGeometry = structuredClone(segment.attributes.geoJSONGeometry)
- // just left the first and last elements
- newGeometry.coordinates.splice(1, newGeometry.coordinates.length - 2)
- // W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(segment, segment.getGeometry(), newGeometry))
- let updateSegmentGeometry = new WazeActionUpdateSegmentGeometry(segment, segment.attributes.geoJSONGeometry, newGeometry)
- updateSegmentGeometry.generateDescription();
- multiAction.doSubAction(W.model, updateSegmentGeometry);
- W.model.actionManager.add(multiAction);
- }
- }
-
- /**
- * Move node to new position
- * @param {Object} node target
- * @param {Array<2>} coords of the new position, array of the wo elements
- */
- moveNode (node, coords) {
- let nodeGeometry = node.getGeometry()
- nodeGeometry.coordinates = coords
-
- let connectedSegObjs = {}
- let emptyObj = {}
-
- node.getSegmentIds().forEach((id) => {
- connectedSegObjs[id] = { ...W.model.segments.getObjectById(id).getGeometry() }
- })
-
- W.model.actionManager.add(
- new WazeActionMoveNode(
- node,
- node.getGeometry(),
- nodeGeometry,
- connectedSegObjs,
- emptyObj
- )
- )
- }
- }
-
- /**
- * @param {Array<number,number>} A point of line
- * @param {Array<number,number>} B point of line
- * @param {Array<number,number>} C point
- * @return {(number|*)[]}
- */
- function findIntersection(A, B, C) {
- // Функция для вычисления углового коэффициента прямой
- function slope(point1, point2) {
- return (point2[1] - point1[1]) / (point2[0] - point1[0]);
- }
-
- // Функция для вычисления c (точка пересечения с осью Y) для уравнения прямой
- function intercept(point, slope) {
- return point[1] - slope * point[0];
- }
-
- // Вычисляем угловой коэффициент для прямой AB
- let mAB = slope(A, B);
- // Вычисляем c для прямой AB
- let cAB = intercept(A, mAB);
-
- // Для перпендикуляра угловой коэффициент будет обратным и противоположным
- let mPerpendicular = -1 / mAB;
- // Вычисляем c для перпендикулярной прямой, проходящей через C
- let cPerpendicular = intercept(C, mPerpendicular);
-
- // Находим точку пересечения прямых
- let x = (cPerpendicular - cAB) / (mAB - mPerpendicular);
- let y = mAB * x + cAB;
-
- return [x, y];
- }
-
-
- /**
- * Find intersection point
- * @param {Number} A
- * @param {Number} B
- * @param {Number} C
- * @param {Number} D
- * @return {Number[]}
- */
- function getIntersectCoordinates (A, B, C, D) {
- // http://rsdn.ru/forum/alg/2589531.hot
- let r = [2]
- r[1] = -1.0 * (C * B - A * D) / (A * A + B * B)
- r[0] = (-r[1] * (B + A) - C + D) / (A - B)
-
- return r
- }
-
- /**
- * Detect direction
- * @param {Number} A
- * @param {Number} B
- * @return {Number}
- */
- function getDeltaDirect (A, B) {
- if (A < B) {
- return 1.0
- } else if (A > B) {
- return -1.0
- }
-
- return 0.0
- }
-
- /**
- * Calculate the approximate distance between two coordinates (lat/lon)
- *
- * © Chris Veness, MIT-licensed,
- * http://www.movable-type.co.uk/scripts/latlong.html#equirectangular
- *
- * @param {number} λ1 first point latitude
- * @param {number} φ1 first point longitude
- * @param {number} λ2 second point latitude
- * @param {number} φ2 second point longitude
- */
- function distance (λ1, φ1, λ2, φ2) {
- let R = 6371000;
- let Δλ = (λ2 - λ1) * Math.PI / 180;
- φ1 = φ1 * Math.PI / 180;
- φ2 = φ2 * Math.PI / 180;
- let x = Δλ * Math.cos((φ1+φ2)/2);
- let y = (φ2-φ1);
- let d = Math.sqrt(x*x + y*y);
- return R * d;
- };
-
- $(document).on('bootstrap.wme', () => {
-
- WazeActionAddNode = require('Waze/Action/AddNode')
- WazeActionMoveNode = require('Waze/Action/MoveNode')
- WazeActionMultiAction = require('Waze/Action/MultiAction');
- WazeActionUpdateSegmentGeometry = require('Waze/Action/UpdateSegmentGeometry')
-
- let Instance = new E85(NAME, SETTINGS)
- Instance.init(BUTTONS)
-
- // setup name for a shortcut section
- WMEUIShortcut.setGroupTitle(NAME, I18n.t(NAME).title)
- })
- })()