Exporte dados do OpenStreetMap usando o editor visual em rete.js

No meu trabalho, frequentemente encontro a tarefa de exportar dados do OpenStreetMap. O OSM é uma incrível fonte de dados de onde você pode obter pelo menos pontos turísticos , pelo menos áreas da cidade , pelo menos ruas para estudos de acessibilidade para pedestres e geralmente qualquer coisa.


Esse é apenas o processo de trabalhar com eles em algum momento que começou a me aborrecer. Para recuperar dados para alguma solicitação não trivial, você precisa estudar a linguagem de solicitação Overpass ou escrever scripts e vasculhar no formato XML do OSM.


Realizando essas manipulações pela centésima vez, pensei em criar uma ferramenta mais simples e mais conveniente. E agora ele está pronto - https://yourmaps.io , o editor visual de descrições de exportação do OpenStreetMap. No editor, você pode passar o mouse sobre um gráfico, cada nó representando uma operação ou filtro em um fluxo de objetos OSM e depois fazer o download do resultado no GeoJSON.


Aqui está um exemplo de um gráfico que seleciona todas as escolas de um determinado município e, em seguida, cria amortecedores de 300 metros em torno deles:



Como resultado do trabalho, obtemos um conjunto de polígonos no formato GeoJSON, que podem ser importados para o QGIS ou outro software.


— , Rete.js, -.


Rete.js


Rete.js — JS , . , .


, Rete . OpenStreetMap — .


imagem


(node), (component). , ( ). data, (, , )


Rete . : , ( ) .


, , , , - .


— , , ( ). , - , - jquery 220, . , .



InputControl Rete, .


var SelectComponent = {
    //  -  HTML ,       ,        this.root   
    template: '<select></select>',
    data() {
        return {
            value: ""
        };
    },
    methods: {
        update() {
            //        
            this.putData(this.ikey, $(this.root).val())
        }
    },
    //         
    mounted() {
        // this.root -  html ,    template, .. select   
        let jqueryRoot = $(this.root)
        //     
        for (let idx = 0; idx < this.values.length; ++idx) {
            let v = this.values[idx]
            jqueryRoot.append($("<option></option>")
                .attr("value", v[0])
                .text(v[1]));
        }
        //       -        ,  
        let currentVal = this.getData(this.ikey)
        if (currentVal === undefined) {
            currentVal = this.defaultValue
            this.putData(this.ikey, this.defaultValue)
        }
        jqueryRoot.val(currentVal);

        const _self = this;
        //          data
        jqueryRoot.change(function() {
            _self.root.update()
        })
    }
}
//          node.addControl(new SelectControl(...))
class SelectControl extends Rete.Control {
    constructor(emitter, key, values, defaultValue) {
        super(key);
        this.key = key;
        this.component = SelectComponent
        //          ,       
        this.props = { emitter, ikey: key, values: values, defaultValue: defaultValue};
    }
}

,
var AddTextFieldComponent = {
   //   -  ,        InputControl
    template: '<button type="button" class="btn btn-outline-light">' +
        '<i class="fa fa-plus-circle"></i>&nbsp;Add Value</button>',
    data() {
        return {
            value: ""
        };
    },
    methods: {
        //    ,    ,  InputControl,   id    
        getCount(node, prefix) {
            let count = 0;
            node.controls.forEach((value, key, map) => {
                if (key.startsWith(prefix) && value instanceof InputControl) {
                    ++count;
                }
            });

            return count;
        },
        //         ,     
        update(e) {
            let count = this.methods.getCount(this.node, this.prefix)
            this.node.addControl(new InputControl(this.editor, this.prefix + count))
            //     ,   Rete      
            this.node.update()
            this.emitter.view.updateConnections(this)
            //         ,      json  ,     
            this.putData(this.iKey, count + 1)
        }
    },
    mounted() {
        const _self = this;
        this.root.onclick = function(event) {
            _self.root.update()
        }
    }
};

class AddTextFieldControl extends Rete.Control {
    constructor(emitter, key, prefix, node, inputPlaceholder) {
        super(key);
        this.key = key;
        this.component = AddTextFieldComponent
        this.props = { emitter, iKey: key, prefix: prefix, node: node, inputPlaceholder: inputPlaceholder};
    }
}

class FilterByTagValueComponent extends Rete.Component {
    constructor(){
        super("Filter_by_Tag_Value");
    }

    builder(node) {
        //         ,       osm. 
        //     Rete  ,         
        var input = new Rete.Input('osm',"Map Data", osmSocket);
        var output = new Rete.Output('osm', "Filtered Map Data", osmSocket);
        //     
        var tagNameInput = new InputControl(this.editor, 'tag_name')
        //       
        var  modeControl = new SelectControl(this.editor,
            "mode",
            [["EQUAL", "=="], ["NOT_EQUAL", "!="], ["GREATER", ">"], ["LESS", "<"], ["GE", ">="], ["LE", ">="]],
            "EQUAL")
        //   
        node.addInput(input)
            .addControl(tagNameInput)
            .addControl(modeControl)
            .addControl(new AddTextFieldControl(this.editor, "tag_valueCount", "tag_value", node, "Tag Value"))
        //       json -  ,     ,    
       //  data.tag_valueCount  AddTextFieldControl,  
        let valuesCount = 1;
        if (node.data.tag_valueCount !== undefined) {
            valuesCount = node.data.tag_valueCount
        }
        //    InputControl
        node.addControl(new InputControl(this.editor, 'tag_value'))
        for (let i = 1; i < valuesCount; ++i) {
            node.addControl(new InputControl(this.editor, 'tag_value' + i))
        }

        return node
            .addOutput(output);
    }
}

:




.


, JSON ( Rete ), , .


OSM


, .


: ( leisure=park, OSM):



— , OSM , , . ( ( Java) — ), , , .


, :



: 500- :



, amenity=school, ( — ), .


, . - .


, , ? : ( ), , , Union. :



… . -, — , .. . , amenity=school , , . , .



, , - . :



. . — 4, 3. .
. .. - -, .


:




Rete.js YourMaps .


No futuro, pretendo adicionar ainda mais lá - por exemplo, a capacidade de baixar dados não apenas do OSM, mas também dos meus arquivos GeoJSON, mais tipos de operações e filtros, etc.


Este serviço me ajuda pessoalmente pessoalmente. Por exemplo, quando um aluno precisa mostrar rapidamente algo no mapa OSM - não preciso mais executar o QGIS e lembro da complexa linguagem de consulta do Overpass, clico no gráfico desejado com alguns movimentos do mouse, ele é processado em alguns segundos e você pode ver imediatamente o resultado.


Espero que seja útil para alguns de vocês. Como sempre, estou pronto para ouvir sugestões e desejos aqui nos comentários, ou você pode enviá-lo para evsmirnov@itmo.ru


All Articles