Package sprouts
It was designed in conjunction with the SwingTree GUI Framework, which has native support for binding to the Sprouts properties.
We recommended using Sprouts as a tool for bridging the gap between
place oriented programming and data oriented programming,
which is especially useful in GUI programming, where data oriented / functional
programming principles are hard to apply due to the inherently place oriented
nature of GUI components.
Sprouts allows you to create lens based properties that make value object based
view models interoperate with your GUI components or other kinds of views.
This is best achieved using the MVI (Model-View-Intent) and MVL (Model-View-Lens) patterns.
Here an example demonstrating how to create
a set of bindable properties from a basic record based
modelling scenario:
View Models:
public record Address(
String street, String city, int postalCode
) {
public Address withStreet(String street) {
return new Address(street, city, postalCode);
}
public Address withCity(String city) {
return new Address(street, city, postalCode);
}
public Address withPostalCode(int postalCode) {
return new Address(street, city, postalCode);
}
}
public record Person(
String forename, String surname, Address address
) {
public Person withForename(String forename) {
return new Person(forename, surname, address);
}
public Person withSurname(String surname) {
return new Person(forename, surname, address);
}
public Person withAddress(Address address) {
return new Person(forename, surname, address);
}
}
View:
public final class PersonView {
public static void main(String[] args) {
Person person = new Person("John", "Doe", new Address("Main Street", "Springfield", 12345));
Var<Person> applicationState = Var.of(person);
var view = new PersonView(applicationState);
// show view
}
public PersonView(Var<Person> vm) {
Var<String> forename = vm.zoomTo(Person::forename, Person::withForename);
Var<String> surname = vm.zoomTo(Person::surname, Person::withSurname);
Val<String> fullName = vm.viewAsString( person -> person.forename() + " " + person.surname() );
Var<Address> address = vm.zoomTo(Person::address, Person::withAddress);
Var<String> street = address.zoomTo(Address::street, Address::withStreet);
Var<String> city = address.zoomTo(Address::city, Address::withCity);
Var<Integer> postalCode = address.zoomTo(Address::postalCode, Address::withPostalCode);
// Use mutable properties in the views GUI...
}
}
For more usage information check out this page,
where you can browse the living documentation of the Sprouts API.
Also check out the SwingTree GUI Framework for more example
code using the Sprouts properties natively.
-
ClassDescriptionAction<D>A functional interface for observing state changes and performing some action in response.Association<K,
V> Defines an association between keys and values as an immutable value object where the values are accessed by the keys in a map-like fashion.This is a marker interface for property event channels.An event is something that can be triggered and reacted throughObservable
s created by theEvent.observable()
method.HasId<I>This is a sort of marker interface forVar
/Val
property item types, which tells the property that it wants to use the object returned by theHasId.id()
method as a unique identifier for its item when determining theValDelegate.change()
type.TheLens
interface defines an access and update operation on an individual part of a nested and immutable data structure.Maybe<T>A checked exception thrown when an item is missing from aMaybe
object.This represents an event that can be observed but not triggered.An immutable value object representing a pair of two generic values.A problem is a wrapper for information describing an issue that can be reported and attached to aResult
.Result<V>Describes how aVars
instance was mutated.A set of constants that describe how a property item has changed.An immutable collection of ordered items of the same typeT
(seeTuple.type()
), whose items can be iterated over and accessed through their indices.
This class can be thought of as an immutable array with an API designed for functional programming and robust handling ofnull
values.
A read only wrapper around an item which can be mapped to a weakly referencedViewable
to then be observed for changes usingAction
s registered through theViewable.onChange(Channel, Action)
method, where theChannel
is used to distinguish between changes from different sources (usually application layers like the view model or the view).
UseVal.view()
to access a simple no-op live view of the item of this property and register change listeners on it to react to state changes.ValDelegate<T>A read only API of a list of read-only viewed properties that can be observed, iterated over, mapped, filtered, turned into a stream, and more.ValsDelegate<T>A mutable wrapper for an item which can also be mapped to a weakly referencedViewable
to be observed for changes usingAction
s registered through theViewable.onChange(Channel, Action)
method, where theChannel
is used to distinguish between changes from different sources (usually application layers like the view model or the view).
UseVal.view()
to access a simple no-op live view of the item of this property and register change listeners on it to react to state changes.A mutable list of mutable properties designed for MVVM, that can be observed for changes through aVals.view()
by registeringSubscriber
types on it likeObserver
s andAction
s.This is a value object representing a unique ID consisting of two numbers, a lineage and succession, allowing you to identify something and also determine the order in which something was created and updated.Viewable<T>A read only live view on a delegated item which can be observed for changes usingAction
s registered through theViewable.onChange(Channel, Action)
method, where theChannel
is used to distinguish between changes from different sources (usually application layers like the view model or the view).Viewables<T>A read-only live view on a delegated list of items which can be observed for changes usingAction
s registered through theViewables.onChange(Action)
method.WeakAction<O,D> A weak action is an extension of theAction
interface which defines a method for clearing the action and a method for retrieving the owner of the action.WeakObserver<O>A weak observer is an extension of theObserver
interface which defines a method for clearing its state (setting it to null) and a method for retrieving the owner of the observer.