]> git.openstreetmap.org Git - rails.git/blobdiff - vendor/assets/iD/iD.js
Update leaflet.locate (fixes #438)
[rails.git] / vendor / assets / iD / iD.js
index 1f2cc30ac3cb12bbce52456c0e5cfc1be5540234..a663c72997d0caa03a01d6b8d1e2e20e08da9fb5 100644 (file)
@@ -2789,7 +2789,7 @@ function d3_geo_clipView(x0, y0, x1, y1) {
             }
             listener.point(b[0], b[1]);
             if (!v) listener.lineEnd();
-          } else {
+          } else if (v) {
             listener.lineStart();
             listener.point(x, y);
           }
@@ -10935,7 +10935,7 @@ module.exports = function(o) {
     return oauth;
 };
 
-},{"store":2,"ohauth":3}],2:[function(require,module,exports){
+},{"ohauth":2,"store":3}],3:[function(require,module,exports){
 /* Copyright (c) 2010-2012 Marcus Westin
  *
  * Permission is hereby granted, free of charge, to any person obtaining a copy
@@ -11112,25 +11112,26 @@ module.exports = function(o) {
        else { this.store = store }
 })();
 
-},{}],3:[function(require,module,exports){
+},{}],2:[function(require,module,exports){
 'use strict';
 
 var hashes = require('jshashes'),
+    xtend = require('xtend'),
     sha1 = new hashes.SHA1();
 
 var ohauth = {};
 
 ohauth.qsString = function(obj) {
     return Object.keys(obj).sort().map(function(key) {
-        return encodeURIComponent(key) + '=' +
-            encodeURIComponent(obj[key]);
+        return ohauth.percentEncode(key) + '=' +
+            ohauth.percentEncode(obj[key]);
     }).join('&');
 };
 
 ohauth.stringQs = function(str) {
     return str.split('&').reduce(function(obj, pair){
         var parts = pair.split('=');
-        obj[parts[0]] = (null === parts[1]) ?
+        obj[decodeURIComponent(parts[0])] = (null === parts[1]) ?
             '' : decodeURIComponent(parts[1]);
         return obj;
     }, {});
@@ -11195,9 +11196,62 @@ ohauth.signature = function(oauth_secret, token_secret, baseString) {
         baseString);
 };
 
+/**
+ * Takes an options object for configuration (consumer_key,
+ * consumer_secret, version, signature_method, token) and returns a
+ * function that generates the Authorization header for given data.
+ *
+ * The returned function takes these parameters:
+ * - method: GET/POST/...
+ * - uri: full URI with protocol, port, path and query string
+ * - extra_params: any extra parameters (that are passed in the POST data),
+ *   can be an object or a from-urlencoded string.
+ *
+ * Returned function returns full OAuth header with "OAuth" string in it.
+ */
+
+ohauth.headerGenerator = function(options) {
+    options = options || {};
+    var consumer_key = options.consumer_key || '',
+        consumer_secret = options.consumer_secret || '',
+        signature_method = options.signature_method || 'HMAC-SHA1',
+        version = options.version || '1.0',
+        token = options.token || '';
+
+    return function(method, uri, extra_params) {
+        method = method.toUpperCase();
+        if (typeof extra_params === 'string' && extra_params.length > 0) {
+            extra_params = ohauth.stringQs(extra_params);
+        }
+
+        var uri_parts = uri.split('?', 2),
+        base_uri = uri_parts[0];
+
+        var query_params = uri_parts.length === 2 ?
+            ohauth.stringQs(uri_parts[1]) : {};
+
+        var oauth_params = {
+            oauth_consumer_key: consumer_key,
+            oauth_signature_method: signature_method,
+            oauth_version: version,
+            oauth_timestamp: ohauth.timestamp(),
+            oauth_nonce: ohauth.nonce()
+        };
+
+        if (token) oauth_params.oauth_token = token;
+
+        var all_params = xtend({}, oauth_params, query_params, extra_params),
+            base_str = ohauth.baseString(method, base_uri, all_params);
+
+        oauth_params.oauth_signature = ohauth.signature(consumer_secret, token, base_str);
+
+        return 'OAuth ' + ohauth.authHeader(oauth_params);
+    };
+};
+
 module.exports = ohauth;
 
-},{"jshashes":4}],4:[function(require,module,exports){
+},{"jshashes":4,"xtend":5}],4:[function(require,module,exports){
 (function(global){/**\r
  * jsHashes - A fast and independent hashing library pure JavaScript implemented (ES5 compliant) for both server and client side\r
  * \r
@@ -12819,9 +12873,49 @@ module.exports = ohauth;
   }( this ));\r
 }()); // IIFE
 })(window)
+},{}],5:[function(require,module,exports){
+var Keys = Object.keys || objectKeys
+
+module.exports = extend
+
+function extend() {
+    var target = {}
+
+    for (var i = 0; i < arguments.length; i++) {
+        var source = arguments[i]
+
+        if (!isObject(source)) {
+            continue
+        }
+
+        var keys = Keys(source)
+
+        for (var j = 0; j < keys.length; j++) {
+            var name = keys[j]
+            target[name] = source[name]
+        }
+    }
+
+    return target
+}
+
+function objectKeys(obj) {
+    var keys = []
+    for (var k in obj) {
+        keys.push(k)
+    }
+    return keys
+}
+
+function isObject(obj) {
+    return obj !== null && typeof obj === "object"
+}
+
 },{}]},{},[1])(1)
 });
-;/******************************************************************************
+;
+
+/******************************************************************************
        rtree.js - General-Purpose Non-Recursive Javascript R-Tree Library
        Version 0.6.2, December 5st 2009
 
@@ -14782,8 +14876,8 @@ if (typeof exports === 'object') {
 (function () {
 'use strict';
 window.iD = function () {
-    locale.en = iD.data.en;
-    locale.current('en');
+    window.locale.en = iD.data.en;
+    window.locale.current('en');
 
     var context = {},
         storage;
@@ -14805,7 +14899,13 @@ window.iD = function () {
         container,
         ui = iD.ui(context),
         map = iD.Map(context),
-        connection = iD.Connection();
+        connection = iD.Connection(),
+        locale = iD.detect().locale,
+        localePath;
+
+    if (locale && iD.data.locales.indexOf(locale) === -1) {
+        locale = locale.split('-')[0];
+    }
 
     connection.on('load.context', function loadContext(err, result) {
         history.merge(result);
@@ -14816,20 +14916,21 @@ window.iD = function () {
         return context;
     };
 
+    context.locale = function(_, path) {
+        locale = _;
+        localePath = path;
+        return context;
+    };
+
     context.ui = function() {
         return function(container) {
             context.container(container);
 
-            var detectedLocale = iD.detect().locale;
-
-            if (iD.data.locales.indexOf(detectedLocale) === -1) {
-                detectedLocale = detectedLocale.split('-')[0];
-            }
-
-            if (detectedLocale !== 'en' && iD.data.locales.indexOf(detectedLocale) !== -1) {
-                d3.json(context.assetPath() + 'locales/' + detectedLocale + '.json', function(err, result) {
-                    locale[detectedLocale] = result;
-                    locale.current(detectedLocale);
+            if (locale && locale !== 'en' && iD.data.locales.indexOf(locale) !== -1) {
+                localePath = localePath || context.assetPath() + 'locales/' + locale + '.json';
+                d3.json(localePath, function(err, result) {
+                    window.locale[locale] = result;
+                    window.locale.current(locale);
                     container.call(ui);
                 });
             } else {
@@ -14864,6 +14965,10 @@ window.iD = function () {
         return history.graph().entity(id);
     };
 
+    context.childNodes = function(way) {
+        return history.graph().childNodes(way);
+    };
+
     context.geometry = function(id) {
         return context.entity(id).geometry(history.graph());
     };
@@ -14996,7 +15101,7 @@ window.iD = function () {
     return d3.rebind(context, dispatch, 'on');
 };
 
-iD.version = '0.0.0-beta1';
+iD.version = '1.0.1';
 
 iD.detect = function() {
     var browser = {};
@@ -15207,7 +15312,7 @@ iD.wikipedia  = function() {
             }), function(d) {
                 var list = d.query.pages[Object.keys(d.query.pages)[0]],
                     translations = {};
-                if (list) {
+                if (list && list.langlinks) {
                     list.langlinks.forEach(function(d) {
                         translations[d.lang] = d['*'];
                     });
@@ -15359,24 +15464,48 @@ iD.geo.dist = function(a, b) {
     return Math.sqrt((x * x) + (y * y));
 };
 
-iD.geo.chooseIndex = function(way, point, context) {
+// Choose the edge with the minimal distance from `point` to its orthogonal
+// projection onto that edge, if such a projection exists, or the distance to
+// the closest vertex on that edge. Returns an object with the `index` of the
+// chosen edge, the chosen `loc` on that edge, and the `distance` to to it.
+iD.geo.chooseEdge = function(nodes, point, projection) {
     var dist = iD.geo.dist,
-        graph = context.graph(),
-        nodes = graph.childNodes(way),
-        projNodes = nodes.map(function(n) { return context.projection(n.loc); });
+        points = nodes.map(function(n) { return projection(n.loc); }),
+        min = Infinity,
+        idx, loc;
+
+    function dot(p, q) {
+        return p[0] * q[0] + p[1] * q[1];
+    }
+
+    for (var i = 0; i < points.length - 1; i++) {
+        var o = points[i],
+            s = [points[i + 1][0] - o[0],
+                 points[i + 1][1] - o[1]],
+            v = [point[0] - o[0],
+                 point[1] - o[1]],
+            proj = dot(v, s) / dot(s, s),
+            p;
+
+        if (proj < 0) {
+            p = o;
+        } else if (proj > 1) {
+            p = points[i + 1];
+        } else {
+            p = [o[0] + proj * s[0], o[1] + proj * s[1]];
+        }
 
-    for (var i = 0, changes = []; i < projNodes.length - 1; i++) {
-        changes[i] =
-            (dist(projNodes[i], point) + dist(point, projNodes[i + 1])) /
-            dist(projNodes[i], projNodes[i + 1]);
+        var d = dist(p, point);
+        if (d < min) {
+            min = d;
+            idx = i + 1;
+            loc = projection.invert(p);
+        }
     }
 
-    var idx = _.indexOf(changes, _.min(changes)),
-        ratio = dist(projNodes[idx], point) / dist(projNodes[idx], projNodes[idx + 1]),
-        loc = iD.geo.interp(nodes[idx].loc, nodes[idx + 1].loc, ratio);
-
     return {
-        index: idx + 1,
+        index: idx,
+        distance: min,
         loc: loc
     };
 };
@@ -15474,6 +15603,56 @@ _.extend(iD.geo.Extent.prototype, {
                 [this[1][0] + dLon, this[1][1] + dLat]);
     }
 });
+// For fixing up rendering of multipolygons with tags on the outer member.
+// https://github.com/systemed/iD/issues/613
+iD.geo.isSimpleMultipolygonOuterMember = function(entity, graph) {
+    if (entity.type !== 'way')
+        return false;
+
+    var parents = graph.parentRelations(entity);
+    if (parents.length !== 1)
+        return false;
+
+    var parent = parents[0];
+    if (!parent.isMultipolygon() || Object.keys(parent.tags).length > 1)
+        return false;
+
+    var members = parent.members, member;
+    for (var i = 0; i < members.length; i++) {
+        member = members[i];
+        if (member.id === entity.id && member.role && member.role !== 'outer')
+            return false; // Not outer member
+        if (member.id !== entity.id && (!member.role || member.role === 'outer'))
+            return false; // Not a simple multipolygon
+    }
+
+    return parent;
+};
+
+iD.geo.simpleMultipolygonOuterMember = function(entity, graph) {
+    if (entity.type !== 'way')
+        return false;
+
+    var parents = graph.parentRelations(entity);
+    if (parents.length !== 1)
+        return false;
+
+    var parent = parents[0];
+    if (!parent.isMultipolygon() || Object.keys(parent.tags).length > 1)
+        return false;
+
+    var members = parent.members, member, outerMember;
+    for (var i = 0; i < members.length; i++) {
+        member = members[i];
+        if (!member.role || member.role === 'outer') {
+            if (outerMember)
+                return false; // Not a simple multipolygon
+            outerMember = member;
+        }
+    }
+
+    return outerMember && graph.hasEntity(outerMember.id);
+};
 iD.actions = {};
 iD.actions.AddEntity = function(way) {
     return function(graph) {
@@ -15840,6 +16019,15 @@ iD.actions.Join = function(ids) {
             b = graph.entity(idB),
             nodes;
 
+        // Prefer to keep an existing way.
+        if (a.isNew() && !b.isNew()) {
+            var tmp = a;
+            a = b;
+            b = tmp;
+            idA = a.id;
+            idB = b.id;
+        }
+
         if (a.first() === b.first()) {
             // a <-- b ==> c
             // Expected result:
@@ -16669,7 +16857,7 @@ iD.behavior.Draw = function(context) {
     function click() {
         var d = datum();
         if (d.type === 'way') {
-            var choice = iD.geo.chooseIndex(d, d3.mouse(context.surface().node()), context),
+            var choice = iD.geo.chooseEdge(context.childNodes(d), d3.mouse(context.surface().node()), context.projection),
                 edge = [d.nodes[choice.index - 1], d.nodes[choice.index]];
             event.clickWay(choice.loc, edge);
 
@@ -16770,7 +16958,7 @@ iD.behavior.DrawWay = function(context, wayId, index, mode, baseGraph) {
         } else if (datum.type === 'node') {
             loc = datum.loc;
         } else if (datum.type === 'way') {
-            loc = iD.geo.chooseIndex(datum, d3.mouse(context.surface().node()), context).loc;
+            loc = iD.geo.chooseEdge(context.childNodes(datum), d3.mouse(context.surface().node()), context.projection).loc;
         }
 
         context.replace(iD.actions.MoveNode(end.id, loc));
@@ -17565,7 +17753,7 @@ iD.modes.DragNode = function(context) {
         if (d.type === 'node' && d.id !== entity.id) {
             loc = d.loc;
         } else if (d.type === 'way') {
-            loc = iD.geo.chooseIndex(d, d3.mouse(context.surface().node()), context).loc;
+            loc = iD.geo.chooseEdge(context.childNodes(d), d3.mouse(context.surface().node()), context.projection).loc;
         }
 
         context.replace(
@@ -17579,7 +17767,7 @@ iD.modes.DragNode = function(context) {
         var d = datum();
 
         if (d.type === 'way') {
-            var choice = iD.geo.chooseIndex(d, d3.mouse(context.surface().node()), context);
+            var choice = iD.geo.chooseEdge(context.childNodes(d), d3.mouse(context.surface().node()), context.projection);
             context.replace(
                 iD.actions.AddMidpoint({ loc: choice.loc, edge: [d.nodes[choice.index - 1], d.nodes[choice.index]] }, entity),
                 connectAnnotation(d));
@@ -17603,9 +17791,13 @@ iD.modes.DragNode = function(context) {
 
         var parentWays = _.pluck(context.graph().parentWays(entity), 'id');
 
-        context.enter(
-            iD.modes.Select(context, parentWays)
-                .suppressMenu(true));
+        if (parentWays.length) {
+            context.enter(
+                iD.modes.Select(context, parentWays)
+                    .suppressMenu(true));
+        } else {
+            context.enter(iD.modes.Browse(context));
+        }
     }
 
     function cancel() {
@@ -18054,8 +18246,8 @@ iD.modes.Select = function(context, selection) {
                 datum = target.datum();
 
             if (datum instanceof iD.Way && !target.classed('fill')) {
-                var choice = iD.geo.chooseIndex(datum,
-                        d3.mouse(context.surface().node()), context),
+                var choice = iD.geo.chooseEdge(context.childNodes(datum),
+                        d3.mouse(context.surface().node()), context.projection),
                     node = iD.Node();
 
                 var prev = datum.nodes[choice.index - 1],
@@ -20074,6 +20266,8 @@ _.extend(iD.Way.prototype, {
 
     isOneWay: function() {
         return this.tags.oneway === 'yes' ||
+            this.tags.oneway === '1' ||
+            this.tags.oneway === '-1' ||
             this.tags.waterway === 'river' ||
             this.tags.waterway === 'stream' ||
             this.tags.junction === 'roundabout';
@@ -20227,7 +20421,7 @@ iD.Way.areaKeys = {
 };
 iD.Background = function(backgroundType) {
 
-    backgroundType = backgroundType || 'layer';
+    backgroundType = backgroundType || 'background';
 
     var tileSize = 256,
         tile = d3.geo.tile(),
@@ -20276,6 +20470,12 @@ iD.Background = function(backgroundType) {
 
     // Update tiles based on current state of `projection`.
     function background(selection) {
+        var layer = selection.selectAll('.' + backgroundType + '-layer')
+            .data([background]);
+
+        layer.enter().append('div')
+            .attr('class', 'layer-layer ' + backgroundType + '-layer', true);
+
         tile.scale(projection.scale() * 2 * Math.PI)
             .translate(projection.translate());
 
@@ -20285,7 +20485,7 @@ iD.Background = function(backgroundType) {
 
         z = Math.max(Math.log(projection.scale() * 2 * Math.PI) / Math.log(2) - 8, 0);
 
-        render(selection);
+        render(layer);
     }
 
     // Derive the tiles onscreen, remove those offscreen and position them.
@@ -20510,31 +20710,25 @@ iD.BackgroundSource.Custom = function() {
 
 iD.BackgroundSource.Custom.data = { 'name': 'Custom' };
 iD.LocalGpx = function(context) {
-    var tileSize = 256,
-        projection,
+    var projection,
         gj = {},
         enable = true,
         size = [0, 0],
-        transformProp = iD.util.prefixCSSProperty('Transform'),
-        path = d3.geo.path().projection(projection),
-        source = d3.functor('');
+        svg;
 
     function render(selection) {
+        svg = selection.selectAll('svg')
+            .data([render]);
 
-        path.projection(projection);
-
-        var surf = selection.selectAll('svg')
-            .data(enable ? [gj] : []);
-
-        surf.exit().remove();
-
-        surf.enter()
+        svg.enter()
             .append('svg')
-            .style('position', 'absolute');
+            .attr('class', 'layer-layer gpx-layer');
+
+        svg.style('display', enable ? 'block' : 'none');
 
-        var paths = surf
+        var paths = svg
             .selectAll('path')
-            .data(function(d) { return [d]; });
+            .data([gj]);
 
         paths
             .enter()
@@ -20542,7 +20736,7 @@ iD.LocalGpx = function(context) {
             .attr('class', 'gpx');
 
         paths
-            .attr('d', path);
+            .attr('d', d3.geo.path().projection(projection));
     }
 
     function toDom(x) {
@@ -20568,8 +20762,8 @@ iD.LocalGpx = function(context) {
     };
 
     render.size = function(_) {
-        if (!arguments.length) return size;
-        size = _;
+        if (!arguments.length) return svg.size();
+        svg.size(_);
         return render;
     };
 
@@ -20616,12 +20810,13 @@ iD.Map = function(context) {
             .on('zoom', zoomPan),
         dblclickEnabled = true,
         transformStart,
+        transformed = false,
         minzoom = 0,
         layers = [
             iD.Background().projection(projection),
             iD.LocalGpx(context).projection(projection),
             iD.Background('overlay').projection(projection)
-            ],
+        ],
         transformProp = iD.util.prefixCSSProperty('Transform'),
         points = iD.svg.Points(roundedProjection, context),
         vertices = iD.svg.Vertices(roundedProjection, context),
@@ -20630,7 +20825,7 @@ iD.Map = function(context) {
         midpoints = iD.svg.Midpoints(roundedProjection, context),
         labels = iD.svg.Labels(roundedProjection, context),
         tail = iD.ui.Tail(),
-        supersurface, surface, layergroup;
+        supersurface, surface;
 
     function map(selection) {
         context.history()
@@ -20645,10 +20840,16 @@ iD.Map = function(context) {
         supersurface = selection.append('div')
             .attr('id', 'supersurface');
 
-        layergroup = supersurface.append('div')
-            .attr('id', 'layer-g');
+        layers.forEach(function(layer) {
+            supersurface.call(layer);
+        });
+
+        // Need a wrapper div because Opera can't cope with an absolutely positioned
+        // SVG element: http://bl.ocks.org/jfirebaugh/6fbfbd922552bf776c16
+        var dataLayer = supersurface.append('div')
+            .attr('class', 'layer-layer layer-data');
 
-        surface = supersurface.append('svg')
+        surface = dataLayer.append('svg')
             .on('mousedown.zoom', function() {
                 if (d3.event.button == 2) {
                     d3.event.stopPropagation();
@@ -20661,14 +20862,14 @@ iD.Map = function(context) {
             .call(iD.svg.Surface(context));
 
         surface.on('mouseover.vertices', function() {
-            if (map.editable() && !isTransformed()) {
+            if (map.editable() && !transformed) {
                 var hover = d3.event.target.__data__;
                 surface.call(vertices.drawHover, context.graph(), hover, map.zoom());
             }
         });
 
         surface.on('mouseout.vertices', function() {
-            if (map.editable() && !isTransformed()) {
+            if (map.editable() && !transformed) {
                 var hover = d3.event.relatedTarget && d3.event.relatedTarget.__data__;
                 surface.call(vertices.drawHover, context.graph(), hover, map.zoom());
             }
@@ -20676,7 +20877,6 @@ iD.Map = function(context) {
 
         map.size(selection.size());
         map.surface = surface;
-        map.layersurface = layergroup;
 
         labels.supersurface(supersurface);
 
@@ -20774,20 +20974,17 @@ iD.Map = function(context) {
                 'translate(' + tX + 'px,' + tY + 'px)' :
                 'translate3d(' + tX + 'px,' + tY + 'px, 0)');
 
+        transformed = true;
         supersurface.style(transformProp, transform);
         queueRedraw();
 
         dispatch.move(map);
     }
 
-    function isTransformed() {
-        var prop = supersurface.style(transformProp);
-        return prop && prop !== 'none';
-    }
-
     function resetTransform() {
-        if (!isTransformed()) return false;
+        if (!transformed) return false;
         supersurface.style(transformProp, '');
+        transformed = false;
         return true;
     }
 
@@ -20810,18 +21007,9 @@ iD.Map = function(context) {
         }
 
         if (!difference) {
-            var sel = layergroup
-                .selectAll('.layer-layer')
-                .data(layers);
-
-            sel.exit().remove();
-
-            sel.enter().append('div')
-                .attr('class', 'layer-layer');
-
-            sel.each(function(layer) {
-                    d3.select(this).call(layer);
-                });
+            layers.forEach(function(layer) {
+                supersurface.call(layer);
+            });
         }
 
         if (map.editable()) {
@@ -20917,8 +21105,8 @@ iD.Map = function(context) {
         var center = map.center();
         dimensions = _;
         surface.size(dimensions);
-        layers.map(function(l) {
-            l.size(dimensions);
+        layers.forEach(function(layer) {
+            layer.size(dimensions);
         });
         projection.clipExtent([[0, 0], dimensions]);
         setCenter(center);
@@ -21073,13 +21261,16 @@ iD.svg = {
                 b,
                 i = 0,
                 offset = dt,
-                segments = [];
+                segments = [],
+                coordinates = graph.childNodes(entity).map(function(n) {
+                    return n.loc;
+                });
+
+            if (entity.tags.oneway === '-1') coordinates.reverse();
 
             d3.geo.stream({
                 type: 'LineString',
-                coordinates: graph.childNodes(entity).map(function(n) {
-                    return n.loc;
-                })
+                coordinates: coordinates
             }, projection.stream({
                 lineStart: function() {},
                 lineEnd: function() {},
@@ -21137,32 +21328,6 @@ iD.svg = {
     }
 };
 iD.svg.Areas = function(projection) {
-    // For fixing up rendering of multipolygons with tags on the outer member.
-    // https://github.com/systemed/iD/issues/613
-    function isSimpleMultipolygonOuterMember(entity, graph) {
-        if (entity.type !== 'way')
-            return false;
-
-        var parents = graph.parentRelations(entity);
-        if (parents.length !== 1)
-            return false;
-
-        var parent = parents[0];
-        if (!parent.isMultipolygon() || Object.keys(parent.tags).length > 1)
-            return false;
-
-        var members = parent.members, member;
-        for (var i = 0; i < members.length; i++) {
-            member = members[i];
-            if (member.id === entity.id && member.role && member.role !== 'outer')
-                return false; // Not outer member
-            if (member.id !== entity.id && (!member.role || member.role === 'outer'))
-                return false; // Not a simple multipolygon
-        }
-
-        return parent;
-    }
-
     // Patterns only work in Firefox when set directly on element
     var patterns = {
         wetland: 'wetland',
@@ -21200,7 +21365,7 @@ iD.svg.Areas = function(projection) {
             var entity = entities[i];
             if (entity.geometry(graph) !== 'area') continue;
 
-            if (multipolygon = isSimpleMultipolygonOuterMember(entity, graph)) {
+            if (multipolygon = iD.geo.isSimpleMultipolygonOuterMember(entity, graph)) {
                 areas[multipolygon.id] = {
                     entity: multipolygon.mergeTags(entity.tags),
                     area: Math.abs(path.area(entity.asGeoJSON(graph, true)))
@@ -21760,33 +21925,6 @@ iD.svg.Lines = function(projection) {
         return as - bs;
     }
 
-    // For fixing up rendering of multipolygons with tags on the outer member.
-    // https://github.com/systemed/iD/issues/613
-    function simpleMultipolygonOuterMember(entity, graph) {
-        if (entity.type !== 'way')
-            return false;
-
-        var parents = graph.parentRelations(entity);
-        if (parents.length !== 1)
-            return false;
-
-        var parent = parents[0];
-        if (!parent.isMultipolygon() || Object.keys(parent.tags).length > 1)
-            return false;
-
-        var members = parent.members, member, outer;
-        for (var i = 0; i < members.length; i++) {
-            member = members[i];
-            if (!member.role || member.role === 'outer') {
-                if (outer)
-                    return false; // Not a simple multipolygon
-                outer = graph.entity(member.id);
-            }
-        }
-
-        return outer;
-    }
-
     return function drawLines(surface, graph, entities, filter) {
         function drawPaths(group, lines, filter, klass, lineString) {
             lines = lines.filter(function(line) {
@@ -21823,7 +21961,7 @@ iD.svg.Lines = function(projection) {
 
         for (var i = 0; i < entities.length; i++) {
             var entity = entities[i],
-                outer = simpleMultipolygonOuterMember(entity, graph);
+                outer = iD.geo.simpleMultipolygonOuterMember(entity, graph);
             if (outer) {
                 lines.push(entity.mergeTags(outer.tags));
             } else if (entity.geometry(graph) === 'line') {
@@ -21850,6 +21988,7 @@ iD.svg.Lines = function(projection) {
             .map(iD.svg.OneWaySegments(projection, graph, 35)));
 
         var oneways = oneway.selectAll('path.oneway')
+            .filter(filter)
             .data(segments, function(d) { return [d.id, d.index]; });
 
         oneways.enter()
@@ -22666,7 +22805,9 @@ iD.ui.Background = function(context) {
             ['top', [0, -1]],
             ['right', [-1, 0]],
             ['bottom', [0, 1]]],
-        layers = context.backgroundSources();
+        layers = context.backgroundSources(),
+        opacityDefault = (context.storage('background-opacity') !== undefined) ?
+            (+context.storage('background-opacity')) : 0.5;
 
     function getSources() {
         var ext = context.map().extent();
@@ -22679,19 +22820,15 @@ iD.ui.Background = function(context) {
     function background(selection) {
 
         function setOpacity(d) {
-            context.map().layersurface.selectAll('.layer-layer')
-                .filter(function(d) { return d == context.map().layers[0]; })
+            context.container().selectAll('.background-layer')
                 .transition()
                 .style('opacity', d)
                 .attr('data-opacity', d);
 
             opacityList.selectAll('li')
-                .classed('selected', false);
+                .classed('selected', function(_) { return _ === d; });
 
-            if (d3.event) {
-                d3.select(this)
-                    .classed('selected', true);
-            }
+            context.storage('background-opacity', d);
         }
 
         function selectLayer() {
@@ -22896,10 +23033,6 @@ iD.ui.Background = function(context) {
             .attr('class', 'opacity')
             .style('opacity', String);
 
-        // Make sure there is an active selection by default
-        opa.select('.opacity-options li:nth-child(2)')
-            .classed('selected', true);
-
         var backgroundList = content
             .append('ul')
             .attr('class', 'toggle-list');
@@ -22986,7 +23119,7 @@ iD.ui.Background = function(context) {
         context.map()
             .on('move.background-update', _.debounce(update, 1000));
         update();
-        setOpacity(0.5);
+        setOpacity(opacityDefault);
 
         var keybinding = d3.keybinding('background');
         keybinding.on(key, toggle);
@@ -23204,7 +23337,7 @@ iD.ui.confirm = function(selection) {
         .on('click.confirm', function() {
             modal.remove();
         })
-        .text('Okay');
+        .text(t('confirm.okay'));
 
     return modal;
 };
@@ -23789,7 +23922,7 @@ iD.ui.intro = function(context) {
         var history = context.history().toJSON(),
             hash = window.location.hash,
             background = context.background().source(),
-            opacity = d3.select('.layer-layer:first-child').style('opacity'),
+            opacity = d3.select('.background-layer').style('opacity'),
             loadedTiles = context.connection().loadedTiles(),
             baseEntities = context.history().graph().base().entities;
 
@@ -23810,7 +23943,7 @@ iD.ui.intro = function(context) {
         var beforeunload = window.onbeforeunload;
         window.onbeforeunload = null;
 
-        d3.select('.layer-layer:first-child').style('opacity', 1);
+        d3.select('.background-layer').style('opacity', 1);
 
         var curtain = d3.curtain();
         selection.call(curtain);
@@ -23824,7 +23957,7 @@ iD.ui.intro = function(context) {
             var s = iD.ui.intro[step](context, reveal)
                 .on('done', function() {
                     entered.filter(function(d) {
-                        return d.name === s.name;
+                        return d.title === s.title;
                     }).classed('finished', true);
                     enter(steps[i + 1]);
                 });
@@ -23834,7 +23967,7 @@ iD.ui.intro = function(context) {
         steps[steps.length - 1].on('startEditing', function() {
             curtain.remove();
             navwrap.remove();
-            d3.select('.layer-layer:first-child').style('opacity', opacity);
+            d3.select('.background-layer').style('opacity', opacity);
             context.connection().toggle(true).flush().loadedTiles(loadedTiles);
             context.history().reset().merge(baseEntities);
             context.background().source(background);
@@ -23856,7 +23989,7 @@ iD.ui.intro = function(context) {
                 .on('click', enter);
 
         entered.append('div').attr('class','icon icon-pre-text apply');
-        entered.append('label').text(function(d) { return d.name; });
+        entered.append('label').text(function(d) { return t(d.title); });
         enter(steps[0]);
 
         function enter (newStep) {
@@ -23871,7 +24004,7 @@ iD.ui.intro = function(context) {
             step.enter();
 
             entered.classed('active', function(d) {
-                return d.name === step.name;
+                return d.title === step.title;
             });
         }
 
@@ -24390,8 +24523,6 @@ iD.ui.preset = function(context, entity, preset) {
 };
 iD.ui.PresetGrid = function(context, entity) {
     var event = d3.dispatch('choose', 'close'),
-        defaultLimit = 9,
-        currentlyDrawn = 9,
         presets,
         autofocus = false;
 
@@ -24427,16 +24558,7 @@ iD.ui.PresetGrid = function(context, entity) {
 
         var grid = gridwrap.append('div')
             .attr('class', 'preset-grid fillL cf')
-            .data([context.presets().defaults(entity, 36).collection]);
-
-        var showMore = gridwrap.append('button')
-            .attr('class', 'fillL show-more')
-            .text(t('inspector.show_more'))
-            .on('click', function() {
-                grid.call(drawGrid, (currentlyDrawn += defaultLimit));
-            });
-
-        grid.call(drawGrid, defaultLimit);
+            .call(drawGrid, context.presets().defaults(entity, 36));
 
         function keydown() {
             // hack to let delete shortcut work when search is autofocused
@@ -24463,7 +24585,6 @@ iD.ui.PresetGrid = function(context, entity) {
             if (d3.event.keyCode === 13 && value.length) {
                 choose(grid.selectAll('.grid-entry:first-child').datum());
             } else {
-                currentlyDrawn = defaultLimit;
                 grid.classed('filtered', value.length);
                 if (value.length) {
                     var results = presets.search(value);
@@ -24471,11 +24592,9 @@ iD.ui.PresetGrid = function(context, entity) {
                         n: results.collection.length,
                         search: value
                     }));
-                    grid.data([results.collection])
-                        .call(drawGrid, defaultLimit);
+                    grid.call(drawGrid, results);
                 } else {
-                    grid.data([context.presets().defaults(entity, 36).collection])
-                        .call(drawGrid, defaultLimit);
+                    grid.call(drawGrid, context.presets().defaults(entity, 36));
                 }
             }
         }
@@ -24508,15 +24627,14 @@ iD.ui.PresetGrid = function(context, entity) {
 
                     subgrid.append('div')
                         .attr('class', 'preset-grid fillL3 cf fl')
-                        .data([d.members.collection])
-                        .call(drawGrid, 1000);
+                        .call(drawGrid, d.members);
 
                     subgrid.style('max-height', '0px')
                         .style('padding-bottom', '0px')
                         .transition()
                         .duration(300)
                         .style('padding-bottom', '20px')
-                        .style('max-height', (d.members.collection.length / 3 * 150) + 200 + 'px');
+                        .style('max-height', (d.members.collection.length * 80) + 200 + 'px');
                 }
 
             // Preset
@@ -24526,8 +24644,6 @@ iD.ui.PresetGrid = function(context, entity) {
             }
         }
 
-        function name(d) { return d.name(); }
-
         // Inserts a div inline after the entry for the provided entity
         // Used for preset descriptions, and for expanding categories
         function insertBox(grid, entity, klass) {
@@ -24554,25 +24670,24 @@ iD.ui.PresetGrid = function(context, entity) {
                 if (d === entity) index = i;
             });
 
-            var insertIndex = index + 3 - index % 3;
-            if (insertIndex > shownIndex) insertIndex ++;
+            if (index >= shownIndex) index++;
 
             var elem = document.createElement('div');
-            grid.node().insertBefore(elem, grid.node().childNodes[insertIndex]);
+            grid.node().insertBefore(elem, grid.node().childNodes[index + 1]);
 
             var newbox = d3.select(elem)
-                .attr('class', 'col12 box-insert ' + klass + ' arrow-' + (index % 3))
+                .attr('class', 'col12 box-insert ' + klass)
                 .datum(entity);
 
             return newbox;
         }
 
-        function drawGrid(selection, limit) {
+        function drawGrid(grid, presets) {
 
             function helpClick(d) {
                 d3.event.stopPropagation();
 
-                var presetinspect = insertBox(selection, d, 'preset-inspect');
+                var presetinspect = insertBox(grid, d, 'preset-inspect');
 
                 if (!presetinspect) return;
 
@@ -24588,23 +24703,18 @@ iD.ui.PresetGrid = function(context, entity) {
                 tagReference.show();
             }
 
-            if (selection.node() === grid.node()) {
-                showMore
-                    .style('display', (selection.data()[0].length > limit) ? 'block' : 'none');
-            }
+            grid.selectAll('.preset-inspect, .subgrid').remove();
 
-            selection.selectAll('.preset-inspect, .subgrid').remove();
-
-            var entries = selection
-                .selectAll('div.grid-entry-wrap')
-                .data(function(d) { return d.slice(0, limit); }, name);
+            var entries = grid
+                .selectAll('.grid-entry-wrap')
+                .data(presets.collection, function(d) { return d.id; });
 
             entries.exit()
                 .remove();
 
             var entered = entries.enter()
                 .append('div')
-                .attr('class','grid-button-wrap col4 grid-entry-wrap')
+                .attr('class','grid-button-wrap col12 grid-entry-wrap')
                 .classed('category', function(d) { return !!d.members; })
                 .classed('current', function(d) { return d === preset; });
 
@@ -24622,15 +24732,15 @@ iD.ui.PresetGrid = function(context, entity) {
 
             var label = buttonInner.append('div')
                 .attr('class','label')
-                .text(name);
+                .text(function(d) { return d.name(); });
 
             entered.filter(function(d) { return !d.members; })
                 .append('button')
                 .attr('tabindex', -1)
                 .attr('class', 'tag-reference-button minor')
-                .on('click', helpClick, selection)
+                .on('click', helpClick)
                 .append('span')
-                    .attr('class', 'icon inspect');
+                .attr('class', 'icon inspect');
 
             entries.order();
         }
@@ -25109,7 +25219,7 @@ iD.ui.Success = function(connection) {
         var m = changeset.comment ?
             changeset.comment.substring(0, 130) : '';
 
-        var message = (m || 'Edited OSM!') + ' ' +
+        var message = (m || t('success.edited_osm')) + ' ' +
             connection.changesetURL(changeset.id);
 
         var links = body.append('div').attr('class','modal-actions cf');
@@ -25129,7 +25239,7 @@ iD.ui.Success = function(connection) {
                 return 'https://twitter.com/intent/tweet?source=webclient&text=' +
                     encodeURIComponent(message);
             })
-            .text('Tweet');
+            .text(t('success.tweet'));
 
         links.append('a')
             .attr('class','col4 facebook')
@@ -25137,7 +25247,7 @@ iD.ui.Success = function(connection) {
             .attr('href', function() {
                 return 'https://facebook.com/sharer/sharer.php?u=' + encodeURIComponent(message);
             })
-            .text('Share on Facebook');
+            .text(t('success.facebook'));
 
         var section = body.append('div').attr('class','modal-section cf');
 
@@ -25146,7 +25256,7 @@ iD.ui.Success = function(connection) {
             .on('click.save', function() {
                 event.cancel();
             })
-            .text('Okay')
+            .text(t('success.okay'))
             .node().focus();
     }
 
@@ -25891,11 +26001,11 @@ iD.ui.preset.address = function(field, context) {
                 var loc = context.projection([
                     (extent[0][0] + extent[1][0]) / 2,
                     (extent[0][1] + extent[1][1]) / 2]),
-                    closest = context.projection(iD.geo.chooseIndex(d, loc, context).loc);
+                    choice = iD.geo.chooseEdge(context.childNodes(d), loc, context.projection);
                 return {
                     title: d.tags.name,
                     value: d.tags.name,
-                    dist: iD.geo.dist(closest, loc)
+                    dist: choice.distance
                 };
             }).sort(function(a, b) {
                 return a.dist - b.dist;
@@ -26526,12 +26636,12 @@ iD.ui.preset.textarea = function(field) {
 
     function change() {
         var t = {};
-        t[field.key] = input.text();
+        t[field.key] = input.property('value');
         event.change(t);
     }
 
     i.tags = function(tags) {
-        input.text(tags[field.key] || '');
+        input.property('value', tags[field.key] || '');
     };
 
     i.focus = function() {
@@ -26673,7 +26783,7 @@ iD.ui.intro.area = function(context, reveal) {
         timeout;
 
     var step = {
-        name: 'Areas'
+        title: 'intro.areas.title'
     };
 
     step.enter = function() {
@@ -26725,7 +26835,7 @@ iD.ui.intro.area = function(context, reveal) {
                 d3.select('.preset-grid-search-wrap input').on('keyup.intro', keySearch);
             }, 500);
         }
-        
+
         function keySearch() {
             var first = d3.select('.grid-button-wrap:first-child');
             if (first.datum().id === 'leisure/playground') {
@@ -26739,8 +26849,6 @@ iD.ui.intro.area = function(context, reveal) {
             reveal('.pane', 'intro.areas.describe');
             context.on('exit.intro', event.done);
         }
-
-
     };
 
     step.exit = function() {
@@ -26760,7 +26868,7 @@ iD.ui.intro.line = function(context, reveal) {
         timeouts = [];
 
     var step = {
-        name: 'Lines'
+        title: 'intro.lines.title'
     };
 
     function one(target, e, f) {
@@ -26895,7 +27003,7 @@ iD.ui.intro.navigation = function(context, reveal) {
         timeouts = [];
 
     var step = {
-        name: 'Navigation'
+        title: 'intro.navigation.title'
     };
 
     function set(f, t) {
@@ -26976,7 +27084,7 @@ iD.ui.intro.point = function(context, reveal) {
         timeouts = [];
 
     var step = {
-        name: 'Points'
+        title: 'intro.points.title'
     };
 
     function setTimeout(f, t) {
@@ -27122,7 +27230,7 @@ iD.ui.intro.startEditing = function(context, reveal) {
         timeouts = [];
 
     var step = {
-        name: 'Start Editing'
+        title: 'intro.startediting.title'
     };
 
     function timeout(f, t) {
@@ -28288,6 +28396,25 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 ]
             ],
             "sourcetag": "ngi-aerial"
+        },
+        {
+            "name": "Lithuania - ORT10LT",
+            "template": "http://mapproxy.openmap.lt/ort10lt/g/{z}/{x}/{y}.jpeg",
+            "extent": [
+                [
+                    21,
+                    53.88
+                ],
+                [
+                    26.85,
+                    56.45
+                ]
+            ],
+            "scaleExtent": [
+                4,
+                18
+            ],
+            "sourcetag": "NŽT ORT10LT"
         }
     ],
     "wikipedia": [
@@ -29746,11 +29873,59 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                     "airport",
                     "aerodrome"
                 ],
+                "fields": [
+                    "ref",
+                    "iata",
+                    "icao",
+                    "operator"
+                ],
                 "tags": {
                     "aeroway": "aerodrome"
                 },
                 "name": "Airport"
             },
+            "aeroway/apron": {
+                "icon": "airport",
+                "geometry": [
+                    "area"
+                ],
+                "terms": [
+                    "ramp"
+                ],
+                "fields": [
+                    "ref",
+                    "surface"
+                ],
+                "tags": {
+                    "aeroway": "apron"
+                },
+                "name": "Apron"
+            },
+            "aeroway/gate": {
+                "icon": "airport",
+                "geometry": [
+                    "point"
+                ],
+                "fields": [
+                    "ref"
+                ],
+                "tags": {
+                    "aeroway": "gate"
+                },
+                "name": "Airport gate"
+            },
+            "aeroway/hangar": {
+                "geometry": [
+                    "area"
+                ],
+                "fields": [
+                    "building_area"
+                ],
+                "tags": {
+                    "aeroway": "hangar"
+                },
+                "name": "Hangar"
+            },
             "aeroway/helipad": {
                 "icon": "heliport",
                 "geometry": [
@@ -29767,6 +29942,54 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 },
                 "name": "Helipad"
             },
+            "aeroway/runway": {
+                "geometry": [
+                    "line",
+                    "area"
+                ],
+                "terms": [
+                    "landing strip"
+                ],
+                "fields": [
+                    "ref",
+                    "surface"
+                ],
+                "tags": {
+                    "aeroway": "runway"
+                },
+                "name": "Runway"
+            },
+            "aeroway/taxiway": {
+                "geometry": [
+                    "line"
+                ],
+                "fields": [
+                    "ref",
+                    "surface"
+                ],
+                "tags": {
+                    "aeroway": "taxiway"
+                },
+                "name": "Taxiway"
+            },
+            "aeroway/terminal": {
+                "geometry": [
+                    "point",
+                    "area"
+                ],
+                "terms": [
+                    "airport",
+                    "aerodrome"
+                ],
+                "fields": [
+                    "operator",
+                    "building_area"
+                ],
+                "tags": {
+                    "aeroway": "terminal"
+                },
+                "name": "Airport terminal"
+            },
             "amenity": {
                 "fields": [
                     "amenity"
@@ -29781,6 +30004,20 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 },
                 "name": "Amenity"
             },
+            "amenity/atm": {
+                "icon": "bank",
+                "fields": [
+                    "operator"
+                ],
+                "geometry": [
+                    "point",
+                    "vertex"
+                ],
+                "tags": {
+                    "amenity": "atm"
+                },
+                "name": "ATM"
+            },
             "amenity/bank": {
                 "icon": "bank",
                 "fields": [
@@ -29954,6 +30191,22 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 },
                 "name": "Cinema"
             },
+            "amenity/college": {
+                "icon": "college",
+                "fields": [
+                    "operator",
+                    "address"
+                ],
+                "geometry": [
+                    "point",
+                    "area"
+                ],
+                "tags": {
+                    "amenity": "college"
+                },
+                "terms": [],
+                "name": "College"
+            },
             "amenity/courthouse": {
                 "fields": [
                     "operator",
@@ -30095,6 +30348,28 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 },
                 "name": "Hospital"
             },
+            "amenity/kindergarten": {
+                "icon": "school",
+                "fields": [
+                    "building_area",
+                    "address"
+                ],
+                "geometry": [
+                    "point",
+                    "vertex",
+                    "area"
+                ],
+                "terms": [
+                    "preschool",
+                    "nursery",
+                    "childcare",
+                    "playgroup"
+                ],
+                "tags": {
+                    "amenity": "kindergarten"
+                },
+                "name": "Kindergarten"
+            },
             "amenity/library": {
                 "icon": "library",
                 "fields": [
@@ -30585,7 +30860,9 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "tags": {
                     "amenity": "university"
                 },
-                "terms": [],
+                "terms": [
+                    "college"
+                ],
                 "name": "University"
             },
             "amenity/waste_basket": {
@@ -31025,7 +31302,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "name": "Foot Path"
             },
             "highway/living_street": {
-                "icon": "highway-residential",
+                "icon": "highway-living-street",
                 "fields": [
                     "oneway",
                     "structure",
@@ -31717,7 +31994,8 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                     "landuse": "farm"
                 },
                 "terms": [],
-                "name": "Farm"
+                "name": "Farm",
+                "icon": "farm"
             },
             "landuse/farmyard": {
                 "geometry": [
@@ -32039,7 +32317,11 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "tags": {
                     "leisure": "playground"
                 },
-                "name": "Playground"
+                "name": "Playground",
+                "terms": [
+                    "jungle gym",
+                    "play area"
+                ]
             },
             "leisure/slipway": {
                 "geometry": [
@@ -32091,6 +32373,25 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 },
                 "name": "Man Made"
             },
+            "man_made/breakwater": {
+                "geometry": [
+                    "line",
+                    "area"
+                ],
+                "tags": {
+                    "man_made": "breakwater"
+                },
+                "name": "Breakwater"
+            },
+            "man_made/cutline": {
+                "geometry": [
+                    "line"
+                ],
+                "tags": {
+                    "man_made": "cutline"
+                },
+                "name": "Cut line"
+            },
             "man_made/lighthouse": {
                 "geometry": [
                     "point",
@@ -32111,6 +32412,19 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 },
                 "name": "Pier"
             },
+            "man_made/pipeline": {
+                "geometry": [
+                    "line"
+                ],
+                "tags": {
+                    "man_made": "pipeline"
+                },
+                "fields": [
+                    "location",
+                    "operator"
+                ],
+                "name": "Pipeline"
+            },
             "man_made/survey_point": {
                 "icon": "monument",
                 "geometry": [
@@ -32125,6 +32439,19 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 ],
                 "name": "Survey Point"
             },
+            "man_made/tower": {
+                "geometry": [
+                    "point",
+                    "area"
+                ],
+                "tags": {
+                    "man_made": "tower"
+                },
+                "fields": [
+                    "towertype"
+                ],
+                "name": "Tower"
+            },
             "man_made/wastewater_plant": {
                 "icon": "water",
                 "geometry": [
@@ -32153,6 +32480,16 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 },
                 "name": "Water Tower"
             },
+            "man_made/water_well": {
+                "geometry": [
+                    "point",
+                    "area"
+                ],
+                "tags": {
+                    "man_made": "water_well"
+                },
+                "name": "Water well"
+            },
             "man_made/water_works": {
                 "icon": "water",
                 "geometry": [
@@ -32467,7 +32804,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "name": "Place"
             },
             "place/city": {
-                "icon": "square",
+                "icon": "city",
                 "geometry": [
                     "point",
                     "area"
@@ -32530,7 +32867,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "name": "Locality"
             },
             "place/town": {
-                "icon": "square-stroked",
+                "icon": "town",
                 "geometry": [
                     "point",
                     "area"
@@ -32541,7 +32878,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "name": "Town"
             },
             "place/village": {
-                "icon": "triangle",
+                "icon": "village",
                 "geometry": [
                     "point",
                     "area"
@@ -34170,7 +34507,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
             "category-landuse": {
                 "geometry": "area",
                 "name": "Land Use",
-                "icon": "category-landuse",
+                "icon": "land-use",
                 "members": [
                     "landuse/residential",
                     "landuse/industrial",
@@ -34474,6 +34811,16 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "type": "combo",
                 "label": "Type"
             },
+            "iata": {
+                "key": "iata",
+                "type": "text",
+                "label": "IATA"
+            },
+            "icao": {
+                "key": "icao",
+                "type": "text",
+                "label": "ICAO"
+            },
             "incline": {
                 "key": "incline",
                 "type": "combo",
@@ -34525,6 +34872,11 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "type": "number",
                 "label": "Levels"
             },
+            "location": {
+                "key": "location",
+                "type": "combo",
+                "label": "Location"
+            },
             "man_made": {
                 "key": "man_made",
                 "type": "combo",
@@ -34726,6 +35078,11 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "type": "combo",
                 "label": "Type"
             },
+            "towertype": {
+                "key": "tower:type",
+                "type": "combo",
+                "label": "Tower type"
+            },
             "tracktype": {
                 "key": "tracktype",
                 "type": "combo",
@@ -44923,7 +45280,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1114
             ]
         },
-        "college": {
+        "city": {
             "12": [
                 0,
                 1140
@@ -44937,7 +45294,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1174
             ]
         },
-        "commercial": {
+        "college": {
             "12": [
                 0,
                 1200
@@ -44951,7 +45308,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1234
             ]
         },
-        "cricket": {
+        "commercial": {
             "12": [
                 0,
                 1260
@@ -44965,7 +45322,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1294
             ]
         },
-        "cross": {
+        "cricket": {
             "12": [
                 0,
                 1320
@@ -44979,7 +45336,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1354
             ]
         },
-        "dam": {
+        "cross": {
             "12": [
                 0,
                 1380
@@ -44993,7 +45350,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1414
             ]
         },
-        "danger": {
+        "dam": {
             "12": [
                 0,
                 1440
@@ -45007,7 +45364,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1474
             ]
         },
-        "disability": {
+        "danger": {
             "12": [
                 0,
                 1500
@@ -45021,7 +45378,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1534
             ]
         },
-        "embassy": {
+        "disability": {
             "12": [
                 0,
                 1560
@@ -45035,7 +45392,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1594
             ]
         },
-        "emergency-telephone": {
+        "embassy": {
             "12": [
                 0,
                 1620
@@ -45049,7 +45406,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1654
             ]
         },
-        "fast-food": {
+        "emergency-telephone": {
             "12": [
                 0,
                 1680
@@ -45063,7 +45420,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1714
             ]
         },
-        "ferry": {
+        "farm": {
             "12": [
                 0,
                 1740
@@ -45077,7 +45434,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1774
             ]
         },
-        "fire-station": {
+        "fast-food": {
             "12": [
                 0,
                 1800
@@ -45091,7 +45448,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1834
             ]
         },
-        "fuel": {
+        "ferry": {
             "12": [
                 0,
                 1860
@@ -45105,7 +45462,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1894
             ]
         },
-        "garden": {
+        "fire-station": {
             "12": [
                 0,
                 1920
@@ -45119,7 +45476,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 1954
             ]
         },
-        "golf": {
+        "fuel": {
             "12": [
                 0,
                 1980
@@ -45133,7 +45490,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2014
             ]
         },
-        "grocery": {
+        "garden": {
             "12": [
                 0,
                 2040
@@ -45147,7 +45504,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2074
             ]
         },
-        "harbor": {
+        "golf": {
             "12": [
                 0,
                 2100
@@ -45161,7 +45518,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2134
             ]
         },
-        "heliport": {
+        "grocery": {
             "12": [
                 0,
                 2160
@@ -45175,7 +45532,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2194
             ]
         },
-        "hospital": {
+        "harbor": {
             "12": [
                 0,
                 2220
@@ -45189,7 +45546,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2254
             ]
         },
-        "industrial": {
+        "heliport": {
             "12": [
                 0,
                 2280
@@ -45203,7 +45560,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2314
             ]
         },
-        "library": {
+        "hospital": {
             "12": [
                 0,
                 2340
@@ -45217,7 +45574,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2374
             ]
         },
-        "lodging": {
+        "industrial": {
             "12": [
                 0,
                 2400
@@ -45231,7 +45588,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2434
             ]
         },
-        "logging": {
+        "land-use": {
             "12": [
                 0,
                 2460
@@ -45245,7 +45602,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2494
             ]
         },
-        "marker": {
+        "library": {
             "12": [
                 0,
                 2520
@@ -45259,7 +45616,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2554
             ]
         },
-        "marker-stroked": {
+        "lodging": {
             "12": [
                 0,
                 2580
@@ -45273,7 +45630,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2614
             ]
         },
-        "monument": {
+        "logging": {
             "12": [
                 0,
                 2640
@@ -45287,7 +45644,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2674
             ]
         },
-        "museum": {
+        "marker": {
             "12": [
                 0,
                 2700
@@ -45301,7 +45658,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2734
             ]
         },
-        "music": {
+        "marker-stroked": {
             "12": [
                 0,
                 2760
@@ -45315,7 +45672,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2794
             ]
         },
-        "oil-well": {
+        "monument": {
             "12": [
                 0,
                 2820
@@ -45329,7 +45686,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2854
             ]
         },
-        "park": {
+        "museum": {
             "12": [
                 0,
                 2880
@@ -45343,7 +45700,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2914
             ]
         },
-        "park2": {
+        "music": {
             "12": [
                 0,
                 2940
@@ -45357,7 +45714,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 2974
             ]
         },
-        "parking": {
+        "oil-well": {
             "12": [
                 0,
                 3000
@@ -45371,7 +45728,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3034
             ]
         },
-        "parking-garage": {
+        "park": {
             "12": [
                 0,
                 3060
@@ -45385,7 +45742,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3094
             ]
         },
-        "pharmacy": {
+        "park2": {
             "12": [
                 0,
                 3120
@@ -45399,7 +45756,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3154
             ]
         },
-        "pitch": {
+        "parking": {
             "12": [
                 0,
                 3180
@@ -45413,7 +45770,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3214
             ]
         },
-        "place-of-worship": {
+        "parking-garage": {
             "12": [
                 0,
                 3240
@@ -45427,7 +45784,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3274
             ]
         },
-        "police": {
+        "pharmacy": {
             "12": [
                 0,
                 3300
@@ -45441,7 +45798,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3334
             ]
         },
-        "post": {
+        "pitch": {
             "12": [
                 0,
                 3360
@@ -45455,7 +45812,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3394
             ]
         },
-        "prison": {
+        "place-of-worship": {
             "12": [
                 0,
                 3420
@@ -45469,7 +45826,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3454
             ]
         },
-        "rail": {
+        "police": {
             "12": [
                 0,
                 3480
@@ -45483,7 +45840,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3514
             ]
         },
-        "rail-above": {
+        "post": {
             "12": [
                 0,
                 3540
@@ -45497,7 +45854,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3574
             ]
         },
-        "rail-underground": {
+        "prison": {
             "12": [
                 0,
                 3600
@@ -45511,7 +45868,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3634
             ]
         },
-        "religious-christian": {
+        "rail": {
             "12": [
                 0,
                 3660
@@ -45525,7 +45882,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3694
             ]
         },
-        "religious-jewish": {
+        "rail-above": {
             "12": [
                 0,
                 3720
@@ -45539,7 +45896,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3754
             ]
         },
-        "religious-muslim": {
+        "rail-underground": {
             "12": [
                 0,
                 3780
@@ -45553,7 +45910,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3814
             ]
         },
-        "restaurant": {
+        "religious-christian": {
             "12": [
                 0,
                 3840
@@ -45567,7 +45924,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3874
             ]
         },
-        "roadblock": {
+        "religious-jewish": {
             "12": [
                 0,
                 3900
@@ -45581,7 +45938,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3934
             ]
         },
-        "school": {
+        "religious-muslim": {
             "12": [
                 0,
                 3960
@@ -45595,7 +45952,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 3994
             ]
         },
-        "shop": {
+        "restaurant": {
             "12": [
                 0,
                 4020
@@ -45609,7 +45966,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4054
             ]
         },
-        "skiing": {
+        "roadblock": {
             "12": [
                 0,
                 4080
@@ -45623,7 +45980,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4114
             ]
         },
-        "slaughterhouse": {
+        "school": {
             "12": [
                 0,
                 4140
@@ -45637,7 +45994,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4174
             ]
         },
-        "soccer": {
+        "shop": {
             "12": [
                 0,
                 4200
@@ -45651,7 +46008,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4234
             ]
         },
-        "square": {
+        "skiing": {
             "12": [
                 0,
                 4260
@@ -45665,7 +46022,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4294
             ]
         },
-        "square-stroked": {
+        "slaughterhouse": {
             "12": [
                 0,
                 4320
@@ -45679,7 +46036,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4354
             ]
         },
-        "star": {
+        "soccer": {
             "12": [
                 0,
                 4380
@@ -45693,7 +46050,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4414
             ]
         },
-        "star-stroked": {
+        "square": {
             "12": [
                 0,
                 4440
@@ -45707,7 +46064,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4474
             ]
         },
-        "swimming": {
+        "square-stroked": {
             "12": [
                 0,
                 4500
@@ -45721,7 +46078,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4534
             ]
         },
-        "telephone": {
+        "star": {
             "12": [
                 0,
                 4560
@@ -45735,7 +46092,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4594
             ]
         },
-        "tennis": {
+        "star-stroked": {
             "12": [
                 0,
                 4620
@@ -45749,7 +46106,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4654
             ]
         },
-        "theatre": {
+        "swimming": {
             "12": [
                 0,
                 4680
@@ -45763,7 +46120,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4714
             ]
         },
-        "toilets": {
+        "telephone": {
             "12": [
                 0,
                 4740
@@ -45777,7 +46134,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4774
             ]
         },
-        "town-hall": {
+        "tennis": {
             "12": [
                 0,
                 4800
@@ -45791,7 +46148,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4834
             ]
         },
-        "triangle": {
+        "theatre": {
             "12": [
                 0,
                 4860
@@ -45805,7 +46162,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4894
             ]
         },
-        "triangle-stroked": {
+        "toilets": {
             "12": [
                 0,
                 4920
@@ -45819,7 +46176,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 4954
             ]
         },
-        "warehouse": {
+        "town": {
             "12": [
                 0,
                 4980
@@ -45833,7 +46190,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 5014
             ]
         },
-        "waste-basket": {
+        "town-hall": {
             "12": [
                 0,
                 5040
@@ -45847,7 +46204,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 5074
             ]
         },
-        "water": {
+        "triangle": {
             "12": [
                 0,
                 5100
@@ -45861,7 +46218,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 5134
             ]
         },
-        "wetland": {
+        "triangle-stroked": {
             "12": [
                 0,
                 5160
@@ -45875,7 +46232,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 5194
             ]
         },
-        "zoo": {
+        "village": {
             "12": [
                 0,
                 5220
@@ -45889,6 +46246,76 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 5254
             ]
         },
+        "warehouse": {
+            "12": [
+                0,
+                5280
+            ],
+            "18": [
+                0,
+                5294
+            ],
+            "24": [
+                0,
+                5314
+            ]
+        },
+        "waste-basket": {
+            "12": [
+                0,
+                5340
+            ],
+            "18": [
+                0,
+                5354
+            ],
+            "24": [
+                0,
+                5374
+            ]
+        },
+        "water": {
+            "12": [
+                0,
+                5400
+            ],
+            "18": [
+                0,
+                5414
+            ],
+            "24": [
+                0,
+                5434
+            ]
+        },
+        "wetland": {
+            "12": [
+                0,
+                5460
+            ],
+            "18": [
+                0,
+                5474
+            ],
+            "24": [
+                0,
+                5494
+            ]
+        },
+        "zoo": {
+            "12": [
+                0,
+                5520
+            ],
+            "18": [
+                0,
+                5534
+            ],
+            "24": [
+                0,
+                5554
+            ]
+        },
         "highway-motorway": {
             "line": [
                 20,
@@ -46204,23 +46631,29 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
     },
     "locales": [
         "af",
+        "ast",
         "bs",
+        "ca",
         "zh",
-        "zh_TW",
+        "zh-TW",
+        "hr",
         "cs",
         "da",
         "nl",
         "fr",
         "de",
-        "hu",
+        "is",
+        "id",
         "it",
         "ja",
         "lv",
         "pl",
         "pt",
+        "pt-BR",
         "ru",
         "sr",
         "sk",
+        "sl",
         "es",
         "sv",
         "tr",
@@ -46457,9 +46890,18 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
             "uploading": "Uploading changes to OpenStreetMap.",
             "unsaved_changes": "You have unsaved changes"
         },
+        "success": {
+            "edited_osm": "Edited OSM!",
+            "facebook": "Share on Facebook",
+            "tweet": "Tweet",
+            "okay": "Okay"
+        },
+        "confirm": {
+            "okay": "Okay"
+        },
         "splash": {
             "welcome": "Welcome to the iD OpenStreetMap editor",
-            "text": "iD is a friendly but powerful tool for contributing to the world's best free world map. This is development version {version}. For more information see {website} and report bugs at {github}.",
+            "text": "iD is a friendly but powerful tool for contributing to the world's best free world map. This is version {version}. For more information see {website} and report bugs at {github}.",
             "walkthrough": "Start the Walkthrough",
             "start": "Edit Now"
         },
@@ -46499,20 +46941,22 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
             "imagery": "# Imagery\n\nAerial imagery is an important resource for mapping. A combination of\nairplane flyovers, satellite views, and freely-compiled sources are available\nin the editor under the 'Background Settings' menu on the left.\n\nBy default a [Bing Maps](http://www.bing.com/maps/) satellite layer is\npresented in the editor, but as you pan and zoom the map to new geographical\nareas, new sources will become available. Some countries, like the United\nStates, France, and Denmark have very high-quality imagery available for some areas.\n\nImagery is sometimes offset from the map data because of a mistake on the\nimagery provider's side. If you see a lot of roads shifted from the background,\ndon't immediately move them all to match the background. Instead you can adjust\nthe imagery so that it matches the existing data by clicking 'Fix alignment' at\nthe bottom of the Background Settings UI.\n",
             "addresses": "# Addresses\n\nAddresses are some of the most useful information for the map.\n\nAlthough addresses are often represented as parts of streets, in OpenStreetMap\nthey're recorded as attributes of buildings and places along streets.\n\nYou can add address information to places mapped as building outlines as well\nas well as those mapped as single points. The optimal source of address\ndata is from an on-the-ground survey or personal knowledge - as with any\nother feature, copying from commercial sources like Google Maps is strictly\nforbidden.\n",
             "inspector": "# Using the Inspector\n\nThe inspector is the user interface element on the right-hand side of the\npage that appears when a feature is selected and allows you to edit its details.\n\n### Selecting a Feature Type\n\nAfter you add a point, line, or area, you can choose what type of feature it\nis, like whether it's a highway or residential road, supermarket or cafe.\nThe inspector will display buttons for common feature types, and you can\nfind others by typing what you're looking for in the search box.\n\nClick the 'i' in the bottom-right-hand corner of a feature type button to\nlearn more about it. Click a button to choose that type.\n\n### Using Forms and Editing Tags\n\nAfter you choose a feature type, or when you select a feature that already\nhas a type assigned, the inspector will display fields with details about\nthe feature like its name and address.\n\nBelow the fields you see, you can click icons to add other details,\nlike [Wikipedia](http://www.wikipedia.org/) information, wheelchair\naccess, and more.\n\nAt the bottom of the inspector, click 'Additional tags' to add arbitrary\nother tags to the element. [Taginfo](http://taginfo.openstreetmap.org/) is a\ngreat resource for learn more about popular tag combinations.\n\nChanges you make in the inspector are automatically applied to the map.\nYou can undo them at any time by clicking the 'Undo' button.\n\n### Closing the Inspector\n\nYou can close the inspector by clicking the close button in the top-right,\npressing the 'Escape' key, or clicking on the map.\n",
-            "buildings": "# Buildings\n\nOpenStreetMap is the world's largest database of buildings. You can create\nand improve this database.\n\n### Selecting\n\nYou can select a building by clicking on its border. This will highlight the\nbuilding and open a small tools menu and a sidebar showing more information\nabout the building.\n\n### Modifying\n\nSometimes buildings are incorrectly placed or have incorrect tags.\n\nTo move an entire building, select it, then click the 'Move' tool. Move your\nmouse to shift the building, and click when it's correctly placed.\n\nTo fix the specific shape of a building, click and drag the nodes that form\nits border into better places.\n\n### Creating\n\nOne of the main questions around adding buildings to the map is that\nOpenStreetMap records buildings both as shapes and points. The rule of thumb\nis to _map a building as a shape whenever possible_, and map companies, homes,\namenities, and other things that operate out of buildings as points placed\nwithin the building shape.\n\nStart drawing a building as a shape by clicking the 'Area' button in the top\nleft of the interface, and end it either by pressing 'Return' on your keyboard\nor clicking on the first node drawn to close the shape.\n\n### Deleting\n\nIf a building is entirely incorrect - you can see that it doesn't exist in satellite\nimagery and ideally have confirmed locally that it's not present - you can delete\nit, which removes it from the map. Be cautious when deleting features -\nlike any other edit, the results are seen by everyone and satellite imagery\nis often out of date, so the road could simply be newly built.\n\nYou can delete a building by clicking on it to select it, then clicking the\ntrash can icon or pressing the 'Delete' key.\n"
+            "buildings": "# Buildings\n\nOpenStreetMap is the world's largest database of buildings. You can create\nand improve this database.\n\n### Selecting\n\nYou can select a building by clicking on its border. This will highlight the\nbuilding and open a small tools menu and a sidebar showing more information\nabout the building.\n\n### Modifying\n\nSometimes buildings are incorrectly placed or have incorrect tags.\n\nTo move an entire building, select it, then click the 'Move' tool. Move your\nmouse to shift the building, and click when it's correctly placed.\n\nTo fix the specific shape of a building, click and drag the nodes that form\nits border into better places.\n\n### Creating\n\nOne of the main questions around adding buildings to the map is that\nOpenStreetMap records buildings both as shapes and points. The rule of thumb\nis to _map a building as a shape whenever possible_, and map companies, homes,\namenities, and other things that operate out of buildings as points placed\nwithin the building shape.\n\nStart drawing a building as a shape by clicking the 'Area' button in the top\nleft of the interface, and end it either by pressing 'Return' on your keyboard\nor clicking on the first node drawn to close the shape.\n\n### Deleting\n\nIf a building is entirely incorrect - you can see that it doesn't exist in satellite\nimagery and ideally have confirmed locally that it's not present - you can delete\nit, which removes it from the map. Be cautious when deleting features -\nlike any other edit, the results are seen by everyone and satellite imagery\nis often out of date, so the building could simply be newly built.\n\nYou can delete a building by clicking on it to select it, then clicking the\ntrash can icon or pressing the 'Delete' key.\n"
         },
         "intro": {
             "navigation": {
+                "title": "Navigation",
                 "drag": "The main map area shows OpenStreetMap data on top of a background. You can navigate by dragging and scrolling, just like any web map. **Drag the map!**",
                 "select": "Map features are represented three ways: using points, lines or areas. All features can be selected by clicking on them. **Click on the point to select it.**",
                 "header": "The header shows us the feature type.",
                 "pane": "When a feature is selected, the feature editor is displayed. The header shows us the feature type and the main pane shows the feature's attributes, such as its name and address. **Close the feature editor with the close button in the top right.**"
             },
             "points": {
+                "title": "Points",
                 "add": "Points can be used to represent features such as shops, restaurants and monuments. They mark a specific location, and describe what's there. **Click the Point button to add a new point.**",
                 "place": "The point can be placed by clicking on the map. **Place the point on top of the building.**",
-                "search": "There many different features that can be represented by points. The point you just added is a Cafe. **Search for 'Cafe' **",
-                "choose": "**Choose Cafe from the grid.**",
+                "search": "There are many different features that can be represented by points. The point you just added is a Cafe. **Search for 'Cafe' **",
+                "choose": "**Choose Cafe from the list.**",
                 "describe": "The point is now marked as a cafe. Using the feature editor, we can add more information about the feature. **Add a name**",
                 "close": "The feature editor can be closed by clicking on the close button. **Close the feature editor**",
                 "reselect": "Often points will already exist, but have mistakes or be incomplete. We can edit existing points. **Select the point you just created.**",
@@ -46521,24 +46965,27 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "delete": "The menu around the point contains operations that can be performed on it, including delete. **Delete the point.**"
             },
             "areas": {
-                "add": "Areas are a more detailed way to represent features. They provide information on the boundaries of the feature. Areas can be used for most features types points can be used for, and are often preferred. **Click the Area button to add a new area.**",
+                "title": "Areas",
+                "add": "Areas are a more detailed way to represent features. They provide information on the boundaries of the feature. Areas can be used for most feature types points can be used for, and are often preferred. **Click the Area button to add a new area.**",
                 "corner": "Areas are drawn by placing nodes that mark the boundary of the area. **Place the starting node on one of the corners of the playground.**",
                 "place": "Draw the area by placing more nodes. Finish the area by clicking on the starting node. **Draw an area for the playground.**",
                 "search": "**Search for Playground.**",
-                "choose": "**Choose Playground from the grid.**",
+                "choose": "**Choose Playground from the list.**",
                 "describe": "**Add a name, and close the feature editor**"
             },
             "lines": {
+                "title": "Lines",
                 "add": "Lines are used to represent features such as roads, railways and rivers. **Click the Line button to add a new line.**",
                 "start": "**Start the line by clicking on the end of the road.**",
                 "intersect": "Click to add more nodes to the line. You can drag the map while drawing if necessary. Roads, and many other types of lines, are part of a larger network. It is important for these lines to be connected properly in order for routing applications to work. **Click on Flower Street, to create an intersection connecting the two lines.**",
                 "finish": "Lines can be finished by clicking on the last node again. **Finish drawing the road.**",
-                "road": "**Select Road from the grid**",
+                "road": "**Select Road from the list**",
                 "residential": "There are different types of roads, the most common of which is Residential. **Choose the Residential road type**",
                 "describe": "**Name the road and close the feature editor.**",
                 "restart": "The road needs to intersect Flower Street."
             },
             "startediting": {
+                "title": "Start Editing",
                 "help": "More documentation and this walkthrough are available here.",
                 "save": "Don't forget to regularly save your changes!",
                 "start": "Start mapping!"
@@ -46691,6 +47138,12 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "historic": {
                     "label": "Type"
                 },
+                "iata": {
+                    "label": "IATA"
+                },
+                "icao": {
+                    "label": "ICAO"
+                },
                 "incline": {
                     "label": "Incline"
                 },
@@ -46719,6 +47172,9 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "levels": {
                     "label": "Levels"
                 },
+                "location": {
+                    "label": "Location"
+                },
                 "man_made": {
                     "label": "Type"
                 },
@@ -46821,6 +47277,9 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 "tourism": {
                     "label": "Type"
                 },
+                "towertype": {
+                    "label": "Tower type"
+                },
                 "tracktype": {
                     "label": "Type"
                 },
@@ -46858,14 +47317,42 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                     "name": "Airport",
                     "terms": "airplane,airport,aerodrome"
                 },
+                "aeroway/apron": {
+                    "name": "Apron",
+                    "terms": "ramp"
+                },
+                "aeroway/gate": {
+                    "name": "Airport gate",
+                    "terms": ""
+                },
+                "aeroway/hangar": {
+                    "name": "Hangar",
+                    "terms": ""
+                },
                 "aeroway/helipad": {
                     "name": "Helipad",
                     "terms": "helicopter,helipad,heliport"
                 },
+                "aeroway/runway": {
+                    "name": "Runway",
+                    "terms": "landing strip"
+                },
+                "aeroway/taxiway": {
+                    "name": "Taxiway",
+                    "terms": ""
+                },
+                "aeroway/terminal": {
+                    "name": "Airport terminal",
+                    "terms": "airport,aerodrome"
+                },
                 "amenity": {
                     "name": "Amenity",
                     "terms": ""
                 },
+                "amenity/atm": {
+                    "name": "ATM",
+                    "terms": ""
+                },
                 "amenity/bank": {
                     "name": "Bank",
                     "terms": "coffer,countinghouse,credit union,depository,exchequer,fund,hoard,investment firm,repository,reserve,reservoir,safe,savings,stock,stockpile,store,storehouse,thrift,treasury,trust company,vault"
@@ -46898,6 +47385,10 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                     "name": "Cinema",
                     "terms": "big screen,bijou,cine,drive-in,film,flicks,motion pictures,movie house,movie theater,moving pictures,nabes,photoplay,picture show,pictures,playhouse,show,silver screen"
                 },
+                "amenity/college": {
+                    "name": "College",
+                    "terms": ""
+                },
                 "amenity/courthouse": {
                     "name": "Courthouse",
                     "terms": ""
@@ -46930,6 +47421,10 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                     "name": "Hospital",
                     "terms": "clinic,emergency room,health service,hospice,infirmary,institution,nursing home,rest home,sanatorium,sanitarium,sick bay,surgery,ward"
                 },
+                "amenity/kindergarten": {
+                    "name": "Kindergarten",
+                    "terms": "preschool,nursery,childcare,playgroup"
+                },
                 "amenity/library": {
                     "name": "Library",
                     "terms": ""
@@ -47008,7 +47503,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 },
                 "amenity/university": {
                     "name": "University",
-                    "terms": ""
+                    "terms": "college"
                 },
                 "amenity/waste_basket": {
                     "name": "Waste Basket",
@@ -47400,7 +47895,7 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                 },
                 "leisure/playground": {
                     "name": "Playground",
-                    "terms": ""
+                    "terms": "jungle gym,play area"
                 },
                 "leisure/slipway": {
                     "name": "Slipway",
@@ -47418,6 +47913,14 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                     "name": "Man Made",
                     "terms": ""
                 },
+                "man_made/breakwater": {
+                    "name": "Breakwater",
+                    "terms": ""
+                },
+                "man_made/cutline": {
+                    "name": "Cut line",
+                    "terms": ""
+                },
                 "man_made/lighthouse": {
                     "name": "Lighthouse",
                     "terms": ""
@@ -47426,10 +47929,18 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                     "name": "Pier",
                     "terms": ""
                 },
+                "man_made/pipeline": {
+                    "name": "Pipeline",
+                    "terms": ""
+                },
                 "man_made/survey_point": {
                     "name": "Survey Point",
                     "terms": ""
                 },
+                "man_made/tower": {
+                    "name": "Tower",
+                    "terms": ""
+                },
                 "man_made/wastewater_plant": {
                     "name": "Wastewater Plant",
                     "terms": "sewage works,sewage treatment plant,water treatment plant,reclamation plant"
@@ -47438,6 +47949,10 @@ iD.introGraph = '{"n185954700":{"id":"n185954700","loc":[-85.642244,41.939081],"
                     "name": "Water Tower",
                     "terms": ""
                 },
+                "man_made/water_well": {
+                    "name": "Water well",
+                    "terms": ""
+                },
                 "man_made/water_works": {
                     "name": "Water Works",
                     "terms": ""