0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-02-11 01:18:24 -05:00

feat(mobile): Add filter to people_picker.dart (#15771)

* Add filter to people_picker.dart

* feat: styling

---------

Co-authored-by: Alex <alex.tran1502@gmail.com>
This commit is contained in:
Justin Forseth 2025-01-29 14:02:54 -07:00 committed by GitHub
parent b287c0cbe8
commit 6e31ac4c75
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
3 changed files with 134 additions and 49 deletions

View file

@ -476,6 +476,7 @@
"search_filter_media_type_video": "Video", "search_filter_media_type_video": "Video",
"search_filter_people": "People", "search_filter_people": "People",
"search_filter_people_title": "Select people", "search_filter_people_title": "Select people",
"search_filter_people_hint": "Filter people",
"search_page_categories": "Categories", "search_page_categories": "Categories",
"search_page_favorites": "Favorites", "search_page_favorites": "Favorites",
"search_page_motion_photos": "Motion Photos", "search_page_motion_photos": "Motion Photos",

View file

@ -16,6 +16,8 @@ class LargeLeadingTile extends StatelessWidget {
this.trailing, this.trailing,
this.selected = false, this.selected = false,
this.disabled = false, this.disabled = false,
this.selectedTileColor,
this.tileColor,
}); });
final Widget leading; final Widget leading;
@ -27,6 +29,9 @@ class LargeLeadingTile extends StatelessWidget {
final Widget? trailing; final Widget? trailing;
final bool selected; final bool selected;
final bool disabled; final bool disabled;
final Color? selectedTileColor;
final Color? tileColor;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return InkWell( return InkWell(
@ -35,8 +40,9 @@ class LargeLeadingTile extends StatelessWidget {
child: Container( child: Container(
decoration: BoxDecoration( decoration: BoxDecoration(
color: selected color: selected
? Theme.of(context).primaryColor.withAlpha(30) ? selectedTileColor ??
: Colors.transparent, Theme.of(context).primaryColor.withAlpha(30)
: tileColor ?? Colors.transparent,
borderRadius: BorderRadius.circular(borderRadius), borderRadius: BorderRadius.circular(borderRadius),
), ),
child: Row( child: Row(

View file

@ -1,9 +1,12 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/extensions/asyncvalue_extensions.dart'; import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
import 'package:immich_mobile/extensions/build_context_extensions.dart'; import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/extensions/theme_extensions.dart';
import 'package:immich_mobile/interfaces/person_api.interface.dart'; import 'package:immich_mobile/interfaces/person_api.interface.dart';
import 'package:immich_mobile/pages/common/large_leading_tile.dart';
import 'package:immich_mobile/providers/search/people.provider.dart'; import 'package:immich_mobile/providers/search/people.provider.dart';
import 'package:immich_mobile/services/api.service.dart'; import 'package:immich_mobile/services/api.service.dart';
import 'package:immich_mobile/utils/image_url_builder.dart'; import 'package:immich_mobile/utils/image_url_builder.dart';
@ -16,63 +19,138 @@ class PeoplePicker extends HookConsumerWidget {
@override @override
Widget build(BuildContext context, WidgetRef ref) { Widget build(BuildContext context, WidgetRef ref) {
var imageSize = 45.0; final formFocus = useFocusNode();
final imageSize = 75.0;
final searchQuery = useState('');
final people = ref.watch(getAllPeopleProvider); final people = ref.watch(getAllPeopleProvider);
final headers = ApiService.getRequestHeaders(); final headers = ApiService.getRequestHeaders();
final selectedPeople = useState<Set<Person>>(filter ?? {}); final selectedPeople = useState<Set<Person>>(filter ?? {});
return people.widgetWhen( return Column(
onData: (people) { children: [
return ListView.builder( Padding(
shrinkWrap: true,
itemCount: people.length,
padding: const EdgeInsets.all(8), padding: const EdgeInsets.all(8),
itemBuilder: (context, index) { child: TextField(
final person = people[index]; focusNode: formFocus,
return Card( onChanged: (value) => searchQuery.value = value,
elevation: 0, onTapOutside: (_) => formFocus.unfocus(),
shape: const RoundedRectangleBorder( decoration: InputDecoration(
borderRadius: BorderRadius.all(Radius.circular(15)), contentPadding: const EdgeInsets.only(left: 24),
filled: true,
fillColor: context.primaryColor.withOpacity(0.1),
hintStyle: context.textTheme.bodyLarge?.copyWith(
color: context.themeData.colorScheme.onSurfaceSecondary,
), ),
child: ListTile( border: OutlineInputBorder(
title: Text( borderRadius: BorderRadius.circular(25),
person.name, borderSide: BorderSide(
style: context.textTheme.bodyLarge, color: context.colorScheme.surfaceContainerHighest,
), ),
leading: SizedBox( ),
height: imageSize, enabledBorder: OutlineInputBorder(
child: Material( borderRadius: BorderRadius.circular(25),
shape: const CircleBorder(side: BorderSide.none), borderSide: BorderSide(
elevation: 3, color: context.colorScheme.surfaceContainerHighest,
child: CircleAvatar(
maxRadius: imageSize / 2,
backgroundImage: NetworkImage(
getFaceThumbnailUrl(person.id),
headers: headers,
),
),
),
), ),
onTap: () { ),
if (selectedPeople.value.contains(person)) { disabledBorder: OutlineInputBorder(
selectedPeople.value.remove(person); borderRadius: BorderRadius.circular(25),
} else { borderSide: BorderSide(
selectedPeople.value.add(person); color: context.colorScheme.surfaceContainerHighest,
} ),
),
focusedBorder: OutlineInputBorder(
borderRadius: BorderRadius.circular(25),
borderSide: BorderSide(
color: context.colorScheme.primary.withAlpha(150),
),
),
prefixIcon: Icon(
Icons.search_rounded,
color: context.colorScheme.primary,
),
hintText: 'search_filter_people_hint'.tr(),
),
),
),
Padding(
padding: const EdgeInsets.only(left: 16.0, right: 16.0, bottom: 0),
child: Divider(
color: context.colorScheme.surfaceContainerHighest,
thickness: 1,
),
),
Expanded(
child: people.widgetWhen(
onData: (people) {
return ListView.builder(
shrinkWrap: true,
itemCount: people
.where(
(person) => person.name
.toLowerCase()
.contains(searchQuery.value.toLowerCase()),
)
.length,
padding: const EdgeInsets.all(8),
itemBuilder: (context, index) {
final person = people
.where(
(person) => person.name
.toLowerCase()
.contains(searchQuery.value.toLowerCase()),
)
.toList()[index];
final isSelected = selectedPeople.value.contains(person);
selectedPeople.value = {...selectedPeople.value}; return Padding(
onSelect(selectedPeople.value); padding: const EdgeInsets.only(bottom: 2.0),
child: LargeLeadingTile(
title: Text(
person.name,
style: context.textTheme.bodyLarge?.copyWith(
fontSize: 20,
fontWeight: FontWeight.w500,
color: isSelected
? context.colorScheme.onPrimary
: context.colorScheme.onSurface,
),
),
leading: SizedBox(
height: imageSize,
child: Material(
shape: const CircleBorder(side: BorderSide.none),
elevation: 3,
child: CircleAvatar(
maxRadius: imageSize / 2,
backgroundImage: NetworkImage(
getFaceThumbnailUrl(person.id),
headers: headers,
),
),
),
),
onTap: () {
if (selectedPeople.value.contains(person)) {
selectedPeople.value.remove(person);
} else {
selectedPeople.value.add(person);
}
selectedPeople.value = {...selectedPeople.value};
onSelect(selectedPeople.value);
},
selected: isSelected,
selectedTileColor: context.primaryColor,
tileColor: context.primaryColor.withAlpha(25),
),
);
}, },
selected: selectedPeople.value.contains(person), );
selectedTileColor: context.primaryColor.withOpacity(0.2), },
shape: const RoundedRectangleBorder( ),
borderRadius: BorderRadius.all(Radius.circular(15)), ),
), ],
),
);
},
);
},
); );
} }
} }