ente/lib/ui/viewer/search/search_widget.dart

192 lines
6.2 KiB
Dart
Raw Normal View History

2022-07-18 04:45:06 +00:00
import 'package:flutter/material.dart';
import 'package:photos/db/files_db.dart';
2022-07-18 04:45:06 +00:00
import 'package:photos/ente_theme_data.dart';
import 'package:photos/models/collection_items.dart';
import 'package:photos/models/file.dart';
2022-08-04 16:16:16 +00:00
import 'package:photos/models/search/album_search_result.dart';
import 'package:photos/models/search/file_search_result.dart';
import 'package:photos/models/search/search_results.dart';
import 'package:photos/services/collections_service.dart';
import 'package:photos/services/user_service.dart';
import 'package:photos/ui/viewer/search/search_results_suggestions.dart';
2022-07-18 04:45:06 +00:00
class SearchIconWidget extends StatefulWidget {
2022-07-28 05:08:20 +00:00
const SearchIconWidget({Key key}) : super(key: key);
2022-07-18 04:45:06 +00:00
@override
State<SearchIconWidget> createState() => _SearchIconWidgetState();
}
class _SearchIconWidgetState extends State<SearchIconWidget> {
2022-07-28 05:08:20 +00:00
@override
void initState() {
super.initState();
}
2022-07-18 04:45:06 +00:00
@override
Widget build(BuildContext context) {
2022-08-05 11:08:25 +00:00
return Hero(
tag: "search icon",
child: IconButton(
onPressed: () {
setState(
() {
Navigator.push(
context,
TransparentRoute(
builder: (BuildContext context) => const SearchWidget(),
),
);
},
);
},
icon: const Icon(Icons.search),
),
);
2022-07-18 04:45:06 +00:00
}
}
2022-07-18 04:45:06 +00:00
class SearchWidget extends StatefulWidget {
final String searchQuery = '';
const SearchWidget({Key key}) : super(key: key);
@override
State<SearchWidget> createState() => _SearchWidgetState();
}
class _SearchWidgetState extends State<SearchWidget> {
final List<SearchResult> results = [];
@override
Widget build(BuildContext context) {
return SafeArea(
2022-08-06 08:15:42 +00:00
child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 1.5),
child: Column(
children: [
const SizedBox(height: 8),
Row(
children: [
Flexible(
child: Container(
color: Theme.of(context).colorScheme.defaultBackgroundColor,
child: TextFormField(
style: Theme.of(context).textTheme.subtitle1,
decoration: InputDecoration(
hintText: 'Search for albums, locations, files...',
filled: true,
contentPadding: const EdgeInsets.symmetric(
horizontal: 16,
vertical: 14,
),
border: UnderlineInputBorder(
borderSide: BorderSide.none,
borderRadius: BorderRadius.circular(8),
),
focusedBorder: const UnderlineInputBorder(
borderSide: BorderSide.none,
),
prefixIcon: Hero(
tag: "search icon",
child: Icon(
Icons.search,
color: Theme.of(context)
.colorScheme
.iconColor
.withOpacity(0.5),
),
),
suffixIcon: IconButton(
onPressed: () {
Navigator.pop(context);
},
icon: Icon(
Icons.close,
color: Theme.of(context)
.colorScheme
.iconColor
.withOpacity(0.5),
),
),
),
2022-08-06 08:15:42 +00:00
onChanged: (value) async {
2022-08-06 09:22:54 +00:00
List<SearchResult> allResults = [];
2022-08-06 08:15:42 +00:00
final collectionResults = await CollectionsService
.instance
.getFilteredCollectionsWithThumbnail(value);
for (CollectionWithThumbnail collectionResult
in collectionResults) {
2022-08-06 09:22:54 +00:00
allResults.add(AlbumSearchResult(collectionResult));
2022-08-06 08:15:42 +00:00
}
final locationResults = await UserService.instance
.getLocationsToMatchedFiles(value);
for (final result in locationResults) {
2022-08-06 09:22:54 +00:00
allResults.add(result);
2022-08-06 08:15:42 +00:00
}
final fileResults = await FilesDB.instance
.getFilesOnFileNameSearch(value);
for (File file in fileResults) {
2022-08-06 09:22:54 +00:00
allResults.add(FileSearchResult(file));
2022-08-06 08:15:42 +00:00
}
setState(() {
results.clear();
2022-08-06 09:22:54 +00:00
results.addAll(allResults);
2022-08-06 08:15:42 +00:00
});
},
autofocus: true,
),
),
),
2022-08-06 08:15:42 +00:00
],
),
results.isNotEmpty
? SearchResultsSuggestionsWidget(results)
: const SizedBox.shrink(),
],
),
),
);
2022-07-18 04:45:06 +00:00
}
}
2022-08-05 11:08:25 +00:00
class TransparentRoute extends PageRoute<void> {
TransparentRoute({
@required this.builder,
RouteSettings settings,
}) : assert(builder != null),
super(settings: settings, fullscreenDialog: false);
final WidgetBuilder builder;
@override
bool get opaque => false;
@override
Color get barrierColor => null;
@override
String get barrierLabel => null;
@override
bool get maintainState => true;
@override
2022-08-05 11:21:15 +00:00
Duration get transitionDuration => const Duration(milliseconds: 350);
2022-08-05 11:08:25 +00:00
@override
Widget buildPage(
BuildContext context,
Animation<double> animation,
Animation<double> secondaryAnimation,
) {
final result = builder(context);
return FadeTransition(
opacity: Tween<double>(begin: 0, end: 1).animate(animation),
child: Semantics(
scopesRoute: true,
explicitChildNodes: true,
child: result,
),
);
}
}