repository and data provider concept
This commit is contained in:
parent
6ad8203b6a
commit
b171fef348
@ -1,4 +1,3 @@
|
|||||||
abstract class DataLoader<TResult> {
|
abstract class DataLoader<TResult> {
|
||||||
|
|
||||||
Future<TResult> fetch();
|
Future<TResult> fetch();
|
||||||
}
|
}
|
||||||
|
@ -1,25 +0,0 @@
|
|||||||
import 'package:hydrated_bloc/hydrated_bloc.dart';
|
|
||||||
|
|
||||||
import '../repository/repository.dart';
|
|
||||||
import 'loadable_state.dart';
|
|
||||||
|
|
||||||
sealed class LoadableHydratedBlocEvent {}
|
|
||||||
class DataRecieved extends LoadableHydratedBlocEvent {}
|
|
||||||
|
|
||||||
abstract class LoadableHydratedBloc<TEvent, TState> extends HydratedBloc<TEvent, LoadableState<TState>> {
|
|
||||||
LoadableHydratedBloc() : super(const LoadableState()) {
|
|
||||||
repository().load();
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
Repository repository();
|
|
||||||
|
|
||||||
|
|
||||||
@override
|
|
||||||
fromJson(Map<String, dynamic> json) => LoadableState(isLoading: true, data: fromStorage(json));
|
|
||||||
@override
|
|
||||||
Map<String, dynamic>? toJson(state) => state.data.toJson();
|
|
||||||
|
|
||||||
TState fromStorage(Map<String, dynamic> json);
|
|
||||||
Map<String, dynamic>? toStorage(TState state);
|
|
||||||
}
|
|
@ -1,14 +1,14 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
|
import 'loadable_state_consumer.dart';
|
||||||
|
|
||||||
class LoadableStateBackgroundLoading extends StatelessWidget {
|
class LoadableStateBackgroundLoading extends StatelessWidget {
|
||||||
final bool visible;
|
final bool visible;
|
||||||
const LoadableStateBackgroundLoading({required this.visible, super.key});
|
const LoadableStateBackgroundLoading({required this.visible, super.key});
|
||||||
|
|
||||||
final Duration animationDuration = const Duration(milliseconds: 200);
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) => AnimatedSwitcher(
|
Widget build(BuildContext context) => AnimatedSwitcher(
|
||||||
duration: animationDuration,
|
duration: LoadableStateConsumer.animationDuration,
|
||||||
transitionBuilder: (Widget child, Animation<double> animation) => SlideTransition(
|
transitionBuilder: (Widget child, Animation<double> animation) => SlideTransition(
|
||||||
position: Tween<Offset>(
|
position: Tween<Offset>(
|
||||||
begin: const Offset(0.0, -1.0),
|
begin: const Offset(0.0, -1.0),
|
||||||
|
@ -2,36 +2,36 @@ import 'package:bloc/bloc.dart';
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:provider/provider.dart';
|
import 'package:provider/provider.dart';
|
||||||
|
|
||||||
|
import '../../utilityWidgets/loadableHydratedBloc/loadable_hydrated_bloc_event.dart';
|
||||||
import '../loadable_state.dart';
|
import '../loadable_state.dart';
|
||||||
import 'loadable_state_background_loading.dart';
|
import 'loadable_state_background_loading.dart';
|
||||||
import 'loadable_state_error_bar.dart';
|
import 'loadable_state_error_bar.dart';
|
||||||
import 'loadable_state_primary_loading.dart';
|
import 'loadable_state_primary_loading.dart';
|
||||||
|
|
||||||
// TODO might be a simpler way
|
class LoadableStateConsumer<TController extends Bloc<LoadableHydratedBlocEvent<TState>, LoadableState<TState>>, TState> extends StatelessWidget {
|
||||||
class LoadableStateConsumer<TController extends Bloc<dynamic, TWrappedState>, TWrappedState extends LoadableState<TState>, TState> extends StatelessWidget {
|
final Widget Function(TState state, bool loading) child;
|
||||||
final Widget Function(TState state) child;
|
|
||||||
const LoadableStateConsumer({required this.child, super.key});
|
const LoadableStateConsumer({required this.child, super.key});
|
||||||
|
|
||||||
static Duration animationDuration = const Duration(milliseconds: 200);
|
static Duration animationDuration = const Duration(milliseconds: 200);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
var state = context.read<TController>().state as LoadableState<TState>;
|
var loadableState = context.watch<TController>().state;
|
||||||
|
|
||||||
return Column(
|
return Column(
|
||||||
children: [
|
children: [
|
||||||
LoadableStateErrorBar(visible: state.showError()),
|
LoadableStateErrorBar(visible: loadableState.showError()),
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Stack(
|
child: Stack(
|
||||||
children: [
|
children: [
|
||||||
LoadableStatePrimaryLoading(visible: state.showPrimaryLoading()),
|
LoadableStatePrimaryLoading(visible: loadableState.showPrimaryLoading()),
|
||||||
LoadableStateBackgroundLoading(visible: state.showBackgroundLoading()),
|
LoadableStateBackgroundLoading(visible: loadableState.showBackgroundLoading()),
|
||||||
|
|
||||||
AnimatedOpacity(
|
AnimatedOpacity(
|
||||||
opacity: state.showContent() ? 1.0 : 0.0,
|
opacity: loadableState.showContent() ? 1.0 : 0.0,
|
||||||
duration: animationDuration,
|
duration: animationDuration,
|
||||||
curve: Curves.easeInOut,
|
curve: Curves.easeInOut,
|
||||||
child: state.showContent() ? child(state.data) : const SizedBox.shrink()
|
child: loadableState.showContent() ? child(loadableState.data, loadableState.isLoading) : const SizedBox.shrink()
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
|
|
||||||
import '../../utilityWidgets/bloc_providing_builder.dart';
|
import '../../utilityWidgets/bloc_module.dart';
|
||||||
import '../bloc/loadable_state_bloc.dart';
|
import '../bloc/loadable_state_bloc.dart';
|
||||||
import '../bloc/loadable_state_state.dart';
|
import '../bloc/loadable_state_state.dart';
|
||||||
|
|
||||||
@ -12,7 +12,7 @@ class LoadableStateErrorBar extends StatelessWidget {
|
|||||||
final Duration animationDuration = const Duration(milliseconds: 200);
|
final Duration animationDuration = const Duration(milliseconds: 200);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) => BlocProvidingBuilder<LoadableStateBloc, LoadableStateState>(
|
Widget build(BuildContext context) => BlocModule<LoadableStateBloc, LoadableStateState>(
|
||||||
create: (context) => LoadableStateBloc(),
|
create: (context) => LoadableStateBloc(),
|
||||||
child: (context, state) => AnimatedSize(
|
child: (context, state) => AnimatedSize(
|
||||||
duration: animationDuration,
|
duration: animationDuration,
|
||||||
|
@ -1,15 +1,15 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
|
import 'loadable_state_consumer.dart';
|
||||||
|
|
||||||
class LoadableStatePrimaryLoading extends StatelessWidget {
|
class LoadableStatePrimaryLoading extends StatelessWidget {
|
||||||
final bool visible;
|
final bool visible;
|
||||||
const LoadableStatePrimaryLoading({required this.visible, super.key});
|
const LoadableStatePrimaryLoading({required this.visible, super.key});
|
||||||
|
|
||||||
final Duration animationDuration = const Duration(milliseconds: 200);
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) => AnimatedOpacity(
|
Widget build(BuildContext context) => AnimatedOpacity(
|
||||||
opacity: visible ? 1.0 : 0.0,
|
opacity: visible ? 1.0 : 0.0,
|
||||||
duration: animationDuration,
|
duration: LoadableStateConsumer.animationDuration,
|
||||||
curve: Curves.easeInOut,
|
curve: Curves.easeInOut,
|
||||||
child: const Center(child: CircularProgressIndicator()),
|
child: const Center(child: CircularProgressIndicator()),
|
||||||
);
|
);
|
||||||
|
@ -1,13 +1 @@
|
|||||||
import '../dataLoader/data_loader.dart';
|
abstract class Repository<TState> {}
|
||||||
|
|
||||||
abstract class Repository {
|
|
||||||
final List<DataLoader> dataLoaders;
|
|
||||||
|
|
||||||
Repository(this.dataLoaders);
|
|
||||||
|
|
||||||
Future<void> load() async {
|
|
||||||
dataLoaders.forEach((element) {
|
|
||||||
element.fetch();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
|
|
||||||
class BlocProvidingBuilder<TBloc extends StateStreamableSource<TState>, TState> extends StatelessWidget {
|
class BlocModule<TBloc extends StateStreamableSource<TState>, TState> extends StatelessWidget {
|
||||||
final TBloc Function(BuildContext context) create;
|
final TBloc Function(BuildContext context) create;
|
||||||
final Widget Function(BuildContext context, TState state) child;
|
final Widget Function(BuildContext context, TState state) child;
|
||||||
const BlocProvidingBuilder({required this.create, required this.child, super.key});
|
const BlocModule({required this.create, required this.child, super.key});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) => BlocProvider<TBloc>(create: create, child: BlocBuilder<TBloc, TState>(builder: child));
|
Widget build(BuildContext context) => BlocProvider<TBloc>(create: create, child: BlocBuilder<TBloc, TState>(builder: child));
|
@ -0,0 +1,31 @@
|
|||||||
|
import 'package:hydrated_bloc/hydrated_bloc.dart';
|
||||||
|
|
||||||
|
import '../../repository/repository.dart';
|
||||||
|
import 'loadable_hydrated_bloc_event.dart';
|
||||||
|
import '../../loadableState/loadable_state.dart';
|
||||||
|
|
||||||
|
abstract class LoadableHydratedBloc<TEvent extends LoadableHydratedBlocEvent<TState>, TState, TRepository extends Repository<TState>> extends HydratedBloc<LoadableHydratedBlocEvent<TState>, LoadableState<TState>> {
|
||||||
|
late TRepository _repository;
|
||||||
|
LoadableHydratedBloc() : super(const LoadableState()) {
|
||||||
|
on<Emit<TState>>((event, emit) => emit(LoadableState(isLoading: event.loading, data: event.state(innerState ?? fromNothing()))));
|
||||||
|
on<ClearState<TState>>((event, emit) => emit(const LoadableState()));
|
||||||
|
|
||||||
|
_repository = repository();
|
||||||
|
loadState();
|
||||||
|
}
|
||||||
|
|
||||||
|
TState? get innerState => state.data;
|
||||||
|
TRepository get repo => _repository;
|
||||||
|
|
||||||
|
@override
|
||||||
|
fromJson(Map<String, dynamic> json) => LoadableState(isLoading: true, data: fromStorage(json));
|
||||||
|
@override
|
||||||
|
Map<String, dynamic>? toJson(LoadableState<TState> state) => state.data == null ? {} : state.data.toJson();
|
||||||
|
|
||||||
|
Future<void> loadState();
|
||||||
|
TRepository repository();
|
||||||
|
|
||||||
|
TState fromNothing();
|
||||||
|
TState fromStorage(Map<String, dynamic> json);
|
||||||
|
Map<String, dynamic>? toStorage(TState state);
|
||||||
|
}
|
@ -0,0 +1,7 @@
|
|||||||
|
class LoadableHydratedBlocEvent<TState> {}
|
||||||
|
class Emit<TState> extends LoadableHydratedBlocEvent<TState> {
|
||||||
|
final TState Function(TState state) state;
|
||||||
|
final bool loading;
|
||||||
|
Emit(this.state, {this.loading = false});
|
||||||
|
}
|
||||||
|
class ClearState<TState> extends LoadableHydratedBlocEvent<TState> {}
|
@ -1,17 +1,32 @@
|
|||||||
import '../../../infrastructure/loadableState/loadable_hydrated_bloc.dart';
|
import '../../../infrastructure/utilityWidgets/loadableHydratedBloc/loadable_hydrated_bloc.dart';
|
||||||
import '../../../infrastructure/repository/repository.dart';
|
import '../../../infrastructure/utilityWidgets/loadableHydratedBloc/loadable_hydrated_bloc_event.dart';
|
||||||
import '../repository/marianum_message_repository.dart';
|
import '../repository/marianum_message_repository.dart';
|
||||||
import 'marianum_message_state.dart';
|
import 'marianum_message_state.dart';
|
||||||
|
|
||||||
class MarianumMessageBloc extends LoadableHydratedBloc<void, MarianumMessageState> {
|
sealed class MarianumMessageEvent extends LoadableHydratedBlocEvent<MarianumMessageState> {}
|
||||||
MarianumMessageBloc();
|
class MessageEvent extends MarianumMessageEvent {}
|
||||||
|
|
||||||
|
class MarianumMessageBloc extends LoadableHydratedBloc<MarianumMessageEvent, MarianumMessageState, MarianumMessageRepository> {
|
||||||
|
MarianumMessageBloc() {
|
||||||
|
on<MessageEvent>((event, emit) {
|
||||||
|
add(Emit((state) => state.copyWith.messageList(messages: [])));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<void> loadState() async {
|
||||||
|
var messages = await repo.getMessages();
|
||||||
|
add(Emit((state) => state.copyWith(messageList: messages)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
MarianumMessageRepository repository() => MarianumMessageRepository();
|
||||||
|
|
||||||
|
@override
|
||||||
|
MarianumMessageState fromNothing() => const MarianumMessageState(messageList: MarianumMessageList(base: "", messages: []));
|
||||||
|
|
||||||
@override
|
@override
|
||||||
MarianumMessageState fromStorage(Map<String, dynamic> json) => MarianumMessageState.fromJson(json);
|
MarianumMessageState fromStorage(Map<String, dynamic> json) => MarianumMessageState.fromJson(json);
|
||||||
@override
|
@override
|
||||||
Map<String, dynamic>? toStorage(MarianumMessageState state) => state.toJson();
|
Map<String, dynamic>? toStorage(MarianumMessageState state) => state.toJson();
|
||||||
|
|
||||||
@override
|
|
||||||
Repository repository() => MarianumMessageRepository(emit);
|
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,11 @@
|
|||||||
import '../../../infrastructure/dataLoader/data_loader.dart';
|
import '../../../infrastructure/dataLoader/data_loader.dart';
|
||||||
|
import '../bloc/marianum_message_state.dart';
|
||||||
|
|
||||||
|
class MarianumMessageGetMessages extends DataLoader<MarianumMessageList> {
|
||||||
|
|
||||||
class MarianumMessageGetMessages extends DataLoader {
|
|
||||||
@override
|
@override
|
||||||
Future fetch() => Future(() => 'Test');
|
Future<MarianumMessageList> fetch() async {
|
||||||
|
await Future.delayed(const Duration(seconds: 3));
|
||||||
|
return const MarianumMessageList(base: '', messages: [MarianumMessage(date: '', name: 'RepoTest', url: '')]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,9 +1,7 @@
|
|||||||
import '../../../infrastructure/loadableState/loadable_state.dart';
|
|
||||||
import '../../../infrastructure/repository/repository.dart';
|
import '../../../infrastructure/repository/repository.dart';
|
||||||
|
import '../bloc/marianum_message_state.dart';
|
||||||
import '../dataProvider/marianum_message_get_messages.dart';
|
import '../dataProvider/marianum_message_get_messages.dart';
|
||||||
|
|
||||||
class MarianumMessageRepository<TState> extends Repository {
|
class MarianumMessageRepository extends Repository<MarianumMessageState> {
|
||||||
MarianumMessageRepository(void Function(LoadableState<TState> content) emit) : super([
|
Future<MarianumMessageList> getMessages() => MarianumMessageGetMessages().fetch();
|
||||||
MarianumMessageGetMessages(),
|
|
||||||
]);
|
|
||||||
}
|
}
|
||||||
|
@ -1,12 +1,14 @@
|
|||||||
import 'dart:developer';
|
import 'dart:developer';
|
||||||
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:provider/provider.dart';
|
||||||
|
|
||||||
import '../../../../../api/mhsl/message/getMessages/getMessagesResponse.dart';
|
import '../../../../../api/mhsl/message/getMessages/getMessagesResponse.dart';
|
||||||
import '../../../../../view/pages/more/message/messageView.dart';
|
import '../../../../../view/pages/more/message/messageView.dart';
|
||||||
import '../../../infrastructure/loadableState/loadable_state.dart';
|
import '../../../infrastructure/loadableState/loadable_state.dart';
|
||||||
import '../../../infrastructure/loadableState/view/loadable_state_consumer.dart';
|
import '../../../infrastructure/loadableState/view/loadable_state_consumer.dart';
|
||||||
import '../../../infrastructure/utilityWidgets/bloc_providing_builder.dart';
|
import '../../../infrastructure/utilityWidgets/bloc_module.dart';
|
||||||
|
import '../../../infrastructure/utilityWidgets/loadableHydratedBloc/loadable_hydrated_bloc_event.dart';
|
||||||
import '../bloc/marianum_message_bloc.dart';
|
import '../bloc/marianum_message_bloc.dart';
|
||||||
import '../bloc/marianum_message_state.dart';
|
import '../bloc/marianum_message_state.dart';
|
||||||
|
|
||||||
@ -14,17 +16,22 @@ class MarianumMessageListView extends StatelessWidget {
|
|||||||
const MarianumMessageListView({super.key});
|
const MarianumMessageListView({super.key});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) => Scaffold(
|
Widget build(BuildContext context) => BlocModule<MarianumMessageBloc, LoadableState<MarianumMessageState>>(
|
||||||
appBar: AppBar(
|
|
||||||
title: const Text('Marianum Message'),
|
|
||||||
),
|
|
||||||
body: BlocProvidingBuilder<MarianumMessageBloc, LoadableState<MarianumMessageState>>(
|
|
||||||
create: (context) => MarianumMessageBloc(),
|
create: (context) => MarianumMessageBloc(),
|
||||||
child: (context, state) {
|
child: (context, state) {
|
||||||
// if(value.primaryLoading()) return const LoadingSpinner();
|
// if(value.primaryLoading()) return const LoadingSpinner();
|
||||||
log(state.toString());
|
log(state.toString());
|
||||||
return LoadableStateConsumer<MarianumMessageBloc, LoadableState<MarianumMessageState>, MarianumMessageState>(
|
return Scaffold(
|
||||||
child: (state) => ListView.builder(
|
appBar: AppBar(
|
||||||
|
title: const Text('Marianum Message'),
|
||||||
|
actions: [
|
||||||
|
IconButton(onPressed: () => context.read<MarianumMessageBloc>().add(MessageEvent()), icon: Icon(Icons.abc)),
|
||||||
|
IconButton(onPressed: () => context.read<MarianumMessageBloc>().add(Emit((state) => MarianumMessageState(messageList: MarianumMessageList(base: "", messages: [MarianumMessage(url: "", name: "Teeest", date: "now")])))), icon: Icon(Icons.add)),
|
||||||
|
IconButton(onPressed: () => context.read<MarianumMessageBloc>().add(ClearState()), icon: Icon(Icons.add))
|
||||||
|
],
|
||||||
|
),
|
||||||
|
body: LoadableStateConsumer<MarianumMessageBloc, MarianumMessageState>(
|
||||||
|
child: (state, loading) => ListView.builder(
|
||||||
itemCount: state.messageList.messages.length,
|
itemCount: state.messageList.messages.length,
|
||||||
itemBuilder: (context, index) {
|
itemBuilder: (context, index) {
|
||||||
var message = state.messageList.messages.toList()[index];
|
var message = state.messageList.messages.toList()[index];
|
||||||
@ -33,7 +40,7 @@ class MarianumMessageListView extends StatelessWidget {
|
|||||||
mainAxisAlignment: MainAxisAlignment.center,
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
children: [Icon(Icons.newspaper)],
|
children: [Icon(Icons.newspaper)],
|
||||||
),
|
),
|
||||||
title: Text(message.name, overflow: TextOverflow.ellipsis),
|
title: Text("${message.name}${loading ? "loading" : "notloading"}", overflow: TextOverflow.ellipsis),
|
||||||
subtitle: Text('vom ${message.date}'),
|
subtitle: Text('vom ${message.date}'),
|
||||||
trailing: const Icon(Icons.arrow_right),
|
trailing: const Icon(Icons.arrow_right),
|
||||||
onTap: () {
|
onTap: () {
|
||||||
@ -42,8 +49,8 @@ class MarianumMessageListView extends StatelessWidget {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
);
|
|
||||||
}
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user