mirror of
				https://github.com/immich-app/immich.git
				synced 2025-11-03 19:17:11 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			176 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			176 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
import 'package:auto_route/auto_route.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/build_context_extensions.dart';
 | 
						|
import 'package:immich_mobile/extensions/translate_extensions.dart';
 | 
						|
import 'package:immich_mobile/pages/common/large_leading_tile.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/images/thumbnail.widget.dart';
 | 
						|
import 'package:immich_mobile/providers/infrastructure/asset.provider.dart';
 | 
						|
import 'package:immich_mobile/routing/router.dart';
 | 
						|
import 'package:immich_mobile/widgets/common/search_field.dart';
 | 
						|
import 'package:immich_mobile/widgets/map/map_thumbnail.dart';
 | 
						|
import 'package:maplibre_gl/maplibre_gl.dart';
 | 
						|
 | 
						|
@RoutePage()
 | 
						|
class DriftPlacePage extends StatelessWidget {
 | 
						|
  const DriftPlacePage({super.key, this.currentLocation});
 | 
						|
 | 
						|
  final LatLng? currentLocation;
 | 
						|
 | 
						|
  @override
 | 
						|
  Widget build(BuildContext context) {
 | 
						|
    final ValueNotifier<String?> search = ValueNotifier(null);
 | 
						|
 | 
						|
    return Scaffold(
 | 
						|
      body: ValueListenableBuilder(
 | 
						|
        valueListenable: search,
 | 
						|
        builder: (context, searchValue, child) {
 | 
						|
          return CustomScrollView(
 | 
						|
            slivers: [
 | 
						|
              _PlaceSliverAppBar(search: search),
 | 
						|
              _Map(search: search, currentLocation: currentLocation),
 | 
						|
              _PlaceList(search: search),
 | 
						|
            ],
 | 
						|
          );
 | 
						|
        },
 | 
						|
      ),
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
class _PlaceSliverAppBar extends HookWidget {
 | 
						|
  const _PlaceSliverAppBar({required this.search});
 | 
						|
 | 
						|
  final ValueNotifier<String?> search;
 | 
						|
 | 
						|
  @override
 | 
						|
  Widget build(BuildContext context) {
 | 
						|
    final searchFocusNode = useFocusNode();
 | 
						|
 | 
						|
    return SliverAppBar(
 | 
						|
      floating: true,
 | 
						|
      pinned: true,
 | 
						|
      snap: false,
 | 
						|
      backgroundColor: context.colorScheme.surfaceContainer,
 | 
						|
      shape: const RoundedRectangleBorder(borderRadius: BorderRadius.all(Radius.circular(5))),
 | 
						|
      automaticallyImplyLeading: search.value == null,
 | 
						|
      centerTitle: true,
 | 
						|
      title: search.value != null
 | 
						|
          ? SearchField(
 | 
						|
              focusNode: searchFocusNode,
 | 
						|
              onTapOutside: (_) => searchFocusNode.unfocus(),
 | 
						|
              onChanged: (value) => search.value = value,
 | 
						|
              filled: true,
 | 
						|
              hintText: 'filter_places'.t(context: context),
 | 
						|
              autofocus: true,
 | 
						|
            )
 | 
						|
          : Text('places'.t(context: context)),
 | 
						|
      actions: [
 | 
						|
        IconButton(
 | 
						|
          icon: Icon(search.value != null ? Icons.close : Icons.search),
 | 
						|
          onPressed: () {
 | 
						|
            search.value = search.value == null ? '' : null;
 | 
						|
          },
 | 
						|
        ),
 | 
						|
      ],
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
class _Map extends StatelessWidget {
 | 
						|
  const _Map({required this.search, this.currentLocation});
 | 
						|
 | 
						|
  final ValueNotifier<String?> search;
 | 
						|
  final LatLng? currentLocation;
 | 
						|
 | 
						|
  @override
 | 
						|
  Widget build(BuildContext context) {
 | 
						|
    return search.value == null
 | 
						|
        ? SliverPadding(
 | 
						|
            padding: const EdgeInsets.all(16.0),
 | 
						|
            sliver: SliverToBoxAdapter(
 | 
						|
              child: SizedBox(
 | 
						|
                height: 200,
 | 
						|
                width: context.width,
 | 
						|
                child: MapThumbnail(
 | 
						|
                  onTap: (_, __) => context.pushRoute(DriftMapRoute(initialLocation: currentLocation)),
 | 
						|
                  zoom: 8,
 | 
						|
                  centre: currentLocation ?? const LatLng(21.44950, -157.91959),
 | 
						|
                  showAttribution: false,
 | 
						|
                  themeMode: context.isDarkTheme ? ThemeMode.dark : ThemeMode.light,
 | 
						|
                ),
 | 
						|
              ),
 | 
						|
            ),
 | 
						|
          )
 | 
						|
        : const SliverToBoxAdapter(child: SizedBox.shrink());
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
class _PlaceList extends ConsumerWidget {
 | 
						|
  const _PlaceList({required this.search});
 | 
						|
 | 
						|
  final ValueNotifier<String?> search;
 | 
						|
 | 
						|
  @override
 | 
						|
  Widget build(BuildContext context, WidgetRef ref) {
 | 
						|
    final places = ref.watch(placesProvider);
 | 
						|
 | 
						|
    return places.when(
 | 
						|
      loading: () => const SliverToBoxAdapter(
 | 
						|
        child: Center(
 | 
						|
          child: Padding(padding: EdgeInsets.all(20.0), child: CircularProgressIndicator()),
 | 
						|
        ),
 | 
						|
      ),
 | 
						|
      error: (error, stack) => SliverToBoxAdapter(
 | 
						|
        child: Center(
 | 
						|
          child: Padding(
 | 
						|
            padding: const EdgeInsets.all(20.0),
 | 
						|
            child: Text(
 | 
						|
              'Error loading places: $error, stack: $stack',
 | 
						|
              style: TextStyle(color: context.colorScheme.error),
 | 
						|
            ),
 | 
						|
          ),
 | 
						|
        ),
 | 
						|
      ),
 | 
						|
      data: (places) {
 | 
						|
        if (search.value != null) {
 | 
						|
          places = places.where((place) {
 | 
						|
            return place.$1.toLowerCase().contains(search.value!.toLowerCase());
 | 
						|
          }).toList();
 | 
						|
        }
 | 
						|
 | 
						|
        return SliverList.builder(
 | 
						|
          itemCount: places.length,
 | 
						|
          itemBuilder: (context, index) {
 | 
						|
            final place = places[index];
 | 
						|
            return _PlaceTile(place: place);
 | 
						|
          },
 | 
						|
        );
 | 
						|
      },
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
class _PlaceTile extends StatelessWidget {
 | 
						|
  const _PlaceTile({required this.place});
 | 
						|
 | 
						|
  final (String, String) place;
 | 
						|
 | 
						|
  @override
 | 
						|
  Widget build(BuildContext context) {
 | 
						|
    return LargeLeadingTile(
 | 
						|
      onTap: () => context.pushRoute(DriftPlaceDetailRoute(place: place.$1)),
 | 
						|
      title: Text(place.$1, style: context.textTheme.titleMedium?.copyWith(fontWeight: FontWeight.w500)),
 | 
						|
      leading: ClipRRect(
 | 
						|
        borderRadius: const BorderRadius.all(Radius.circular(20)),
 | 
						|
        child: SizedBox(
 | 
						|
          width: 80,
 | 
						|
          height: 80,
 | 
						|
          child: Thumbnail.remote(remoteId: place.$2, fit: BoxFit.cover),
 | 
						|
        ),
 | 
						|
      ),
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 |