import 'dart:math'; import 'package:aves/utils/geo_utils.dart'; import 'package:equatable/equatable.dart'; import 'package:flutter/foundation.dart'; import 'package:latlong2/latlong.dart'; @immutable class ZoomedBounds extends Equatable { final LatLng sw, ne; final double zoom, rotation; // returns [southwestLng, southwestLat, northeastLng, northeastLat], as expected by Fluster List get boundingBox => [sw.longitude, sw.latitude, ne.longitude, ne.latitude]; LatLng get center => getLatLngCenter([sw, ne]); @override List get props => [sw, ne, zoom, rotation]; const ZoomedBounds({ required this.sw, required this.ne, required this.zoom, required this.rotation, }); static const _collocationMaxDeltaThreshold = 360 / (2 << 19); factory ZoomedBounds.fromPoints({ required Set points, double collocationZoom = 20, }) { var west = .0, south = .0, east = .0, north = .0; var zoom = collocationZoom; if (points.isNotEmpty) { final first = points.first; west = first.longitude; south = first.latitude; east = first.longitude; north = first.latitude; for (var point in points) { final lng = point.longitude; final lat = point.latitude; if (lng < west) west = lng; if (lat < south) south = lat; if (lng > east) east = lng; if (lat > north) north = lat; } final boundsDelta = max(north - south, east - west); if (boundsDelta > _collocationMaxDeltaThreshold) { zoom = max(1, log(360) / ln2 - log(boundsDelta) / ln2); } } return ZoomedBounds( sw: LatLng(south, west), ne: LatLng(north, east), zoom: zoom, rotation: 0, ); } bool contains(LatLng point) { final lat = point.latitude; final lng = point.longitude; final south = sw.latitude; final north = ne.latitude; final west = sw.longitude; final east = ne.longitude; return (south <= lat && lat <= north) && (west <= east ? (west <= lng && lng <= east) : (west <= lng || lng <= east)); } }