flutter_rearch 1.0.0-dev.0 flutter_rearch: ^1.0.0-dev.0 copied to clipboard
A re-imagined declarative approach to application design and architecture
import 'dart:math';
import 'dart:ui';
import 'package:flutter/material.dart';
import 'package:flutter_mimir/flutter_mimir.dart';
import 'package:flutter_rearch/flutter_rearch.dart';
import 'package:rearch/rearch.dart';
void main() => runApp(const TodoApp());
/// Represents the [MimirIndex] that contains the todos.
Future<MimirIndex> indexAsyncCapsule(CapsuleHandle use) async {
final instance = await Mimir.defaultInstance;
return instance.openIndex('todos', primaryKey: 'timestamp');
}
/// Allows for the [indexAsyncCapsule] to more easily be warmed up
/// for use in [indexCapsule].
AsyncValue<MimirIndex> indexWarmUpCapsule(CapsuleHandle use) {
final future = use(indexAsyncCapsule);
return use.future(future);
}
/// Acts as a proxy to the warmed-up [indexAsyncCapsule].
MimirIndex indexCapsule(CapsuleHandle use) {
return use(indexWarmUpCapsule).dataOrElse(
() => throw StateError('indexWarmUpCapsule was not warmed up!'),
);
}
/// Represents an item in the todos list.
typedef Todo = ({
int timestamp, // milliseconds since epoch
String title,
String? description,
bool completed,
});
/// Utilities for handling conversion between [Todo] and [MimirDocument]s.
extension TodoDocumentUtilities on Todo {
/// Creates a [MimirDocument] from a [Todo].
MimirDocument toDocument() {
return {
'timestamp': timestamp,
'title': title,
'description': description,
'completed': completed,
};
}
/// Creates a [Todo] from a [MimirDocument].
static Todo toTodo(MimirDocument document) {
return (
timestamp: document['timestamp']!,
title: document['title']!,
description: document['description'],
completed: document['completed']!,
);
}
}
/// Provides a way to create/update and delete todos.
({
void Function(Todo) updateTodo,
void Function(int) deleteTodo,
}) todoListManagerCapsule(CapsuleHandle use) {
final index = use(indexCapsule);
return (
updateTodo: (todo) => index.addDocument(todo.toDocument()),
deleteTodo: (timestamp) => index.deleteDocument(timestamp.toString()),
);
}
/// Represents the filter for a list of todos.
typedef TodoListFilter = ({
String query,
bool completionStatus,
});
/// Represents the current filter to search with
/// ('' as a query string represents no current query).
({
TodoListFilter filter,
void Function(String) setQueryString,
void Function() toggleCompletionStatus,
}) filterCapsule(CapsuleHandle use) {
final (query, setQuery) = use.state('');
final (completionStatus, setCompletionStatus) = use.state(false);
return (
filter: (query: query, completionStatus: completionStatus),
setQueryString: setQuery,
toggleCompletionStatus: () => setCompletionStatus(!completionStatus),
);
}
/// Represents the todos list using the filter from the [filterCapsule].
AsyncValue<List<Todo>> todoListCapsule(CapsuleHandle use) {
final index = use(indexCapsule);
final (
filter: (:query, :completionStatus),
setQueryString: _,
toggleCompletionStatus: _
) = use(filterCapsule);
// When query is null/empty, it does not affect the search.
final documentsStream = use.memo(
() => index.searchStream(
query: query,
filter: Mimir.where('completed', isEqualTo: completionStatus.toString()),
),
[index, query, completionStatus],
);
final todoDocumentsState = use.stream(documentsStream);
return todoDocumentsState.map(
(todoDocs) => todoDocs.map(TodoDocumentUtilities.toTodo).toList()
..sort((todo1, todo2) => todo1.timestamp.compareTo(todo2.timestamp)),
);
}
/// Represents the length of the [todoListCapsule].
AsyncValue<int> todoListLengthCapsule(CapsuleHandle use) =>
use(todoListCapsule).map((todos) => todos.length);
/// {@template TodoApp}
/// Wraps around [MaterialApp] and is the entry point [Widget] of the app.
/// {@endtemplate}
class TodoApp extends StatelessWidget {
/// {@macro TodoApp}
const TodoApp({super.key});
@override
Widget build(BuildContext context) {
return RearchBootstrapper(
child: MaterialApp(
title: 'Rearch Todos',
theme: ThemeData.light(useMaterial3: true),
darkTheme: ThemeData.dark(useMaterial3: true),
home: const GlobalWarmUps(child: Body()),
),
);
}
}
/// {@template GlobalWarmUps}
/// Warms up all of the global warm up capsules so that the rest of the app
/// doesn't have to individually handle failure states.
/// {@endtemplate}
final class GlobalWarmUps extends RearchConsumer {
/// {@macro GlobalWarmUps}
const GlobalWarmUps({required this.child, super.key});
/// The [Widget] to show when all warm up capsules are [AsyncData]s.
final Widget child;
@override
Widget build(BuildContext context, WidgetHandle use) {
return [
use(indexWarmUpCapsule),
].toWarmUpWidget(
child: child,
loading: const Center(child: CircularProgressIndicator.adaptive()),
errorBuilder: (errors) => Column(
children: [
for (final AsyncError(:error, :stackTrace) in errors)
Text('$error\n$stackTrace'),
],
),
);
}
}
/// {@template Body}
/// Wraps around [Scaffold] and serves as the main body of the application.
/// {@endtemplate}
class Body extends RearchConsumer {
/// {@macro Body}
const Body({super.key});
@override
Widget build(BuildContext context, WidgetHandle use) {
const searchBarHeight = 56.0;
const animationDuration = Duration(milliseconds: 125);
final (
filter: (query: _, :completionStatus),
setQueryString: _,
:toggleCompletionStatus,
) = use(filterCapsule);
final completionText = completionStatus ? 'completed' : 'incomplete';
final todoListLengthState = use(todoListLengthCapsule);
final todoListLength = todoListLengthState.dataOr(0);
final statusWidget = switch (todoListLengthState) {
AsyncLoading() => const CircularProgressIndicator.adaptive(),
AsyncError(:final error) => Text('$error'),
AsyncData(data: final length) when length == 0 =>
Text('No $completionText todos found'),
_ => null,
};
final (:updateTodo, deleteTodo: _) = use(todoListManagerCapsule);
final bottomHeightAnimationController =
use.animationController(duration: animationDuration);
final (isSearching, setIsSearching) = use.state(false);
use.effect(
() {
if (isSearching) {
bottomHeightAnimationController.forward();
} else {
bottomHeightAnimationController.reverse();
}
return null;
},
[isSearching, bottomHeightAnimationController],
);
return Scaffold(
body: AnimatedBuilder(
animation: bottomHeightAnimationController,
builder: (context, _) {
final bottomHeight =
bottomHeightAnimationController.value * searchBarHeight;
return Stack(
children: [
// The dynamic background
const Positioned.fill(child: DynamicBackground()),
// The main todos content
Positioned.fill(
child: CustomScrollView(
slivers: [
SliverToBoxAdapter(
child: SizedBox(
height: kToolbarHeight + bottomHeight + 12,
),
),
SliverSafeArea(
sliver: SliverList.builder(
itemCount: todoListLength,
itemBuilder: (context, index) => TodoItem(index: index),
),
),
],
),
),
// The app bar
Positioned(
left: 0,
right: 0,
top: 0,
child: CustomAppBar(
bottomHeight: bottomHeight,
completionStatus: completionStatus,
toggleCompletionStatus: toggleCompletionStatus,
toggleIsSearching: () => setIsSearching(!isSearching),
isSearching: isSearching,
updateTodo: updateTodo,
animationDuration: animationDuration,
),
),
// The info/status widget at the bottom
if (statusWidget != null)
Positioned(
left: 0,
right: 0,
bottom: max(MediaQuery.paddingOf(context).bottom, 16),
child: Center(
child: Card(
elevation: 8,
child: Padding(
padding: const EdgeInsets.all(16),
child: statusWidget,
),
),
),
),
],
);
},
),
);
}
}
/// {@template CustomAppBar}
/// The custom [AppBar] featured in the application.
/// {@endtemplate}
class CustomAppBar extends StatelessWidget {
/// {@macro CustomAppBar}
const CustomAppBar({
required this.bottomHeight,
required this.completionStatus,
required this.toggleCompletionStatus,
required this.isSearching,
required this.toggleIsSearching,
required this.updateTodo,
required this.animationDuration,
super.key,
});
/// The height of the [AppBar.bottom] (this should be an animated value).
final double bottomHeight;
/// Whether we are displaying completed or incomplete todos.
final bool completionStatus;
/// Callback that toggles the type of todos we are displaying.
final void Function() toggleCompletionStatus;
/// Whether or not the AppBar should be displayed with the [SearchBar].
final bool isSearching;
/// Toggle for [isSearching].
final void Function() toggleIsSearching;
/// Function that creates/updates a given [Todo].
final void Function(Todo) updateTodo;
/// The [Duration] of the [SearchBar] open/close animation.
final Duration animationDuration;
@override
Widget build(BuildContext context) {
return SizedBox(
height: MediaQuery.paddingOf(context).top + kToolbarHeight + bottomHeight,
child: ClipRRect(
borderRadius: const BorderRadius.vertical(
bottom: Radius.circular(16),
),
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 4, sigmaY: 4),
child: AppBar(
title: const Text(
'rearch todos',
style: TextStyle(fontWeight: FontWeight.w900),
),
elevation: 2,
scrolledUnderElevation: 2,
backgroundColor:
Theme.of(context).colorScheme.surface.withOpacity(0.7),
actions: [
IconButton(
tooltip: completionStatus
? 'Show incomplete todos'
: 'Show completed todos',
icon: Icon(
completionStatus
? Icons.task_alt_rounded
: Icons.radio_button_unchecked_rounded,
),
onPressed: toggleCompletionStatus,
),
IconButton(
tooltip: 'Search todos',
icon: const Icon(Icons.search_rounded),
onPressed: toggleIsSearching,
),
IconButton(
tooltip: 'Create todo',
icon: const Icon(Icons.edit_rounded),
onPressed: () => showCreateTodoDialog(context, updateTodo),
),
],
bottom: PreferredSize(
preferredSize: Size.fromHeight(bottomHeight),
child: SizedBox(
height: bottomHeight,
child: AnimatedSwitcher(
duration: animationDuration,
child:
isSearching ? SearchBar(close: toggleIsSearching) : null,
),
),
),
),
),
),
);
}
}
/// {@template SearchBar}
/// Displays the search bar at the top of the application
/// and mutates the [filterCapsule].
/// {@endtemplate}
class SearchBar extends RearchConsumer {
/// {@macro SearchBar}
const SearchBar({required this.close, super.key});
/// Callback that will close the search bar.
final void Function() close;
@override
Widget build(BuildContext context, WidgetHandle use) {
final textController = use.textEditingController();
final (
filter: _,
:setQueryString,
toggleCompletionStatus: _,
) = use(filterCapsule);
return Padding(
padding: const EdgeInsets.only(left: 8, right: 8, bottom: 8),
child: TextField(
controller: textController,
onChanged: setQueryString,
decoration: InputDecoration(
contentPadding: EdgeInsets.zero,
border: const OutlineInputBorder(
borderRadius: BorderRadius.all(Radius.circular(100)),
),
prefixIcon: const Icon(Icons.search_rounded),
suffixIcon: Row(
mainAxisSize: MainAxisSize.min,
children: [
IconButton(
icon: const Icon(Icons.cancel_rounded),
onPressed: () {
if (textController.text != '') {
textController.text = '';
setQueryString('');
} else {
close();
}
},
),
],
),
),
),
);
}
}
/// {@template TodoItem}
/// Displays a singular [Todo] item from the [todoListCapsule].
/// This example is slightly over-engineered to showcase how to reduce rebuilds
/// when dealing with slightly larger lists of data.
/// {@endtemplate}
class TodoItem extends RearchConsumer {
/// {@macro TodoItem}
const TodoItem({required this.index, super.key});
/// The [index] of this [TodoItem] in the [todoListCapsule].
final int index;
@override
Widget build(BuildContext context, WidgetHandle use) {
// The following uses a more advanced technique in rearch: inline capsules.
// This is similar to `select` in other state management frameworks, but
// inline capsules are much more powerful because they are full capsules.
// Please read the documentation for more.
final (:title, :description, :timestamp, :completed) = use(
todoListCapsule.map(
(asyncList) => asyncList.dataOrElse(
() => throw StateError(
'In order to display a TodoItem, the todo list must have data!',
),
)[index],
),
);
final (:updateTodo, :deleteTodo) = use(todoListManagerCapsule);
void delete() => deleteTodo(timestamp);
void toggleCompletionStatus() {
updateTodo(
(
title: title,
description: description,
timestamp: timestamp,
completed: !completed,
),
);
}
return Padding(
key: ValueKey(timestamp),
padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 2),
child: Card(
color: Theme.of(context).colorScheme.surface.withOpacity(0.6),
child: ListTile(
title: Text(title),
subtitle: description != null ? Text(description) : null,
leading: Icon(
completed
? Icons.task_alt_rounded
: Icons.radio_button_unchecked_rounded,
),
onTap: toggleCompletionStatus,
onLongPress: () => showDeletionConfirmationDialog(context, delete),
),
),
);
}
}
/// {@template DynamicBackground}
/// Displays the bubbly dynamic background effect.
/// {@endtemplate}
class DynamicBackground extends RearchConsumer {
/// {@macro DynamicBackground}
const DynamicBackground({super.key});
@override
Widget build(BuildContext context, WidgetHandle use) {
const color1 = Color(0xFFD223E4);
const color2 = Color(0xFF0157F5);
const avgCircleRadius = 0.07;
final numCirclesToFillScreen = 1 / (pi * pow(avgCircleRadius, 2));
final goalCircleCount = numCirclesToFillScreen / 2;
// We need to use this more advanced side effect since we need to be able
// to grab the most up-to-date copy of the circles when we check to see
// if we actually need to add a new circle (if we used the regular state
// effect, the closure would capture an outdated copy of the state).
final (getCircles, setCircles) =
use.stateGetterSetter(<SplashCircleProperties>{});
final addCircle = use.memo(
() => (SplashCircleProperties circle) {
if (getCircles().length >= goalCircleCount) return;
setCircles({...getCircles(), circle});
},
[getCircles, goalCircleCount, setCircles],
);
final removeCircle = use.memo(
() => (int id) {
setCircles({
...getCircles().where((circle) => circle.id != id),
});
},
[getCircles, setCircles],
);
use.effect(
() {
final random = Random();
final circleStream = Stream.periodic(
const Duration(milliseconds: 50),
(i) {
return (
id: i,
centerX: random.nextDouble(),
centerY: random.nextDouble(),
color: Color.lerp(color1, color2, random.nextDouble())!
.withOpacity(0.3),
radius: avgCircleRadius +
avgCircleRadius * (random.nextDouble() - 0.5),
appear: Duration(
seconds: 2 + (random.nextDouble() * 3).round(),
),
disappear: Duration(
seconds: 2 + (random.nextDouble() * 3).round(),
),
remove: () => removeCircle(i),
);
},
);
final subscription = circleStream.listen(addCircle);
return subscription.cancel;
},
[addCircle, removeCircle, color1, color2, avgCircleRadius],
);
return LayoutBuilder(
builder: (context, constraints) {
return Stack(
children: [
for (final circle in getCircles())
Positioned(
key: ValueKey(circle.id),
left: (circle.centerX - circle.radius) * constraints.maxWidth,
top: (circle.centerY - circle.radius) * constraints.maxHeight,
child: AnimatedSplashCircle(
color: circle.color,
radius: circle.radius * constraints.maxHeight,
appear: circle.appear,
disappear: circle.disappear,
remove: circle.remove,
),
),
BackdropFilter(
filter: ImageFilter.blur(sigmaX: 24, sigmaY: 24),
child: const SizedBox.expand(),
),
],
);
},
);
}
}
/// Represents the mathematical properties of an [AnimatedSplashCircle]
/// in a [DynamicBackground].
typedef SplashCircleProperties = ({
int id,
double centerX,
double centerY,
Color color,
double radius,
Duration appear,
Duration disappear,
void Function() remove,
});
/// {@template AnimatedSplashCircle}
/// An individual animated circle in the [DynamicBackground].
/// {@endtemplate}
class AnimatedSplashCircle extends RearchConsumer {
/// {@macro AnimatedSplashCircle}
const AnimatedSplashCircle({
required this.color,
required this.radius,
required this.appear,
required this.disappear,
required this.remove,
super.key,
});
/// The color of this circle.
final Color color;
/// The radius of this circle.
final double radius;
/// The time it takes for the circle to fully appear.
final Duration appear;
/// The time it takes for the circle to fully disappear.
final Duration disappear;
/// A callback that removes this circle from the [DynamicBackground]
/// for when its animations complete.
final void Function() remove;
@override
Widget build(BuildContext context, WidgetHandle use) {
final controller = use.animationController(
duration: appear,
reverseDuration: disappear,
);
use.effect(
() {
controller.forward();
return null;
},
[controller],
);
final animation = use.memo(
() {
return CurvedAnimation(
parent: controller,
curve: Curves.easeInOutQuint,
reverseCurve: Curves.linear,
);
},
[controller],
);
use.effect(
() {
void statusListener(AnimationStatus status) {
switch (status) {
case AnimationStatus.completed:
controller.reverse();
case AnimationStatus.dismissed:
remove();
case _:
break;
}
}
animation.addStatusListener(statusListener);
return animation.dispose;
},
[controller, remove, animation],
);
return ScaleTransition(
scale: animation,
child: CircleAvatar(
backgroundColor: color,
radius: radius,
),
);
}
}
/// Shows a dialog that enables users to create todos.
Future<void> showCreateTodoDialog(
BuildContext context,
void Function(Todo) todoCreator,
) {
var (title, description) = ('', '');
return showDialog(
context: context,
builder: (context) {
return AlertDialog(
icon: const Icon(Icons.edit_rounded),
title: const Text('Create Todo'),
content: Column(
mainAxisSize: MainAxisSize.min,
children: [
TextField(
onChanged: (newTitle) => title = newTitle,
decoration: const InputDecoration(labelText: 'Title'),
),
TextField(
onChanged: (newDescription) => description = newDescription,
decoration: const InputDecoration(labelText: 'Description'),
),
],
),
actions: [
TextButton(
onPressed: Navigator.of(context).pop,
child: const Text('Cancel'),
),
TextButton(
onPressed: () {
todoCreator(
(
title: title,
description: description == '' ? null : description,
completed: false,
timestamp: DateTime.now().millisecondsSinceEpoch,
),
);
Navigator.of(context).pop();
},
child: const Text('Save'),
),
],
);
},
);
}
/// Shows a deletion confirmation dialog.
Future<void> showDeletionConfirmationDialog(
BuildContext context,
void Function() delete,
) {
return showDialog(
context: context,
builder: (context) {
return AlertDialog(
icon: const Icon(Icons.delete_rounded),
title: const Text('Delete Todo'),
content: const Text('Are you sure you want to delete this todo?'),
actions: [
TextButton(
onPressed: Navigator.of(context).pop,
child: const Text('Cancel'),
),
TextButton(
onPressed: () {
delete();
Navigator.of(context).pop();
},
child: const Text('Delete'),
),
],
);
},
);
}