mirror of
https://github.com/immich-app/immich.git
synced 2026-05-21 07:06:31 -04:00
4d6a50c2cb
* refactor: app metadata * refactor to per row store * cleanup * more test * review changes * more refactor * refactor * migrate primary color * migrate dynamic theme * migrate colorfulInterface * cleanup providers * migrate cleanup * migrate mapconfig --------- Co-authored-by: shenlong-tanwen <139912620+shalong-tanwen@users.noreply.github.com>
142 lines
4.9 KiB
Dart
142 lines
4.9 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
import 'package:immich_mobile/domain/models/events.model.dart';
|
|
import 'package:immich_mobile/domain/models/metadata_key.dart';
|
|
import 'package:immich_mobile/domain/utils/event_stream.dart';
|
|
import 'package:immich_mobile/infrastructure/repositories/timeline.repository.dart';
|
|
import 'package:immich_mobile/providers/infrastructure/map.provider.dart';
|
|
import 'package:immich_mobile/providers/infrastructure/metadata.provider.dart';
|
|
import 'package:immich_mobile/providers/map/map_state.provider.dart';
|
|
import 'package:maplibre_gl/maplibre_gl.dart';
|
|
|
|
class MapState {
|
|
final ThemeMode themeMode;
|
|
final LatLngBounds bounds;
|
|
final bool onlyFavorites;
|
|
final bool includeArchived;
|
|
final bool withPartners;
|
|
final int relativeDays;
|
|
|
|
const MapState({
|
|
this.themeMode = ThemeMode.system,
|
|
required this.bounds,
|
|
this.onlyFavorites = false,
|
|
this.includeArchived = false,
|
|
this.withPartners = false,
|
|
this.relativeDays = 0,
|
|
});
|
|
|
|
@override
|
|
bool operator ==(covariant MapState other) {
|
|
return bounds == other.bounds;
|
|
}
|
|
|
|
@override
|
|
int get hashCode => bounds.hashCode;
|
|
|
|
MapState copyWith({
|
|
LatLngBounds? bounds,
|
|
ThemeMode? themeMode,
|
|
bool? onlyFavorites,
|
|
bool? includeArchived,
|
|
bool? withPartners,
|
|
int? relativeDays,
|
|
}) {
|
|
return MapState(
|
|
bounds: bounds ?? this.bounds,
|
|
themeMode: themeMode ?? this.themeMode,
|
|
onlyFavorites: onlyFavorites ?? this.onlyFavorites,
|
|
includeArchived: includeArchived ?? this.includeArchived,
|
|
withPartners: withPartners ?? this.withPartners,
|
|
relativeDays: relativeDays ?? this.relativeDays,
|
|
);
|
|
}
|
|
|
|
TimelineMapOptions toOptions() => TimelineMapOptions(
|
|
bounds: bounds,
|
|
onlyFavorites: onlyFavorites,
|
|
includeArchived: includeArchived,
|
|
withPartners: withPartners,
|
|
relativeDays: relativeDays,
|
|
);
|
|
}
|
|
|
|
class MapStateNotifier extends Notifier<MapState> {
|
|
MapStateNotifier();
|
|
|
|
bool setBounds(LatLngBounds bounds) {
|
|
if (state.bounds == bounds) {
|
|
return false;
|
|
}
|
|
state = state.copyWith(bounds: bounds);
|
|
return true;
|
|
}
|
|
|
|
void switchTheme(ThemeMode mode) {
|
|
// TODO: Remove this line when map theme provider is removed
|
|
// Until then, keep both in sync as MapThemeOverride uses map state provider
|
|
// ref.read(appSettingsServiceProvider).setSetting(AppSettingsEnum.mapThemeMode, mode.index);
|
|
ref.read(mapStateNotifierProvider.notifier).switchTheme(mode);
|
|
state = state.copyWith(themeMode: mode);
|
|
}
|
|
|
|
void switchFavoriteOnly(bool isFavoriteOnly) {
|
|
ref.read(metadataProvider).write(MetadataKey.mapShowFavoriteOnly, isFavoriteOnly);
|
|
state = state.copyWith(onlyFavorites: isFavoriteOnly);
|
|
EventStream.shared.emit(const MapMarkerReloadEvent());
|
|
}
|
|
|
|
void switchIncludeArchived(bool isIncludeArchived) {
|
|
ref.read(metadataProvider).write(MetadataKey.mapIncludeArchived, isIncludeArchived);
|
|
state = state.copyWith(includeArchived: isIncludeArchived);
|
|
EventStream.shared.emit(const MapMarkerReloadEvent());
|
|
}
|
|
|
|
void switchWithPartners(bool isWithPartners) {
|
|
ref.read(metadataProvider).write(MetadataKey.mapWithPartners, isWithPartners);
|
|
state = state.copyWith(withPartners: isWithPartners);
|
|
EventStream.shared.emit(const MapMarkerReloadEvent());
|
|
}
|
|
|
|
void setRelativeTime(int relativeDays) {
|
|
ref.read(metadataProvider).write(MetadataKey.mapRelativeDate, relativeDays);
|
|
state = state.copyWith(relativeDays: relativeDays);
|
|
EventStream.shared.emit(const MapMarkerReloadEvent());
|
|
}
|
|
|
|
@override
|
|
MapState build() {
|
|
final mapConfig = ref.read(appConfigProvider.select((config) => config.map));
|
|
return MapState(
|
|
themeMode: mapConfig.themeMode,
|
|
onlyFavorites: mapConfig.favoritesOnly,
|
|
includeArchived: mapConfig.includeArchived,
|
|
withPartners: mapConfig.withPartners,
|
|
relativeDays: mapConfig.relativeDays,
|
|
bounds: LatLngBounds(northeast: const LatLng(0, 0), southwest: const LatLng(0, 0)),
|
|
);
|
|
}
|
|
}
|
|
|
|
// This provider watches the markers from the map service and serves the markers.
|
|
// It should be used only after the map service provider is overridden
|
|
final mapMarkerProvider = FutureProvider.family<Map<String, dynamic>, LatLngBounds?>((ref, bounds) async {
|
|
final mapService = ref.watch(mapServiceProvider);
|
|
final markers = await mapService.getMarkers(bounds);
|
|
final features = List.filled(markers.length, const <String, dynamic>{});
|
|
for (int i = 0; i < markers.length; i++) {
|
|
final marker = markers[i];
|
|
features[i] = {
|
|
'type': 'Feature',
|
|
'id': marker.assetId,
|
|
'geometry': {
|
|
'type': 'Point',
|
|
'coordinates': [marker.location.longitude, marker.location.latitude],
|
|
},
|
|
};
|
|
}
|
|
return {'type': 'FeatureCollection', 'features': features};
|
|
}, dependencies: [mapServiceProvider]);
|
|
|
|
final mapStateProvider = NotifierProvider<MapStateNotifier, MapState>(MapStateNotifier.new);
|