From 495301410c41c96723dad026b92d2c79f88cf97c Mon Sep 17 00:00:00 2001 From: Joonas Kerttula Date: Thu, 29 Feb 2024 14:22:20 +0200 Subject: [PATCH] [google_maps_flutter_web] Add clustering support --- .../google_maps_flutter_web/AUTHORS | 1 + .../google_maps_flutter_web/CHANGELOG.md | 4 + .../google_maps_flutter_web/README.md | 13 ++ .../google_maps_controller_test.mocks.dart | 43 ----- .../google_maps_plugin_test.dart | 4 +- .../google_maps_plugin_test.mocks.dart | 51 +++--- .../marker_clustering_test.dart | 170 ++++++++++++++++++ .../integration_test/markers_test.dart | 8 +- .../example/pubspec.yaml | 2 +- .../example/web/index.html | 1 + .../lib/google_maps_flutter_web.dart | 1 + .../lib/src/convert.dart | 12 +- .../lib/src/google_maps_controller.dart | 46 ++++- .../lib/src/google_maps_flutter_web.dart | 13 ++ .../lib/src/google_maps_inspector_web.dart | 22 ++- .../lib/src/marker.dart | 9 +- .../lib/src/marker_clustering.dart | 153 ++++++++++++++++ .../lib/src/marker_clustering_js_interop.dart | 112 ++++++++++++ .../lib/src/markers.dart | 59 ++++-- .../google_maps_flutter_web/pubspec.yaml | 5 +- 20 files changed, 622 insertions(+), 107 deletions(-) create mode 100644 packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/marker_clustering_test.dart create mode 100644 packages/google_maps_flutter/google_maps_flutter_web/lib/src/marker_clustering.dart create mode 100644 packages/google_maps_flutter/google_maps_flutter_web/lib/src/marker_clustering_js_interop.dart diff --git a/packages/google_maps_flutter/google_maps_flutter_web/AUTHORS b/packages/google_maps_flutter/google_maps_flutter_web/AUTHORS index b5b0e84d4734..906ab219beee 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/AUTHORS +++ b/packages/google_maps_flutter/google_maps_flutter_web/AUTHORS @@ -65,3 +65,4 @@ Anton Borries Alex Li Rahul Raj <64.rahulraj@gmail.com> Justin Baumann +Joonas Kerttula diff --git a/packages/google_maps_flutter/google_maps_flutter_web/CHANGELOG.md b/packages/google_maps_flutter/google_maps_flutter_web/CHANGELOG.md index 1df35f0c3c04..ac62fea05bc5 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/CHANGELOG.md +++ b/packages/google_maps_flutter/google_maps_flutter_web/CHANGELOG.md @@ -1,3 +1,7 @@ +## 0.5.7 + +* Adds support for marker clustering. + ## 0.5.6+1 * Fixes an issue where `dart:js_interop` object literal factories did not diff --git a/packages/google_maps_flutter/google_maps_flutter_web/README.md b/packages/google_maps_flutter/google_maps_flutter_web/README.md index b52da95119cb..f057847578cc 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/README.md +++ b/packages/google_maps_flutter/google_maps_flutter_web/README.md @@ -50,6 +50,19 @@ To request multiple libraries, separate them with commas: Now you should be able to use the Google Maps plugin normally. +## Marker clustering + +If you need marker clustering support, modify the tag to load the [js-markerclusterer](https://github.com/googlemaps/js-markerclusterer#install) library, like so: + +```html + + + + + + +``` + ## Limitations of the web version The following map options are not available in web, because the map doesn't rotate there: diff --git a/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/google_maps_controller_test.mocks.dart b/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/google_maps_controller_test.mocks.dart index c85599934ba4..ee6ffe02a420 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/google_maps_controller_test.mocks.dart +++ b/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/google_maps_controller_test.mocks.dart @@ -42,7 +42,6 @@ class MockCirclesController extends _i1.Mock implements _i3.CirclesController { returnValue: <_i4.CircleId, _i3.CircleController>{}, returnValueForMissingStub: <_i4.CircleId, _i3.CircleController>{}, ) as Map<_i4.CircleId, _i3.CircleController>); - @override _i2.GMap get googleMap => (super.noSuchMethod( Invocation.getter(#googleMap), @@ -55,7 +54,6 @@ class MockCirclesController extends _i1.Mock implements _i3.CirclesController { Invocation.getter(#googleMap), ), ) as _i2.GMap); - @override set googleMap(_i2.GMap? _googleMap) => super.noSuchMethod( Invocation.setter( @@ -64,14 +62,12 @@ class MockCirclesController extends _i1.Mock implements _i3.CirclesController { ), returnValueForMissingStub: null, ); - @override int get mapId => (super.noSuchMethod( Invocation.getter(#mapId), returnValue: 0, returnValueForMissingStub: 0, ) as int); - @override set mapId(int? _mapId) => super.noSuchMethod( Invocation.setter( @@ -80,7 +76,6 @@ class MockCirclesController extends _i1.Mock implements _i3.CirclesController { ), returnValueForMissingStub: null, ); - @override void addCircles(Set<_i4.Circle>? circlesToAdd) => super.noSuchMethod( Invocation.method( @@ -89,7 +84,6 @@ class MockCirclesController extends _i1.Mock implements _i3.CirclesController { ), returnValueForMissingStub: null, ); - @override void changeCircles(Set<_i4.Circle>? circlesToChange) => super.noSuchMethod( Invocation.method( @@ -98,7 +92,6 @@ class MockCirclesController extends _i1.Mock implements _i3.CirclesController { ), returnValueForMissingStub: null, ); - @override void removeCircles(Set<_i4.CircleId>? circleIdsToRemove) => super.noSuchMethod( @@ -108,7 +101,6 @@ class MockCirclesController extends _i1.Mock implements _i3.CirclesController { ), returnValueForMissingStub: null, ); - @override void bindToMap( int? mapId, @@ -137,7 +129,6 @@ class MockPolygonsController extends _i1.Mock returnValue: <_i4.PolygonId, _i3.PolygonController>{}, returnValueForMissingStub: <_i4.PolygonId, _i3.PolygonController>{}, ) as Map<_i4.PolygonId, _i3.PolygonController>); - @override _i2.GMap get googleMap => (super.noSuchMethod( Invocation.getter(#googleMap), @@ -150,7 +141,6 @@ class MockPolygonsController extends _i1.Mock Invocation.getter(#googleMap), ), ) as _i2.GMap); - @override set googleMap(_i2.GMap? _googleMap) => super.noSuchMethod( Invocation.setter( @@ -159,14 +149,12 @@ class MockPolygonsController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override int get mapId => (super.noSuchMethod( Invocation.getter(#mapId), returnValue: 0, returnValueForMissingStub: 0, ) as int); - @override set mapId(int? _mapId) => super.noSuchMethod( Invocation.setter( @@ -175,7 +163,6 @@ class MockPolygonsController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void addPolygons(Set<_i4.Polygon>? polygonsToAdd) => super.noSuchMethod( Invocation.method( @@ -184,7 +171,6 @@ class MockPolygonsController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void changePolygons(Set<_i4.Polygon>? polygonsToChange) => super.noSuchMethod( Invocation.method( @@ -193,7 +179,6 @@ class MockPolygonsController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void removePolygons(Set<_i4.PolygonId>? polygonIdsToRemove) => super.noSuchMethod( @@ -203,7 +188,6 @@ class MockPolygonsController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void bindToMap( int? mapId, @@ -232,7 +216,6 @@ class MockPolylinesController extends _i1.Mock returnValue: <_i4.PolylineId, _i3.PolylineController>{}, returnValueForMissingStub: <_i4.PolylineId, _i3.PolylineController>{}, ) as Map<_i4.PolylineId, _i3.PolylineController>); - @override _i2.GMap get googleMap => (super.noSuchMethod( Invocation.getter(#googleMap), @@ -245,7 +228,6 @@ class MockPolylinesController extends _i1.Mock Invocation.getter(#googleMap), ), ) as _i2.GMap); - @override set googleMap(_i2.GMap? _googleMap) => super.noSuchMethod( Invocation.setter( @@ -254,14 +236,12 @@ class MockPolylinesController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override int get mapId => (super.noSuchMethod( Invocation.getter(#mapId), returnValue: 0, returnValueForMissingStub: 0, ) as int); - @override set mapId(int? _mapId) => super.noSuchMethod( Invocation.setter( @@ -270,7 +250,6 @@ class MockPolylinesController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void addPolylines(Set<_i4.Polyline>? polylinesToAdd) => super.noSuchMethod( Invocation.method( @@ -279,7 +258,6 @@ class MockPolylinesController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void changePolylines(Set<_i4.Polyline>? polylinesToChange) => super.noSuchMethod( @@ -289,7 +267,6 @@ class MockPolylinesController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void removePolylines(Set<_i4.PolylineId>? polylineIdsToRemove) => super.noSuchMethod( @@ -299,7 +276,6 @@ class MockPolylinesController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void bindToMap( int? mapId, @@ -327,7 +303,6 @@ class MockMarkersController extends _i1.Mock implements _i3.MarkersController { returnValue: <_i4.MarkerId, _i3.MarkerController>{}, returnValueForMissingStub: <_i4.MarkerId, _i3.MarkerController>{}, ) as Map<_i4.MarkerId, _i3.MarkerController>); - @override _i2.GMap get googleMap => (super.noSuchMethod( Invocation.getter(#googleMap), @@ -340,7 +315,6 @@ class MockMarkersController extends _i1.Mock implements _i3.MarkersController { Invocation.getter(#googleMap), ), ) as _i2.GMap); - @override set googleMap(_i2.GMap? _googleMap) => super.noSuchMethod( Invocation.setter( @@ -349,14 +323,12 @@ class MockMarkersController extends _i1.Mock implements _i3.MarkersController { ), returnValueForMissingStub: null, ); - @override int get mapId => (super.noSuchMethod( Invocation.getter(#mapId), returnValue: 0, returnValueForMissingStub: 0, ) as int); - @override set mapId(int? _mapId) => super.noSuchMethod( Invocation.setter( @@ -365,7 +337,6 @@ class MockMarkersController extends _i1.Mock implements _i3.MarkersController { ), returnValueForMissingStub: null, ); - @override void addMarkers(Set<_i4.Marker>? markersToAdd) => super.noSuchMethod( Invocation.method( @@ -374,7 +345,6 @@ class MockMarkersController extends _i1.Mock implements _i3.MarkersController { ), returnValueForMissingStub: null, ); - @override void changeMarkers(Set<_i4.Marker>? markersToChange) => super.noSuchMethod( Invocation.method( @@ -383,7 +353,6 @@ class MockMarkersController extends _i1.Mock implements _i3.MarkersController { ), returnValueForMissingStub: null, ); - @override void removeMarkers(Set<_i4.MarkerId>? markerIdsToRemove) => super.noSuchMethod( @@ -393,7 +362,6 @@ class MockMarkersController extends _i1.Mock implements _i3.MarkersController { ), returnValueForMissingStub: null, ); - @override void showMarkerInfoWindow(_i4.MarkerId? markerId) => super.noSuchMethod( Invocation.method( @@ -402,7 +370,6 @@ class MockMarkersController extends _i1.Mock implements _i3.MarkersController { ), returnValueForMissingStub: null, ); - @override void hideMarkerInfoWindow(_i4.MarkerId? markerId) => super.noSuchMethod( Invocation.method( @@ -411,7 +378,6 @@ class MockMarkersController extends _i1.Mock implements _i3.MarkersController { ), returnValueForMissingStub: null, ); - @override bool isInfoWindowShown(_i4.MarkerId? markerId) => (super.noSuchMethod( Invocation.method( @@ -421,7 +387,6 @@ class MockMarkersController extends _i1.Mock implements _i3.MarkersController { returnValue: false, returnValueForMissingStub: false, ) as bool); - @override void bindToMap( int? mapId, @@ -456,7 +421,6 @@ class MockTileOverlaysController extends _i1.Mock Invocation.getter(#googleMap), ), ) as _i2.GMap); - @override set googleMap(_i2.GMap? _googleMap) => super.noSuchMethod( Invocation.setter( @@ -465,14 +429,12 @@ class MockTileOverlaysController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override int get mapId => (super.noSuchMethod( Invocation.getter(#mapId), returnValue: 0, returnValueForMissingStub: 0, ) as int); - @override set mapId(int? _mapId) => super.noSuchMethod( Invocation.setter( @@ -481,7 +443,6 @@ class MockTileOverlaysController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void addTileOverlays(Set<_i4.TileOverlay>? tileOverlaysToAdd) => super.noSuchMethod( @@ -491,7 +452,6 @@ class MockTileOverlaysController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void changeTileOverlays(Set<_i4.TileOverlay>? tileOverlays) => super.noSuchMethod( @@ -501,7 +461,6 @@ class MockTileOverlaysController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void removeTileOverlays(Set<_i4.TileOverlayId>? tileOverlayIds) => super.noSuchMethod( @@ -511,7 +470,6 @@ class MockTileOverlaysController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void clearTileCache(_i4.TileOverlayId? tileOverlayId) => super.noSuchMethod( Invocation.method( @@ -520,7 +478,6 @@ class MockTileOverlaysController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void bindToMap( int? mapId, diff --git a/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/google_maps_plugin_test.dart b/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/google_maps_plugin_test.dart index 36b4d11e07d0..b84b267c0425 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/google_maps_plugin_test.dart +++ b/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/google_maps_plugin_test.dart @@ -361,10 +361,10 @@ void main() { }); testWidgets('isMarkerInfoWindowShown', (WidgetTester tester) async { - when(controller.isInfoWindowShown(any)).thenReturn(true); - const MarkerId markerId = MarkerId('testing-123'); + when(controller.isInfoWindowShown(markerId)).thenReturn(true); + await plugin.isMarkerInfoWindowShown(markerId, mapId: mapId); verify(controller.isInfoWindowShown(markerId)); diff --git a/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/google_maps_plugin_test.mocks.dart b/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/google_maps_plugin_test.mocks.dart index 3f84b40adc88..582d6ddc1dee 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/google_maps_plugin_test.mocks.dart +++ b/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/google_maps_plugin_test.mocks.dart @@ -9,6 +9,7 @@ import 'package:google_maps/google_maps.dart' as _i5; import 'package:google_maps_flutter_platform_interface/google_maps_flutter_platform_interface.dart' as _i2; import 'package:google_maps_flutter_web/google_maps_flutter_web.dart' as _i4; +import 'package:google_maps_flutter_web/src/marker_clustering.dart' as _i6; import 'package:mockito/mockito.dart' as _i1; // ignore_for_file: type=lint @@ -94,42 +95,37 @@ class MockGoogleMapController extends _i1.Mock Invocation.getter(#configuration), ), ) as _i2.MapConfiguration); - @override - _i3.StreamController<_i2.MapEvent> get stream => (super.noSuchMethod( + _i3.StreamController<_i2.MapEvent> get stream => (super.noSuchMethod( Invocation.getter(#stream), - returnValue: _FakeStreamController_1<_i2.MapEvent>( + returnValue: _FakeStreamController_1<_i2.MapEvent>( this, Invocation.getter(#stream), ), returnValueForMissingStub: - _FakeStreamController_1<_i2.MapEvent>( + _FakeStreamController_1<_i2.MapEvent>( this, Invocation.getter(#stream), ), - ) as _i3.StreamController<_i2.MapEvent>); - + ) as _i3.StreamController<_i2.MapEvent>); @override - _i3.Stream<_i2.MapEvent> get events => (super.noSuchMethod( + _i3.Stream<_i2.MapEvent> get events => (super.noSuchMethod( Invocation.getter(#events), - returnValue: _i3.Stream<_i2.MapEvent>.empty(), - returnValueForMissingStub: _i3.Stream<_i2.MapEvent>.empty(), - ) as _i3.Stream<_i2.MapEvent>); - + returnValue: _i3.Stream<_i2.MapEvent>.empty(), + returnValueForMissingStub: _i3.Stream<_i2.MapEvent>.empty(), + ) as _i3.Stream<_i2.MapEvent>); @override bool get isInitialized => (super.noSuchMethod( Invocation.getter(#isInitialized), returnValue: false, returnValueForMissingStub: false, ) as bool); - @override List<_i5.MapTypeStyle> get styles => (super.noSuchMethod( Invocation.getter(#styles), returnValue: <_i5.MapTypeStyle>[], returnValueForMissingStub: <_i5.MapTypeStyle>[], ) as List<_i5.MapTypeStyle>); - @override void debugSetOverrides({ _i4.DebugCreateMapFunction? createMap, @@ -138,6 +134,7 @@ class MockGoogleMapController extends _i1.Mock _i4.CirclesController? circles, _i4.PolygonsController? polygons, _i4.PolylinesController? polylines, + _i6.ClusterManagersController? clusterManagers, _i4.TileOverlaysController? tileOverlays, }) => super.noSuchMethod( @@ -151,12 +148,12 @@ class MockGoogleMapController extends _i1.Mock #circles: circles, #polygons: polygons, #polylines: polylines, + #clusterManagers: clusterManagers, #tileOverlays: tileOverlays, }, ), returnValueForMissingStub: null, ); - @override void init() => super.noSuchMethod( Invocation.method( @@ -165,7 +162,6 @@ class MockGoogleMapController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void updateMapConfiguration(_i2.MapConfiguration? update) => super.noSuchMethod( @@ -175,7 +171,6 @@ class MockGoogleMapController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void updateStyles(List<_i5.MapTypeStyle>? styles) => super.noSuchMethod( Invocation.method( @@ -184,7 +179,6 @@ class MockGoogleMapController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override _i3.Future<_i2.LatLngBounds> getVisibleRegion() => (super.noSuchMethod( Invocation.method( @@ -207,7 +201,6 @@ class MockGoogleMapController extends _i1.Mock ), )), ) as _i3.Future<_i2.LatLngBounds>); - @override _i3.Future<_i2.ScreenCoordinate> getScreenCoordinate(_i2.LatLng? latLng) => (super.noSuchMethod( @@ -232,7 +225,6 @@ class MockGoogleMapController extends _i1.Mock ), )), ) as _i3.Future<_i2.ScreenCoordinate>); - @override _i3.Future<_i2.LatLng> getLatLng(_i2.ScreenCoordinate? screenCoordinate) => (super.noSuchMethod( @@ -255,7 +247,6 @@ class MockGoogleMapController extends _i1.Mock ), )), ) as _i3.Future<_i2.LatLng>); - @override _i3.Future moveCamera(_i2.CameraUpdate? cameraUpdate) => (super.noSuchMethod( @@ -266,7 +257,6 @@ class MockGoogleMapController extends _i1.Mock returnValue: _i3.Future.value(), returnValueForMissingStub: _i3.Future.value(), ) as _i3.Future); - @override _i3.Future getZoomLevel() => (super.noSuchMethod( Invocation.method( @@ -276,7 +266,6 @@ class MockGoogleMapController extends _i1.Mock returnValue: _i3.Future.value(0.0), returnValueForMissingStub: _i3.Future.value(0.0), ) as _i3.Future); - @override void updateCircles(_i2.CircleUpdates? updates) => super.noSuchMethod( Invocation.method( @@ -285,7 +274,6 @@ class MockGoogleMapController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void updatePolygons(_i2.PolygonUpdates? updates) => super.noSuchMethod( Invocation.method( @@ -294,7 +282,6 @@ class MockGoogleMapController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void updatePolylines(_i2.PolylineUpdates? updates) => super.noSuchMethod( Invocation.method( @@ -303,7 +290,6 @@ class MockGoogleMapController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void updateMarkers(_i2.MarkerUpdates? updates) => super.noSuchMethod( Invocation.method( @@ -312,7 +298,15 @@ class MockGoogleMapController extends _i1.Mock ), returnValueForMissingStub: null, ); - + @override + void updateClusterManagers(_i2.ClusterManagerUpdates? updates) => + super.noSuchMethod( + Invocation.method( + #updateClusterManagers, + [updates], + ), + returnValueForMissingStub: null, + ); @override void updateTileOverlays(Set<_i2.TileOverlay>? newOverlays) => super.noSuchMethod( @@ -322,7 +316,6 @@ class MockGoogleMapController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void clearTileCache(_i2.TileOverlayId? id) => super.noSuchMethod( Invocation.method( @@ -331,7 +324,6 @@ class MockGoogleMapController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void showInfoWindow(_i2.MarkerId? markerId) => super.noSuchMethod( Invocation.method( @@ -340,7 +332,6 @@ class MockGoogleMapController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override void hideInfoWindow(_i2.MarkerId? markerId) => super.noSuchMethod( Invocation.method( @@ -349,7 +340,6 @@ class MockGoogleMapController extends _i1.Mock ), returnValueForMissingStub: null, ); - @override bool isInfoWindowShown(_i2.MarkerId? markerId) => (super.noSuchMethod( Invocation.method( @@ -359,7 +349,6 @@ class MockGoogleMapController extends _i1.Mock returnValue: false, returnValueForMissingStub: false, ) as bool); - @override void dispose() => super.noSuchMethod( Invocation.method( diff --git a/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/marker_clustering_test.dart b/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/marker_clustering_test.dart new file mode 100644 index 000000000000..c34d9c533117 --- /dev/null +++ b/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/marker_clustering_test.dart @@ -0,0 +1,170 @@ +// Copyright 2013 The Flutter Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +// ignore_for_file: unnecessary_nullable_for_final_variable_declarations + +import 'dart:async'; + +import 'package:flutter/material.dart'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:google_maps_flutter_platform_interface/google_maps_flutter_platform_interface.dart'; +import 'package:integration_test/integration_test.dart'; + +void main() { + IntegrationTestWidgetsFlutterBinding.ensureInitialized(); + + GoogleMapsFlutterPlatform.instance.enableDebugInspection(); + final GoogleMapsFlutterPlatform plugin = GoogleMapsFlutterPlatform.instance; + final GoogleMapsInspectorPlatform inspector = + GoogleMapsInspectorPlatform.instance!; + + const LatLng mapCenter = LatLng(20, 20); + const CameraPosition initialCameraPosition = + CameraPosition(target: mapCenter); + + group('MarkersController', () { + const int testMapId = 33930; + + testWidgets('Marker clustering', (WidgetTester tester) async { + const ClusterManagerId clusterManagerId = ClusterManagerId('cluster 1'); + + final Set clusterManagers = { + const ClusterManager(clusterManagerId: clusterManagerId), + }; + + // Create the marker with clusterManagerId. + final Set initialMarkers = { + const Marker( + markerId: MarkerId('1'), + position: mapCenter, + clusterManagerId: clusterManagerId), + const Marker( + markerId: MarkerId('2'), + position: mapCenter, + clusterManagerId: clusterManagerId), + }; + + final Completer mapIdCompleter = Completer(); + + await _pumpMap( + tester, + plugin.buildViewWithConfiguration( + testMapId, (int id) => mapIdCompleter.complete(id), + widgetConfiguration: const MapWidgetConfiguration( + initialCameraPosition: initialCameraPosition, + textDirection: TextDirection.ltr, + ), + mapObjects: MapObjects( + clusterManagers: clusterManagers, markers: initialMarkers))); + + final int mapId = await mapIdCompleter.future; + expect(mapId, equals(testMapId)); + + addTearDown(() => plugin.dispose(mapId: mapId)); + + final LatLng latlon = await plugin + .getLatLng(const ScreenCoordinate(x: 0, y: 0), mapId: mapId); + debugPrint(latlon.toString()); + + final List clusters = + await waitForValueMatchingPredicate>( + tester, + () async => inspector.getClusters( + mapId: mapId, clusterManagerId: clusterManagerId), + (List clusters) => clusters.isNotEmpty) ?? + []; + + expect(clusters.length, 1); + expect(clusters[0].markerIds.length, 2); + + // Copy only the first marker with null clusterManagerId. + // This means that both markers should be removed from the cluster. + final Set updatedMarkers = { + _copyMarkerWithClusterManagerId(initialMarkers.first, null) + }; + + final MarkerUpdates markerUpdates = + MarkerUpdates.from(initialMarkers, updatedMarkers); + await plugin.updateMarkers(markerUpdates, mapId: mapId); + + final List updatedClusters = + await waitForValueMatchingPredicate>( + tester, + () async => inspector.getClusters( + mapId: mapId, clusterManagerId: clusterManagerId), + (List clusters) => clusters.isNotEmpty) ?? + []; + + expect(updatedClusters.length, 0); + }); + }); +} + +// Repeatedly checks an asynchronous value against a test condition, waiting +// one frame between each check, returing the value if it passes the predicate +// before [maxTries] is reached. +// +// Returns null if the predicate is never satisfied. +// +// This is useful for cases where the Maps SDK has some internally +// asynchronous operation that we don't have visibility into (e.g., native UI +// animations). +Future waitForValueMatchingPredicate(WidgetTester tester, + Future Function() getValue, bool Function(T) predicate, + {int maxTries = 100}) async { + for (int i = 0; i < maxTries; i++) { + final T value = await getValue(); + if (predicate(value)) { + return value; + } + await tester.pump(); + } + return null; +} + +Marker _copyMarkerWithClusterManagerId( + Marker marker, ClusterManagerId? clusterManagerId) { + return Marker( + markerId: marker.markerId, + alpha: marker.alpha, + anchor: marker.anchor, + consumeTapEvents: marker.consumeTapEvents, + draggable: marker.draggable, + flat: marker.flat, + icon: marker.icon, + infoWindow: marker.infoWindow, + position: marker.position, + rotation: marker.rotation, + visible: marker.visible, + zIndex: marker.zIndex, + onTap: marker.onTap, + onDragStart: marker.onDragStart, + onDrag: marker.onDrag, + onDragEnd: marker.onDragEnd, + clusterManagerId: clusterManagerId, + ); +} + +/// Pumps a [map] widget in [tester] of a certain [size], then waits until it settles. +Future _pumpMap(WidgetTester tester, Widget map, + [Size size = const Size.square(200)]) async { + await tester.pumpWidget(_wrapMap(map, size)); + await tester.pumpAndSettle(); +} + +/// Wraps a [map] in a bunch of widgets so it renders in all platforms. +/// +/// An optional [size] can be passed. +Widget _wrapMap(Widget map, [Size size = const Size.square(200)]) { + return MaterialApp( + home: Scaffold( + body: Center( + child: SizedBox.fromSize( + size: size, + child: map, + ), + ), + ), + ); +} diff --git a/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/markers_test.dart b/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/markers_test.dart index d965435b3a55..81e1eb265574 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/markers_test.dart +++ b/packages/google_maps_flutter/google_maps_flutter_web/example/integration_test/markers_test.dart @@ -11,6 +11,7 @@ import 'package:flutter_test/flutter_test.dart'; import 'package:google_maps/google_maps.dart' as gmaps; import 'package:google_maps_flutter_platform_interface/google_maps_flutter_platform_interface.dart'; import 'package:google_maps_flutter_web/google_maps_flutter_web.dart'; +import 'package:google_maps_flutter_web/src/marker_clustering.dart'; // ignore: implementation_imports import 'package:google_maps_flutter_web/src/utils.dart'; import 'package:http/http.dart' as http; @@ -25,12 +26,17 @@ void main() { group('MarkersController', () { late StreamController> events; late MarkersController controller; + late ClusterManagersController clusterManagersController; late gmaps.GMap map; setUp(() { events = StreamController>(); - controller = MarkersController(stream: events); + + clusterManagersController = ClusterManagersController(stream: events); + controller = MarkersController( + stream: events, clusterManagersController: clusterManagersController); map = gmaps.GMap(createDivElement()); + clusterManagersController.bindToMap(123, map); controller.bindToMap(123, map); }); diff --git a/packages/google_maps_flutter/google_maps_flutter_web/example/pubspec.yaml b/packages/google_maps_flutter/google_maps_flutter_web/example/pubspec.yaml index 28381575ea0a..a7222dfb12ab 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/example/pubspec.yaml +++ b/packages/google_maps_flutter/google_maps_flutter_web/example/pubspec.yaml @@ -9,7 +9,7 @@ environment: dependencies: flutter: sdk: flutter - google_maps_flutter_platform_interface: ^2.5.0 + google_maps_flutter_platform_interface: ^2.6.0 google_maps_flutter_web: path: ../ web: ^0.5.0 diff --git a/packages/google_maps_flutter/google_maps_flutter_web/example/web/index.html b/packages/google_maps_flutter/google_maps_flutter_web/example/web/index.html index 3121d189b913..65f0633e6d47 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/example/web/index.html +++ b/packages/google_maps_flutter/google_maps_flutter_web/example/web/index.html @@ -7,6 +7,7 @@ Browser Tests + diff --git a/packages/google_maps_flutter/google_maps_flutter_web/lib/google_maps_flutter_web.dart b/packages/google_maps_flutter/google_maps_flutter_web/lib/google_maps_flutter_web.dart index fe44d41aa484..cda20cf9f526 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/lib/google_maps_flutter_web.dart +++ b/packages/google_maps_flutter/google_maps_flutter_web/lib/google_maps_flutter_web.dart @@ -24,6 +24,7 @@ import 'package:web/web.dart'; import 'src/dom_window_extension.dart'; import 'src/google_maps_inspector_web.dart'; import 'src/map_styler.dart'; +import 'src/marker_clustering.dart'; import 'src/third_party/to_screen_location/to_screen_location.dart'; import 'src/types.dart'; import 'src/utils.dart'; diff --git a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/convert.dart b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/convert.dart index 6702d5f520b8..9e37a14a528a 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/convert.dart +++ b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/convert.dart @@ -169,20 +169,22 @@ gmaps.LatLng _latLngToGmLatLng(LatLng latLng) { return gmaps.LatLng(latLng.latitude, latLng.longitude); } -LatLng _gmLatLngToLatLng(gmaps.LatLng latLng) { +/// Converts [gmaps.LatLng] to [LatLng]. +LatLng gmLatLngToLatLng(gmaps.LatLng latLng) { return LatLng(latLng.lat.toDouble(), latLng.lng.toDouble()); } -LatLngBounds _gmLatLngBoundsTolatLngBounds(gmaps.LatLngBounds latLngBounds) { +/// Converts a [gmaps.LatLngBounds] into a [LatLngBounds]. +LatLngBounds gmLatLngBoundsTolatLngBounds(gmaps.LatLngBounds latLngBounds) { return LatLngBounds( - southwest: _gmLatLngToLatLng(latLngBounds.southWest), - northeast: _gmLatLngToLatLng(latLngBounds.northEast), + southwest: gmLatLngToLatLng(latLngBounds.southWest), + northeast: gmLatLngToLatLng(latLngBounds.northEast), ); } CameraPosition _gmViewportToCameraPosition(gmaps.GMap map) { return CameraPosition( - target: _gmLatLngToLatLng(map.center ?? _nullGmapsLatLng), + target: gmLatLngToLatLng(map.center ?? _nullGmapsLatLng), bearing: map.heading?.toDouble() ?? 0, tilt: map.tilt?.toDouble() ?? 0, zoom: map.zoom?.toDouble() ?? 0, diff --git a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/google_maps_controller.dart b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/google_maps_controller.dart index c60dd92a0aca..c2fe750de89f 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/google_maps_controller.dart +++ b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/google_maps_controller.dart @@ -29,12 +29,17 @@ class GoogleMapController { _polygons = mapObjects.polygons, _polylines = mapObjects.polylines, _circles = mapObjects.circles, + _clusterManagers = mapObjects.clusterManagers, _tileOverlays = mapObjects.tileOverlays, _lastMapConfiguration = mapConfiguration { _circlesController = CirclesController(stream: _streamController); _polygonsController = PolygonsController(stream: _streamController); _polylinesController = PolylinesController(stream: _streamController); - _markersController = MarkersController(stream: _streamController); + _clusterManagersController = + ClusterManagersController(stream: _streamController); + _markersController = MarkersController( + stream: _streamController, + clusterManagersController: _clusterManagersController!); _tileOverlaysController = TileOverlaysController(); _updateStylesFromConfiguration(mapConfiguration); @@ -60,7 +65,9 @@ class GoogleMapController { final Set _polygons; final Set _polylines; final Set _circles; + final Set _clusterManagers; Set _tileOverlays; + // The configuration passed by the user, before converting to gmaps. // Caching this allows us to re-create the map faithfully when needed. MapConfiguration _lastMapConfiguration = const MapConfiguration(); @@ -118,13 +125,20 @@ class GoogleMapController { PolygonsController? _polygonsController; PolylinesController? _polylinesController; MarkersController? _markersController; + ClusterManagersController? _clusterManagersController; TileOverlaysController? _tileOverlaysController; + // Keeps track if _attachGeometryControllers has been called or not. bool _controllersBoundToMap = false; // Keeps track if the map is moving or not. bool _mapIsMoving = false; + /// The ClusterManagersController of this Map. Only for integration testing. + @visibleForTesting + ClusterManagersController? get clusterManagersController => + _clusterManagersController; + /// Overrides certain properties to install mocks defined during testing. @visibleForTesting void debugSetOverrides({ @@ -134,6 +148,7 @@ class GoogleMapController { CirclesController? circles, PolygonsController? polygons, PolylinesController? polylines, + ClusterManagersController? clusterManagers, TileOverlaysController? tileOverlays, }) { _overrideCreateMap = createMap; @@ -142,6 +157,7 @@ class GoogleMapController { _circlesController = circles ?? _circlesController; _polygonsController = polygons ?? _polygonsController; _polylinesController = polylines ?? _polylinesController; + _clusterManagersController = clusterManagers ?? _clusterManagersController; _tileOverlaysController = tileOverlays ?? _tileOverlaysController; } @@ -197,6 +213,8 @@ class GoogleMapController { _attachMapEvents(map); _attachGeometryControllers(map); + _initClustering(_clusterManagers); + // Now attach the geometry, traffic and any other layers... _renderInitialGeometry(); _setTrafficLayer(map, _lastMapConfiguration.trafficEnabled ?? false); @@ -211,13 +229,13 @@ class GoogleMapController { map.onClick.listen((gmaps.IconMouseEvent event) { assert(event.latLng != null); _streamController.add( - MapTapEvent(_mapId, _gmLatLngToLatLng(event.latLng!)), + MapTapEvent(_mapId, gmLatLngToLatLng(event.latLng!)), ); }); map.onRightclick.listen((gmaps.MapMouseEvent event) { assert(event.latLng != null); _streamController.add( - MapLongPressEvent(_mapId, _gmLatLngToLatLng(event.latLng!)), + MapLongPressEvent(_mapId, gmLatLngToLatLng(event.latLng!)), ); }); map.onBoundsChanged.listen((void _) { @@ -251,6 +269,8 @@ class GoogleMapController { 'Cannot attach a map to a null PolylinesController instance.'); assert(_markersController != null, 'Cannot attach a map to a null MarkersController instance.'); + assert(_clusterManagersController != null, + 'Cannot attach a map to a null ClusterManagersController instance.'); assert(_tileOverlaysController != null, 'Cannot attach a map to a null TileOverlaysController instance.'); @@ -258,11 +278,16 @@ class GoogleMapController { _polygonsController!.bindToMap(_mapId, map); _polylinesController!.bindToMap(_mapId, map); _markersController!.bindToMap(_mapId, map); + _clusterManagersController!.bindToMap(_mapId, map); _tileOverlaysController!.bindToMap(_mapId, map); _controllersBoundToMap = true; } + void _initClustering(Set clusterManagers) { + _clusterManagersController!.addClusterManagers(clusterManagers); + } + // Renders the initial sets of geometry. void _renderInitialGeometry() { assert( @@ -369,7 +394,7 @@ class GoogleMapController { await Future.value(_googleMap!.bounds) ?? _nullGmapsLatLngBounds; - return _gmLatLngBoundsTolatLngBounds(bounds); + return gmLatLngBoundsTolatLngBounds(bounds); } /// Returns the [ScreenCoordinate] for a given viewport [LatLng]. @@ -390,7 +415,7 @@ class GoogleMapController { final gmaps.LatLng latLng = _pixelToLatLng(_googleMap!, screenCoordinate.x, screenCoordinate.y); - return _gmLatLngToLatLng(latLng); + return gmLatLngToLatLng(latLng); } /// Applies a `cameraUpdate` to the current viewport. @@ -447,6 +472,16 @@ class GoogleMapController { _markersController?.removeMarkers(updates.markerIdsToRemove); } + /// Applies [ClusterManagerUpdates] to the currently managed cluster managers. + void updateClusterManagers(ClusterManagerUpdates updates) { + assert(_clusterManagersController != null, + 'Cannot update markers after dispose().'); + _clusterManagersController + ?.addClusterManagers(updates.clusterManagersToAdd); + _clusterManagersController + ?.removeClusterManagers(updates.clusterManagerIdsToRemove); + } + /// Updates the set of [TileOverlay]s. void updateTileOverlays(Set newOverlays) { final MapsObjectUpdates updates = @@ -498,6 +533,7 @@ class GoogleMapController { _polygonsController = null; _polylinesController = null; _markersController = null; + _clusterManagersController = null; _tileOverlaysController = null; _streamController.close(); } diff --git a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/google_maps_flutter_web.dart b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/google_maps_flutter_web.dart index 805887f0d7f5..72c1119ec2e7 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/google_maps_flutter_web.dart +++ b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/google_maps_flutter_web.dart @@ -98,6 +98,14 @@ class GoogleMapsPlugin extends GoogleMapsFlutterPlatform { _map(mapId).updateTileOverlays(newTileOverlays); } + @override + Future updateClusterManagers( + ClusterManagerUpdates clusterManagerUpdates, { + required int mapId, + }) async { + _map(mapId).updateClusterManagers(clusterManagerUpdates); + } + @override Future clearTileCache( TileOverlayId tileOverlayId, { @@ -280,6 +288,10 @@ class GoogleMapsPlugin extends GoogleMapsFlutterPlatform { } @override + Stream onClusterTap({required int mapId}) { + return _events(mapId).whereType(); + } + Future getStyleError({required int mapId}) async { return _map(mapId).lastStyleError; } @@ -339,6 +351,7 @@ class GoogleMapsPlugin extends GoogleMapsFlutterPlatform { void enableDebugInspection() { GoogleMapsInspectorPlatform.instance = GoogleMapsInspectorWeb( (int mapId) => _map(mapId).configuration, + (int mapId) => _map(mapId).clusterManagersController, ); } } diff --git a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/google_maps_inspector_web.dart b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/google_maps_inspector_web.dart index 6d9553152395..98b474309584 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/google_maps_inspector_web.dart +++ b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/google_maps_inspector_web.dart @@ -3,17 +3,25 @@ // found in the LICENSE file. import 'package:google_maps_flutter_platform_interface/google_maps_flutter_platform_interface.dart'; +import 'marker_clustering.dart'; /// Function that gets the [MapConfiguration] for a given `mapId`. typedef ConfigurationProvider = MapConfiguration Function(int mapId); +/// Function that gets the [ClusterManagersController] for a given `mapId`. +typedef ClusterManagersControllerProvider = ClusterManagersController? Function( + int mapId); + /// This platform implementation allows inspecting the running maps. class GoogleMapsInspectorWeb extends GoogleMapsInspectorPlatform { /// Build an "inspector" that is able to look into maps. - GoogleMapsInspectorWeb(ConfigurationProvider configurationProvider) - : _configurationProvider = configurationProvider; + GoogleMapsInspectorWeb(ConfigurationProvider configurationProvider, + ClusterManagersControllerProvider clusterManagersControllerProvider) + : _configurationProvider = configurationProvider, + _clusterManagersControllerProvider = clusterManagersControllerProvider; final ConfigurationProvider _configurationProvider; + final ClusterManagersControllerProvider _clusterManagersControllerProvider; @override Future areBuildingsEnabled({required int mapId}) async { @@ -85,4 +93,14 @@ class GoogleMapsInspectorWeb extends GoogleMapsInspectorPlatform { Future isTrafficEnabled({required int mapId}) async { return _configurationProvider(mapId).trafficEnabled ?? false; } + + @override + Future> getClusters({ + required int mapId, + required ClusterManagerId clusterManagerId, + }) async { + return _clusterManagersControllerProvider(mapId) + ?.getClusters(clusterManagerId) ?? + []; + } } diff --git a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/marker.dart b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/marker.dart index c1b0772a1394..518dce6de775 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/marker.dart +++ b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/marker.dart @@ -15,9 +15,11 @@ class MarkerController { LatLngCallback? onDrag, LatLngCallback? onDragEnd, VoidCallback? onTap, + ClusterManagerId? clusterManagerId, }) : _marker = marker, _infoWindow = infoWindow, - _consumeTapEvents = consumeTapEvents { + _consumeTapEvents = consumeTapEvents, + _clusterManagerId = clusterManagerId { if (onTap != null) { marker.onClick.listen((gmaps.MapMouseEvent event) { onTap.call(); @@ -47,6 +49,8 @@ class MarkerController { final bool _consumeTapEvents; + final ClusterManagerId? _clusterManagerId; + final gmaps.InfoWindow? _infoWindow; bool _infoWindowShown = false; @@ -57,6 +61,9 @@ class MarkerController { /// Returns `true` if the [gmaps.InfoWindow] associated to this marker is being shown. bool get infoWindowShown => _infoWindowShown; + /// Returns [ClusterManagerId] if marker belongs to cluster. + ClusterManagerId? get clusterManagerId => _clusterManagerId; + /// Returns the [gmaps.Marker] associated to this controller. gmaps.Marker? get marker => _marker; diff --git a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/marker_clustering.dart b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/marker_clustering.dart new file mode 100644 index 000000000000..aa2bb890de98 --- /dev/null +++ b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/marker_clustering.dart @@ -0,0 +1,153 @@ +// Copyright 2013 The Flutter Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +import 'dart:async'; + +import 'package:google_maps/google_maps.dart' as gmaps; +import 'package:google_maps_flutter_platform_interface/google_maps_flutter_platform_interface.dart'; + +import '../google_maps_flutter_web.dart'; +import 'marker_clustering_js_interop.dart'; +import 'types.dart'; + +/// A controller class for managing marker clustering. +/// +/// This class maps [ClusterManager] objects to javascript [MarkerClusterer] +/// objects and provides an interface for adding and removing markers from +/// clusters. +class ClusterManagersController extends GeometryController { + /// Creates a new [ClusterManagersController] instance. + /// + /// The [stream] parameter is a required [StreamController] used for + /// emitting map events. + ClusterManagersController( + {required StreamController> stream}) + : _streamController = stream, + _clusterManagerIdToMarkerClusterer = + {}; + + // The stream over which cluster managers broadcast their events + final StreamController> _streamController; + + // A cache of [MarkerClusterer]s indexed by their [ClusterManagerId]. + final Map + _clusterManagerIdToMarkerClusterer; + + /// Adds a set of [ClusterManager] objects to the cache. + void addClusterManagers(Set clusterManagersToAdd) { + clusterManagersToAdd.forEach(_addClusterManager); + } + + void _addClusterManager(ClusterManager clusterManager) { + final MarkerClusterer markerClusterer = createMarkerClusterer( + googleMap, + (gmaps.MapMouseEvent event, MarkerClustererCluster cluster, + gmaps.GMap map) => + _clusterClicked( + clusterManager.clusterManagerId, event, cluster, map)); + + _clusterManagerIdToMarkerClusterer[clusterManager.clusterManagerId] = + markerClusterer; + markerClusterer.onAdd(); + } + + /// Removes a set of [ClusterManagerId]s from the cache. + void removeClusterManagers(Set clusterManagerIdsToRemove) { + clusterManagerIdsToRemove.forEach(_removeClusterManager); + } + + void _removeClusterManager(ClusterManagerId clusterManagerId) { + final MarkerClusterer? markerClusterer = + _clusterManagerIdToMarkerClusterer[clusterManagerId]; + if (markerClusterer != null) { + markerClusterer.clearMarkers(true); + markerClusterer.onRemove(); + } + _clusterManagerIdToMarkerClusterer.remove(clusterManagerId); + } + + /// Adds given [gmaps.Marker] to the [MarkerClusterer] with given + /// [ClusterManagerId]. + void addItem(ClusterManagerId clusterManagerId, gmaps.Marker marker) { + final MarkerClusterer? markerClusterer = + _clusterManagerIdToMarkerClusterer[clusterManagerId]; + if (markerClusterer != null) { + markerClusterer.addMarker(marker, true); + markerClusterer.dirty = true; + } + } + + /// Removes given [gmaps.Marker] from the [MarkerClusterer] with given + /// [ClusterManagerId]. + void removeItem(ClusterManagerId clusterManagerId, gmaps.Marker? marker) { + if (marker != null) { + final MarkerClusterer? markerClusterer = + _clusterManagerIdToMarkerClusterer[clusterManagerId]; + if (markerClusterer != null) { + markerClusterer.removeMarker(marker, true); + markerClusterer.dirty = true; + } + } + } + + /// Renders all cluster managers that are marked as dirty. + /// + /// This mechanism is used to batch changes to the cluster managers to avoid + /// unnecessary re-renders. + /// [MarkerClusterer] is marked as dirty when a marker is added or removed. + void renderDirty() { + _clusterManagerIdToMarkerClusterer.values + .where((MarkerClusterer markerClusterer) => markerClusterer.dirty) + .forEach((MarkerClusterer markerClusterer) { + markerClusterer.dirty = false; + markerClusterer.render(); + }); + } + + /// Returns list of clusters in [MarkerClusterer] with given + /// [ClusterManagerId]. + List getClusters(ClusterManagerId clusterManagerId) { + final MarkerClusterer? markerClusterer = + _clusterManagerIdToMarkerClusterer[clusterManagerId]; + if (markerClusterer != null) { + return markerClusterer.clusters + .map((MarkerClustererCluster cluster) => + _convertCluster(clusterManagerId, cluster)) + .toList(); + } + return []; + } + + void _clusterClicked( + ClusterManagerId clusterManagerId, + gmaps.MapMouseEvent event, + MarkerClustererCluster markerClustererCluster, + gmaps.GMap map) { + if (markerClustererCluster.count > 0 && + markerClustererCluster.bounds != null) { + final Cluster cluster = + _convertCluster(clusterManagerId, markerClustererCluster); + _streamController.add(ClusterTapEvent(mapId, cluster)); + } + } + + /// Converts [MarkerClustererCluster] to [Cluster]. + Cluster _convertCluster(ClusterManagerId clusterManagerId, + MarkerClustererCluster markerClustererCluster) { + final LatLng position = gmLatLngToLatLng(markerClustererCluster.position); + final LatLngBounds bounds = + gmLatLngBoundsTolatLngBounds(markerClustererCluster.bounds!); + + final List markerIds = markerClustererCluster.markers + .map((gmaps.Marker marker) => + MarkerId(marker.get('markerId')! as String)) + .toList(); + return Cluster( + clusterManagerId, + markerIds, + position: position, + bounds: bounds, + ); + } +} diff --git a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/marker_clustering_js_interop.dart b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/marker_clustering_js_interop.dart new file mode 100644 index 000000000000..9e6d6579c866 --- /dev/null +++ b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/marker_clustering_js_interop.dart @@ -0,0 +1,112 @@ +import 'package:google_maps/google_maps.dart' as gmaps; +import 'package:js/js.dart'; + +/// A typedef representing a callback function for handling cluster tap events. +typedef ClusterClickHandler = void Function( + gmaps.MapMouseEvent, MarkerClustererCluster, gmaps.GMap); + +/// The [MarkerClustererOptions] object used to initialize [MarkerClusterer]. +@JS() +@anonymous +abstract class MarkerClustererOptions { + /// Constructs a new [MarkerClustererOptions] object. + external factory MarkerClustererOptions(); + + /// Returns the [gmaps.GMap] object. + external gmaps.GMap? get map; + + /// Sets the [gmaps.GMap] object. + external set map(gmaps.GMap? map); + + /// Returns the list of [gmaps.Marker] objects. + external List get markers; + + /// Sets the list of [gmaps.Marker] objects. + external set markers(List? markers); + + /// Returns the onClusterClick handler. + external ClusterClickHandler? get onClusterClick; + + /// Sets the onClusterClick. + external set onClusterClick(ClusterClickHandler? handler); +} + +/// The cluster object handled by the [MarkerClusterer]. +@JS('markerClusterer.Cluster') +abstract class MarkerClustererCluster { + /// Getter for the cluster marker. + external gmaps.Marker get marker; + + /// List of markers in the cluster. + external List get markers; + + /// The bounds of the cluster. + external gmaps.LatLngBounds? get bounds; + + /// The position of the cluster marker. + external gmaps.LatLng get position; + + /// Get the count of **visible** markers. + external int get count; + + /// Deletes the cluster. + external void delete(); + + /// Adds a marker to the cluster. + external void push(gmaps.Marker marker); +} + +/// The [MarkerClusterer] object used to cluster markers on the map. +@JS('markerClusterer.MarkerClusterer') +class MarkerClusterer { + /// Constructs a new [MarkerClusterer] object. + external MarkerClusterer(MarkerClustererOptions options); + + /// Adds a marker to be clustered by the [MarkerClusterer]. + external void addMarker(gmaps.Marker marker, bool? noDraw); + + /// Adds a list of markers to be clustered by the [MarkerClusterer]. + external void addMarkers(List? markers, bool? noDraw); + + /// Removes a marker from the [MarkerClusterer]. + external bool removeMarker(gmaps.Marker marker, bool? noDraw); + + /// Removes a list of markers from the [MarkerClusterer]. + external bool removeMarkers(List? markers, bool? noDraw); + + /// Clears all the markers from the [MarkerClusterer]. + external void clearMarkers(bool? noDraw); + + /// Called when the [MarkerClusterer] is added to the map. + external void onAdd(); + + /// Called when the [MarkerClusterer] is removed from the map. + external void onRemove(); + + /// Returns the list of clusters. + external List get clusters; + + /// Recalculates and draws all the marker clusters. + external void render(); + + /// Flag to control the need for re-rendering the cluster after bulk changes. + bool dirty = false; +} + +/// Creates [MarkerClusterer] object with given [gmaps.GMap] and +/// [ClusterClickHandler]. +MarkerClusterer createMarkerClusterer( + gmaps.GMap map, ClusterClickHandler onClusterClickHandler) { + return MarkerClusterer(_createClusterOptions(map, onClusterClickHandler)); +} + +/// Creates [MarkerClustererOptions] object with given [gmaps.GMap] and +/// [ClusterClickHandler]. +MarkerClustererOptions _createClusterOptions( + gmaps.GMap map, ClusterClickHandler onClusterClickHandler) { + final MarkerClustererOptions options = MarkerClustererOptions() + ..map = map + ..onClusterClick = allowInterop(onClusterClickHandler); + + return options; +} diff --git a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/markers.dart b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/markers.dart index 26cb94f9ad28..343ebbe149b9 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/lib/src/markers.dart +++ b/packages/google_maps_flutter/google_maps_flutter_web/lib/src/markers.dart @@ -9,7 +9,9 @@ class MarkersController extends GeometryController { /// Initialize the cache. The [StreamController] comes from the [GoogleMapController], and is shared with other controllers. MarkersController({ required StreamController> stream, + required ClusterManagersController clusterManagersController, }) : _streamController = stream, + _clusterManagersController = clusterManagersController, _markerIdToController = {}; // A cache of [MarkerController]s indexed by their [MarkerId]. @@ -18,6 +20,8 @@ class MarkersController extends GeometryController { // The stream over which markers broadcast their events final StreamController> _streamController; + final ClusterManagersController _clusterManagersController; + /// Returns the cache of [MarkerController]s. Test only. @visibleForTesting Map get markers => _markerIdToController; @@ -27,6 +31,7 @@ class MarkersController extends GeometryController { /// Wraps each [Marker] into its corresponding [MarkerController]. void addMarkers(Set markersToAdd) { markersToAdd.forEach(_addMarker); + _clusterManagersController.renderDirty(); } void _addMarker(Marker marker) { @@ -53,9 +58,19 @@ class MarkersController extends GeometryController { final gmaps.MarkerOptions markerOptions = _markerOptionsFromMarker(marker, currentMarker); - final gmaps.Marker gmMarker = gmaps.Marker(markerOptions)..map = googleMap; + + final gmaps.Marker gmMarker = gmaps.Marker(markerOptions); + + gmMarker.set('markerId', marker.markerId.value); + + if (marker.clusterManagerId != null) { + _clusterManagersController.addItem(marker.clusterManagerId!, gmMarker); + } else { + gmMarker.map = googleMap; + } final MarkerController controller = MarkerController( marker: gmMarker, + clusterManagerId: marker.clusterManagerId, infoWindow: gmInfoWindow, consumeTapEvents: marker.consumeTapEvents, onTap: () { @@ -78,32 +93,48 @@ class MarkersController extends GeometryController { /// Updates a set of [Marker] objects with new options. void changeMarkers(Set markersToChange) { markersToChange.forEach(_changeMarker); + _clusterManagersController.renderDirty(); } void _changeMarker(Marker marker) { final MarkerController? markerController = _markerIdToController[marker.markerId]; if (markerController != null) { - final gmaps.MarkerOptions markerOptions = _markerOptionsFromMarker( - marker, - markerController.marker, - ); - final gmaps.InfoWindowOptions? infoWindow = - _infoWindowOptionsFromMarker(marker); - markerController.update( - markerOptions, - newInfoWindowContent: infoWindow?.content as HTMLElement?, - ); + final ClusterManagerId? oldClusterManagerId = + markerController.clusterManagerId; + final ClusterManagerId? newClusterManagerId = marker.clusterManagerId; + + if (oldClusterManagerId != newClusterManagerId) { + // If clusterManagerId changes. Remove existing marker and create new one. + _removeMarker(marker.markerId); + _addMarker(marker); + } else { + final gmaps.MarkerOptions markerOptions = _markerOptionsFromMarker( + marker, + markerController.marker, + ); + final gmaps.InfoWindowOptions? infoWindow = + _infoWindowOptionsFromMarker(marker); + markerController.update( + markerOptions, + newInfoWindowContent: infoWindow?.content as HTMLElement?, + ); + } } } /// Removes a set of [MarkerId]s from the cache. void removeMarkers(Set markerIdsToRemove) { markerIdsToRemove.forEach(_removeMarker); + _clusterManagersController.renderDirty(); } void _removeMarker(MarkerId markerId) { final MarkerController? markerController = _markerIdToController[markerId]; + if (markerController?.clusterManagerId != null) { + _clusterManagersController.removeItem( + markerController!.clusterManagerId!, markerController.marker); + } markerController?.remove(); _markerIdToController.remove(markerId); } @@ -151,7 +182,7 @@ class MarkersController extends GeometryController { void _onMarkerDragStart(MarkerId markerId, gmaps.LatLng latLng) { _streamController.add(MarkerDragStartEvent( mapId, - _gmLatLngToLatLng(latLng), + gmLatLngToLatLng(latLng), markerId, )); } @@ -159,7 +190,7 @@ class MarkersController extends GeometryController { void _onMarkerDrag(MarkerId markerId, gmaps.LatLng latLng) { _streamController.add(MarkerDragEvent( mapId, - _gmLatLngToLatLng(latLng), + gmLatLngToLatLng(latLng), markerId, )); } @@ -167,7 +198,7 @@ class MarkersController extends GeometryController { void _onMarkerDragEnd(MarkerId markerId, gmaps.LatLng latLng) { _streamController.add(MarkerDragEndEvent( mapId, - _gmLatLngToLatLng(latLng), + gmLatLngToLatLng(latLng), markerId, )); } diff --git a/packages/google_maps_flutter/google_maps_flutter_web/pubspec.yaml b/packages/google_maps_flutter/google_maps_flutter_web/pubspec.yaml index 6f59a0433c75..232059500127 100644 --- a/packages/google_maps_flutter/google_maps_flutter_web/pubspec.yaml +++ b/packages/google_maps_flutter/google_maps_flutter_web/pubspec.yaml @@ -2,7 +2,7 @@ name: google_maps_flutter_web description: Web platform implementation of google_maps_flutter repository: https://github.com/flutter/packages/tree/main/packages/google_maps_flutter/google_maps_flutter_web issue_tracker: https://github.com/flutter/flutter/issues?q=is%3Aissue+is%3Aopen+label%3A%22p%3A+maps%22 -version: 0.5.6+1 +version: 0.5.7 environment: sdk: ^3.3.0 @@ -23,7 +23,8 @@ dependencies: flutter_web_plugins: sdk: flutter google_maps: ^7.1.0 - google_maps_flutter_platform_interface: ^2.5.0 + google_maps_flutter_platform_interface: ^2.6.0 + js: ^0.6.7 sanitize_html: ^2.0.0 stream_transform: ^2.0.0 web: ^0.5.0