Stateful map controller for Flutter Map. Manage markers, lines and polygons.
This is a fork from synw's map_controller package made because the project has been abandoned. This new and improved version supports the latest version of the flutter_map package. If you need a feature or a fix you can open an issue on the forked repository.
import 'dart:async';
import 'package:flutter/material.dart';
import 'package:flutter_map/flutter_map.dart';
import 'package:latlong2/latlong.dart';
import 'package:map_controller_plus/map_controller_plus.dart';
class MapPage extends StatefulWidget {
@override
State<MapPage> createState() => _MapPageState();
}
class _MapPageState extends State<MapPage> {
late final MapController mapController;
late final StatefulMapController statefulMapController;
late final StreamSubscription<StatefulMapControllerStateChange> sub;
@override
void initState() {
super.initState();
// intialize the controllers
mapController = MapController();
statefulMapController = StatefulMapController(mapController: mapController);
/// [Important] listen to the changefeed to rebuild the map on changes:
/// this will rebuild the map when for example addMarker or any method
/// that mutates the map assets is called
sub = statefulMapController.changeFeed.listen((change) => setState(() {}));
}
@override
void dispose() {
sub.cancel();
mapController.dispose();
super.dispose();
}
@override
Widget build(BuildContext context) {
return Scaffold(
body: SafeArea(
child: Stack(children: <Widget>[
FlutterMap(
mapController: mapController,
options: MapOptions(center: LatLng(48.853831, 2.348722), zoom: 11.0),
children: [
MarkerLayer(markers: statefulMapController.markers),
PolylineLayer(polylines: statefulMapController.lines),
PolygonLayer(polygons: statefulMapController.polygons),
],
),
// ...
])),
);
}
}
Api for the StatefulMapController class
zoom
: get the current zoom valuezoomIn()
: increase the zoom level by 1zoomOut()
: decrease the zoom level by 1zoomTo()
: zoom to the provided value
center
: get the current centerLatLng
valuecenterOnPoint()
: center on theLatLng
value
addMarker()
: add a named marker on the mapaddMarkers()
: add several named markers on the mapremoveMarker()
: remove a named marker from the mapremoveMarkers()
: remove several named markers from the mapmarkers
: get the markers that are on the mapnamedMarkers
: get the markers with their names that are on the mapgetMarker()
: return the marker with the corresponding namegetMarkers()
: return the markers with the corresponding names
New in 0.7: the stateful makers hold their own state and can be mutated
statefulMapController.addStatefulMarker(
name: "some marker",
statefulMarker: StatefulMarker(
height: 80.0,
width: 120.0,
state: <String, dynamic>{"showText": false},
point: LatLng(48.853831, 2.348722),
builder: (BuildContext context, Map<String, dynamic> state) {
Widget w;
final markerIcon = IconButton(
icon: const Icon(Icons.location_on),
onPressed: () => statefulMapController.mutateMarker(
name: "some marker",
property: "showText",
value: !(state["showText"] as bool)));
if (state["showText"] == true) {
w = Column(children: <Widget>[
markerIcon,
Container(
color: Colors.white,
child: Padding(
padding: const EdgeInsets.all(5.0),
child: Text(place.name, textScaleFactor: 1.3))),
]);
} else {
w = markerIcon;
}
return w;
})
);
addLine()
: add a line on the maplines
: get the lines that are on the map
addPolygon
: add a polygon on the mappolygons
: get the polygons that are on the map
A changefeed is available: it's a stream with all state changes from the map controller. Use it to update the map when a change occurs:
statefulMapController.changeFeed.listen((change) => setState(() {}));
The map controller can draw on the map from geojson data:
void loadData() async {
print("Loading geojson data");
final data = await rootBundle.loadString('assets/airports.geojson');
await statefulMapController.fromGeoJson(data,
markerIcon: Icon(Icons.local_airport), verbose: true);
}
@override
void initState() {
super.initState();
mapController = MapController();
statefulMapController = StatefulMapController(mapController: mapController);
loadData();
sub = statefulMapController.changeFeed.listen((change) => setState(() {}));
}
Some predefined tile layers are available.
FlutterMap(
mapController: mapController,
options: MapOptions(
center: LatLng(48.853831, 2.348722),
zoom: 11.0,
),
children: [
TileLayer(
urlTemplate: 'https://tile.openstreetmap.org/{z}/{x}/{y}.png',
subdomains: const ['a', 'b', 'c'],
userAgentPackageName: 'dev.fleaflet.flutter_map.example',
),
MarkerLayer(markers: statefulMapController.markers),
// ...
],
)