lens4d
A functional lens library for Dart, enabling type-safe, composable data access and modification.
Overview
Lenses are a functional programming pattern that provide a clean, type-safe way to focus on, extract from, and modify deeply nested data structures. Originally from Haskell and popularized in languages like Scala and Kotlin, lenses solve the problem of working with immutable data in an elegant, composable way.
This library is inspired by and adapted from the lens system in http4k Kotlin library, bringing the power of functional lenses to Dart.
Why Lenses?
Working with immutable data often leads to verbose, error-prone code:
// Without lenses - verbose and fragile
final newUri = uri.replace(
queryParameters: {
...uri.queryParameters,
'search': 'flutter',
'limit': '20',
}
);
With lenses, the same operation becomes clean and composable:
// With lenses - clean and type-safe
final newUri = uri.havingAll([
Query.required('search').of('flutter'),
Query.integer().required('limit').of(20),
]);
Features
- ✅ Type-safe data access - Extract and modify values with compile-time type safety
- ✅ Functional composition - Chain operations with
having()andhavingAll() - ✅ Bidirectional lenses - Both read from and write to data structures
- ✅ Type conversions - Built-in conversions for integers, booleans, dates, and more
- ✅ Error handling - Comprehensive failure handling with detailed error information
- ✅ Zero dependencies - Pure Dart implementation
Quick Start
Installation
Add lens4d to your pubspec.yaml:
dependencies:
lens4d: ^1.0.0
Basic Usage
Here's a practical example working with URI query parameters:
import 'package:lens4d/lens4d.dart';
/// Define a Query lens for Uri objects
final Query = BiDiLensSpec<Uri, String>(
'query',
const StringParam(),
LensGet<Uri, String>((name, uri) {
final value = uri.queryParameters[name];
return value != null ? [value] : [];
}),
LensSet<Uri, String>((name, values, uri) {
final newParams = Map<String, String>.from(uri.queryParameters);
if (values.isEmpty) {
newParams.remove(name);
} else {
newParams[name] = values.first;
}
return uri.replace(queryParameters: newParams);
}),
);
void main() {
final uri = Uri.parse('https://api.example.com/search?q=dart&limit=10');
// Extract values with type safety
final searchTerm = Query.required('q')(uri); // 'dart'
final limit = Query.integer().required('limit')(uri); // 10 (as int)
final page = Query.integer().optional('page')(uri); // null
// Single modification
final newUri = uri.having(Query.required('q').of('flutter'));
// Multiple modifications
final complexUri = uri.havingAll([
Query.required('search').of('dart programming'),
Query.integer().required('limit').of(50),
Query.boolean().required('debug').of(true),
]);
// Error handling
try {
Query.required('missing')(uri);
} on LensFailure catch (e) {
print('Parameter not found: ${e.failures.first}');
}
}
Type Conversions
Lenses support automatic type conversions, which are extensible via extension methods:
// String conversions
Query.required('name') // String (default)
Query.string().required('name') // String (explicit)
Query.nonEmptyString().required('title') // Non-empty string
// Numeric conversions
Query.integer().required('count') // int
Query.decimal().required('price') // double
// Other conversions
Query.boolean().required('enabled') // bool
Query.dateTime().required('created') // DateTime
Functional Composition
The having() and havingAll() extension methods enable clean functional composition:
// Chain single modifications
final result = data
.having(someLens.of(value1))
.having(anotherLens.of(value2));
// Apply multiple modifications at once
final result = data.havingAll([
lens1.of(value1),
lens2.of(value2),
lens3.of(value3),
]);
API Overview
BiDiLensSpec<IN, OUT>- Bidirectional lens specification for extraction and injectionLensSpec<IN, OUT>- Unidirectional lens for extraction onlyBiDiLens<IN, OUT>- Concrete bidirectional lens instanceLens<IN, OUT>- Concrete unidirectional lens instance
Core Methods
required(name)- Extract required value, throw on missingoptional(name)- Extract optional value, return null if missingdefaulted(name, defaultValue)- Extract with fallback valueinject(value, target)- Inject value into targetof(value)- Create injection function for use withhaving()
Extension Methods
having(modifier)- Apply single lens modifierhavingAll(modifiers)- Apply multiple lens modifiers- Type conversions:
integer(),boolean(),dateTime(), etc.
Contributing
Contributions are welcome! Please feel free to submit a PR.
License
This project is licensed under the Apache2 License - see the LICENSE file for details.
Acknowledgments
- Inspired by the http4k Kotlin library
- Based on functional programming patterns from Haskell and other FP languages
- Thanks to the Dart community for feedback and contributions