Allow Re-Ordering of Sources in a Column (#245)

It is now possible to re-order sources in a column by dragging them into
a new position. To achieve this a new `position` field was added to the
`sources` column, which contains the index of the source in the column.
We also added a new `updateSourcePositions` function, which is used to
sort the sources locally and update `position` field in the database
afterwards. The dragging is handled via a `ReorderableListView` widget.
Last but not least the selection of sources from the database was
changed, so order them by the `position` column and if the column is
`null` by the `createdAt` data as it was before.
This commit is contained in:
Rico Berger
2025-04-19 15:08:55 +02:00
committed by GitHub
parent 2c712f6a07
commit bf95846e6c
4 changed files with 186 additions and 130 deletions

View File

@@ -10,11 +10,7 @@ import 'package:feeddeck/models/deck.dart';
import 'package:feeddeck/models/source.dart';
import 'package:feeddeck/utils/api_exception.dart';
enum FDAppStatus {
uninitialized,
authenticated,
unauthenticated,
}
enum FDAppStatus { uninitialized, authenticated, unauthenticated }
/// [AppRepository] is the repository for our app. The repository is responsible
/// for managing the state of our app, this includes the authentication status
@@ -82,10 +78,7 @@ class AppRepository with ChangeNotifier {
///
/// If the user was signed in successfully, we run the same logic as in the
/// [init] function, to set the active deck for the user.
Future<void> signInWithPassword(
String email,
String password,
) async {
Future<void> signInWithPassword(String email, String password) async {
await Supabase.instance.client.auth.signInWithPassword(
email: email,
password: password,
@@ -115,9 +108,7 @@ class AppRepository with ChangeNotifier {
///
/// If the user was signed in successfully, we run the same logic as in the
/// [init] function, to set the active deck for the user.
Future<void> signInWithCallback(
Uri uri,
) async {
Future<void> signInWithCallback(Uri uri) async {
await Supabase.instance.client.auth.getSessionFromUrl(
uri,
storeSession: true,
@@ -145,17 +136,16 @@ class AppRepository with ChangeNotifier {
/// create a new deck for the user with the given name. After the deck was
/// created, the deck is set as the users active deck and the deck is added to
/// the list of decks.
Future<void> createDeck(
String name,
) async {
final data = await Supabase.instance.client
.from('decks')
.insert({
'name': name,
'userId': Supabase.instance.client.auth.currentUser!.id,
})
.select()
.single();
Future<void> createDeck(String name) async {
final data =
await Supabase.instance.client
.from('decks')
.insert({
'name': name,
'userId': Supabase.instance.client.auth.currentUser!.id,
})
.select()
.single();
final newDeck = FDDeck.fromJson(data);
@@ -182,13 +172,11 @@ class AppRepository with ChangeNotifier {
/// a [deckId] and a [name] as parameters. The function calls the Supabase
/// client to update the name of the deck. After the deck was updated, the
/// deck is also updated in the list of decks.
Future<void> updateDeck(
String deckId,
String name,
) async {
Future<void> updateDeck(String deckId, String name) async {
await Supabase.instance.client
.from('decks')
.update({'name': name}).eq('id', deckId);
.update({'name': name})
.eq('id', deckId);
for (var i = 0; i < _decks.length; i++) {
if (_decks[i].id == deckId) {
@@ -205,9 +193,7 @@ class AppRepository with ChangeNotifier {
/// the deck. After the deck was deleted, the deck is also removed from the
/// list of decks. If the deleted deck was the active deck, the active deck is
/// set to `null`.
Future<void> deleteDeck(
String deckId,
) async {
Future<void> deleteDeck(String deckId) async {
await Supabase.instance.client.from('decks').delete().eq('id', deckId);
_decks.removeWhere((deck) => deck.id == deckId);
@@ -223,9 +209,7 @@ class AppRepository with ChangeNotifier {
/// to get the columns for the deck and all sources for each column. After the
/// columns and sources are fetched, the active deck is set to the provided
/// deckId and the columns and sources are stored in the repository.
Future<void> selectDeck(
String deckId,
) async {
Future<void> selectDeck(String deckId) async {
final columns = await getColumns(deckId);
for (final column in columns) {
column.sources = await getSources(column.id);
@@ -243,18 +227,18 @@ class AppRepository with ChangeNotifier {
/// function takes a [name] as parameter. The function calls the Supabase
/// client to create a new column for the active deck with the given name.
/// Finally the newly created column is added to the list of columns.
Future<void> createColumn(
String name,
) async {
final data = await Supabase.instance.client.from('columns').insert({
'deckId': _activeDeckId,
'userId': Supabase.instance.client.auth.currentUser!.id,
'name': name,
'position': _columns.length,
}).select();
Future<void> createColumn(String name) async {
final data =
await Supabase.instance.client.from('columns').insert({
'deckId': _activeDeckId,
'userId': Supabase.instance.client.auth.currentUser!.id,
'name': name,
'position': _columns.length,
}).select();
final newColumn =
List<FDColumn>.from(data.map((column) => FDColumn.fromJson(column)));
final newColumn = List<FDColumn>.from(
data.map((column) => FDColumn.fromJson(column)),
);
_columns.addAll(newColumn);
notifyListeners();
}
@@ -262,9 +246,7 @@ class AppRepository with ChangeNotifier {
/// [getColumns] is called to get all columns for the deck with the provided
/// [deckId]. The function calls the Supabase client to get all columns for
/// the deck. The function returns a list of [FDColumn]s.
Future<List<FDColumn>> getColumns(
String deckId,
) async {
Future<List<FDColumn>> getColumns(String deckId) async {
final data = await Supabase.instance.client
.from('columns')
.select('id, name, position')
@@ -276,9 +258,7 @@ class AppRepository with ChangeNotifier {
/// [deleteColumn] is called to delete a column with the provided [columnId].
/// The function calls the Supabase client to delete the column. After the
/// column was deleted, the column is also removed from the list of columns.
Future<void> deleteColumn(
String columnId,
) async {
Future<void> deleteColumn(String columnId) async {
await Supabase.instance.client.from('columns').delete().eq('id', columnId);
_columns.removeWhere((column) => column.id == columnId);
@@ -289,13 +269,11 @@ class AppRepository with ChangeNotifier {
/// The function takes a [name] as parameter. The function calls the Supabase
/// client to update the name of the column. After the column was updated, the
/// column is also updated in the list of columns.
Future<void> updateColumn(
String columnId,
String name,
) async {
Future<void> updateColumn(String columnId, String name) async {
await Supabase.instance.client
.from('columns')
.update({'name': name}).eq('id', columnId);
.update({'name': name})
.eq('id', columnId);
for (var i = 0; i < _columns.length; i++) {
if (_columns[i].id == columnId) {
@@ -311,22 +289,15 @@ class AppRepository with ChangeNotifier {
/// with the provided [index1] and [index2]. The function calls the Supabase
/// client to update the positions of the columns. After the columns were
/// updated, the columns are also updated in the list of columns.
Future<void> updateColumnPositions(
int index1,
int index2,
) async {
Future<void> updateColumnPositions(int index1, int index2) async {
await Supabase.instance.client
.from('columns')
.update({'position': _columns[index2].position}).eq(
'id',
_columns[index1].id,
);
.update({'position': _columns[index2].position})
.eq('id', _columns[index1].id);
await Supabase.instance.client
.from('columns')
.update({'position': _columns[index1].position}).eq(
'id',
_columns[index2].id,
);
.update({'position': _columns[index1].position})
.eq('id', _columns[index2].id);
final tmp = _columns[index1];
_columns[index1] = _columns[index2];
@@ -340,13 +311,19 @@ class AppRepository with ChangeNotifier {
/// [getSources] is called to get all sources for the column with the provided
/// [columnId]. The function calls the Supabase client to get all sources for
/// the column. The function returns a list of [FDSource]s.
Future<List<FDSource>> getSources(
String columnId,
) async {
///
/// The returned list of sources is ordered by the `position` field. Since the
/// position field was added later there might be columns where the field is
/// `null`, which will come after all columns with a `position`. The source
/// where the position is `null` will be ordered by the `createdAt` date. This
/// should retain the order as before the `position` field was added and
/// should also work for new sources, which are added without a position.
Future<List<FDSource>> getSources(String columnId) async {
final data = await Supabase.instance.client
.from('sources')
.select('id, type, title, options, link, icon')
.eq('columnId', columnId)
.order('position', ascending: true, nullsFirst: false)
.order('createdAt', ascending: true);
return List<FDSource>.from(data.map((source) => FDSource.fromJson(source)));
}
@@ -355,10 +332,7 @@ class AppRepository with ChangeNotifier {
/// The function calls the Supabase client to delete the source. After the
/// source was deleted, the source is also removed from the list of sources of
/// the column with the provided [columnId].
Future<void> deleteSource(
String columnId,
String sourceId,
) async {
Future<void> deleteSource(String columnId, String sourceId) async {
await Supabase.instance.client.from('sources').delete().eq('id', sourceId);
/// It could take some time before we can retrieve the items after a source
@@ -423,4 +397,51 @@ class AppRepository with ChangeNotifier {
notifyListeners();
}
/// [updateSourcePositions] can be used to reorder the list of sources for a
/// column. To achieve this order the list of sources for the provided
/// [columnId] locally and update the `position` field of each source
/// afterwards in the database.
///
/// We have to check if the user drags a source from top to bottom ([start]
/// is lower then [current]) or from the bottom to the top ([start] is greater
/// then [current]), to apply a different logic for the reordering.
Future<void> updateSourcePositions(
String columnId,
int start,
int current,
) async {
final columnIndex = _columns.indexWhere((column) => column.id == columnId);
if (columnIndex == -1) {
return;
}
if (start < current) {
int end = current - 1;
FDSource startItem = _columns[columnIndex].sources[start];
int i = 0;
int local = start;
do {
_columns[columnIndex].sources[local] =
_columns[columnIndex].sources[++local];
i++;
} while (i < end - start);
_columns[columnIndex].sources[end] = startItem;
} else if (start > current) {
FDSource startItem = _columns[columnIndex].sources[start];
for (int i = start; i > current; i--) {
_columns[columnIndex].sources[i] = _columns[columnIndex].sources[i - 1];
}
_columns[columnIndex].sources[current] = startItem;
}
for (var i = 0; i < _columns[columnIndex].sources.length; i++) {
await Supabase.instance.client
.from('sources')
.update({'position': i})
.eq('id', _columns[columnIndex].sources[i].id);
}
notifyListeners();
}
}

View File

@@ -32,28 +32,66 @@ class ColumnLayoutHeaderSettingsSources extends StatefulWidget {
class _ColumnLayoutHeaderSettingsSourcesState
extends State<ColumnLayoutHeaderSettingsSources> {
/// [_proxyDecorator] is used to highlight the source which is currently
/// draged by the user.
Widget _proxyDecorator(Widget child, int index, Animation<double> animation) {
return Material(
elevation: 0,
color: Colors.transparent,
child: Stack(
children: [
Positioned(
top: 0,
left: 0,
right: 0,
bottom: 16,
child: Material(
borderRadius: BorderRadius.circular(16),
elevation: 24,
color: Colors.transparent,
),
),
child,
],
),
);
}
/// [_buildSourcesList] returns a list of all sources of the current column.
/// If the list of sources is empty it will return a [Container].
///
/// Each source in the list also contains a delete item, which can be used to
/// remove the source from the current column.
List<Widget> _buildSourcesList() {
Widget _buildSourcesList() {
if (widget.column.sources.isEmpty) {
return [Container()];
return Container();
}
List<Widget> columns = [];
return ReorderableListView.builder(
shrinkWrap: true,
buildDefaultDragHandles: false,
physics: const NeverScrollableScrollPhysics(),
onReorder: (int start, int current) {
final AppRepository appRepository = Provider.of<AppRepository>(
context,
listen: false,
);
for (var i = 0; i < widget.column.sources.length; i++) {
columns.add(
SourceListItem(
appRepository.updateSourcePositions(widget.column.id, start, current);
},
proxyDecorator: (Widget child, int index, Animation<double> animation) {
return _proxyDecorator(child, index, animation);
},
itemCount: widget.column.sources.length,
itemBuilder: (context, index) {
return SourceListItem(
key: Key(widget.column.sources[index].id),
columnId: widget.column.id,
source: widget.column.sources[i],
),
);
}
return columns;
sourceIndex: index,
source: widget.column.sources[index],
);
},
);
}
/// [_showAddSource] shows the [AddSource] widget within a modal bottom sheet
@@ -92,14 +130,12 @@ class _ColumnLayoutHeaderSettingsSourcesState
mainAxisAlignment: MainAxisAlignment.start,
children: [
ConstrainedBox(
constraints: const BoxConstraints(
maxHeight: 275,
),
constraints: const BoxConstraints(maxHeight: 275),
child: ListView(
padding: EdgeInsets.zero,
shrinkWrap: true,
children: [
..._buildSourcesList(),
_buildSourcesList(),
ElevatedButton.icon(
style: ElevatedButton.styleFrom(
backgroundColor: Constants.secondary,
@@ -116,9 +152,7 @@ class _ColumnLayoutHeaderSettingsSourcesState
),
label: const Text('Add Source'),
onPressed: () => _showAddSource(),
icon: const Icon(
Icons.add,
),
icon: const Icon(Icons.add),
),
],
),

View File

@@ -14,10 +14,12 @@ class SourceListItem extends StatefulWidget {
const SourceListItem({
super.key,
required this.columnId,
required this.sourceIndex,
required this.source,
});
final String columnId;
final int sourceIndex;
final FDSource source;
@override
State<SourceListItem> createState() => _SourceListItemState();
@@ -35,17 +37,16 @@ class _SourceListItemState extends State<SourceListItem> {
builder: (BuildContext context) {
return AlertDialog(
insetPadding: EdgeInsets.symmetric(
horizontal: MediaQuery.of(context).size.width >=
(Constants.centeredFormMaxWidth +
2 * Constants.spacingMiddle)
? (MediaQuery.of(context).size.width -
Constants.centeredFormMaxWidth) /
2
: Constants.spacingMiddle,
),
title: const Text(
'Delete Source',
horizontal:
MediaQuery.of(context).size.width >=
(Constants.centeredFormMaxWidth +
2 * Constants.spacingMiddle)
? (MediaQuery.of(context).size.width -
Constants.centeredFormMaxWidth) /
2
: Constants.spacingMiddle,
),
title: const Text('Delete Source'),
content: const Text(
'Do you really want to delete this source? This can not be undone and will also delete all items and bookmarks related to this source.',
),
@@ -61,12 +62,13 @@ class _SourceListItemState extends State<SourceListItem> {
),
TextButton(
onPressed: _isLoading ? null : () => _deleteSource(),
child: _isLoading
? const ElevatedButtonProgressIndicator()
: const Text(
'Delete',
style: TextStyle(color: Constants.error),
),
child:
_isLoading
? const ElevatedButtonProgressIndicator()
: const Text(
'Delete',
style: TextStyle(color: Constants.error),
),
),
],
);
@@ -110,19 +112,13 @@ class _SourceListItemState extends State<SourceListItem> {
Widget build(BuildContext context) {
return Card(
color: Constants.secondary,
margin: const EdgeInsets.only(
bottom: Constants.spacingSmall,
),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(12),
),
margin: const EdgeInsets.only(bottom: Constants.spacingSmall),
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(12)),
child: Column(
crossAxisAlignment: CrossAxisAlignment.stretch,
children: [
Padding(
padding: const EdgeInsets.all(
Constants.spacingMiddle,
),
padding: const EdgeInsets.all(Constants.spacingMiddle),
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
@@ -135,14 +131,10 @@ class _SourceListItemState extends State<SourceListItem> {
.replaceAll(Characters(''), Characters('\u{200B}'))
.toString(),
maxLines: 1,
style: const TextStyle(
overflow: TextOverflow.ellipsis,
),
style: const TextStyle(overflow: TextOverflow.ellipsis),
),
Text(
Characters(
widget.source.type.toLocalizedString(),
)
Characters(widget.source.type.toLocalizedString())
.replaceAll(Characters(''), Characters('\u{200B}'))
.toString(),
maxLines: 1,
@@ -156,11 +148,14 @@ class _SourceListItemState extends State<SourceListItem> {
),
IconButton(
onPressed: () => _showDeleteDialog(),
icon: _isLoading
? const ElevatedButtonProgressIndicator()
: const Icon(
Icons.delete,
),
icon:
_isLoading
? const ElevatedButtonProgressIndicator()
: const Icon(Icons.delete),
),
ReorderableDragStartListener(
index: widget.sourceIndex,
child: Icon(Icons.drag_handle),
),
],
),

View File

@@ -0,0 +1,6 @@
------------------------------------------------------------------------------------------------------------------------
-- Add a new "postion" column to the "sources" table, which is used to store the position of a source within a column.
-- This is required, so that users are able to sort the sources within a column.
------------------------------------------------------------------------------------------------------------------------
ALTER TABLE "sources"
ADD COLUMN "position" SMALLINT DEFAULT NULL;