mirror of
				https://github.com/immich-app/immich.git
				synced 2025-11-04 03:39:37 -05:00 
			
		
		
		
	feat(mobile): improve explore page and allow metadata search (#2097)
This commit is contained in:
		
							parent
							
								
									f0e272d0f2
								
							
						
					
					
						commit
						0854737be2
					
				@ -258,5 +258,9 @@
 | 
				
			|||||||
  "motion_photos_page_title": "Motion Photos",
 | 
					  "motion_photos_page_title": "Motion Photos",
 | 
				
			||||||
  "search_page_motion_photos": "Motion Photos",
 | 
					  "search_page_motion_photos": "Motion Photos",
 | 
				
			||||||
  "search_page_recently_added": "Recently added",
 | 
					  "search_page_recently_added": "Recently added",
 | 
				
			||||||
  "search_page_categories": "Categories"
 | 
					  "search_page_categories": "Categories",
 | 
				
			||||||
 | 
					  "search_page_screenshots": "Screenshots",
 | 
				
			||||||
 | 
					  "search_page_selfies": "Selfies",
 | 
				
			||||||
 | 
					  "search_suggestion_list_smart_search_hint_1": "Smart search is enabled by default, to search for metadata use the syntax ",
 | 
				
			||||||
 | 
					  "search_suggestion_list_smart_search_hint_2": "m:your-search-term"
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
										
											Binary file not shown.
										
									
								
							| 
		 Before Width: | Height: | Size: 536 KiB  | 
@ -454,6 +454,10 @@ class EmailInput extends StatelessWidget {
 | 
				
			|||||||
        labelText: 'login_form_label_email'.tr(),
 | 
					        labelText: 'login_form_label_email'.tr(),
 | 
				
			||||||
        border: const OutlineInputBorder(),
 | 
					        border: const OutlineInputBorder(),
 | 
				
			||||||
        hintText: 'login_form_email_hint'.tr(),
 | 
					        hintText: 'login_form_email_hint'.tr(),
 | 
				
			||||||
 | 
					        hintStyle: const TextStyle(
 | 
				
			||||||
 | 
					          fontWeight: FontWeight.normal,
 | 
				
			||||||
 | 
					          fontSize: 14,
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
      ),
 | 
					      ),
 | 
				
			||||||
      validator: _validateInput,
 | 
					      validator: _validateInput,
 | 
				
			||||||
      autovalidateMode: AutovalidateMode.always,
 | 
					      autovalidateMode: AutovalidateMode.always,
 | 
				
			||||||
@ -487,6 +491,10 @@ class PasswordInput extends StatelessWidget {
 | 
				
			|||||||
        labelText: 'login_form_label_password'.tr(),
 | 
					        labelText: 'login_form_label_password'.tr(),
 | 
				
			||||||
        border: const OutlineInputBorder(),
 | 
					        border: const OutlineInputBorder(),
 | 
				
			||||||
        hintText: 'login_form_password_hint'.tr(),
 | 
					        hintText: 'login_form_password_hint'.tr(),
 | 
				
			||||||
 | 
					        hintStyle: const TextStyle(
 | 
				
			||||||
 | 
					          fontWeight: FontWeight.normal,
 | 
				
			||||||
 | 
					          fontSize: 14,
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
      ),
 | 
					      ),
 | 
				
			||||||
      autofillHints: const [AutofillHints.password],
 | 
					      autofillHints: const [AutofillHints.password],
 | 
				
			||||||
      keyboardType: TextInputType.text,
 | 
					      keyboardType: TextInputType.text,
 | 
				
			||||||
 | 
				
			|||||||
@ -18,7 +18,7 @@ class SearchResultPageNotifier extends StateNotifier<SearchResultPageState> {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  final SearchService _searchService;
 | 
					  final SearchService _searchService;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  void search(String searchTerm) async {
 | 
					  void search(String searchTerm, {bool clipEnable = true}) async {
 | 
				
			||||||
    state = state.copyWith(
 | 
					    state = state.copyWith(
 | 
				
			||||||
      searchResult: [],
 | 
					      searchResult: [],
 | 
				
			||||||
      isError: false,
 | 
					      isError: false,
 | 
				
			||||||
@ -26,7 +26,10 @@ class SearchResultPageNotifier extends StateNotifier<SearchResultPageState> {
 | 
				
			|||||||
      isSuccess: false,
 | 
					      isSuccess: false,
 | 
				
			||||||
    );
 | 
					    );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    List<Asset>? assets = await _searchService.searchAsset(searchTerm);
 | 
					    List<Asset>? assets = await _searchService.searchAsset(
 | 
				
			||||||
 | 
					      searchTerm,
 | 
				
			||||||
 | 
					      clipEnable: clipEnable,
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (assets != null) {
 | 
					    if (assets != null) {
 | 
				
			||||||
      state = state.copyWith(
 | 
					      state = state.copyWith(
 | 
				
			||||||
 | 
				
			|||||||
@ -29,11 +29,16 @@ class SearchService {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  Future<List<Asset>?> searchAsset(String searchTerm) async {
 | 
					  Future<List<Asset>?> searchAsset(
 | 
				
			||||||
 | 
					    String searchTerm, {
 | 
				
			||||||
 | 
					    bool clipEnable = true,
 | 
				
			||||||
 | 
					  }) async {
 | 
				
			||||||
    // TODO search in local DB: 1. when offline, 2. to find local assets
 | 
					    // TODO search in local DB: 1. when offline, 2. to find local assets
 | 
				
			||||||
    try {
 | 
					    try {
 | 
				
			||||||
      final SearchResponseDto? results = await _apiService.searchApi
 | 
					      final SearchResponseDto? results = await _apiService.searchApi.search(
 | 
				
			||||||
          .search(query: searchTerm, clip: true);
 | 
					        query: searchTerm,
 | 
				
			||||||
 | 
					        clip: clipEnable,
 | 
				
			||||||
 | 
					      );
 | 
				
			||||||
      if (results == null) {
 | 
					      if (results == null) {
 | 
				
			||||||
        return null;
 | 
					        return null;
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
 | 
				
			|||||||
@ -6,7 +6,7 @@ import 'package:immich_mobile/shared/models/store.dart';
 | 
				
			|||||||
class CuratedRow extends StatelessWidget {
 | 
					class CuratedRow extends StatelessWidget {
 | 
				
			||||||
  final List<CuratedContent> content;
 | 
					  final List<CuratedContent> content;
 | 
				
			||||||
  final double imageSize;
 | 
					  final double imageSize;
 | 
				
			||||||
  
 | 
					
 | 
				
			||||||
  /// Callback with the content and the index when tapped
 | 
					  /// Callback with the content and the index when tapped
 | 
				
			||||||
  final Function(CuratedContent, int)? onTap;
 | 
					  final Function(CuratedContent, int)? onTap;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -19,7 +19,6 @@ class CuratedRow extends StatelessWidget {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
  Widget build(BuildContext context) {
 | 
					  Widget build(BuildContext context) {
 | 
				
			||||||
 | 
					 | 
				
			||||||
    // Guard empty [content]
 | 
					    // Guard empty [content]
 | 
				
			||||||
    if (content.isEmpty) {
 | 
					    if (content.isEmpty) {
 | 
				
			||||||
      // Return empty thumbnail
 | 
					      // Return empty thumbnail
 | 
				
			||||||
 | 
				
			|||||||
@ -22,8 +22,7 @@ class ExploreGrid extends StatelessWidget {
 | 
				
			|||||||
          width: 100,
 | 
					          width: 100,
 | 
				
			||||||
          child: ThumbnailWithInfo(
 | 
					          child: ThumbnailWithInfo(
 | 
				
			||||||
            textInfo: '',
 | 
					            textInfo: '',
 | 
				
			||||||
            onTap: () {
 | 
					            onTap: () {},
 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
@ -42,9 +41,10 @@ class ExploreGrid extends StatelessWidget {
 | 
				
			|||||||
        return ThumbnailWithInfo(
 | 
					        return ThumbnailWithInfo(
 | 
				
			||||||
          imageUrl: thumbnailRequestUrl,
 | 
					          imageUrl: thumbnailRequestUrl,
 | 
				
			||||||
          textInfo: content.label,
 | 
					          textInfo: content.label,
 | 
				
			||||||
 | 
					          borderRadius: 0,
 | 
				
			||||||
          onTap: () {
 | 
					          onTap: () {
 | 
				
			||||||
            AutoRouter.of(context).push(
 | 
					            AutoRouter.of(context).push(
 | 
				
			||||||
              SearchResultRoute(searchTerm: content.label),
 | 
					              SearchResultRoute(searchTerm: 'm:${content.label}'),
 | 
				
			||||||
            );
 | 
					            );
 | 
				
			||||||
          },
 | 
					          },
 | 
				
			||||||
        );
 | 
					        );
 | 
				
			||||||
@ -52,5 +52,4 @@ class ExploreGrid extends StatelessWidget {
 | 
				
			|||||||
      itemCount: curatedContent.length,
 | 
					      itemCount: curatedContent.length,
 | 
				
			||||||
    );
 | 
					    );
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -30,7 +30,10 @@ class SearchBar extends HookConsumerWidget with PreferredSizeWidget {
 | 
				
			|||||||
              },
 | 
					              },
 | 
				
			||||||
              icon: const Icon(Icons.arrow_back_ios_rounded),
 | 
					              icon: const Icon(Icons.arrow_back_ios_rounded),
 | 
				
			||||||
            )
 | 
					            )
 | 
				
			||||||
          : const Icon(Icons.search_rounded),
 | 
					          : const Icon(
 | 
				
			||||||
 | 
					              Icons.search_rounded,
 | 
				
			||||||
 | 
					              size: 20,
 | 
				
			||||||
 | 
					            ),
 | 
				
			||||||
      title: TextField(
 | 
					      title: TextField(
 | 
				
			||||||
        controller: searchTermController,
 | 
					        controller: searchTermController,
 | 
				
			||||||
        focusNode: searchFocusNode,
 | 
					        focusNode: searchFocusNode,
 | 
				
			||||||
@ -55,6 +58,8 @@ class SearchBar extends HookConsumerWidget with PreferredSizeWidget {
 | 
				
			|||||||
          hintText: 'search_bar_hint'.tr(),
 | 
					          hintText: 'search_bar_hint'.tr(),
 | 
				
			||||||
          hintStyle: Theme.of(context).textTheme.titleSmall?.copyWith(
 | 
					          hintStyle: Theme.of(context).textTheme.titleSmall?.copyWith(
 | 
				
			||||||
                color: Theme.of(context).colorScheme.onSurface.withOpacity(0.5),
 | 
					                color: Theme.of(context).colorScheme.onSurface.withOpacity(0.5),
 | 
				
			||||||
 | 
					                fontWeight: FontWeight.w500,
 | 
				
			||||||
 | 
					                fontSize: 14,
 | 
				
			||||||
              ),
 | 
					              ),
 | 
				
			||||||
          enabledBorder: const UnderlineInputBorder(
 | 
					          enabledBorder: const UnderlineInputBorder(
 | 
				
			||||||
            borderSide: BorderSide(color: Colors.transparent),
 | 
					            borderSide: BorderSide(color: Colors.transparent),
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										31
									
								
								mobile/lib/modules/search/ui/search_result_grid.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								mobile/lib/modules/search/ui/search_result_grid.dart
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,31 @@
 | 
				
			|||||||
 | 
					import 'package:flutter/material.dart';
 | 
				
			||||||
 | 
					import 'package:hooks_riverpod/hooks_riverpod.dart';
 | 
				
			||||||
 | 
					import 'package:immich_mobile/modules/home/ui/asset_grid/thumbnail_image.dart';
 | 
				
			||||||
 | 
					import 'package:immich_mobile/shared/models/asset.dart';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class SearchResultGrid extends HookConsumerWidget {
 | 
				
			||||||
 | 
					  const SearchResultGrid({super.key, required this.assets});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  final List<Asset> assets;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  @override
 | 
				
			||||||
 | 
					  Widget build(BuildContext context, WidgetRef ref) {
 | 
				
			||||||
 | 
					    return GridView.builder(
 | 
				
			||||||
 | 
					      gridDelegate: const SliverGridDelegateWithFixedCrossAxisCount(
 | 
				
			||||||
 | 
					        crossAxisCount: 4,
 | 
				
			||||||
 | 
					        childAspectRatio: 1,
 | 
				
			||||||
 | 
					        crossAxisSpacing: 4,
 | 
				
			||||||
 | 
					        mainAxisSpacing: 4,
 | 
				
			||||||
 | 
					      ),
 | 
				
			||||||
 | 
					      itemCount: assets.length,
 | 
				
			||||||
 | 
					      itemBuilder: (context, index) {
 | 
				
			||||||
 | 
					        final asset = assets[index];
 | 
				
			||||||
 | 
					        return ThumbnailImage(
 | 
				
			||||||
 | 
					          asset: asset,
 | 
				
			||||||
 | 
					          assetList: assets,
 | 
				
			||||||
 | 
					          useGrayBoxPlaceholder: true,
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@ -1,3 +1,4 @@
 | 
				
			|||||||
 | 
					import 'package:easy_localization/easy_localization.dart';
 | 
				
			||||||
import 'package:flutter/material.dart';
 | 
					import 'package:flutter/material.dart';
 | 
				
			||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
 | 
					import 'package:hooks_riverpod/hooks_riverpod.dart';
 | 
				
			||||||
import 'package:immich_mobile/modules/search/providers/search_page_state.provider.dart';
 | 
					import 'package:immich_mobile/modules/search/providers/search_page_state.provider.dart';
 | 
				
			||||||
@ -12,6 +13,7 @@ class SearchSuggestionList extends ConsumerWidget {
 | 
				
			|||||||
    final searchTerm = ref.watch(searchPageStateProvider).searchTerm;
 | 
					    final searchTerm = ref.watch(searchPageStateProvider).searchTerm;
 | 
				
			||||||
    final searchSuggestion =
 | 
					    final searchSuggestion =
 | 
				
			||||||
        ref.watch(searchPageStateProvider).searchSuggestion;
 | 
					        ref.watch(searchPageStateProvider).searchSuggestion;
 | 
				
			||||||
 | 
					    var isDarkTheme = Theme.of(context).brightness == Brightness.dark;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return Container(
 | 
					    return Container(
 | 
				
			||||||
      color: searchTerm.isEmpty
 | 
					      color: searchTerm.isEmpty
 | 
				
			||||||
@ -19,13 +21,38 @@ class SearchSuggestionList extends ConsumerWidget {
 | 
				
			|||||||
          : Theme.of(context).scaffoldBackgroundColor,
 | 
					          : Theme.of(context).scaffoldBackgroundColor,
 | 
				
			||||||
      child: CustomScrollView(
 | 
					      child: CustomScrollView(
 | 
				
			||||||
        slivers: [
 | 
					        slivers: [
 | 
				
			||||||
 | 
					          SliverToBoxAdapter(
 | 
				
			||||||
 | 
					            child: Container(
 | 
				
			||||||
 | 
					              color: isDarkTheme ? Colors.grey[800] : Colors.grey[100],
 | 
				
			||||||
 | 
					              child: Padding(
 | 
				
			||||||
 | 
					                padding: const EdgeInsets.all(16.0),
 | 
				
			||||||
 | 
					                child: RichText(
 | 
				
			||||||
 | 
					                  text: TextSpan(
 | 
				
			||||||
 | 
					                    children: [
 | 
				
			||||||
 | 
					                      TextSpan(
 | 
				
			||||||
 | 
					                        text: 'search_suggestion_list_smart_search_hint_1'.tr(),
 | 
				
			||||||
 | 
					                        style: Theme.of(context).textTheme.bodyMedium,
 | 
				
			||||||
 | 
					                      ),
 | 
				
			||||||
 | 
					                      TextSpan(
 | 
				
			||||||
 | 
					                        text: 'search_suggestion_list_smart_search_hint_2'.tr(),
 | 
				
			||||||
 | 
					                        style: Theme.of(context).textTheme.bodyMedium?.copyWith(
 | 
				
			||||||
 | 
					                              color: Theme.of(context).primaryColor,
 | 
				
			||||||
 | 
					                              fontWeight: FontWeight.bold,
 | 
				
			||||||
 | 
					                            ),
 | 
				
			||||||
 | 
					                      )
 | 
				
			||||||
 | 
					                    ],
 | 
				
			||||||
 | 
					                  ),
 | 
				
			||||||
 | 
					                ),
 | 
				
			||||||
 | 
					              ),
 | 
				
			||||||
 | 
					            ),
 | 
				
			||||||
 | 
					          ),
 | 
				
			||||||
          SliverFillRemaining(
 | 
					          SliverFillRemaining(
 | 
				
			||||||
            hasScrollBody: true,
 | 
					            hasScrollBody: true,
 | 
				
			||||||
            child: ListView.builder(
 | 
					            child: ListView.builder(
 | 
				
			||||||
              itemBuilder: ((context, index) {
 | 
					              itemBuilder: ((context, index) {
 | 
				
			||||||
                return ListTile(
 | 
					                return ListTile(
 | 
				
			||||||
                  onTap: () {
 | 
					                  onTap: () {
 | 
				
			||||||
                    onSubmitted(searchSuggestion[index]);
 | 
					                    onSubmitted("m:${searchSuggestion[index]}");
 | 
				
			||||||
                  },
 | 
					                  },
 | 
				
			||||||
                  title: Text(searchSuggestion[index]),
 | 
					                  title: Text(searchSuggestion[index]),
 | 
				
			||||||
                );
 | 
					                );
 | 
				
			||||||
 | 
				
			|||||||
@ -1,13 +1,16 @@
 | 
				
			|||||||
import 'package:cached_network_image/cached_network_image.dart';
 | 
					import 'package:cached_network_image/cached_network_image.dart';
 | 
				
			||||||
import 'package:flutter/material.dart';
 | 
					import 'package:flutter/material.dart';
 | 
				
			||||||
import 'package:immich_mobile/shared/models/store.dart';
 | 
					import 'package:immich_mobile/shared/models/store.dart';
 | 
				
			||||||
 | 
					import 'package:immich_mobile/utils/capitalize_first_letter.dart';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ignore: must_be_immutable
 | 
				
			||||||
class ThumbnailWithInfo extends StatelessWidget {
 | 
					class ThumbnailWithInfo extends StatelessWidget {
 | 
				
			||||||
  const ThumbnailWithInfo({
 | 
					  ThumbnailWithInfo({
 | 
				
			||||||
    Key? key,
 | 
					    Key? key,
 | 
				
			||||||
    required this.textInfo,
 | 
					    required this.textInfo,
 | 
				
			||||||
    this.imageUrl,
 | 
					    this.imageUrl,
 | 
				
			||||||
    this.noImageIcon,
 | 
					    this.noImageIcon,
 | 
				
			||||||
 | 
					    this.borderRadius = 10,
 | 
				
			||||||
    required this.onTap,
 | 
					    required this.onTap,
 | 
				
			||||||
  }) : super(key: key);
 | 
					  }) : super(key: key);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -15,6 +18,7 @@ class ThumbnailWithInfo extends StatelessWidget {
 | 
				
			|||||||
  final String? imageUrl;
 | 
					  final String? imageUrl;
 | 
				
			||||||
  final Function onTap;
 | 
					  final Function onTap;
 | 
				
			||||||
  final IconData? noImageIcon;
 | 
					  final IconData? noImageIcon;
 | 
				
			||||||
 | 
					  double borderRadius;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
  Widget build(BuildContext context) {
 | 
					  Widget build(BuildContext context) {
 | 
				
			||||||
@ -29,12 +33,12 @@ class ThumbnailWithInfo extends StatelessWidget {
 | 
				
			|||||||
        children: [
 | 
					        children: [
 | 
				
			||||||
          Container(
 | 
					          Container(
 | 
				
			||||||
            decoration: BoxDecoration(
 | 
					            decoration: BoxDecoration(
 | 
				
			||||||
              borderRadius: BorderRadius.circular(25),
 | 
					              borderRadius: BorderRadius.circular(borderRadius),
 | 
				
			||||||
              color: isDarkMode ? Colors.grey[900] : Colors.grey[100],
 | 
					              color: isDarkMode ? Colors.grey[900] : Colors.grey[100],
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
            child: imageUrl != null
 | 
					            child: imageUrl != null
 | 
				
			||||||
                ? ClipRRect(
 | 
					                ? ClipRRect(
 | 
				
			||||||
                    borderRadius: BorderRadius.circular(20),
 | 
					                    borderRadius: BorderRadius.circular(borderRadius),
 | 
				
			||||||
                    child: CachedNetworkImage(
 | 
					                    child: CachedNetworkImage(
 | 
				
			||||||
                      width: double.infinity,
 | 
					                      width: double.infinity,
 | 
				
			||||||
                      height: double.infinity,
 | 
					                      height: double.infinity,
 | 
				
			||||||
@ -55,15 +59,32 @@ class ThumbnailWithInfo extends StatelessWidget {
 | 
				
			|||||||
                    ),
 | 
					                    ),
 | 
				
			||||||
                  ),
 | 
					                  ),
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
 | 
					          Container(
 | 
				
			||||||
 | 
					            decoration: BoxDecoration(
 | 
				
			||||||
 | 
					              borderRadius: BorderRadius.circular(borderRadius),
 | 
				
			||||||
 | 
					              color: Colors.white,
 | 
				
			||||||
 | 
					              gradient: LinearGradient(
 | 
				
			||||||
 | 
					                begin: FractionalOffset.topCenter,
 | 
				
			||||||
 | 
					                end: FractionalOffset.bottomCenter,
 | 
				
			||||||
 | 
					                colors: [
 | 
				
			||||||
 | 
					                  Colors.grey.withOpacity(0.0),
 | 
				
			||||||
 | 
					                  textInfo == ''
 | 
				
			||||||
 | 
					                      ? Colors.black.withOpacity(0.1)
 | 
				
			||||||
 | 
					                      : Colors.black.withOpacity(0.5),
 | 
				
			||||||
 | 
					                ],
 | 
				
			||||||
 | 
					                stops: const [0.0, 1.0],
 | 
				
			||||||
 | 
					              ),
 | 
				
			||||||
 | 
					            ),
 | 
				
			||||||
 | 
					          ),
 | 
				
			||||||
          Positioned(
 | 
					          Positioned(
 | 
				
			||||||
            bottom: 12,
 | 
					            bottom: 12,
 | 
				
			||||||
            left: 14,
 | 
					            left: 14,
 | 
				
			||||||
            child: Text(
 | 
					            child: Text(
 | 
				
			||||||
              textInfo,
 | 
					              textInfo == '' ? textInfo : textInfo.capitalizeFirstLetter(),
 | 
				
			||||||
              style: const TextStyle(
 | 
					              style: const TextStyle(
 | 
				
			||||||
                color: Colors.white,
 | 
					                color: Colors.white,
 | 
				
			||||||
                fontWeight: FontWeight.bold,
 | 
					                fontWeight: FontWeight.bold,
 | 
				
			||||||
                fontSize: 12,
 | 
					                fontSize: 14,
 | 
				
			||||||
              ),
 | 
					              ),
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
 | 
				
			|||||||
@ -27,6 +27,7 @@ class SearchPage extends HookConsumerWidget {
 | 
				
			|||||||
        ref.watch(getCuratedObjectProvider);
 | 
					        ref.watch(getCuratedObjectProvider);
 | 
				
			||||||
    var isDarkTheme = Theme.of(context).brightness == Brightness.dark;
 | 
					    var isDarkTheme = Theme.of(context).brightness == Brightness.dark;
 | 
				
			||||||
    double imageSize = MediaQuery.of(context).size.width / 3;
 | 
					    double imageSize = MediaQuery.of(context).size.width / 3;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    TextStyle categoryTitleStyle = const TextStyle(
 | 
					    TextStyle categoryTitleStyle = const TextStyle(
 | 
				
			||||||
      fontWeight: FontWeight.bold,
 | 
					      fontWeight: FontWeight.bold,
 | 
				
			||||||
      fontSize: 14.0,
 | 
					      fontSize: 14.0,
 | 
				
			||||||
@ -46,7 +47,11 @@ class SearchPage extends HookConsumerWidget {
 | 
				
			|||||||
      searchFocusNode.unfocus();
 | 
					      searchFocusNode.unfocus();
 | 
				
			||||||
      ref.watch(searchPageStateProvider.notifier).disableSearch();
 | 
					      ref.watch(searchPageStateProvider.notifier).disableSearch();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      AutoRouter.of(context).push(SearchResultRoute(searchTerm: searchTerm));
 | 
					      AutoRouter.of(context).push(
 | 
				
			||||||
 | 
					        SearchResultRoute(
 | 
				
			||||||
 | 
					          searchTerm: searchTerm,
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
 | 
					      );
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    buildPlaces() {
 | 
					    buildPlaces() {
 | 
				
			||||||
@ -67,7 +72,9 @@ class SearchPage extends HookConsumerWidget {
 | 
				
			|||||||
            imageSize: imageSize,
 | 
					            imageSize: imageSize,
 | 
				
			||||||
            onTap: (content, index) {
 | 
					            onTap: (content, index) {
 | 
				
			||||||
              AutoRouter.of(context).push(
 | 
					              AutoRouter.of(context).push(
 | 
				
			||||||
                SearchResultRoute(searchTerm: content.label),
 | 
					                SearchResultRoute(
 | 
				
			||||||
 | 
					                  searchTerm: 'm:${content.label}',
 | 
				
			||||||
 | 
					                ),
 | 
				
			||||||
              );
 | 
					              );
 | 
				
			||||||
            },
 | 
					            },
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
@ -99,7 +106,9 @@ class SearchPage extends HookConsumerWidget {
 | 
				
			|||||||
            imageSize: imageSize,
 | 
					            imageSize: imageSize,
 | 
				
			||||||
            onTap: (content, index) {
 | 
					            onTap: (content, index) {
 | 
				
			||||||
              AutoRouter.of(context).push(
 | 
					              AutoRouter.of(context).push(
 | 
				
			||||||
                SearchResultRoute(searchTerm: content.label),
 | 
					                SearchResultRoute(
 | 
				
			||||||
 | 
					                  searchTerm: 'm:${content.label}',
 | 
				
			||||||
 | 
					                ),
 | 
				
			||||||
              );
 | 
					              );
 | 
				
			||||||
            },
 | 
					            },
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
@ -131,7 +140,7 @@ class SearchPage extends HookConsumerWidget {
 | 
				
			|||||||
                    children: [
 | 
					                    children: [
 | 
				
			||||||
                      Text(
 | 
					                      Text(
 | 
				
			||||||
                        "search_page_places",
 | 
					                        "search_page_places",
 | 
				
			||||||
                        style: Theme.of(context).textTheme.titleMedium,
 | 
					                        style: Theme.of(context).textTheme.titleSmall,
 | 
				
			||||||
                      ).tr(),
 | 
					                      ).tr(),
 | 
				
			||||||
                      TextButton(
 | 
					                      TextButton(
 | 
				
			||||||
                        child: Text(
 | 
					                        child: Text(
 | 
				
			||||||
@ -162,7 +171,7 @@ class SearchPage extends HookConsumerWidget {
 | 
				
			|||||||
                    children: [
 | 
					                    children: [
 | 
				
			||||||
                      Text(
 | 
					                      Text(
 | 
				
			||||||
                        "search_page_things",
 | 
					                        "search_page_things",
 | 
				
			||||||
                        style: Theme.of(context).textTheme.titleMedium,
 | 
					                        style: Theme.of(context).textTheme.titleSmall,
 | 
				
			||||||
                      ).tr(),
 | 
					                      ).tr(),
 | 
				
			||||||
                      TextButton(
 | 
					                      TextButton(
 | 
				
			||||||
                        child: Text(
 | 
					                        child: Text(
 | 
				
			||||||
@ -186,7 +195,7 @@ class SearchPage extends HookConsumerWidget {
 | 
				
			|||||||
                  padding: const EdgeInsets.symmetric(horizontal: 16),
 | 
					                  padding: const EdgeInsets.symmetric(horizontal: 16),
 | 
				
			||||||
                  child: Text(
 | 
					                  child: Text(
 | 
				
			||||||
                    'search_page_your_activity',
 | 
					                    'search_page_your_activity',
 | 
				
			||||||
                    style: Theme.of(context).textTheme.titleMedium,
 | 
					                    style: Theme.of(context).textTheme.titleSmall,
 | 
				
			||||||
                  ).tr(),
 | 
					                  ).tr(),
 | 
				
			||||||
                ),
 | 
					                ),
 | 
				
			||||||
                ListTile(
 | 
					                ListTile(
 | 
				
			||||||
@ -201,13 +210,7 @@ class SearchPage extends HookConsumerWidget {
 | 
				
			|||||||
                    const FavoritesRoute(),
 | 
					                    const FavoritesRoute(),
 | 
				
			||||||
                  ),
 | 
					                  ),
 | 
				
			||||||
                ),
 | 
					                ),
 | 
				
			||||||
                const Padding(
 | 
					                const CategoryDivider(),
 | 
				
			||||||
                  padding: EdgeInsets.only(
 | 
					 | 
				
			||||||
                    left: 72,
 | 
					 | 
				
			||||||
                    right: 16,
 | 
					 | 
				
			||||||
                  ),
 | 
					 | 
				
			||||||
                  child: Divider(),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                ListTile(
 | 
					                ListTile(
 | 
				
			||||||
                  leading: Icon(
 | 
					                  leading: Icon(
 | 
				
			||||||
                    Icons.schedule_outlined,
 | 
					                    Icons.schedule_outlined,
 | 
				
			||||||
@ -226,9 +229,36 @@ class SearchPage extends HookConsumerWidget {
 | 
				
			|||||||
                  padding: const EdgeInsets.symmetric(horizontal: 16.0),
 | 
					                  padding: const EdgeInsets.symmetric(horizontal: 16.0),
 | 
				
			||||||
                  child: Text(
 | 
					                  child: Text(
 | 
				
			||||||
                    'search_page_categories',
 | 
					                    'search_page_categories',
 | 
				
			||||||
                    style: Theme.of(context).textTheme.titleMedium,
 | 
					                    style: Theme.of(context).textTheme.titleSmall,
 | 
				
			||||||
                  ).tr(),
 | 
					                  ).tr(),
 | 
				
			||||||
                ),
 | 
					                ),
 | 
				
			||||||
 | 
					                ListTile(
 | 
				
			||||||
 | 
					                  title: Text('Screenshots', style: categoryTitleStyle).tr(),
 | 
				
			||||||
 | 
					                  leading: Icon(
 | 
				
			||||||
 | 
					                    Icons.screenshot,
 | 
				
			||||||
 | 
					                    color: categoryIconColor,
 | 
				
			||||||
 | 
					                  ),
 | 
				
			||||||
 | 
					                  onTap: () => AutoRouter.of(context).push(
 | 
				
			||||||
 | 
					                    SearchResultRoute(
 | 
				
			||||||
 | 
					                      searchTerm: 'screenshots',
 | 
				
			||||||
 | 
					                    ),
 | 
				
			||||||
 | 
					                  ),
 | 
				
			||||||
 | 
					                ),
 | 
				
			||||||
 | 
					                const CategoryDivider(),
 | 
				
			||||||
 | 
					                ListTile(
 | 
				
			||||||
 | 
					                  title: Text('search_page_selfies', style: categoryTitleStyle)
 | 
				
			||||||
 | 
					                      .tr(),
 | 
				
			||||||
 | 
					                  leading: Icon(
 | 
				
			||||||
 | 
					                    Icons.photo_camera_front_outlined,
 | 
				
			||||||
 | 
					                    color: categoryIconColor,
 | 
				
			||||||
 | 
					                  ),
 | 
				
			||||||
 | 
					                  onTap: () => AutoRouter.of(context).push(
 | 
				
			||||||
 | 
					                    SearchResultRoute(
 | 
				
			||||||
 | 
					                      searchTerm: 'selfies',
 | 
				
			||||||
 | 
					                    ),
 | 
				
			||||||
 | 
					                  ),
 | 
				
			||||||
 | 
					                ),
 | 
				
			||||||
 | 
					                const CategoryDivider(),
 | 
				
			||||||
                ListTile(
 | 
					                ListTile(
 | 
				
			||||||
                  title: Text('search_page_videos', style: categoryTitleStyle)
 | 
					                  title: Text('search_page_videos', style: categoryTitleStyle)
 | 
				
			||||||
                      .tr(),
 | 
					                      .tr(),
 | 
				
			||||||
@ -240,13 +270,7 @@ class SearchPage extends HookConsumerWidget {
 | 
				
			|||||||
                    const AllVideosRoute(),
 | 
					                    const AllVideosRoute(),
 | 
				
			||||||
                  ),
 | 
					                  ),
 | 
				
			||||||
                ),
 | 
					                ),
 | 
				
			||||||
                const Padding(
 | 
					                const CategoryDivider(),
 | 
				
			||||||
                  padding: EdgeInsets.only(
 | 
					 | 
				
			||||||
                    left: 72,
 | 
					 | 
				
			||||||
                    right: 16,
 | 
					 | 
				
			||||||
                  ),
 | 
					 | 
				
			||||||
                  child: Divider(),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                ListTile(
 | 
					                ListTile(
 | 
				
			||||||
                  title: Text(
 | 
					                  title: Text(
 | 
				
			||||||
                    'search_page_motion_photos',
 | 
					                    'search_page_motion_photos',
 | 
				
			||||||
@ -270,3 +294,20 @@ class SearchPage extends HookConsumerWidget {
 | 
				
			|||||||
    );
 | 
					    );
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class CategoryDivider extends StatelessWidget {
 | 
				
			||||||
 | 
					  const CategoryDivider({super.key});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  @override
 | 
				
			||||||
 | 
					  Widget build(BuildContext context) {
 | 
				
			||||||
 | 
					    return const Padding(
 | 
				
			||||||
 | 
					      padding: EdgeInsets.only(
 | 
				
			||||||
 | 
					        left: 72,
 | 
				
			||||||
 | 
					        right: 16,
 | 
				
			||||||
 | 
					      ),
 | 
				
			||||||
 | 
					      child: Divider(
 | 
				
			||||||
 | 
					        height: 0,
 | 
				
			||||||
 | 
					      ),
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -6,12 +6,30 @@ import 'package:hooks_riverpod/hooks_riverpod.dart';
 | 
				
			|||||||
import 'package:immich_mobile/modules/home/ui/asset_grid/immich_asset_grid.dart';
 | 
					import 'package:immich_mobile/modules/home/ui/asset_grid/immich_asset_grid.dart';
 | 
				
			||||||
import 'package:immich_mobile/modules/search/providers/search_page_state.provider.dart';
 | 
					import 'package:immich_mobile/modules/search/providers/search_page_state.provider.dart';
 | 
				
			||||||
import 'package:immich_mobile/modules/search/providers/search_result_page.provider.dart';
 | 
					import 'package:immich_mobile/modules/search/providers/search_result_page.provider.dart';
 | 
				
			||||||
 | 
					import 'package:immich_mobile/modules/search/ui/search_result_grid.dart';
 | 
				
			||||||
import 'package:immich_mobile/modules/search/ui/search_suggestion_list.dart';
 | 
					import 'package:immich_mobile/modules/search/ui/search_suggestion_list.dart';
 | 
				
			||||||
import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
 | 
					import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class SearchType {
 | 
				
			||||||
 | 
					  SearchType({required this.isClip, required this.searchTerm});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  final bool isClip;
 | 
				
			||||||
 | 
					  final String searchTerm;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					SearchType _getSearchType(String searchTerm) {
 | 
				
			||||||
 | 
					  if (searchTerm.startsWith('m:')) {
 | 
				
			||||||
 | 
					    return SearchType(isClip: false, searchTerm: searchTerm.substring(2));
 | 
				
			||||||
 | 
					  } else {
 | 
				
			||||||
 | 
					    return SearchType(isClip: true, searchTerm: searchTerm);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class SearchResultPage extends HookConsumerWidget {
 | 
					class SearchResultPage extends HookConsumerWidget {
 | 
				
			||||||
  const SearchResultPage({Key? key, required this.searchTerm})
 | 
					  const SearchResultPage({
 | 
				
			||||||
      : super(key: key);
 | 
					    Key? key,
 | 
				
			||||||
 | 
					    required this.searchTerm,
 | 
				
			||||||
 | 
					  }) : super(key: key);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  final String searchTerm;
 | 
					  final String searchTerm;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -20,6 +38,8 @@ class SearchResultPage extends HookConsumerWidget {
 | 
				
			|||||||
    final searchTermController = useTextEditingController(text: "");
 | 
					    final searchTermController = useTextEditingController(text: "");
 | 
				
			||||||
    final isNewSearch = useState(false);
 | 
					    final isNewSearch = useState(false);
 | 
				
			||||||
    final currentSearchTerm = useState(searchTerm);
 | 
					    final currentSearchTerm = useState(searchTerm);
 | 
				
			||||||
 | 
					    final isDarkTheme = Theme.of(context).brightness == Brightness.dark;
 | 
				
			||||||
 | 
					    final isDisplayDateGroup = useState(true);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    FocusNode? searchFocusNode;
 | 
					    FocusNode? searchFocusNode;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -27,9 +47,16 @@ class SearchResultPage extends HookConsumerWidget {
 | 
				
			|||||||
      () {
 | 
					      () {
 | 
				
			||||||
        searchFocusNode = FocusNode();
 | 
					        searchFocusNode = FocusNode();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        var searchType = _getSearchType(searchTerm);
 | 
				
			||||||
 | 
					        searchType.isClip
 | 
				
			||||||
 | 
					            ? isDisplayDateGroup.value = false
 | 
				
			||||||
 | 
					            : isDisplayDateGroup.value = true;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        Future.delayed(
 | 
					        Future.delayed(
 | 
				
			||||||
          Duration.zero,
 | 
					          Duration.zero,
 | 
				
			||||||
          () => ref.read(searchResultPageProvider.notifier).search(searchTerm),
 | 
					          () => ref
 | 
				
			||||||
 | 
					              .read(searchResultPageProvider.notifier)
 | 
				
			||||||
 | 
					              .search(searchType.searchTerm, clipEnable: searchType.isClip),
 | 
				
			||||||
        );
 | 
					        );
 | 
				
			||||||
        return () => searchFocusNode?.dispose();
 | 
					        return () => searchFocusNode?.dispose();
 | 
				
			||||||
      },
 | 
					      },
 | 
				
			||||||
@ -41,7 +68,15 @@ class SearchResultPage extends HookConsumerWidget {
 | 
				
			|||||||
      searchFocusNode?.unfocus();
 | 
					      searchFocusNode?.unfocus();
 | 
				
			||||||
      isNewSearch.value = false;
 | 
					      isNewSearch.value = false;
 | 
				
			||||||
      currentSearchTerm.value = newSearchTerm;
 | 
					      currentSearchTerm.value = newSearchTerm;
 | 
				
			||||||
      ref.watch(searchResultPageProvider.notifier).search(newSearchTerm);
 | 
					
 | 
				
			||||||
 | 
					      var searchType = _getSearchType(newSearchTerm);
 | 
				
			||||||
 | 
					      searchType.isClip
 | 
				
			||||||
 | 
					          ? isDisplayDateGroup.value = false
 | 
				
			||||||
 | 
					          : isDisplayDateGroup.value = true;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      ref
 | 
				
			||||||
 | 
					          .watch(searchResultPageProvider.notifier)
 | 
				
			||||||
 | 
					          .search(searchType.searchTerm, clipEnable: searchType.isClip);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    buildTextField() {
 | 
					    buildTextField() {
 | 
				
			||||||
@ -74,6 +109,12 @@ class SearchResultPage extends HookConsumerWidget {
 | 
				
			|||||||
          focusedBorder: const UnderlineInputBorder(
 | 
					          focusedBorder: const UnderlineInputBorder(
 | 
				
			||||||
            borderSide: BorderSide(color: Colors.transparent),
 | 
					            borderSide: BorderSide(color: Colors.transparent),
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
 | 
					          hintStyle: TextStyle(
 | 
				
			||||||
 | 
					            fontWeight: FontWeight.bold,
 | 
				
			||||||
 | 
					            fontSize: 16.0,
 | 
				
			||||||
 | 
					            color:
 | 
				
			||||||
 | 
					                isDarkTheme ? Colors.grey[500] : Colors.black.withOpacity(0.5),
 | 
				
			||||||
 | 
					          ),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@ -121,11 +162,16 @@ class SearchResultPage extends HookConsumerWidget {
 | 
				
			|||||||
        return const Center(child: ImmichLoadingIndicator());
 | 
					        return const Center(child: ImmichLoadingIndicator());
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 | 
				
			||||||
      if (searchResultPageState.isSuccess) {
 | 
					      if (searchResultPageState.isSuccess) {
 | 
				
			||||||
        return ImmichAssetGrid(
 | 
					        if (isDisplayDateGroup.value) {
 | 
				
			||||||
 | 
					          return ImmichAssetGrid(
 | 
				
			||||||
            assets: allSearchAssets,
 | 
					            assets: allSearchAssets,
 | 
				
			||||||
        );
 | 
					          );
 | 
				
			||||||
 | 
					        } else {
 | 
				
			||||||
 | 
					          return SearchResultGrid(
 | 
				
			||||||
 | 
					            assets: allSearchAssets,
 | 
				
			||||||
 | 
					          );
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      return const SizedBox();
 | 
					      return const SizedBox();
 | 
				
			||||||
 | 
				
			|||||||
@ -144,14 +144,9 @@ class _$AppRouter extends RootStackRouter {
 | 
				
			|||||||
      );
 | 
					      );
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    RecentlyAddedRoute.name: (routeData) {
 | 
					    RecentlyAddedRoute.name: (routeData) {
 | 
				
			||||||
      return CustomPage<dynamic>(
 | 
					      return MaterialPageX<dynamic>(
 | 
				
			||||||
        routeData: routeData,
 | 
					        routeData: routeData,
 | 
				
			||||||
        child: const RecentlyAddedPage(),
 | 
					        child: const RecentlyAddedPage(),
 | 
				
			||||||
        transitionsBuilder: TransitionsBuilders.noTransition,
 | 
					 | 
				
			||||||
        durationInMilliseconds: 200,
 | 
					 | 
				
			||||||
        reverseDurationInMilliseconds: 200,
 | 
					 | 
				
			||||||
        opaque: true,
 | 
					 | 
				
			||||||
        barrierDismissible: false,
 | 
					 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    AssetSelectionRoute.name: (routeData) {
 | 
					    AssetSelectionRoute.name: (routeData) {
 | 
				
			||||||
 | 
				
			|||||||
@ -79,6 +79,7 @@ ThemeData immichLightTheme = ThemeData(
 | 
				
			|||||||
    ),
 | 
					    ),
 | 
				
			||||||
    titleSmall: TextStyle(
 | 
					    titleSmall: TextStyle(
 | 
				
			||||||
      fontSize: 16.0,
 | 
					      fontSize: 16.0,
 | 
				
			||||||
 | 
					      fontWeight: FontWeight.bold,
 | 
				
			||||||
    ),
 | 
					    ),
 | 
				
			||||||
    titleMedium: TextStyle(
 | 
					    titleMedium: TextStyle(
 | 
				
			||||||
      fontSize: 18.0,
 | 
					      fontSize: 18.0,
 | 
				
			||||||
@ -176,6 +177,7 @@ ThemeData immichDarkTheme = ThemeData(
 | 
				
			|||||||
    ),
 | 
					    ),
 | 
				
			||||||
    titleSmall: const TextStyle(
 | 
					    titleSmall: const TextStyle(
 | 
				
			||||||
      fontSize: 16.0,
 | 
					      fontSize: 16.0,
 | 
				
			||||||
 | 
					      fontWeight: FontWeight.bold,
 | 
				
			||||||
    ),
 | 
					    ),
 | 
				
			||||||
    titleMedium: const TextStyle(
 | 
					    titleMedium: const TextStyle(
 | 
				
			||||||
      fontSize: 18.0,
 | 
					      fontSize: 18.0,
 | 
				
			||||||
@ -185,7 +187,6 @@ ThemeData immichDarkTheme = ThemeData(
 | 
				
			|||||||
      fontSize: 26.0,
 | 
					      fontSize: 26.0,
 | 
				
			||||||
      fontWeight: FontWeight.bold,
 | 
					      fontWeight: FontWeight.bold,
 | 
				
			||||||
    ),
 | 
					    ),
 | 
				
			||||||
 | 
					 | 
				
			||||||
  ),
 | 
					  ),
 | 
				
			||||||
  cardColor: Colors.grey[900],
 | 
					  cardColor: Colors.grey[900],
 | 
				
			||||||
  elevatedButtonTheme: ElevatedButtonThemeData(
 | 
					  elevatedButtonTheme: ElevatedButtonThemeData(
 | 
				
			||||||
 | 
				
			|||||||
@ -187,6 +187,16 @@ class MockAssetNotifier extends _i1.Mock implements _i2.AssetNotifier {
 | 
				
			|||||||
        returnValueForMissingStub: _i5.Future<void>.value(),
 | 
					        returnValueForMissingStub: _i5.Future<void>.value(),
 | 
				
			||||||
      ) as _i5.Future<void>);
 | 
					      ) as _i5.Future<void>);
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
 | 
					  _i5.Future<void> getAllAsset({bool? clear = false}) => (super.noSuchMethod(
 | 
				
			||||||
 | 
					        Invocation.method(
 | 
				
			||||||
 | 
					          #getAllAsset,
 | 
				
			||||||
 | 
					          [],
 | 
				
			||||||
 | 
					          {#clear: clear},
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
 | 
					        returnValue: _i5.Future<void>.value(),
 | 
				
			||||||
 | 
					        returnValueForMissingStub: _i5.Future<void>.value(),
 | 
				
			||||||
 | 
					      ) as _i5.Future<void>);
 | 
				
			||||||
 | 
					  @override
 | 
				
			||||||
  _i5.Future<void> clearAllAsset() => (super.noSuchMethod(
 | 
					  _i5.Future<void> clearAllAsset() => (super.noSuchMethod(
 | 
				
			||||||
        Invocation.method(
 | 
					        Invocation.method(
 | 
				
			||||||
          #clearAllAsset,
 | 
					          #clearAllAsset,
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user