elbe 0.1.40 elbe: ^0.1.40 copied to clipboard
a toolkit for simplifying frontend development
Elbe is a Flutter UI toolkit and collection of tools that runs on all platforms.
Description #
Elbe is designed to provide a collection of reusable UI components and utilities for building beautiful and responsive Flutter applications. It is based on Flutter, a popular cross-platform framework for building mobile, web, and desktop applications.
an online demo/documentation can be accessed here: DEMO
Features #
- Reusable UI components (minimalist and customizable)
- Responsive design
- Cross-platform compatibility
- state management
bit
- theming
- routing
- logging
- Open source and free to use
Installation #
To use Elbe in your Flutter project, follow these steps:
-
Open your project's
pubspec.yaml
file. -
Add the following line to the
dependencies
section:dependencies: elbe: ^<latest version>
-
Run
flutter pub get
to fetch the package.
Usage #
Import the Elbe package in your Flutter project
1. define a router
final router = GoRouter(
routes: [
GoRoute(
path: '/',
builder: (context, state) => YourHomePage();
)]);
2. define the app
define and call your app in your main.dart
file
import 'package:elbe/elbe.dart';
void main() async => runApp(const MyApp());
class YourApp extends StatelessWidget {
const YourApp({super.key});
@override
Widget build(BuildContext context) =>
ElbeApp(
router: router,
theme: ThemeData.preset(color: Colors.blue));
}
you are now ready to build your app with elbe
guide #
state management (bit) #
elbe defines its own state management system. This is partly based on the popular bloc
library, but aims to limit the amount of boilerplate code needed.
how it works #
- a
bit
can be defined by the user- it includes both:
- a value
- the logic that can be carried out on this state (
BitControl
)
- the bit is injected into the
BuildContext
and can be accessed within the subtree - the bit will update its dependent children, once its value changes
- the value of a bit can be in one of three states:
loading
: this signifies that no data is currently availableerror
: the logic ran into an error that requires user interaction- examples would be: connection lost, access denied
data
: the value of the bit reflects usable data.- NOTE: In contrast to the
bloc
library, differentiating between different kinds of data has to be done outside of the core state management- the easiest option is to include flags within the value
- NOTE: In contrast to the
- it includes both:
usage #
1. defining a bit
note that there are different types of BitControl
s. These allow for more complex behaviours. For the sake of simplicity, we will stick to the basic MapMsgBitControl
.
use the worker
to carry out complex (async) operations. The bit will be initiated in the loading
state, and the worker is called.
class CounterBit extends MapMsgBitControl<int> {
static const builder = MapMsgBitBuilder<int, CounterBit>.make;
CounterBit({int? initial})
: super.worker((_) async => initial ?? 0);
/// add one to the current state. this internally calles init
/// and thus updates the UI
addOne() =>
state.whenOrNull(
onError: (_) => this.emit(0) //reset on error,
onData: (v) => this.emit(v + 1));
}
2. injecting bit
once you have defined your bit, you need to integrate it into the build tree.
Do this by using the BitProvider
widget within the tree
BitProvider(
create: (_) => CounterBit(initial: 42),
child: ...)
3. using the bit
now you are ready to use your bit (within the subtree) 🎉. The builder
function of the bit will return a Widget that automatically updates your UI when the value of the bit
changes
ConfigBit.builder(
onLoading: (bit, loading) => CircularProgressIndicator.adaptive(),
onError: (bit, error) => Text("Error: $error"),
onData: (bit, value) =>
Button.minor(
label: "current: $value",
onTap: () => bit.addOne()
)
),