created widgets for new location section and surface them on UI
This commit is contained in:
parent
ff598424e0
commit
3203dc380e
2 changed files with 321 additions and 0 deletions
315
lib/ui/viewer/search_tab/locations_section.dart
Normal file
315
lib/ui/viewer/search_tab/locations_section.dart
Normal file
|
@ -0,0 +1,315 @@
|
|||
import "dart:async";
|
||||
import "dart:math";
|
||||
import "dart:ui";
|
||||
|
||||
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 LocationsSection extends StatefulWidget {
|
||||
final List<GenericSearchResult> locationsSearchResults;
|
||||
const LocationsSection(this.locationsSearchResults, {super.key});
|
||||
|
||||
@override
|
||||
State<LocationsSection> createState() => _LocationsSectionState();
|
||||
}
|
||||
|
||||
class _LocationsSectionState extends State<LocationsSection> {
|
||||
late List<GenericSearchResult> _locationsSearchResults;
|
||||
final streamSubscriptions = <StreamSubscription>[];
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
_locationsSearchResults = widget.locationsSearchResults;
|
||||
|
||||
final streamsToListenTo = SectionType.location.sectionUpdateEvents();
|
||||
for (Stream<Event> stream in streamsToListenTo) {
|
||||
streamSubscriptions.add(
|
||||
stream.listen((event) async {
|
||||
_locationsSearchResults = (await SectionType.location.getData(
|
||||
context,
|
||||
limit: kSearchSectionLimit,
|
||||
)) as List<GenericSearchResult>;
|
||||
setState(() {});
|
||||
}),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
for (var subscriptions in streamSubscriptions) {
|
||||
subscriptions.cancel();
|
||||
}
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
@override
|
||||
void didUpdateWidget(covariant LocationsSection oldWidget) {
|
||||
super.didUpdateWidget(oldWidget);
|
||||
_locationsSearchResults = widget.locationsSearchResults;
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
if (_locationsSearchResults.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.location.sectionTitle(context),
|
||||
style: textTheme.largeBold,
|
||||
),
|
||||
const SizedBox(height: 24),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(left: 4),
|
||||
child: Text(
|
||||
SectionType.location.getEmptyStateText(context),
|
||||
style: textTheme.smallMuted,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
const SizedBox(width: 8),
|
||||
const SearchSectionEmptyCTAIcon(SectionType.location),
|
||||
],
|
||||
),
|
||||
);
|
||||
} else {
|
||||
return Padding(
|
||||
padding: const EdgeInsets.symmetric(vertical: 8),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
SectionHeader(
|
||||
SectionType.location,
|
||||
hasMore:
|
||||
(_locationsSearchResults.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: _locationsSearchResults
|
||||
.map(
|
||||
(locationSearchResult) =>
|
||||
LocationRecommendation(locationSearchResult),
|
||||
)
|
||||
.toList(),
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
class LocationRecommendation extends StatelessWidget {
|
||||
static const _width = 100.0;
|
||||
static const _height = 123.0;
|
||||
static const _thumbnailBorderWidth = 1.0;
|
||||
static const _outerCornerRadius = 12.0;
|
||||
static const _cornerSmoothing = 1.0;
|
||||
static const _sideOfThumbnail = 90.0;
|
||||
static const _outerStrokeWidth = 1.0;
|
||||
//This is the space between this widget's boundary and the border stroke of
|
||||
//thumbnail.
|
||||
static const _outerPadding = 4.0;
|
||||
final GenericSearchResult locationSearchResult;
|
||||
const LocationRecommendation(this.locationSearchResult, {super.key});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final heroTag = locationSearchResult.heroTag() +
|
||||
(locationSearchResult.previewThumbnail()?.tag ?? "");
|
||||
final enteTextTheme = getEnteTextTheme(context);
|
||||
return Padding(
|
||||
padding:
|
||||
EdgeInsets.symmetric(horizontal: max(0, 2.5 - _outerStrokeWidth)),
|
||||
child: GestureDetector(
|
||||
onTap: () {
|
||||
RecentSearches().add(locationSearchResult.name());
|
||||
if (locationSearchResult.onResultTap != null) {
|
||||
locationSearchResult.onResultTap!(context);
|
||||
} else {
|
||||
routeToPage(
|
||||
context,
|
||||
SearchResultPage(locationSearchResult),
|
||||
);
|
||||
}
|
||||
},
|
||||
child: Stack(
|
||||
alignment: Alignment.center,
|
||||
children: [
|
||||
ClipSmoothRect(
|
||||
radius: SmoothBorderRadius(
|
||||
cornerRadius: _outerCornerRadius + _outerStrokeWidth,
|
||||
cornerSmoothing: _cornerSmoothing,
|
||||
),
|
||||
child: Container(
|
||||
color: Colors.white.withOpacity(0.1),
|
||||
width: _width + _outerStrokeWidth * 2,
|
||||
height: _height + _outerStrokeWidth * 2,
|
||||
),
|
||||
),
|
||||
SizedBox(
|
||||
width: _width,
|
||||
height: _height,
|
||||
// height: 145,
|
||||
|
||||
child: 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: _outerCornerRadius,
|
||||
cornerSmoothing: _cornerSmoothing,
|
||||
),
|
||||
child: Stack(
|
||||
clipBehavior: Clip.none,
|
||||
children: [
|
||||
ImageFiltered(
|
||||
imageFilter: ImageFilter.blur(sigmaX: 24, sigmaY: 24),
|
||||
child: locationSearchResult.previewThumbnail() != null
|
||||
? ThumbnailWidget(
|
||||
locationSearchResult.previewThumbnail()!,
|
||||
shouldShowArchiveStatus: false,
|
||||
shouldShowSyncStatus: false,
|
||||
)
|
||||
: const NoThumbnailWidget(),
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.fromLTRB(
|
||||
_outerPadding,
|
||||
_outerPadding,
|
||||
_outerPadding,
|
||||
_outerPadding,
|
||||
),
|
||||
child: Column(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
Stack(
|
||||
alignment: Alignment.center,
|
||||
children: [
|
||||
ClipSmoothRect(
|
||||
radius: SmoothBorderRadius(
|
||||
cornerRadius:
|
||||
_outerCornerRadius - _outerPadding,
|
||||
cornerSmoothing: _cornerSmoothing,
|
||||
),
|
||||
child: Container(
|
||||
color: Colors.black.withOpacity(0.1),
|
||||
width: _sideOfThumbnail +
|
||||
_thumbnailBorderWidth * 2,
|
||||
height: _sideOfThumbnail +
|
||||
_thumbnailBorderWidth * 2,
|
||||
),
|
||||
),
|
||||
SizedBox(
|
||||
width: _sideOfThumbnail,
|
||||
height: _sideOfThumbnail,
|
||||
child: locationSearchResult
|
||||
.previewThumbnail() !=
|
||||
null
|
||||
? Hero(
|
||||
tag: heroTag,
|
||||
child: ClipSmoothRect(
|
||||
radius: SmoothBorderRadius(
|
||||
cornerRadius: _outerCornerRadius -
|
||||
_outerPadding -
|
||||
_thumbnailBorderWidth,
|
||||
cornerSmoothing: _cornerSmoothing,
|
||||
),
|
||||
clipBehavior:
|
||||
Clip.antiAliasWithSaveLayer,
|
||||
child: ThumbnailWidget(
|
||||
locationSearchResult
|
||||
.previewThumbnail()!,
|
||||
shouldShowArchiveStatus: false,
|
||||
shouldShowSyncStatus: false,
|
||||
),
|
||||
),
|
||||
)
|
||||
: const NoThumbnailWidget(),
|
||||
),
|
||||
],
|
||||
),
|
||||
Expanded(
|
||||
child: Column(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
children: [
|
||||
Text(
|
||||
locationSearchResult.name(),
|
||||
style: enteTextTheme.small
|
||||
.copyWith(color: Colors.white),
|
||||
maxLines: 1,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
Positioned(
|
||||
top: 8,
|
||||
left: 8,
|
||||
child: Container(
|
||||
width: 16,
|
||||
height: 16,
|
||||
decoration: BoxDecoration(
|
||||
color: Colors.black.withOpacity(0.6),
|
||||
borderRadius: BorderRadius.circular(8),
|
||||
),
|
||||
child: const Icon(
|
||||
Icons.location_on_sharp,
|
||||
color: Colors.white,
|
||||
size: 11,
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
|
@ -14,6 +14,7 @@ import "package:photos/ui/viewer/search/search_suggestions.dart";
|
|||
import "package:photos/ui/viewer/search/tab_empty_state.dart";
|
||||
import 'package:photos/ui/viewer/search_tab/albums_section.dart';
|
||||
import "package:photos/ui/viewer/search_tab/descriptions_section.dart";
|
||||
import "package:photos/ui/viewer/search_tab/locations_section.dart";
|
||||
import "package:photos/ui/viewer/search_tab/moments_section.dart";
|
||||
|
||||
class SearchTab extends StatefulWidget {
|
||||
|
@ -114,6 +115,11 @@ class _AllSearchSectionsState extends State<AllSearchSections> {
|
|||
snapshot.data!.elementAt(index)
|
||||
as List<GenericSearchResult>,
|
||||
);
|
||||
case SectionType.location:
|
||||
return LocationsSection(
|
||||
snapshot.data!.elementAt(index)
|
||||
as List<GenericSearchResult>,
|
||||
);
|
||||
default:
|
||||
return SearchSection(
|
||||
sectionType: searchTypes[index],
|
||||
|
|
Loading…
Add table
Reference in a new issue