2022-02-27 19:43:29 +01:00
|
|
|
import 'package:flutter/material.dart';
|
2022-06-25 20:46:51 +02:00
|
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
2024-10-30 20:27:13 +01:00
|
|
|
import 'package:immich_mobile/extensions/string_extensions.dart';
|
2024-09-24 14:50:21 +02:00
|
|
|
import 'package:immich_mobile/interfaces/asset.interface.dart';
|
2024-05-01 04:36:40 +02:00
|
|
|
import 'package:immich_mobile/models/search/search_filter.model.dart';
|
|
|
|
import 'package:immich_mobile/entities/asset.entity.dart';
|
2024-10-30 20:27:13 +01:00
|
|
|
import 'package:immich_mobile/models/search/search_result.model.dart';
|
2024-05-02 22:59:14 +02:00
|
|
|
import 'package:immich_mobile/providers/api.provider.dart';
|
2024-09-24 14:50:21 +02:00
|
|
|
import 'package:immich_mobile/repositories/asset.repository.dart';
|
2024-05-02 22:59:14 +02:00
|
|
|
import 'package:immich_mobile/services/api.service.dart';
|
2024-04-26 07:49:31 +02:00
|
|
|
import 'package:logging/logging.dart';
|
2022-07-13 14:23:48 +02:00
|
|
|
import 'package:openapi/api.dart';
|
2022-02-27 19:43:29 +01:00
|
|
|
|
2022-07-13 14:23:48 +02:00
|
|
|
final searchServiceProvider = Provider(
|
|
|
|
(ref) => SearchService(
|
|
|
|
ref.watch(apiServiceProvider),
|
2024-09-24 14:50:21 +02:00
|
|
|
ref.watch(assetRepositoryProvider),
|
2022-07-13 14:23:48 +02:00
|
|
|
),
|
|
|
|
);
|
2022-06-25 20:46:51 +02:00
|
|
|
|
2022-02-27 19:43:29 +01:00
|
|
|
class SearchService {
|
2022-07-13 14:23:48 +02:00
|
|
|
final ApiService _apiService;
|
2024-09-24 14:50:21 +02:00
|
|
|
final IAssetRepository _assetRepository;
|
2022-08-03 07:04:34 +02:00
|
|
|
|
2024-04-26 07:49:31 +02:00
|
|
|
final _log = Logger("SearchService");
|
2024-09-24 14:50:21 +02:00
|
|
|
SearchService(this._apiService, this._assetRepository);
|
2022-02-27 19:43:29 +01:00
|
|
|
|
2024-04-01 16:45:11 +02:00
|
|
|
Future<List<String>?> getSearchSuggestions(
|
|
|
|
SearchSuggestionType type, {
|
|
|
|
String? country,
|
|
|
|
String? state,
|
|
|
|
String? make,
|
|
|
|
String? model,
|
2023-03-28 17:34:06 +02:00
|
|
|
}) async {
|
2022-03-02 23:44:24 +01:00
|
|
|
try {
|
2024-04-01 16:45:11 +02:00
|
|
|
return await _apiService.searchApi.getSearchSuggestions(
|
|
|
|
type,
|
|
|
|
country: country,
|
|
|
|
state: state,
|
|
|
|
make: make,
|
|
|
|
model: model,
|
2023-03-28 17:34:06 +02:00
|
|
|
);
|
2024-04-01 16:45:11 +02:00
|
|
|
} catch (e) {
|
|
|
|
debugPrint("[ERROR] [getSearchSuggestions] ${e.toString()}");
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-30 20:27:13 +01:00
|
|
|
Future<SearchResult?> search(SearchFilter filter, int page) async {
|
2024-04-01 16:45:11 +02:00
|
|
|
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,
|
|
|
|
isFavorite: filter.display.isFavorite,
|
|
|
|
isNotInAlbum: filter.display.isNotInAlbum,
|
|
|
|
personIds: filter.people.map((e) => e.id).toList(),
|
|
|
|
type: type,
|
|
|
|
page: page,
|
|
|
|
size: 1000,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
} else {
|
2024-11-20 20:47:25 +01:00
|
|
|
response = await _apiService.searchApi.searchAssets(
|
2024-04-01 16:45:11 +02:00
|
|
|
MetadataSearchDto(
|
|
|
|
originalFileName:
|
|
|
|
filter.filename != null && filter.filename!.isNotEmpty
|
|
|
|
? filter.filename
|
|
|
|
: null,
|
|
|
|
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,
|
|
|
|
isFavorite: filter.display.isFavorite,
|
|
|
|
isNotInAlbum: filter.display.isNotInAlbum,
|
|
|
|
personIds: filter.people.map((e) => e.id).toList(),
|
|
|
|
type: type,
|
|
|
|
page: page,
|
|
|
|
size: 1000,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (response == null) {
|
2023-02-04 21:42:42 +01:00
|
|
|
return null;
|
|
|
|
}
|
2024-04-01 16:45:11 +02:00
|
|
|
|
2024-10-30 20:27:13 +01:00
|
|
|
return SearchResult(
|
|
|
|
assets: await _assetRepository.getAllByRemoteId(
|
|
|
|
response.assets.items.map((e) => e.id),
|
|
|
|
),
|
|
|
|
nextPage: response.assets.nextPage?.toInt(),
|
|
|
|
);
|
2024-04-26 07:49:31 +02:00
|
|
|
} catch (error, stackTrace) {
|
|
|
|
_log.severe("Failed to search for assets", error, stackTrace);
|
2022-03-02 23:44:24 +01:00
|
|
|
}
|
2024-04-01 16:45:11 +02:00
|
|
|
return null;
|
2022-03-02 23:44:24 +01:00
|
|
|
}
|
2022-03-16 16:19:31 +01:00
|
|
|
|
2024-04-26 07:49:31 +02:00
|
|
|
Future<List<SearchExploreResponseDto>?> getExploreData() async {
|
2022-03-27 21:58:54 +02:00
|
|
|
try {
|
2024-04-26 07:49:31 +02:00
|
|
|
return await _apiService.searchApi.getExploreData();
|
|
|
|
} catch (error, stackTrace) {
|
|
|
|
_log.severe("Failed to getExploreData", error, stackTrace);
|
2022-03-27 21:58:54 +02:00
|
|
|
}
|
2024-04-26 07:49:31 +02:00
|
|
|
return null;
|
2022-03-27 21:58:54 +02:00
|
|
|
}
|
2024-05-01 05:14:33 +02:00
|
|
|
|
|
|
|
Future<List<AssetResponseDto>?> getAllPlaces() async {
|
|
|
|
try {
|
|
|
|
return await _apiService.searchApi.getAssetsByCity();
|
|
|
|
} catch (error, stackTrace) {
|
|
|
|
_log.severe("Failed to getAllPlaces", error, stackTrace);
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
2022-02-27 19:43:29 +01:00
|
|
|
}
|