1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-24 20:52:44 +01:00
immich/mobile/lib/modules/search/views/search_result_page.dart

195 lines
6.3 KiB
Dart
Raw Normal View History

2022-03-02 23:44:24 +01:00
import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
2022-03-02 23:44:24 +01:00
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
2022-10-01 10:33:06 +02:00
import 'package:immich_mobile/modules/home/ui/asset_grid/immich_asset_grid.dart';
2022-03-02 23:44:24 +01:00
import 'package:immich_mobile/modules/search/providers/search_page_state.provider.dart';
import 'package:immich_mobile/modules/search/providers/search_result_page.provider.dart';
2022-03-02 23:44:24 +01:00
import 'package:immich_mobile/modules/search/ui/search_suggestion_list.dart';
import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart';
import 'package:immich_mobile/modules/settings/services/app_settings.service.dart';
import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
2022-03-02 23:44:24 +01:00
class SearchResultPage extends HookConsumerWidget {
const SearchResultPage({Key? key, required this.searchTerm})
: super(key: key);
2022-03-02 23:44:24 +01:00
final String searchTerm;
@override
Widget build(BuildContext context, WidgetRef ref) {
final searchTermController = useTextEditingController(text: "");
final isNewSearch = useState(false);
final currentSearchTerm = useState(searchTerm);
FocusNode? searchFocusNode;
useEffect(
() {
searchFocusNode = FocusNode();
Future.delayed(
Duration.zero,
() => ref.read(searchResultPageProvider.notifier).search(searchTerm),
);
return () => searchFocusNode?.dispose();
},
[],
);
2022-03-02 23:44:24 +01:00
onSearchSubmitted(String newSearchTerm) {
2022-03-02 23:44:24 +01:00
debugPrint("Re-Search with $newSearchTerm");
searchFocusNode?.unfocus();
2022-03-02 23:44:24 +01:00
isNewSearch.value = false;
currentSearchTerm.value = newSearchTerm;
ref.watch(searchResultPageProvider.notifier).search(newSearchTerm);
2022-03-02 23:44:24 +01:00
}
buildTextField() {
2022-03-02 23:44:24 +01:00
return TextField(
controller: searchTermController,
focusNode: searchFocusNode,
autofocus: false,
onTap: () {
searchTermController.clear();
ref.watch(searchPageStateProvider.notifier).setSearchTerm("");
searchFocusNode?.requestFocus();
2022-03-02 23:44:24 +01:00
},
textInputAction: TextInputAction.search,
onSubmitted: (searchTerm) {
if (searchTerm.isNotEmpty) {
searchTermController.clear();
onSearchSubmitted(searchTerm);
2022-03-02 23:44:24 +01:00
} else {
isNewSearch.value = false;
}
},
onChanged: (value) {
ref.watch(searchPageStateProvider.notifier).setSearchTerm(value);
},
decoration: InputDecoration(
hintText: 'search_result_page_new_search_hint'.tr(),
enabledBorder: const UnderlineInputBorder(
2022-03-02 23:44:24 +01:00
borderSide: BorderSide(color: Colors.transparent),
),
focusedBorder: const UnderlineInputBorder(
2022-03-02 23:44:24 +01:00
borderSide: BorderSide(color: Colors.transparent),
),
),
);
}
buildChip() {
2022-03-02 23:44:24 +01:00
return Chip(
label: Wrap(
spacing: 5,
runAlignment: WrapAlignment.center,
crossAxisAlignment: WrapCrossAlignment.center,
alignment: WrapAlignment.center,
2022-03-02 23:44:24 +01:00
children: [
Text(
currentSearchTerm.value,
style: TextStyle(
color: Theme.of(context).primaryColor,
fontSize: 13,
fontWeight: FontWeight.bold,
),
maxLines: 1,
2022-03-02 23:44:24 +01:00
),
Icon(
Icons.close_rounded,
color: Theme.of(context).primaryColor,
size: 20,
),
],
),
backgroundColor: Theme.of(context).primaryColor.withAlpha(50),
);
}
buildSearchResult() {
var searchResultPageState = ref.watch(searchResultPageProvider);
var searchResultRenderList = ref.watch(searchRenderListProvider);
var allSearchAssets = ref.watch(searchResultPageProvider).searchResult;
var settings = ref.watch(appSettingsServiceProvider);
final assetsPerRow = settings.getSetting(AppSettingsEnum.tilesPerRow);
final showStorageIndicator =
settings.getSetting(AppSettingsEnum.storageIndicator);
2022-03-02 23:44:24 +01:00
if (searchResultPageState.isError) {
return Padding(
padding: const EdgeInsets.all(12),
child: const Text("common_server_error").tr(),
);
2022-03-02 23:44:24 +01:00
}
if (searchResultPageState.isLoading) {
return const Center(child: ImmichLoadingIndicator());
2022-03-02 23:44:24 +01:00
}
if (searchResultPageState.isSuccess) {
return searchResultRenderList.when(
data: (result) {
return ImmichAssetGrid(
allAssets: allSearchAssets,
renderList: result,
assetsPerRow: assetsPerRow,
showStorageIndicator: showStorageIndicator,
);
},
error: (err, stack) {
return Text("$err");
},
loading: () {
return const CircularProgressIndicator();
},
);
2022-03-02 23:44:24 +01:00
}
return const SizedBox();
2022-03-02 23:44:24 +01:00
}
return Scaffold(
appBar: AppBar(
leading: IconButton(
splashRadius: 20,
onPressed: () {
if (isNewSearch.value) {
isNewSearch.value = false;
} else {
AutoRouter.of(context).pop(true);
}
},
icon: const Icon(Icons.arrow_back_ios_rounded),
),
title: GestureDetector(
onTap: () {
isNewSearch.value = true;
searchFocusNode?.requestFocus();
},
child: isNewSearch.value ? buildTextField() : buildChip(),
),
2022-03-02 23:44:24 +01:00
centerTitle: false,
),
body: GestureDetector(
onTap: () {
if (searchFocusNode != null) {
searchFocusNode?.unfocus();
}
2022-03-02 23:44:24 +01:00
ref.watch(searchPageStateProvider.notifier).disableSearch();
},
child: Stack(
children: [
buildSearchResult(),
if (isNewSearch.value)
SearchSuggestionList(onSubmitted: onSearchSubmitted),
2022-03-02 23:44:24 +01:00
],
),
),
);
}
}