2024-01-15 16:26:13 +01:00
|
|
|
import 'dart:math';
|
|
|
|
|
|
|
|
import 'package:auto_route/auto_route.dart';
|
|
|
|
import 'package:easy_localization/easy_localization.dart';
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter_hooks/flutter_hooks.dart';
|
|
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
|
|
import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
|
|
|
|
import 'package:immich_mobile/extensions/build_context_extensions.dart';
|
|
|
|
import 'package:immich_mobile/extensions/maplibrecontroller_extensions.dart';
|
2024-05-07 06:04:21 +02:00
|
|
|
import 'package:immich_mobile/widgets/map/map_theme_override.dart';
|
2024-01-15 16:26:13 +01:00
|
|
|
import 'package:maplibre_gl/maplibre_gl.dart';
|
2024-05-07 06:04:21 +02:00
|
|
|
import 'package:immich_mobile/utils/map_utils.dart';
|
2024-01-15 16:26:13 +01:00
|
|
|
|
2024-01-15 17:50:33 +01:00
|
|
|
@RoutePage<LatLng?>()
|
2024-01-15 16:26:13 +01:00
|
|
|
class MapLocationPickerPage extends HookConsumerWidget {
|
|
|
|
final LatLng initialLatLng;
|
|
|
|
|
|
|
|
const MapLocationPickerPage({
|
|
|
|
super.key,
|
|
|
|
this.initialLatLng = const LatLng(0, 0),
|
|
|
|
});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context, WidgetRef ref) {
|
|
|
|
final selectedLatLng = useValueNotifier<LatLng>(initialLatLng);
|
2024-07-02 20:43:52 +02:00
|
|
|
final controller = useRef<MaplibreMapController?>(null);
|
2024-01-15 16:26:13 +01:00
|
|
|
final marker = useRef<Symbol?>(null);
|
|
|
|
|
|
|
|
Future<void> onStyleLoaded() async {
|
|
|
|
marker.value = await controller.value?.addMarkerAtLatLng(initialLatLng);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> onMapClick(Point<num> point, LatLng centre) async {
|
|
|
|
selectedLatLng.value = centre;
|
|
|
|
controller.value?.animateCamera(CameraUpdate.newLatLng(centre));
|
|
|
|
if (marker.value != null) {
|
|
|
|
await controller.value
|
|
|
|
?.updateSymbol(marker.value!, SymbolOptions(geometry: centre));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void onClose([LatLng? selected]) {
|
2024-05-14 21:07:31 +02:00
|
|
|
context.maybePop(selected);
|
2024-01-15 16:26:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> getCurrentLocation() async {
|
2024-02-08 04:07:43 +01:00
|
|
|
var (currentLocation, _) =
|
2024-01-15 17:50:33 +01:00
|
|
|
await MapUtils.checkPermAndGetLocation(context);
|
2024-02-08 04:07:43 +01:00
|
|
|
|
2024-01-15 16:26:13 +01:00
|
|
|
if (currentLocation == null) {
|
|
|
|
return;
|
|
|
|
}
|
2024-02-08 04:07:43 +01:00
|
|
|
|
2024-01-15 17:50:33 +01:00
|
|
|
var currentLatLng =
|
|
|
|
LatLng(currentLocation.latitude, currentLocation.longitude);
|
2024-01-15 16:26:13 +01:00
|
|
|
selectedLatLng.value = currentLatLng;
|
|
|
|
controller.value?.animateCamera(CameraUpdate.newLatLng(currentLatLng));
|
|
|
|
}
|
|
|
|
|
|
|
|
return MapThemeOveride(
|
|
|
|
mapBuilder: (style) => Builder(
|
|
|
|
builder: (ctx) => Scaffold(
|
|
|
|
backgroundColor: ctx.themeData.cardColor,
|
|
|
|
appBar: _AppBar(onClose: onClose),
|
|
|
|
extendBodyBehindAppBar: true,
|
2024-02-08 04:07:43 +01:00
|
|
|
primary: true,
|
|
|
|
body: style.widgetWhen(
|
|
|
|
onData: (style) => Container(
|
|
|
|
clipBehavior: Clip.antiAliasWithSaveLayer,
|
|
|
|
decoration: const BoxDecoration(
|
|
|
|
borderRadius: BorderRadius.only(
|
|
|
|
bottomLeft: Radius.circular(40),
|
|
|
|
bottomRight: Radius.circular(40),
|
2024-01-15 16:26:13 +01:00
|
|
|
),
|
|
|
|
),
|
2024-07-02 20:43:52 +02:00
|
|
|
child: MaplibreMap(
|
2024-02-08 04:07:43 +01:00
|
|
|
initialCameraPosition:
|
|
|
|
CameraPosition(target: initialLatLng, zoom: 12),
|
|
|
|
styleString: style,
|
|
|
|
onMapCreated: (mapController) =>
|
|
|
|
controller.value = mapController,
|
|
|
|
onStyleLoadedCallback: onStyleLoaded,
|
|
|
|
onMapClick: onMapClick,
|
|
|
|
dragEnabled: false,
|
|
|
|
tiltGesturesEnabled: false,
|
|
|
|
myLocationEnabled: false,
|
|
|
|
attributionButtonMargins: const Point(20, 15),
|
2024-01-15 16:26:13 +01:00
|
|
|
),
|
2024-02-08 04:07:43 +01:00
|
|
|
),
|
|
|
|
),
|
|
|
|
bottomNavigationBar: _BottomBar(
|
|
|
|
selectedLatLng: selectedLatLng,
|
|
|
|
onUseLocation: () => onClose(selectedLatLng.value),
|
|
|
|
onGetCurrentLocation: getCurrentLocation,
|
2024-01-15 16:26:13 +01:00
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class _AppBar extends StatelessWidget implements PreferredSizeWidget {
|
|
|
|
final Function() onClose;
|
|
|
|
|
|
|
|
const _AppBar({required this.onClose});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return Padding(
|
2024-02-08 04:07:43 +01:00
|
|
|
padding: const EdgeInsets.only(top: 25),
|
|
|
|
child: Align(
|
|
|
|
alignment: Alignment.centerLeft,
|
|
|
|
child: ElevatedButton(
|
|
|
|
onPressed: onClose,
|
|
|
|
style: ElevatedButton.styleFrom(
|
|
|
|
shape: const CircleBorder(),
|
2024-01-15 16:26:13 +01:00
|
|
|
),
|
2024-02-08 04:07:43 +01:00
|
|
|
child: const Icon(Icons.arrow_back_ios_new_rounded),
|
2024-01-15 16:26:13 +01:00
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Size get preferredSize => const Size.fromHeight(100);
|
|
|
|
}
|
|
|
|
|
|
|
|
class _BottomBar extends StatelessWidget {
|
|
|
|
final ValueNotifier<LatLng> selectedLatLng;
|
|
|
|
final Function() onUseLocation;
|
|
|
|
final Function() onGetCurrentLocation;
|
|
|
|
|
|
|
|
const _BottomBar({
|
|
|
|
required this.selectedLatLng,
|
|
|
|
required this.onUseLocation,
|
|
|
|
required this.onGetCurrentLocation,
|
|
|
|
});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return SizedBox(
|
2024-02-08 04:07:43 +01:00
|
|
|
height: 150 + context.padding.bottom,
|
|
|
|
child: Padding(
|
|
|
|
padding: EdgeInsets.only(bottom: context.padding.bottom),
|
|
|
|
child: Column(
|
|
|
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
children: [
|
|
|
|
Row(
|
|
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
|
|
children: [
|
|
|
|
const Icon(Icons.public, size: 18),
|
|
|
|
const SizedBox(width: 15),
|
|
|
|
ValueListenableBuilder(
|
|
|
|
valueListenable: selectedLatLng,
|
|
|
|
builder: (_, value, __) => Text(
|
|
|
|
"${value.latitude.toStringAsFixed(4)}, ${value.longitude.toStringAsFixed(4)}",
|
|
|
|
),
|
2024-01-15 16:26:13 +01:00
|
|
|
),
|
2024-02-08 04:07:43 +01:00
|
|
|
],
|
|
|
|
),
|
|
|
|
Row(
|
|
|
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
|
|
|
children: [
|
|
|
|
ElevatedButton(
|
|
|
|
onPressed: onUseLocation,
|
|
|
|
child:
|
|
|
|
const Text("map_location_picker_page_use_location").tr(),
|
|
|
|
),
|
|
|
|
ElevatedButton(
|
|
|
|
onPressed: onGetCurrentLocation,
|
|
|
|
child: const Icon(Icons.my_location),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
2024-01-15 16:26:13 +01:00
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|