2024-07-08 18:57:03 +02:00
|
|
|
import 'dart:async';
|
|
|
|
import 'dart:ui';
|
|
|
|
|
2024-07-14 19:42:39 +02:00
|
|
|
import 'package:clock/clock.dart';
|
2024-07-08 18:57:03 +02:00
|
|
|
import 'package:copy_with/copy_with.dart';
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter/services.dart';
|
|
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
|
|
import 'package:google_maps_cluster_manager/google_maps_cluster_manager.dart';
|
|
|
|
import 'package:google_maps_flutter/google_maps_flutter.dart';
|
2024-07-14 19:42:39 +02:00
|
|
|
import 'package:intl/intl.dart' as intl;
|
2024-07-08 18:57:03 +02:00
|
|
|
import 'package:kiwi/kiwi.dart';
|
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:nc_photos/account.dart';
|
2024-07-14 19:42:39 +02:00
|
|
|
import 'package:nc_photos/app_localizations.dart';
|
2024-07-08 18:57:03 +02:00
|
|
|
import 'package:nc_photos/bloc_util.dart';
|
|
|
|
import 'package:nc_photos/controller/account_controller.dart';
|
|
|
|
import 'package:nc_photos/di_container.dart';
|
|
|
|
import 'package:nc_photos/entity/collection.dart';
|
|
|
|
import 'package:nc_photos/entity/collection/content_provider/ad_hoc.dart';
|
|
|
|
import 'package:nc_photos/entity/image_location/repo.dart';
|
|
|
|
import 'package:nc_photos/exception_event.dart';
|
2024-07-14 19:42:39 +02:00
|
|
|
import 'package:nc_photos/k.dart' as k;
|
2024-07-08 18:57:03 +02:00
|
|
|
import 'package:nc_photos/snack_bar_manager.dart';
|
2024-07-14 19:42:39 +02:00
|
|
|
import 'package:nc_photos/stream_extension.dart';
|
|
|
|
import 'package:nc_photos/theme.dart';
|
2024-07-08 18:57:03 +02:00
|
|
|
import 'package:nc_photos/widget/collection_browser.dart';
|
2024-07-14 19:42:39 +02:00
|
|
|
import 'package:nc_photos/widget/measure.dart';
|
2024-07-08 18:57:03 +02:00
|
|
|
import 'package:np_codegen/np_codegen.dart';
|
2024-07-14 19:42:39 +02:00
|
|
|
import 'package:np_datetime/np_datetime.dart';
|
2024-07-08 18:57:03 +02:00
|
|
|
import 'package:to_string/to_string.dart';
|
|
|
|
|
|
|
|
part 'map_browser.g.dart';
|
|
|
|
part 'map_browser/bloc.dart';
|
|
|
|
part 'map_browser/state_event.dart';
|
|
|
|
part 'map_browser/type.dart';
|
|
|
|
part 'map_browser/view.dart';
|
|
|
|
|
|
|
|
class MapBrowser extends StatelessWidget {
|
|
|
|
static const routeName = "/map-browser";
|
|
|
|
|
|
|
|
static Route buildRoute() => MaterialPageRoute(
|
|
|
|
builder: (_) => const MapBrowser(),
|
|
|
|
);
|
|
|
|
|
|
|
|
const MapBrowser({super.key});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return BlocProvider(
|
|
|
|
create: (_) => _Bloc(
|
|
|
|
KiwiContainer().resolve(),
|
|
|
|
account: context.read<AccountController>().account,
|
2024-07-14 19:42:39 +02:00
|
|
|
)..add(const _LoadData()),
|
2024-07-08 18:57:03 +02:00
|
|
|
child: const _WrappedMapBrowser(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class _WrappedMapBrowser extends StatelessWidget {
|
|
|
|
const _WrappedMapBrowser();
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return AnnotatedRegion<SystemUiOverlayStyle>(
|
|
|
|
value: SystemUiOverlayStyle.dark,
|
|
|
|
child: Scaffold(
|
|
|
|
body: MultiBlocListener(
|
|
|
|
listeners: [
|
|
|
|
_BlocListenerT<ExceptionEvent?>(
|
|
|
|
selector: (state) => state.error,
|
|
|
|
listener: (context, error) {
|
|
|
|
if (error != null) {
|
|
|
|
SnackBarManager().showSnackBarForException(error.error);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
),
|
|
|
|
],
|
2024-07-14 19:42:39 +02:00
|
|
|
child: Stack(
|
|
|
|
children: [
|
|
|
|
const _MapView(),
|
|
|
|
Positioned.directional(
|
|
|
|
textDirection: Directionality.of(context),
|
|
|
|
top: MediaQuery.of(context).padding.top + 8,
|
|
|
|
end: 8,
|
|
|
|
child: const _DateRangeToggle(),
|
|
|
|
),
|
|
|
|
_BlocSelector<bool>(
|
|
|
|
selector: (state) => state.isShowDataRangeControlPanel,
|
|
|
|
builder: (context, isShowAnyPanel) => Positioned.fill(
|
|
|
|
child: isShowAnyPanel
|
|
|
|
? GestureDetector(
|
|
|
|
onTap: () {
|
|
|
|
context.addEvent(const _CloseControlPanel());
|
|
|
|
},
|
|
|
|
)
|
|
|
|
: const SizedBox.shrink(),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
Positioned(
|
|
|
|
left: 8,
|
|
|
|
right: 8,
|
|
|
|
top: MediaQuery.of(context).padding.top + 8,
|
|
|
|
child: _BlocSelector<bool>(
|
|
|
|
selector: (state) => state.isShowDataRangeControlPanel,
|
|
|
|
builder: (context, isShowDataRangeControlPanel) =>
|
|
|
|
_PanelContainer(
|
|
|
|
isShow: isShowDataRangeControlPanel,
|
|
|
|
child: const _DateRangeControlPanel(),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
2024-07-08 18:57:03 +02:00
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef _BlocBuilder = BlocBuilder<_Bloc, _State>;
|
|
|
|
// typedef _BlocListener = BlocListener<_Bloc, _State>;
|
|
|
|
typedef _BlocListenerT<T> = BlocListenerT<_Bloc, _State, T>;
|
2024-07-14 19:42:39 +02:00
|
|
|
typedef _BlocSelector<T> = BlocSelector<_Bloc, _State, T>;
|
2024-07-08 18:57:03 +02:00
|
|
|
|
|
|
|
extension on BuildContext {
|
|
|
|
_Bloc get bloc => read<_Bloc>();
|
|
|
|
// _State get state => bloc.state;
|
|
|
|
void addEvent(_Event event) => bloc.add(event);
|
|
|
|
}
|