ente/mobile/lib/ui/viewer/search_tab/people_section.dart

345 lines
12 KiB
Dart

import "dart:async";
import "package:flutter/material.dart";
import "package:photos/core/constants.dart";
import "package:photos/events/event.dart";
import "package:photos/models/file/file.dart";
import "package:photos/models/ml/face/person.dart";
import "package:photos/models/search/generic_search_result.dart";
import "package:photos/models/search/recent_searches.dart";
import "package:photos/models/search/search_constants.dart";
import "package:photos/models/search/search_result.dart";
import "package:photos/models/search/search_types.dart";
import "package:photos/theme/ente_theme.dart";
import "package:photos/ui/settings/ml/machine_learning_settings_page.dart";
import "package:photos/ui/viewer/file/no_thumbnail_widget.dart";
import "package:photos/ui/viewer/file/thumbnail_widget.dart";
import "package:photos/ui/viewer/people/add_person_action_sheet.dart";
import "package:photos/ui/viewer/people/people_page.dart";
import "package:photos/ui/viewer/search/result/people_section_all_page.dart";
import 'package:photos/ui/viewer/search/result/person_face_widget.dart';
import "package:photos/ui/viewer/search/result/search_result_page.dart";
import "package:photos/ui/viewer/search/search_section_cta.dart";
import "package:photos/utils/navigation_util.dart";
class PeopleSection extends StatefulWidget {
final SectionType sectionType = SectionType.face;
final List<SearchResult> examples;
final int limit;
const PeopleSection({
super.key,
required this.examples,
this.limit = 7,
});
@override
State<PeopleSection> createState() => _PeopleSectionState();
}
class _PeopleSectionState extends State<PeopleSection> {
late List<SearchResult> _examples;
final streamSubscriptions = <StreamSubscription>[];
@override
void initState() {
super.initState();
_examples = widget.examples;
final streamsToListenTo = widget.sectionType.sectionUpdateEvents();
for (Stream<Event> stream in streamsToListenTo) {
streamSubscriptions.add(
stream.listen((event) async {
_examples = await widget.sectionType.getData(
context,
limit: kSearchSectionLimit,
);
setState(() {});
}),
);
}
}
@override
void dispose() {
for (var subscriptions in streamSubscriptions) {
subscriptions.cancel();
}
super.dispose();
}
@override
void didUpdateWidget(covariant PeopleSection oldWidget) {
super.didUpdateWidget(oldWidget);
_examples = widget.examples;
}
@override
Widget build(BuildContext context) {
debugPrint("Building section for ${widget.sectionType.name}");
final shouldShowMore = _examples.length >= widget.limit - 1;
final textTheme = getEnteTextTheme(context);
return _examples.isNotEmpty
? GestureDetector(
behavior: HitTestBehavior.opaque,
onTap: () {
if (shouldShowMore) {
routeToPage(
context,
const PeopleSectionAllPage(),
);
}
},
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Padding(
padding: const EdgeInsets.all(12),
child: Text(
widget.sectionType.sectionTitle(context),
style: textTheme.largeBold,
),
),
shouldShowMore
? Padding(
padding: const EdgeInsets.all(12),
child: Icon(
Icons.chevron_right_outlined,
color: getEnteColorScheme(context).strokeMuted,
),
)
: const SizedBox.shrink(),
],
),
const SizedBox(height: 2),
SearchExampleRow(_examples, widget.sectionType),
],
),
)
: GestureDetector(
behavior: HitTestBehavior.opaque,
onTap: () {
routeToPage(
context,
const MachineLearningSettingsPage(),
);
},
child: Padding(
padding: const EdgeInsets.only(left: 16, right: 8),
child: Row(
children: [
Expanded(
child: Padding(
padding: const EdgeInsets.symmetric(vertical: 12),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
widget.sectionType.sectionTitle(context),
style: textTheme.largeBold,
),
const SizedBox(height: 24),
Text(
widget.sectionType.getEmptyStateText(context),
style: textTheme.smallMuted,
),
],
),
),
),
const SizedBox(width: 8),
SearchSectionEmptyCTAIcon(widget.sectionType),
],
),
),
);
}
}
class SearchExampleRow extends StatelessWidget {
final SectionType sectionType;
final List<SearchResult> examples;
const SearchExampleRow(this.examples, this.sectionType, {super.key});
@override
Widget build(BuildContext context) {
return SizedBox(
height: 128,
child: ListView.separated(
padding: const EdgeInsets.symmetric(horizontal: 6),
physics: const BouncingScrollPhysics(),
scrollDirection: Axis.horizontal,
itemCount: examples.length,
itemBuilder: (context, index) {
return PersonSearchExample(
searchResult: examples[index],
);
},
separatorBuilder: (context, index) => const SizedBox(width: 3),
),
);
}
}
class PersonSearchExample extends StatelessWidget {
final SearchResult searchResult;
final double size;
const PersonSearchExample({
super.key,
required this.searchResult,
this.size = 102,
});
@override
Widget build(BuildContext context) {
final borderRadius = 82 * (size / 102);
final bool isCluster = (searchResult.type() == ResultType.faces &&
int.tryParse(searchResult.name()) != null);
return GestureDetector(
onTap: () {
RecentSearches().add(searchResult.name());
final genericSearchResult = searchResult as GenericSearchResult;
if (genericSearchResult.onResultTap != null) {
genericSearchResult.onResultTap!(context);
} else {
routeToPage(
context,
SearchResultPage(searchResult),
);
}
},
child: Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.center,
mainAxisAlignment: MainAxisAlignment.center,
children: [
Stack(
clipBehavior: Clip.none,
alignment: Alignment.center,
children: [
ClipPath(
clipper: ShapeBorderClipper(
shape: ContinuousRectangleBorder(
borderRadius: BorderRadius.circular(borderRadius),
),
),
child: Container(
width: size,
height: size,
decoration: BoxDecoration(
color: getEnteColorScheme(context).strokeFaint,
),
),
),
SizedBox(
width: size - 2,
height: size - 2,
child: searchResult.previewThumbnail() != null
? ClipPath(
clipper: ShapeBorderClipper(
shape: ContinuousRectangleBorder(
borderRadius:
BorderRadius.circular(borderRadius - 1),
),
),
child: searchResult.type() != ResultType.faces
? ThumbnailWidget(
searchResult.previewThumbnail()!,
shouldShowSyncStatus: false,
)
: FaceSearchResult(searchResult),
)
: ClipPath(
clipper: ShapeBorderClipper(
shape: ContinuousRectangleBorder(
borderRadius: BorderRadius.circular(81),
),
),
child: const NoThumbnailWidget(
addBorder: false,
),
),
),
],
),
isCluster
? GestureDetector(
behavior: HitTestBehavior.translucent,
onTap: () async {
final result = await showAssignPersonAction(
context,
clusterID: searchResult.name(),
);
if (result != null && result is (PersonEntity, EnteFile)) {
// ignore: unawaited_futures
routeToPage(
context,
PeoplePage(
person: result.$1,
searchResult: null,
),
);
} else if (result != null && result is PersonEntity) {
// ignore: unawaited_futures
routeToPage(
context,
PeoplePage(
person: result,
searchResult: null,
),
);
}
},
child: Padding(
padding: const EdgeInsets.only(top: 6, bottom: 0),
child: Text(
"Add name",
maxLines: 1,
textAlign: TextAlign.center,
overflow: TextOverflow.ellipsis,
style: getEnteTextTheme(context).small,
),
),
)
: Padding(
padding: const EdgeInsets.only(top: 6, bottom: 0),
child: Text(
searchResult.name(),
maxLines: 1,
textAlign: TextAlign.center,
overflow: TextOverflow.ellipsis,
style: getEnteTextTheme(context).small,
),
),
],
),
);
}
}
class FaceSearchResult extends StatelessWidget {
final SearchResult searchResult;
const FaceSearchResult(this.searchResult, {super.key});
@override
Widget build(BuildContext context) {
final params = (searchResult as GenericSearchResult).params;
return PersonFaceWidget(
searchResult.previewThumbnail()!,
personId: params[kPersonParamID],
clusterID: params[kClusterParamId],
key: params.containsKey(kPersonWidgetKey)
? ValueKey(params[kPersonWidgetKey])
: null,
);
}
}