使用dragListener在Google地图上创建可编辑的多边形

3

我希望能够使用拖动监听器实现可编辑的多个多边形。我能够绘制多个多边形,但不知道如何使其可编辑。

我能够移动当前多边形的标记,但是当我尝试移动之前的多边形标记时,应用程序会崩溃。我尝试通过保存多边形列表来解决这个问题,但无法拖动标记。

请在此处查看我的代码。

@Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if (readyToGo()) {
            setContentView(R.layout.activity_maps);
            // Obtain the SupportMapFragment and get notified when the map is ready to be used.
            SupportMapFragment mapFragment = (SupportMapFragment) getSupportFragmentManager()
                    .findFragmentById(R.id.map);
            mapFragment.getMapAsync(this);

            if (savedInstanceState == null) {
                mapFragment.getMapAsync(this);
            }

            mapFragment.getMapAsync(this);
        }
    }

    @Override
    public void onMapReady(GoogleMap googleMap) {
        mMap = googleMap;
        CameraUpdate center =
                CameraUpdateFactory.newLatLng(new LatLng(40.76793169992044,
                        -73.98180484771729));
        CameraUpdate zoom = CameraUpdateFactory.zoomTo(15);
        mMap.moveCamera(center);
        mMap.animateCamera(zoom);
        mMap.setIndoorEnabled(false);

        mMap.setOnMapClickListener(new GoogleMap.OnMapClickListener() {
            @Override
            public void onMapClick(LatLng latLng) {

                Marker marker = mMap.addMarker(new MarkerOptions().position(latLng).draggable(true));
                marker.setTag(latLng);
                markerList.add(marker);
                points.add(latLng);
                drawPolygon(points);
            }
        });


        mMap.setOnMarkerDragListener(new GoogleMap.OnMarkerDragListener() {
            @Override
            public void onMarkerDragStart(Marker marker) {

            }

            @Override
            public void onMarkerDrag(Marker marker) {
                updateMarkerLocation(marker, false);
            }

            @Override
            public void onMarkerDragEnd(Marker marker) {
                updateMarkerLocation(marker, true);

            }
        });

    }

    public void closePolygon(View view) {

    }

    public void newPolygon(View view) {

//
        points.clear();
        markerList.clear();
        polygon = null;
//        mMap.clear();
    }

    private void updateMarkerLocation(Marker marker, boolean calculate) {
        LatLng latLng = (LatLng) marker.getTag();
        int position = points.indexOf(latLng);
        points.set(position, marker.getPosition());
        marker.setTag(marker.getPosition());
        drawPolygon(points);

    }

    private void drawPolygon(List<LatLng> latLngList) {
        if (polygon != null) {
            polygon.remove();
        }
        polygonOptions = new PolygonOptions();
        polygonOptions.addAll(latLngList);
        polygon = mMap.addPolygon(polygonOptions);

    }

}

我能否获取一个Android示例代码,其中绘制了多边形并且标记是可移动的? - Tara
@Tara https://github.com/arpit999/Polygon 祝您愉快 :) - Arpit Patel
@Tara 你是拿着标记笔并拖动它吗? - Arpit Patel
我想要实现拖拽功能。 - Tara
试着拿住标记并拖动它。在我检查过的项目上它有效。 - Arpit Patel
显示剩余5条评论
1个回答

4

基本上,这种方法将标记和点作为与每个多边形相关联的集合保留。它通过假设在 5 个标记后创建一个新的多边形(等同于添加多边形)来简化事情。

更新:为了使用在 github 布局中定义的“新建多边形”按钮。按钮侦听器仅设置一个标志,并且不使用 size=5 检查,而是用该标志替换检查。

维护从任何标记到其对应列表的映射,供 updateMarkerLocation 方法使用。

所有这些都是基于任何标记都由地图 API getId() 提供唯一 id 的事实,实际上是类似于 "m7" 的字符串。

我列出了更新的部分:

// Map a marker id to its corresponding list (represented by the root marker id)
HashMap<String,String> markerToList = new HashMap<>();

// A list of markers for each polygon (designated by the marker root).
HashMap<String,List<Marker>> polygonMarkers = new HashMap<>();

// A list of polygon points for each polygon (designed by the marker root).
HashMap<String,List<LatLng>> polygonPoints = new HashMap<>();

// List of polygons (designated by marker root).
HashMap<String,Polygon> polygons = new HashMap<>();

// The active polygon (designated by marker root) - polygon added to.
String markerListKey;

// Flag used to record when the 'New Polygon' button is pressed.  Next map
// click starts a new polygon.
boolean newPolygon = false;

@Override
public void onMapReady(GoogleMap googleMap) {
    mMap = googleMap;
    CameraUpdate center =
            CameraUpdateFactory.newLatLng(new LatLng(40.76793169992044,
                    -73.98180484771729));
    CameraUpdate zoom = CameraUpdateFactory.zoomTo(15);
    mMap.moveCamera(center);
    mMap.animateCamera(zoom);
    mMap.setIndoorEnabled(false);

    Button b = findViewById(R.id.bt_new_polygon);
    b.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            newPolygon = true;
        }
    });

    mMap.setOnMapClickListener(new GoogleMap.OnMapClickListener() {
        @Override
        public void onMapClick(LatLng latLng) {

            Marker marker = mMap.addMarker(new MarkerOptions().position(latLng).draggable(true));
            marker.setTag(latLng);

            // Special case for very first marker.
            if (polygonMarkers.size() == 0) {
                polygonMarkers.put(marker.getId(),new ArrayList<Marker>());
                // only 0 or 1 polygons so just add it to new one or existing one.
                markerList = new ArrayList<>();
                points = new ArrayList<>();
                polygonMarkers.put(marker.getId(),markerList);
                polygonPoints.put(marker.getId(),points);
                markerListKey = marker.getId();
            }

            if (newPolygon) {
                newPolygon = false;
                markerList = new ArrayList<>();
                points = new ArrayList<>();
                polygonMarkers.put(marker.getId(),markerList);
                polygonPoints.put(marker.getId(),points);
                markerListKey = marker.getId();
            }

            markerList.add(marker);
            points.add(latLng);
            markerToList.put(marker.getId(),markerListKey);

            drawPolygon(markerListKey, points);
        }
    });


private void updateMarkerLocation(Marker marker, boolean calculate) {

    // Use the marker to figure out which polygon list to use...
    List<LatLng> pts = polygonPoints.get(markerToList.get(marker.getId()));
    
    // This is much the same except use the retrieved point list.
    LatLng latLng = (LatLng) marker.getTag();
    int position = pts.indexOf(latLng);
    pts.set(position, marker.getPosition());
    marker.setTag(marker.getPosition());
    drawPolygon(markerToList.get(marker.getId()),pts);

}

private void drawPolygon(String mKey, List<LatLng> latLngList) {

    // Use the existing polygon (if any) for the root marker.
    Polygon polygon = polygons.get(mKey);
    if (polygon != null) {
        polygon.remove();
    }
    polygonOptions = new PolygonOptions();
    polygonOptions.addAll(latLngList);
    polygon = mMap.addPolygon(polygonOptions);
    
    // And update the list for the root marker.
    polygons.put(mKey,polygon);
}

初始

通过在地图上单击,初始添加了3个多边形的集合...

enter image description here

修改

然后显示一个图像,其中每个多边形中的一个点被拉伸...

enter image description here


谢谢你的回答,我会尝试并告诉你结果。 - Arpit Patel
这就是我需要的,但有一个问题:它仅限于5个标记。我想在按钮点击后获得新的多边形。 - Arpit Patel
查看了 GitHub 项目中的布局,并为 bt_new_polygon 添加了按钮侦听器,它仅设置一个标志。在下一张地图单击时,将启动一个新的多边形-而不是使用多边形大小。不太确定用户期望的用户体验,但希望可以为您提供一些东西来使用。也不确定关闭按钮应该如何工作。 - user2711811

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接