|
|
|
|
@ -10,7 +10,7 @@ import '../../core/ui/error_indicator.dart';
|
|
|
|
|
import '../view_models/home_viewmodel.dart';
|
|
|
|
|
import 'home_title.dart';
|
|
|
|
|
|
|
|
|
|
class HomeScreen extends StatelessWidget {
|
|
|
|
|
class HomeScreen extends StatefulWidget {
|
|
|
|
|
const HomeScreen({
|
|
|
|
|
super.key,
|
|
|
|
|
required this.viewModel,
|
|
|
|
|
@ -18,6 +18,30 @@ class HomeScreen extends StatelessWidget {
|
|
|
|
|
|
|
|
|
|
final HomeViewModel viewModel;
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
State<HomeScreen> createState() => _HomeScreenState();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class _HomeScreenState extends State<HomeScreen> {
|
|
|
|
|
@override
|
|
|
|
|
void initState() {
|
|
|
|
|
super.initState();
|
|
|
|
|
widget.viewModel.deleteBooking.addListener(_onResult);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
void didUpdateWidget(covariant HomeScreen oldWidget) {
|
|
|
|
|
super.didUpdateWidget(oldWidget);
|
|
|
|
|
oldWidget.viewModel.deleteBooking.removeListener(_onResult);
|
|
|
|
|
widget.viewModel.deleteBooking.addListener(_onResult);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
void dispose() {
|
|
|
|
|
widget.viewModel.deleteBooking.removeListener(_onResult);
|
|
|
|
|
super.dispose();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
|
return Scaffold(
|
|
|
|
|
@ -33,26 +57,26 @@ class HomeScreen extends StatelessWidget {
|
|
|
|
|
top: true,
|
|
|
|
|
bottom: true,
|
|
|
|
|
child: ListenableBuilder(
|
|
|
|
|
listenable: viewModel.load,
|
|
|
|
|
listenable: widget.viewModel.load,
|
|
|
|
|
builder: (context, child) {
|
|
|
|
|
if (viewModel.load.running) {
|
|
|
|
|
if (widget.viewModel.load.running) {
|
|
|
|
|
return const Center(
|
|
|
|
|
child: CircularProgressIndicator(),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (viewModel.load.error) {
|
|
|
|
|
if (widget.viewModel.load.error) {
|
|
|
|
|
return ErrorIndicator(
|
|
|
|
|
title: AppLocalization.of(context).errorWhileLoadingHome,
|
|
|
|
|
label: AppLocalization.of(context).tryAgain,
|
|
|
|
|
onPressed: viewModel.load.execute,
|
|
|
|
|
onPressed: widget.viewModel.load.execute,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return child!;
|
|
|
|
|
},
|
|
|
|
|
child: ListenableBuilder(
|
|
|
|
|
listenable: viewModel,
|
|
|
|
|
listenable: widget.viewModel,
|
|
|
|
|
builder: (context, _) {
|
|
|
|
|
return CustomScrollView(
|
|
|
|
|
slivers: [
|
|
|
|
|
@ -62,16 +86,20 @@ class HomeScreen extends StatelessWidget {
|
|
|
|
|
vertical: Dimens.of(context).paddingScreenVertical,
|
|
|
|
|
horizontal: Dimens.of(context).paddingScreenHorizontal,
|
|
|
|
|
),
|
|
|
|
|
child: HomeHeader(viewModel: viewModel),
|
|
|
|
|
child: HomeHeader(viewModel: widget.viewModel),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
SliverList.builder(
|
|
|
|
|
itemCount: viewModel.bookings.length,
|
|
|
|
|
itemCount: widget.viewModel.bookings.length,
|
|
|
|
|
itemBuilder: (_, index) => _Booking(
|
|
|
|
|
key: ValueKey(index),
|
|
|
|
|
booking: viewModel.bookings[index],
|
|
|
|
|
onTap: () => context.push(
|
|
|
|
|
Routes.bookingWithId(viewModel.bookings[index].id)),
|
|
|
|
|
key: ValueKey(widget.viewModel.bookings[index].id),
|
|
|
|
|
booking: widget.viewModel.bookings[index],
|
|
|
|
|
onTap: () => context.push(Routes.bookingWithId(
|
|
|
|
|
widget.viewModel.bookings[index].id)),
|
|
|
|
|
onDismissed: (_) =>
|
|
|
|
|
widget.viewModel.deleteBooking.execute(
|
|
|
|
|
widget.viewModel.bookings[index].id,
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
)
|
|
|
|
|
],
|
|
|
|
|
@ -82,6 +110,26 @@ class HomeScreen extends StatelessWidget {
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void _onResult() {
|
|
|
|
|
if (widget.viewModel.deleteBooking.completed) {
|
|
|
|
|
widget.viewModel.deleteBooking.clearResult();
|
|
|
|
|
ScaffoldMessenger.of(context).showSnackBar(
|
|
|
|
|
SnackBar(
|
|
|
|
|
content: Text(AppLocalization.of(context).bookingDeleted),
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (widget.viewModel.deleteBooking.error) {
|
|
|
|
|
widget.viewModel.deleteBooking.clearResult();
|
|
|
|
|
ScaffoldMessenger.of(context).showSnackBar(
|
|
|
|
|
SnackBar(
|
|
|
|
|
content: Text(AppLocalization.of(context).errorWhileDeletingBooking),
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class _Booking extends StatelessWidget {
|
|
|
|
|
@ -89,37 +137,44 @@ class _Booking extends StatelessWidget {
|
|
|
|
|
super.key,
|
|
|
|
|
required this.booking,
|
|
|
|
|
required this.onTap,
|
|
|
|
|
required this.onDismissed,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
final BookingSummary booking;
|
|
|
|
|
final GestureTapCallback onTap;
|
|
|
|
|
final DismissDirectionCallback onDismissed;
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
|
return InkWell(
|
|
|
|
|
onTap: onTap,
|
|
|
|
|
child: Padding(
|
|
|
|
|
padding: EdgeInsets.symmetric(
|
|
|
|
|
horizontal: Dimens.of(context).paddingScreenHorizontal,
|
|
|
|
|
vertical: Dimens.paddingVertical,
|
|
|
|
|
),
|
|
|
|
|
child: Column(
|
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
|
children: [
|
|
|
|
|
Text(
|
|
|
|
|
booking.name,
|
|
|
|
|
style: Theme.of(context).textTheme.titleLarge,
|
|
|
|
|
),
|
|
|
|
|
Text(
|
|
|
|
|
dateFormatStartEnd(
|
|
|
|
|
DateTimeRange(
|
|
|
|
|
start: booking.startDate,
|
|
|
|
|
end: booking.endDate,
|
|
|
|
|
return Dismissible(
|
|
|
|
|
key: ValueKey(booking.id),
|
|
|
|
|
direction: DismissDirection.endToStart,
|
|
|
|
|
onDismissed: onDismissed,
|
|
|
|
|
child: InkWell(
|
|
|
|
|
onTap: onTap,
|
|
|
|
|
child: Padding(
|
|
|
|
|
padding: EdgeInsets.symmetric(
|
|
|
|
|
horizontal: Dimens.of(context).paddingScreenHorizontal,
|
|
|
|
|
vertical: Dimens.paddingVertical,
|
|
|
|
|
),
|
|
|
|
|
child: Column(
|
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
|
children: [
|
|
|
|
|
Text(
|
|
|
|
|
booking.name,
|
|
|
|
|
style: Theme.of(context).textTheme.titleLarge,
|
|
|
|
|
),
|
|
|
|
|
Text(
|
|
|
|
|
dateFormatStartEnd(
|
|
|
|
|
DateTimeRange(
|
|
|
|
|
start: booking.startDate,
|
|
|
|
|
end: booking.endDate,
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
style: Theme.of(context).textTheme.bodyLarge,
|
|
|
|
|
),
|
|
|
|
|
style: Theme.of(context).textTheme.bodyLarge,
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
|