Desenvolvimento de um APP (software) para mapeamento automatizado

Olá a todos.

Estou tentando desenvolver um sistema próprio, onde a ideia é carregar o mapa do google maps em um frame, e selecionar uma área aleatória através de pontos (coordenadas) com o mouse ou via touchscreen. Esta área selecionada será usada para configurar rotas de voo, para um drone fotografar o local selecionado, seguindo uma configuração pré-determinada, de forma autônoma.

Isso não é nada novo. Existem APPs que fazem esse tipo de tarefa, tais como:

  • Drone Harmony
  • DRONE DEPLOY
  • MapPilotPRO

Pesquisei sobre uma forma de desenvolver essa ideia, mas não estou chegando a lugar algum. Ou não estou sabendo usar as palavras corretas em minha pesquisa.

Nos APPs acima, dependendo da necessidade, tornam-se bem caros. E realmente gostaria de desenvolver esse conhecimento. Se o APP existe, então é possível de fazer outro.

A linguagem de programação que mais tenho familiaridade é o JAVA, mas nada me impede de aprender algo novo, pois o mais importante é entender a lógica de programação.

Bom dia!

Cara, acredito que codar isso em Javascript/Typescript é beeeem mais simples do que em Java.
Você pode usar o LeafLet ao invés do Google Maps por conta do custo também.

Um pequeno ex de como mostrar e mapa e desenhar na tela, usando React:

import React, { useRef } from "react";
import { MapContainer, TileLayer } from "react-leaflet";
import { FeatureGroup } from "react-leaflet";
import { EditControl } from "react-leaflet-draw";
import "leaflet/dist/leaflet.css";
import "leaflet-draw/dist/leaflet.draw.css";

const App = () => {
  const mapRef = useRef();

  const handleDrawCreate = (e) => {
    const layer = e.layer;
    const { _latlngs } = layer;
    const coordinates = _latlngs[0];
    console.log(coordinates);
  };

  return (
    <div>
      <MapContainer center={[51.505, -0.09]} zoom={13} style={{ height: "100vh", width: "100%" }}>
        <TileLayer
          url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"
          attribution='&copy; <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors'
        />
        <FeatureGroup>
          <EditControl
            position="topright"
            onCreated={handleDrawCreate}
            draw={{
              rectangle: false,
              polyline: false,
              circle: false,
              marker: false,
              polygon: true,
            }}
          />
        </FeatureGroup>
      </MapContainer>
    </div>
  );
};

export default App;

Exemplo da aplicação rodando:

Achei algumas empresas que fazem integração com drones:

https://developer.dji.com/doc/mobile-sdk-tutorial/en/


Como você comentou que usa Java, o back pode ser em SpringBoot. Mas se quiser migrar pro JS, existe o NestJS (node), que tá bem legal:

Acredito que se vc empacar em algo, o ChatGPT consegue te dar uma mão forte nisso.

  • sdks:

Achei um exemplo de um cara fazendo uma espécie de coordenadas a se seguir:



Boa sorte!

Em JavaFX tem o Gluon Maps com uma versão livre.

Exemplo do Codigo para Desktop/Mobile/Raspberry e afins

/*
 * Copyright (c) 2021, Gluon
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *     * Redistributions of source code must retain the above copyright
 * notice, this list of conditions and the following disclaimer.
 *     * Redistributions in binary form must reproduce the above copyright
 * notice, this list of conditions and the following disclaimer in the
 * documentation and/or other materials provided with the distribution.
 *     * Neither the name of Gluon, any associated website, nor the
 * names of its contributors may be used to endorse or promote products
 * derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL GLUON BE LIABLE FOR ANY
 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */
package com.gluonhq;

import com.gluonhq.attach.position.Position;
import com.gluonhq.attach.position.PositionService;
import com.gluonhq.attach.util.Platform;
import com.gluonhq.attach.util.Services;
import com.gluonhq.maps.MapLayer;
import com.gluonhq.maps.MapPoint;
import com.gluonhq.maps.MapView;
import javafx.application.Application;
import javafx.beans.property.ReadOnlyObjectProperty;
import javafx.geometry.Pos;
import javafx.geometry.Rectangle2D;
import javafx.scene.Scene;
import javafx.scene.control.Label;
import javafx.scene.layout.BorderPane;
import javafx.scene.paint.Color;
import javafx.scene.shape.Circle;
import javafx.stage.Screen;
import javafx.stage.Stage;

public class HelloMaps extends Application {

    private static final int DEFAULT_ZOOM = 3;
    private static final double DEFAULT_LAT = 50.0;
    private static final double DEFAULT_LONG = 4.0;

    private MapPoint mapPoint;

    @Override
    public void start(Stage stage) {
        MapView view = new MapView();
        view.addLayer(positionLayer());
        view.setZoom(DEFAULT_ZOOM);
        Scene scene;
        if (Platform.isDesktop()) {
            scene = new Scene(view, 600, 700);
            stage.setTitle("Gluon Maps Demo");
        } else {
            BorderPane bp = new BorderPane();
            bp.setCenter(view);
            final Label label = new Label("Gluon Maps Demo");
            label.setAlignment(Pos.CENTER);
            label.setMaxWidth(Double.MAX_VALUE);
            label.setStyle("-fx-background-color: dimgrey; -fx-text-fill: white;");
            bp.setTop(label);
            Rectangle2D bounds = Screen.getPrimary().getVisualBounds();
            scene = new Scene(bp, bounds.getWidth(), bounds.getHeight());
        }
        stage.setScene(scene);
        stage.show();

        view.flyTo(1., mapPoint, 2.);
    }

    private MapLayer positionLayer() {
        return Services.get(PositionService.class)
                .map(positionService -> {
                    positionService.start();

                    ReadOnlyObjectProperty<Position> positionProperty = positionService.positionProperty();
                    Position position = positionProperty.get();
                    if (position == null) {
                        position = new Position(DEFAULT_LAT, DEFAULT_LONG);
                    }
                    mapPoint = new MapPoint(position.getLatitude(), position.getLongitude());

                    PoiLayer answer = new PoiLayer();
                    answer.addPoint(mapPoint, new Circle(7, Color.RED));

                    positionProperty.addListener(e -> {
                        Position pos = positionProperty.get();
                        mapPoint.update(pos.getLatitude(), pos.getLongitude());
                    });
                    return answer;
                })
                .orElseGet(() -> {
                    PoiLayer answer = new PoiLayer();
                    mapPoint = new MapPoint(DEFAULT_LAT, DEFAULT_LONG);
                    answer.addPoint(mapPoint, new Circle(7, Color.RED));
                    return answer;
                });
    }

    public static void main(String[] args) {
        launch(args);
    }
}