mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-31 10:37:11 -04:00 
			
		
		
		
	* feat: search by description * wip: mobile * wip: mobile ui * wip: mobile search logic * feat: using f_unaccent * icon to fit with text search
		
			
				
	
	
		
			142 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			142 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| import 'package:flutter/material.dart';
 | |
| import 'package:hooks_riverpod/hooks_riverpod.dart';
 | |
| import 'package:immich_mobile/extensions/string_extensions.dart';
 | |
| import 'package:immich_mobile/interfaces/asset.interface.dart';
 | |
| import 'package:immich_mobile/models/search/search_filter.model.dart';
 | |
| import 'package:immich_mobile/entities/asset.entity.dart';
 | |
| import 'package:immich_mobile/models/search/search_result.model.dart';
 | |
| import 'package:immich_mobile/providers/api.provider.dart';
 | |
| import 'package:immich_mobile/repositories/asset.repository.dart';
 | |
| import 'package:immich_mobile/services/api.service.dart';
 | |
| import 'package:logging/logging.dart';
 | |
| import 'package:openapi/api.dart';
 | |
| 
 | |
| final searchServiceProvider = Provider(
 | |
|   (ref) => SearchService(
 | |
|     ref.watch(apiServiceProvider),
 | |
|     ref.watch(assetRepositoryProvider),
 | |
|   ),
 | |
| );
 | |
| 
 | |
| class SearchService {
 | |
|   final ApiService _apiService;
 | |
|   final IAssetRepository _assetRepository;
 | |
| 
 | |
|   final _log = Logger("SearchService");
 | |
|   SearchService(this._apiService, this._assetRepository);
 | |
| 
 | |
|   Future<List<String>?> getSearchSuggestions(
 | |
|     SearchSuggestionType type, {
 | |
|     String? country,
 | |
|     String? state,
 | |
|     String? make,
 | |
|     String? model,
 | |
|   }) async {
 | |
|     try {
 | |
|       return await _apiService.searchApi.getSearchSuggestions(
 | |
|         type,
 | |
|         country: country,
 | |
|         state: state,
 | |
|         make: make,
 | |
|         model: model,
 | |
|       );
 | |
|     } catch (e) {
 | |
|       debugPrint("[ERROR] [getSearchSuggestions] ${e.toString()}");
 | |
|       return [];
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   Future<SearchResult?> search(SearchFilter filter, int page) async {
 | |
|     try {
 | |
|       SearchResponseDto? response;
 | |
|       AssetTypeEnum? type;
 | |
|       if (filter.mediaType == AssetType.image) {
 | |
|         type = AssetTypeEnum.IMAGE;
 | |
|       } else if (filter.mediaType == AssetType.video) {
 | |
|         type = AssetTypeEnum.VIDEO;
 | |
|       }
 | |
| 
 | |
|       if (filter.context != null && filter.context!.isNotEmpty) {
 | |
|         response = await _apiService.searchApi.searchSmart(
 | |
|           SmartSearchDto(
 | |
|             query: filter.context!,
 | |
|             country: filter.location.country,
 | |
|             state: filter.location.state,
 | |
|             city: filter.location.city,
 | |
|             make: filter.camera.make,
 | |
|             model: filter.camera.model,
 | |
|             takenAfter: filter.date.takenAfter,
 | |
|             takenBefore: filter.date.takenBefore,
 | |
|             isArchived: filter.display.isArchive ? true : null,
 | |
|             isFavorite: filter.display.isFavorite ? true : null,
 | |
|             isNotInAlbum: filter.display.isNotInAlbum ? true : null,
 | |
|             personIds: filter.people.map((e) => e.id).toList(),
 | |
|             type: type,
 | |
|             page: page,
 | |
|             size: 1000,
 | |
|           ),
 | |
|         );
 | |
|       } else {
 | |
|         response = await _apiService.searchApi.searchAssets(
 | |
|           MetadataSearchDto(
 | |
|             originalFileName:
 | |
|                 filter.filename != null && filter.filename!.isNotEmpty
 | |
|                     ? filter.filename
 | |
|                     : null,
 | |
|             country: filter.location.country,
 | |
|             description:
 | |
|                 filter.description != null && filter.description!.isNotEmpty
 | |
|                     ? filter.description
 | |
|                     : null,
 | |
|             state: filter.location.state,
 | |
|             city: filter.location.city,
 | |
|             make: filter.camera.make,
 | |
|             model: filter.camera.model,
 | |
|             takenAfter: filter.date.takenAfter,
 | |
|             takenBefore: filter.date.takenBefore,
 | |
|             isArchived: filter.display.isArchive ? true : null,
 | |
|             isFavorite: filter.display.isFavorite ? true : null,
 | |
|             isNotInAlbum: filter.display.isNotInAlbum ? true : null,
 | |
|             personIds: filter.people.map((e) => e.id).toList(),
 | |
|             type: type,
 | |
|             page: page,
 | |
|             size: 1000,
 | |
|           ),
 | |
|         );
 | |
|       }
 | |
| 
 | |
|       if (response == null || response.assets.items.isEmpty) {
 | |
|         return null;
 | |
|       }
 | |
| 
 | |
|       return SearchResult(
 | |
|         assets: await _assetRepository.getAllByRemoteId(
 | |
|           response.assets.items.map((e) => e.id),
 | |
|         ),
 | |
|         nextPage: response.assets.nextPage?.toInt(),
 | |
|       );
 | |
|     } catch (error, stackTrace) {
 | |
|       _log.severe("Failed to search for assets", error, stackTrace);
 | |
|     }
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   Future<List<SearchExploreResponseDto>?> getExploreData() async {
 | |
|     try {
 | |
|       return await _apiService.searchApi.getExploreData();
 | |
|     } catch (error, stackTrace) {
 | |
|       _log.severe("Failed to getExploreData", error, stackTrace);
 | |
|     }
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   Future<List<AssetResponseDto>?> getAllPlaces() async {
 | |
|     try {
 | |
|       return await _apiService.searchApi.getAssetsByCity();
 | |
|     } catch (error, stackTrace) {
 | |
|       _log.severe("Failed to getAllPlaces", error, stackTrace);
 | |
|     }
 | |
|     return null;
 | |
|   }
 | |
| }
 |