mirror of
https://github.com/austinried/subtracks.git
synced 2026-02-10 15:02:42 +01:00
context menu base and move query to state
This commit is contained in:
@@ -1,29 +1,28 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_hooks/flutter_hooks.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||
import 'package:infinite_scroll_pagination/infinite_scroll_pagination.dart';
|
||||
|
||||
import '../../../database/query.dart';
|
||||
import '../../../sources/models.dart';
|
||||
import '../../hooks/use_on_source.dart';
|
||||
import '../../hooks/use_paging_controller.dart';
|
||||
import '../../state/database.dart';
|
||||
import '../../state/lists.dart';
|
||||
import '../../state/source.dart';
|
||||
import '../menus.dart';
|
||||
import 'items.dart';
|
||||
|
||||
const kPageSize = 60;
|
||||
|
||||
class AlbumsGrid extends HookConsumerWidget {
|
||||
const AlbumsGrid({
|
||||
super.key,
|
||||
required this.query,
|
||||
});
|
||||
|
||||
final AlbumsQuery query;
|
||||
const AlbumsGrid({super.key});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context, WidgetRef ref) {
|
||||
final db = ref.watch(databaseProvider);
|
||||
final query = ref.watch(albumsQueryProvider);
|
||||
|
||||
final controller = usePagingController<int, Album>(
|
||||
getNextPageKey: (state) =>
|
||||
state.lastPageIsEmpty ? null : state.nextIntPageKey,
|
||||
@@ -36,8 +35,8 @@ class AlbumsGrid extends HookConsumerWidget {
|
||||
),
|
||||
);
|
||||
|
||||
useOnSourceChange(ref, (_) => controller.refresh());
|
||||
useOnSourceSync(ref, controller.refresh);
|
||||
useValueChanged(query, (_, _) => controller.refresh());
|
||||
|
||||
return PagingListener(
|
||||
controller: controller,
|
||||
@@ -50,7 +49,9 @@ class AlbumsGrid extends HookConsumerWidget {
|
||||
gridDelegate: SliverGridDelegateWithFixedCrossAxisCount(
|
||||
crossAxisCount: 3,
|
||||
),
|
||||
showNoMoreItemsIndicatorAsGridChild: false,
|
||||
builderDelegate: PagedChildBuilderDelegate<Album>(
|
||||
noMoreItemsIndicatorBuilder: (context) => FabPadding(),
|
||||
itemBuilder: (context, item, index) => AlbumGridTile(
|
||||
album: item,
|
||||
onTap: () async {
|
||||
@@ -64,3 +65,14 @@ class AlbumsGrid extends HookConsumerWidget {
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class AlbumsGridFilters extends HookConsumerWidget {
|
||||
const AlbumsGridFilters({super.key});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context, WidgetRef ref) {
|
||||
return ListView(
|
||||
children: [],
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -9,6 +9,7 @@ import '../../hooks/use_on_source.dart';
|
||||
import '../../hooks/use_paging_controller.dart';
|
||||
import '../../state/database.dart';
|
||||
import '../../state/source.dart';
|
||||
import '../menus.dart';
|
||||
import 'items.dart';
|
||||
|
||||
const kPageSize = 30;
|
||||
@@ -46,6 +47,7 @@ class AlbumsList extends HookConsumerWidget {
|
||||
state: state,
|
||||
fetchNextPage: fetchNextPage,
|
||||
builderDelegate: PagedChildBuilderDelegate<Album>(
|
||||
noMoreItemsIndicatorBuilder: (context) => FabPadding(),
|
||||
itemBuilder: (context, item, index) {
|
||||
final tile = AlbumListTile(
|
||||
album: item,
|
||||
|
||||
@@ -10,6 +10,7 @@ import '../../hooks/use_on_source.dart';
|
||||
import '../../hooks/use_paging_controller.dart';
|
||||
import '../../state/database.dart';
|
||||
import '../../state/source.dart';
|
||||
import '../menus.dart';
|
||||
import 'items.dart';
|
||||
|
||||
const kPageSize = 30;
|
||||
@@ -45,6 +46,7 @@ class ArtistsList extends HookConsumerWidget {
|
||||
state: state,
|
||||
fetchNextPage: fetchNextPage,
|
||||
builderDelegate: PagedChildBuilderDelegate<AristListItem>(
|
||||
noMoreItemsIndicatorBuilder: (context) => FabPadding(),
|
||||
itemBuilder: (context, item, index) {
|
||||
final (:artist, :albumCount) = item;
|
||||
|
||||
|
||||
@@ -10,6 +10,7 @@ import '../../hooks/use_on_source.dart';
|
||||
import '../../hooks/use_paging_controller.dart';
|
||||
import '../../state/database.dart';
|
||||
import '../../state/source.dart';
|
||||
import '../menus.dart';
|
||||
import 'items.dart';
|
||||
|
||||
const kPageSize = 30;
|
||||
@@ -45,6 +46,7 @@ class PlaylistsList extends HookConsumerWidget {
|
||||
state: state,
|
||||
fetchNextPage: fetchNextPage,
|
||||
builderDelegate: PagedChildBuilderDelegate<Playlist>(
|
||||
noMoreItemsIndicatorBuilder: (context) => FabPadding(),
|
||||
itemBuilder: (context, item, index) {
|
||||
return PlaylistListTile(
|
||||
playlist: item,
|
||||
|
||||
@@ -1,4 +1,5 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_hooks/flutter_hooks.dart';
|
||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||
import 'package:infinite_scroll_pagination/infinite_scroll_pagination.dart';
|
||||
|
||||
@@ -8,6 +9,7 @@ import '../../hooks/use_on_source.dart';
|
||||
import '../../hooks/use_paging_controller.dart';
|
||||
import '../../state/database.dart';
|
||||
import '../../state/source.dart';
|
||||
import '../menus.dart';
|
||||
|
||||
const kPageSize = 30;
|
||||
|
||||
@@ -37,8 +39,8 @@ class SongsList extends HookConsumerWidget {
|
||||
),
|
||||
);
|
||||
|
||||
useOnSourceChange(ref, (_) => controller.refresh());
|
||||
useOnSourceSync(ref, controller.refresh);
|
||||
useValueChanged(query, (_, _) => controller.refresh());
|
||||
|
||||
return PagingListener(
|
||||
controller: controller,
|
||||
@@ -47,6 +49,7 @@ class SongsList extends HookConsumerWidget {
|
||||
state: state,
|
||||
fetchNextPage: fetchNextPage,
|
||||
builderDelegate: PagedChildBuilderDelegate<SongListItem>(
|
||||
noMoreItemsIndicatorBuilder: (context) => FabPadding(),
|
||||
itemBuilder: itemBuilder,
|
||||
),
|
||||
);
|
||||
|
||||
76
lib/app/ui/menus.dart
Normal file
76
lib/app/ui/menus.dart
Normal file
@@ -0,0 +1,76 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:material_symbols_icons/symbols.dart';
|
||||
|
||||
Future<void> showContextMenu({
|
||||
required BuildContext context,
|
||||
required WidgetBuilder listBuilder,
|
||||
}) => showModalBottomSheet(
|
||||
context: context,
|
||||
useRootNavigator: true,
|
||||
isScrollControlled: true,
|
||||
builder: (context) => DraggableScrollableSheet(
|
||||
expand: false,
|
||||
snap: true,
|
||||
initialChildSize: 0.3,
|
||||
minChildSize: 0.3,
|
||||
maxChildSize: 0.4,
|
||||
builder: (context, scrollController) => PrimaryScrollController(
|
||||
controller: scrollController,
|
||||
child: listBuilder(context),
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
class ContextMenuList extends StatelessWidget {
|
||||
const ContextMenuList({
|
||||
super.key,
|
||||
required this.children,
|
||||
});
|
||||
|
||||
final List<Widget> children;
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return ListView(
|
||||
children: children,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class FabFilter extends StatelessWidget {
|
||||
const FabFilter({
|
||||
super.key,
|
||||
required this.listBuilder,
|
||||
});
|
||||
|
||||
final WidgetBuilder listBuilder;
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return FloatingActionButton(
|
||||
onPressed: () {
|
||||
showContextMenu(
|
||||
context: context,
|
||||
listBuilder: listBuilder,
|
||||
);
|
||||
},
|
||||
child: Icon(
|
||||
Symbols.filter_list_rounded,
|
||||
weight: 500,
|
||||
opticalSize: 28,
|
||||
size: 28,
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class FabPadding extends StatelessWidget {
|
||||
const FabPadding({
|
||||
super.key,
|
||||
});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return const SizedBox(height: 86);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user