diff --git a/assets/lottie/books-staple.json b/assets/lottie/books-staple.json new file mode 100644 index 0000000..a732ed2 --- /dev/null +++ b/assets/lottie/books-staple.json @@ -0,0 +1 @@ +{"v":"5.5.7","meta":{"g":"LottieFiles AE 0.1.20","a":"","k":"","d":"","tc":""},"fr":25,"ip":2,"op":126,"w":1920,"h":1259,"nm":"arcax_3","ddd":0,"assets":[{"id":"image_0","w":342,"h":98,"u":"","p":"","e":1},{"id":"image_1","w":365,"h":116,"u":"","p":"","e":1},{"id":"image_2","w":383,"h":144,"u":"","p":"","e":1},{"id":"image_3","w":387,"h":122,"u":"","p":"","e":1},{"id":"image_4","w":458,"h":262,"u":"","p":"","e":1}],"layers":[{"ddd":0,"ind":1,"ty":2,"nm":"red_book","refId":"image_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":52,"s":[-30]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":58,"s":[10]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":62,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":66,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":68,"s":[3]},{"t":70,"s":[0]}],"ix":10},"p":{"s":true,"x":{"a":0,"k":940.074,"ix":3},"y":{"a":1,"k":[{"i":{"x":[0.958],"y":[0.625]},"o":{"x":[0.611],"y":[0.247]},"t":60,"s":[297.056]},{"i":{"x":[0.667],"y":[1.008]},"o":{"x":[0.19],"y":[0.736]},"t":66,"s":[492.056]},{"i":{"x":[0.856],"y":[0.636]},"o":{"x":[0.333],"y":[0.008]},"t":68,"s":[474.556]},{"t":70,"s":[492.056]}],"ix":4}},"a":{"a":0,"k":[170.953,48.738,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":52,"s":[50,50,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":58,"s":[110,110,100]},{"t":62,"s":[100,100,100]}],"ix":6}},"ao":0,"ip":52,"op":117,"st":52,"bm":0},{"ddd":0,"ind":2,"ty":2,"nm":"green_book","refId":"image_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":35,"s":[-30]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":41,"s":[10]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":45,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":49,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":51,"s":[3]},{"t":54,"s":[0]}],"ix":10},"p":{"s":true,"x":{"a":0,"k":974.604,"ix":3},"y":{"a":1,"k":[{"i":{"x":[0.958],"y":[0.625]},"o":{"x":[0.611],"y":[0.247]},"t":43,"s":[353.977]},{"i":{"x":[0.667],"y":[1.008]},"o":{"x":[0.19],"y":[0.758]},"t":49,"s":[548.977]},{"i":{"x":[0.856],"y":[0.438]},"o":{"x":[0.333],"y":[0.013]},"t":51,"s":[531.977]},{"t":54,"s":[548.977]}],"ix":4}},"a":{"a":0,"k":[182.464,57.946,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":35,"s":[50,50,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":41,"s":[110,110,100]},{"t":45,"s":[100,100,100]}],"ix":6}},"ao":0,"ip":35,"op":118,"st":35,"bm":0},{"ddd":0,"ind":3,"ty":2,"nm":"yellow_book","refId":"image_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":23,"s":[-30]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":29,"s":[10]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":33,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":37,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":39,"s":[3]},{"t":42,"s":[0]}],"ix":10},"p":{"s":true,"x":{"a":0,"k":955.609,"ix":3},"y":{"a":1,"k":[{"i":{"x":[0.958],"y":[0.625]},"o":{"x":[0.611],"y":[0.247]},"t":31,"s":[430.363]},{"i":{"x":[0.667],"y":[1.007]},"o":{"x":[0.19],"y":[0.613]},"t":37,"s":[625.363]},{"i":{"x":[0.856],"y":[0.545]},"o":{"x":[0.333],"y":[0.01]},"t":39,"s":[604.363]},{"t":42,"s":[625.363]}],"ix":4}},"a":{"a":0,"k":[191.137,71.54,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":23,"s":[50,50,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":29,"s":[110,110,100]},{"t":33,"s":[100,100,100]}],"ix":6}},"ao":0,"ip":23,"op":119,"st":23,"bm":0},{"ddd":0,"ind":4,"ty":2,"nm":"red_book_slim","refId":"image_3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":12,"s":[-30]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":18,"s":[10]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":22,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":26,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":28,"s":[3]},{"t":31,"s":[0]}],"ix":10},"p":{"s":true,"x":{"a":0,"k":981.177,"ix":3},"y":{"a":1,"k":[{"i":{"x":[0.958],"y":[0.6]},"o":{"x":[0.611],"y":[0.263]},"t":20,"s":[488.389]},{"i":{"x":[0.667],"y":[1.009]},"o":{"x":[0.19],"y":[0.776]},"t":26,"s":[671.389]},{"i":{"x":[0.856],"y":[0.425]},"o":{"x":[0.333],"y":[0.013]},"t":28,"s":[654.789]},{"t":31,"s":[671.389]}],"ix":4}},"a":{"a":0,"k":[193.197,60.965,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":12,"s":[50,50,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":18,"s":[110,110,100]},{"t":22,"s":[100,100,100]}],"ix":6}},"ao":0,"ip":12,"op":120,"st":12,"bm":0},{"ddd":0,"ind":5,"ty":2,"nm":"green_book_big","refId":"image_4","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":2,"s":[-30]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":8,"s":[10]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":12,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":16,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":18,"s":[2]},{"t":21,"s":[0]}],"ix":10},"p":{"s":true,"x":{"a":0,"k":956.458,"ix":3},"y":{"a":1,"k":[{"i":{"x":[0.958],"y":[0.575]},"o":{"x":[0.611],"y":[0.28]},"t":10,"s":[569.951]},{"i":{"x":[0.667],"y":[1.008]},"o":{"x":[0.19],"y":[0.76]},"t":16,"s":[741.951]},{"i":{"x":[0.856],"y":[0.632]},"o":{"x":[0.333],"y":[0.008]},"t":18,"s":[725]},{"t":21,"s":[750.951]}],"ix":4}},"a":{"a":0,"k":[228.907,130.619,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":2,"s":[50,50,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":8,"s":[110,110,100]},{"t":12,"s":[100,100,100]}],"ix":6}},"ao":0,"ip":2,"op":121,"st":2,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/assets/translations/de-DE.json b/assets/translations/de-DE.json index 62ba41a..811c475 100644 --- a/assets/translations/de-DE.json +++ b/assets/translations/de-DE.json @@ -68,9 +68,16 @@ "hint": "Title oder ISBN", "title": "Titelsuche" }, - "reset": "TODO", - "reset_password": "TODO", - "reset_password_text": "TODO", + "timeline": { + "sorting": { + "started-in": "Gestartet in", + "ended-in": "Fertig in" + }, + "empty": "Es sind noch keine Bücher in deiner Bibliothek vorhanden, um hier etwas anzuzeigen." + }, + "reset": "Zurücksetzen", + "reset_password": "Password zurücksetzen", + "reset_password_text": "Wir versenden ein Email an deine registrierte Adresse, wo du weitere Instruktionen zum Zurücksetzen findest.", "search": { "empty": { "action": "Online suchen", diff --git a/assets/translations/en-US.json b/assets/translations/en-US.json index 26936c4..ecca667 100644 --- a/assets/translations/en-US.json +++ b/assets/translations/en-US.json @@ -69,6 +69,13 @@ "hint": "Title or ISBN", "title": "Title search" }, + "timeline": { + "sorting": { + "started-in": "Started in", + "ended-in": "Finished in" + }, + "empty": "There are no books in your library yet to build this view." + }, "reset": "Reset", "reset_password": "Reset Password", "reset_password_text": "A link will be sent to your registered email address with instructions on how to reset your password.", diff --git a/lib/src/data/settings/settings_repository.dart b/lib/src/data/settings/settings_repository.dart index 0c9d107..b289312 100644 --- a/lib/src/data/settings/settings_repository.dart +++ b/lib/src/data/settings/settings_repository.dart @@ -1,4 +1,5 @@ import 'package:dantex/src/data/book/book_sort_strategy.dart'; +import 'package:dantex/src/ui/timeline/timeline_sort.dart'; import 'package:flutter/material.dart'; abstract class SettingsRepository { @@ -14,4 +15,7 @@ abstract class SettingsRepository { void setIsRandomBooksEnabled({required bool isRandomBooksEnabled}); bool isRandomBooksEnabled(); + + void setTimelineSortStrategy(TimelineSortStrategy sort); + TimelineSortStrategy getTimelineSortStrategy(); } diff --git a/lib/src/data/settings/shared_preferences_settings_repository.dart b/lib/src/data/settings/shared_preferences_settings_repository.dart index 881a935..f88e411 100644 --- a/lib/src/data/settings/shared_preferences_settings_repository.dart +++ b/lib/src/data/settings/shared_preferences_settings_repository.dart @@ -1,5 +1,6 @@ import 'package:dantex/src/data/book/book_sort_strategy.dart'; import 'package:dantex/src/data/settings/settings_repository.dart'; +import 'package:dantex/src/ui/timeline/timeline_sort.dart'; import 'package:flutter/material.dart'; import 'package:rxdart/rxdart.dart'; import 'package:shared_preferences/shared_preferences.dart'; @@ -11,6 +12,7 @@ class SharedPreferencesSettingsRepository implements SettingsRepository { static const _keyRandomBooksEnabled = 'key_random_books_enabled'; static const _keyThemeMode = 'key_theme_mode'; static const _keySortStrategy = 'key_sort_strategy'; + static const _keyTimelineSortStrategy = 'key_timeline_sort_strategy'; final BehaviorSubject _themeModeSubject = BehaviorSubject(); @@ -73,4 +75,18 @@ class SharedPreferencesSettingsRepository implements SettingsRepository { Stream observeThemeMode() { return _themeModeSubject.stream; } + + @override + TimelineSortStrategy getTimelineSortStrategy() { + final int timelineSortStrategyOrdinal = + _sp.getInt(_keyTimelineSortStrategy) ?? 0; + return TimelineSortStrategy.values[timelineSortStrategyOrdinal]; + } + + @override + Future setTimelineSortStrategy( + TimelineSortStrategy sortStrategy, + ) async { + await _sp.setInt(_keyTimelineSortStrategy, sortStrategy.index); + } } diff --git a/lib/src/data/timeline/timeline.dart b/lib/src/data/timeline/timeline.dart new file mode 100644 index 0000000..248f52c --- /dev/null +++ b/lib/src/data/timeline/timeline.dart @@ -0,0 +1,78 @@ +import 'package:collection/collection.dart'; +import 'package:dantex/src/data/book/book_repository.dart'; +import 'package:dantex/src/data/book/entity/book.dart'; +import 'package:dantex/src/data/settings/settings_repository.dart'; + +import 'package:dantex/src/ui/timeline/timeline_sort.dart'; +import 'package:rxdart/rxdart.dart'; + +class Timeline { + final SettingsRepository _settingsRepository; + final BookRepository _bookRepository; + + final BehaviorSubject _sortStrategySubject; + + Stream get sortStrategy => _sortStrategySubject.stream; + + Timeline(this._settingsRepository, this._bookRepository) + : _sortStrategySubject = BehaviorSubject.seeded( + _settingsRepository.getTimelineSortStrategy(), + ); + + void setSortStrategy(TimelineSortStrategy sortStrategy) { + _sortStrategySubject.add(sortStrategy); + _settingsRepository.setTimelineSortStrategy(sortStrategy); + } + + Stream> getTimelineData() { + return _sortStrategySubject.flatMap(_createTimelineForSortStrategy); + } + + Stream> _createTimelineForSortStrategy( + TimelineSortStrategy sortStrategy, + ) { + return _bookRepository.getAllBooks().map((books) { + return books + .where( + (element) { + return switch (sortStrategy) { + TimelineSortStrategy.byStartDate => element.startDate != null, + TimelineSortStrategy.byEndState => element.endDate != null, + }; + }, + ) + .groupListsBy( + (e) { + return switch (_sortStrategySubject.value) { + TimelineSortStrategy.byStartDate => DateTime( + e.startDate!.year, + e.startDate!.month, + ), + TimelineSortStrategy.byEndState => DateTime( + e.endDate!.year, + e.endDate!.month, + ), + }; + }, + ) + .entries + // Sort descending + .sorted((a, b) => a.key.isBefore(b.key) ? 1 : -1) + .map( + (MapEntry> entry) => + TimelineMonthGrouping(month: entry.key, books: entry.value), + ) + .toList(); + }); + } +} + +class TimelineMonthGrouping { + final DateTime month; + final List books; + + TimelineMonthGrouping({ + required this.month, + required this.books, + }); +} diff --git a/lib/src/providers/service.dart b/lib/src/providers/service.dart index 84de100..b975341 100644 --- a/lib/src/providers/service.dart +++ b/lib/src/providers/service.dart @@ -6,6 +6,7 @@ import 'package:dantex/src/data/isbn/isbn_scanner_service.dart'; import 'package:dantex/src/data/logging/error_only_filter.dart'; import 'package:dantex/src/data/logging/firebase_log_output.dart'; import 'package:dantex/src/data/search/search.dart'; +import 'package:dantex/src/data/timeline/timeline.dart'; import 'package:dantex/src/providers/api.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/widgets.dart'; @@ -43,6 +44,14 @@ Search search(SearchRef ref) { ); } +@riverpod +Timeline timeline(TimelineRef ref) { + return Timeline( + ref.read(settingsRepositoryProvider), + ref.read(bookRepositoryProvider), + ); +} + @riverpod SharedPreferences sharedPreferences(SharedPreferencesRef ref) => throw UnimplementedError(); diff --git a/lib/src/ui/core/dante_components.dart b/lib/src/ui/core/dante_components.dart index 63976f6..588b8d7 100644 --- a/lib/src/ui/core/dante_components.dart +++ b/lib/src/ui/core/dante_components.dart @@ -3,6 +3,7 @@ import 'package:flutter/services.dart'; class DanteDivider extends StatelessWidget { final double width; + const DanteDivider({this.width = 0.5, super.key}); @override @@ -111,6 +112,7 @@ class DanteTextField extends StatelessWidget { class DanteOutlinedButton extends StatelessWidget { final Widget child; final void Function()? onPressed; + const DanteOutlinedButton({required this.child, super.key, this.onPressed}); @override @@ -122,3 +124,25 @@ class DanteOutlinedButton extends StatelessWidget { ); } } + +class DanteOutlinedCard extends StatelessWidget { + final Widget child; + + const DanteOutlinedCard({required this.child, super.key}); + + @override + Widget build(BuildContext context) { + return Card( + clipBehavior: Clip.hardEdge, + elevation: 0, + shape: RoundedRectangleBorder( + side: BorderSide( + color: Theme.of(context).colorScheme.outline, + width: 0.5, + ), + borderRadius: const BorderRadius.all(Radius.circular(12)), + ), + child: child, + ); + } +} diff --git a/lib/src/ui/core/lottie_view.dart b/lib/src/ui/core/lottie_view.dart new file mode 100644 index 0000000..9f950a4 --- /dev/null +++ b/lib/src/ui/core/lottie_view.dart @@ -0,0 +1,33 @@ +import 'package:flutter/material.dart'; +import 'package:lottie/lottie.dart'; + +class LottieView extends StatelessWidget { + final String lottieAsset; + final String text; + + const LottieView({ + required this.lottieAsset, + required this.text, + super.key, + }); + + @override + Widget build(BuildContext context) { + return Padding( + padding: const EdgeInsets.all(16.0), + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + LottieBuilder.asset(lottieAsset), + Text( + text, + style: Theme.of(context).textTheme.titleMedium?.copyWith( + color: Theme.of(context).colorScheme.onSurface, + ), + textAlign: TextAlign.center, + ), + ], + ), + ); + } +} diff --git a/lib/src/ui/core/themed_app_bar.dart b/lib/src/ui/core/themed_app_bar.dart index 0d7dc30..7cea404 100644 --- a/lib/src/ui/core/themed_app_bar.dart +++ b/lib/src/ui/core/themed_app_bar.dart @@ -2,22 +2,14 @@ import 'package:flutter/material.dart'; class ThemedAppBar extends StatelessWidget implements PreferredSizeWidget { final List? actions; - final bool automaticallyImplyLeading; final Widget? leading; final Widget? title; - final ShapeBorder? border; - final Color? shadowColor; - final double elevation; const ThemedAppBar({ super.key, this.actions, - this.automaticallyImplyLeading = true, this.leading, this.title, - this.border, - this.shadowColor, - this.elevation = 0.0, }); @override @@ -27,14 +19,15 @@ class ThemedAppBar extends StatelessWidget implements PreferredSizeWidget { Widget build(BuildContext context) { return AppBar( actions: actions, - automaticallyImplyLeading: automaticallyImplyLeading, - backgroundColor: Theme.of(context).colorScheme.primaryContainer, + backgroundColor: Theme.of(context).colorScheme.background, + // Disable coloring of action bar on scroll + surfaceTintColor: Theme.of(context).colorScheme.background, + scrolledUnderElevation: 4, + shadowColor: Theme.of(context).colorScheme.onBackground, centerTitle: true, - elevation: elevation, + elevation: 0, leading: leading, title: title, - shape: border, - shadowColor: shadowColor, ); } } diff --git a/lib/src/ui/settings/contributors_page.dart b/lib/src/ui/settings/contributors_page.dart index e128f6b..f640e9e 100644 --- a/lib/src/ui/settings/contributors_page.dart +++ b/lib/src/ui/settings/contributors_page.dart @@ -1,4 +1,5 @@ import 'package:cached_network_image/cached_network_image.dart'; +import 'package:dantex/src/ui/core/dante_components.dart'; import 'package:dantex/src/util/url_launcher.dart'; import 'package:easy_localization/easy_localization.dart'; import 'package:flutter/material.dart'; @@ -90,16 +91,7 @@ class _ContributorCard extends StatelessWidget { @override Widget build(BuildContext context) { - return Card( - clipBehavior: Clip.hardEdge, - elevation: 0, - shape: RoundedRectangleBorder( - side: BorderSide( - color: Theme.of(context).colorScheme.outline, - width: 0.5, - ), - borderRadius: const BorderRadius.all(Radius.circular(12)), - ), + return DanteOutlinedCard( child: switch (_contribution) { _Contributor() => _buildContributor( context, diff --git a/lib/src/ui/timeline/timeline_page.dart b/lib/src/ui/timeline/timeline_page.dart index 6e1fc6a..a0ee401 100644 --- a/lib/src/ui/timeline/timeline_page.dart +++ b/lib/src/ui/timeline/timeline_page.dart @@ -1,14 +1,205 @@ +import 'package:collection/collection.dart'; +import 'package:dantex/src/data/book/entity/book.dart'; +import 'package:dantex/src/data/timeline/timeline.dart'; +import 'package:dantex/src/providers/app_router.dart'; +import 'package:dantex/src/providers/service.dart'; +import 'package:dantex/src/ui/book/book_image.dart'; +import 'package:dantex/src/ui/core/dante_components.dart'; +import 'package:dantex/src/ui/core/lottie_view.dart'; +import 'package:dantex/src/ui/core/themed_app_bar.dart'; +import 'package:dantex/src/ui/timeline/timeline_sort.dart'; +import 'package:dantex/src/util/extensions.dart'; +import 'package:dantex/src/util/layout_utils.dart'; +import 'package:easy_localization/easy_localization.dart'; import 'package:flutter/material.dart'; +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:go_router/go_router.dart'; +import 'package:timeline_tile/timeline_tile.dart'; -class TimelinePage extends StatelessWidget { +class TimelinePage extends ConsumerWidget { const TimelinePage({super.key}); @override - Widget build(BuildContext context) { - return const Material( - child: Center( - child: Text('TODO Implement Timeline'), + Widget build(BuildContext context, WidgetRef ref) { + final Timeline timeline = ref.watch(timelineProvider); + + return Scaffold( + appBar: ThemedAppBar( + title: _buildSortingInput(timeline), + ), + body: LayoutBuilder( + builder: (context, constraints) { + return StreamBuilder>( + stream: timeline.getTimelineData(), + builder: (context, snapshot) { + if (!snapshot.hasData) { + return const Center( + child: CircularProgressIndicator.adaptive(), + ); + } + + final List data = snapshot.data!; + + if (data.isEmpty) { + return _buildEmptyScreen(); + } + + return _buildTimeline( + context, + data, + getDeviceFormFactor(constraints), + ); + }, + ); + }, + ), + ); + } + + Widget _buildEmptyScreen() { + return LottieView( + lottieAsset: 'assets/lottie/books-staple.json', + text: 'timeline.empty'.tr(), + ); + } + + Widget _buildTimeline( + BuildContext context, + List grouping, + DeviceFormFactor formFactor, + ) { + return Center( + child: ListView( + children: [ + ...grouping + .map( + (group) => [ + _buildMonthTile(context, group.month, formFactor), + ...group.books.map( + (book) => _buildBookTimelineTile(context, book, formFactor), + ), + ], + ) + .flattened, + const SizedBox(height: 16), + ], + ), + ); + } + + Widget _buildSortingInput(Timeline timeline) { + return StreamBuilder( + stream: timeline.sortStrategy, + builder: (context, snapshot) { + final TimelineSortStrategy sortStrategy = + snapshot.data ?? TimelineSortStrategy.byEndState; + + return SegmentedButton( + onSelectionChanged: (Set selection) { + timeline.setSortStrategy(selection.first); + }, + showSelectedIcon: false, + segments: [ + ButtonSegment( + value: TimelineSortStrategy.byEndState, + label: Text('timeline.sorting.ended-in'.tr()), + icon: const Icon(Icons.check), + ), + ButtonSegment( + value: TimelineSortStrategy.byStartDate, + label: Text('timeline.sorting.started-in'.tr()), + icon: const Icon(Icons.book_outlined), + ), + ], + selected: {sortStrategy}, + ); + }, + ); + } + + Widget _buildBookTimelineTile( + BuildContext context, + Book book, + DeviceFormFactor formFactor, + ) { + final (indicatorSize, tileHeight) = _bookTimelineSizes(formFactor); + + return TimelineTile( + alignment: TimelineAlign.center, + lineXY: 0.5, + beforeLineStyle: LineStyle( + color: Theme.of(context).colorScheme.primary, + thickness: 2, + ), + indicatorStyle: IndicatorStyle( + drawGap: true, + color: Colors.black, + width: indicatorSize, + height: indicatorSize, + indicator: GestureDetector( + onTap: () => context.go( + DanteRoute.bookDetail.navigationUrl.replaceAll( + ':bookId', + book.id, + ), + ), + child: BookImage( + book.thumbnailAddress, + size: indicatorSize, + ), + ), + ), + endChild: SizedBox(height: tileHeight), + ); + } + + Widget _buildMonthTile( + BuildContext context, + DateTime month, + DeviceFormFactor formFactor, + ) { + final Size indicatorSize = _monthTimelineSize(formFactor); + + return TimelineTile( + alignment: TimelineAlign.center, + lineXY: 0.5, + beforeLineStyle: LineStyle( + color: Theme.of(context).colorScheme.primary, + thickness: 2, + ), + indicatorStyle: IndicatorStyle( + drawGap: true, + color: Colors.black, + width: indicatorSize.width, + height: indicatorSize.height, + indicator: DanteOutlinedCard( + child: Center( + child: Padding( + padding: const EdgeInsets.all(8.0), + child: Text( + month.formatWithMonthAndYear(), + style: Theme.of(context).textTheme.titleMedium, + ), + ), + ), + ), ), ); } + + (double indicatorSize, double tileHeight) _bookTimelineSizes( + DeviceFormFactor formFactor, + ) { + return switch (formFactor) { + DeviceFormFactor.phone => (96, 150), + DeviceFormFactor() => (128, 200), + }; + } + + Size _monthTimelineSize(DeviceFormFactor formFactor) { + return switch (formFactor) { + DeviceFormFactor.phone => const Size(192, 48), + DeviceFormFactor() => const Size(256, 64), + }; + } } diff --git a/lib/src/ui/timeline/timeline_sort.dart b/lib/src/ui/timeline/timeline_sort.dart new file mode 100644 index 0000000..e4fd4a8 --- /dev/null +++ b/lib/src/ui/timeline/timeline_sort.dart @@ -0,0 +1,4 @@ +enum TimelineSortStrategy { + byEndState, + byStartDate, +} diff --git a/lib/src/util/extensions.dart b/lib/src/util/extensions.dart index d7916f5..a0701fd 100644 --- a/lib/src/util/extensions.dart +++ b/lib/src/util/extensions.dart @@ -1,5 +1,7 @@ import 'dart:ui'; +import 'package:easy_localization/easy_localization.dart'; + extension HexColor on String { /// String is in the format "aabbcc" or "ffaabbcc" with an optional leading "#". Color toColor() { @@ -9,3 +11,11 @@ extension HexColor on String { return Color(int.parse(buffer.toString(), radix: 16)); } } + +final DateFormat _dfMonth = DateFormat('MMMM yyyy'); + +extension DateTimeX on DateTime { + String formatWithMonthAndYear() { + return _dfMonth.format(this); + } +} diff --git a/pubspec.lock b/pubspec.lock index 7d5f8ea..1e23286 100644 --- a/pubspec.lock +++ b/pubspec.lock @@ -1178,6 +1178,14 @@ packages: url: "https://pub.dev" source: hosted version: "0.6.0" + timeline_tile: + dependency: "direct main" + description: + name: timeline_tile + sha256: "85ec2023c67137397c2812e3e848b2fb20b410b67cd9aff304bb5480c376fc0c" + url: "https://pub.dev" + source: hosted + version: "2.0.0" timing: dependency: transitive description: diff --git a/pubspec.yaml b/pubspec.yaml index d8a0d6f..b1a37af 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -46,6 +46,7 @@ dependencies: intl: ^0.18.1 carousel_slider: ^4.2.1 flex_color_picker: ^3.3.0 + timeline_tile: ^2.0.0 dev_dependencies: flutter_test: @@ -76,5 +77,6 @@ flutter: assets: - assets/data/ - assets/logo/ + - assets/lottie/ - assets/images/ - assets/translations/