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

240 lines
8.6 KiB
Dart
Raw Normal View History

import "dart:async";
import "package:figma_squircle/figma_squircle.dart";
import "package:flutter/material.dart";
import "package:photos/core/constants.dart";
import "package:photos/events/event.dart";
import "package:photos/models/search/generic_search_result.dart";
import "package:photos/models/search/recent_searches.dart";
import "package:photos/models/search/search_types.dart";
import "package:photos/theme/ente_theme.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/search/result/search_result_page.dart";
import "package:photos/ui/viewer/search/search_section_cta.dart";
import "package:photos/ui/viewer/search_tab/section_header.dart";
import "package:photos/utils/navigation_util.dart";
class DescriptionsSection extends StatefulWidget {
final List<GenericSearchResult> descriptionsSearchResults;
const DescriptionsSection(this.descriptionsSearchResults, {super.key});
@override
State<DescriptionsSection> createState() => _DescriptionsSectionState();
}
class _DescriptionsSectionState extends State<DescriptionsSection> {
late List<GenericSearchResult> _descriptionsSearchResults;
final streamSubscriptions = <StreamSubscription>[];
@override
void initState() {
super.initState();
_descriptionsSearchResults = widget.descriptionsSearchResults;
final streamsToListenTo = SectionType.fileCaption.sectionUpdateEvents();
for (Stream<Event> stream in streamsToListenTo) {
streamSubscriptions.add(
stream.listen((event) async {
_descriptionsSearchResults = (await SectionType.fileCaption.getData(
context,
limit: kSearchSectionLimit,
)) as List<GenericSearchResult>;
setState(() {});
}),
);
}
}
@override
void dispose() {
for (var subscriptions in streamSubscriptions) {
subscriptions.cancel();
}
super.dispose();
}
@override
void didUpdateWidget(covariant DescriptionsSection oldWidget) {
super.didUpdateWidget(oldWidget);
_descriptionsSearchResults = widget.descriptionsSearchResults;
}
@override
Widget build(BuildContext context) {
if (_descriptionsSearchResults.isEmpty) {
final textTheme = getEnteTextTheme(context);
return Padding(
padding: const EdgeInsets.only(left: 12, right: 8),
child: Row(
children: [
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
SectionType.fileCaption.sectionTitle(context),
style: textTheme.largeBold,
),
const SizedBox(height: 24),
Padding(
padding: const EdgeInsets.only(left: 4),
child: Text(
SectionType.fileCaption.getEmptyStateText(context),
style: textTheme.smallMuted,
),
),
],
),
),
const SizedBox(width: 8),
const SearchSectionEmptyCTAIcon(SectionType.fileCaption),
],
),
);
} else {
return Padding(
padding: const EdgeInsets.symmetric(vertical: 8),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
SectionHeader(
SectionType.fileCaption,
hasMore: (_descriptionsSearchResults.length >=
kSearchSectionLimit - 1),
),
const SizedBox(height: 2),
SizedBox(
child: SingleChildScrollView(
clipBehavior: Clip.none,
padding: const EdgeInsets.symmetric(horizontal: 4.5),
physics: const BouncingScrollPhysics(),
scrollDirection: Axis.horizontal,
child: Row(
crossAxisAlignment: CrossAxisAlignment.start,
children: _descriptionsSearchResults
.map(
(descriptionSearchResult) =>
DescriptionRecommendation(descriptionSearchResult),
)
.toList(),
),
),
),
],
),
);
}
}
}
class DescriptionRecommendation extends StatelessWidget {
final GenericSearchResult descriptionSearchResult;
const DescriptionRecommendation(this.descriptionSearchResult, {super.key});
@override
Widget build(BuildContext context) {
final heroTag = descriptionSearchResult.heroTag() +
(descriptionSearchResult.previewThumbnail()?.tag ?? "");
final enteTextTheme = getEnteTextTheme(context);
return Padding(
padding: const EdgeInsets.symmetric(horizontal: 2.5),
child: GestureDetector(
onTap: () {
RecentSearches().add(descriptionSearchResult.name());
if (descriptionSearchResult.onResultTap != null) {
descriptionSearchResult.onResultTap!(context);
} else {
routeToPage(
context,
SearchResultPage(descriptionSearchResult),
);
}
},
child: SizedBox(
width: 100,
height: 150,
child: Stack(
clipBehavior: Clip.none,
alignment: Alignment.center,
children: [
Container(
decoration: const BoxDecoration(
boxShadow: [
BoxShadow(
blurRadius: 15,
offset: Offset(0, 7.5),
color: Color.fromRGBO(68, 68, 68, 0.1),
),
],
),
child: ClipSmoothRect(
radius: SmoothBorderRadius(
cornerRadius: 7.5,
cornerSmoothing: 1,
),
child: Container(
color: Theme.of(context).colorScheme.brightness ==
Brightness.light
? const Color(0xFFFFFFFF)
: const Color(0xFF181818),
child: Column(
children: [
Padding(
padding: const EdgeInsets.fromLTRB(9, 9, 9, 0),
child: SizedBox(
width: 82,
height: 82,
child: descriptionSearchResult.previewThumbnail() !=
null
? Hero(
tag: heroTag,
child: ClipSmoothRect(
radius: SmoothBorderRadius(
cornerRadius: 7.5,
cornerSmoothing: 1,
),
child: ThumbnailWidget(
descriptionSearchResult
.previewThumbnail()!,
shouldShowArchiveStatus: false,
shouldShowSyncStatus: false,
),
),
)
: const NoThumbnailWidget(),
),
),
Expanded(
child: Padding(
padding: const EdgeInsets.symmetric(
horizontal: 5,
vertical: 10,
),
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Text(
descriptionSearchResult.name(),
style: enteTextTheme.smallMuted,
maxLines: 2,
overflow: TextOverflow.ellipsis,
textAlign: TextAlign.center,
),
],
),
),
),
],
),
),
),
),
],
),
),
),
);
}
}