I have an application that manages (bank) accounts. In my data model, I have defined an observable list for the accounts:
private final ObservableList<Account> accounts = observableArrayList();
Each account has a list of cashflows, which is implemented via a property (also being an observable list):
// in Account class:
private final SimpleListProperty<Cashflow> cashflows = new SimpleListProperty<>(observableArrayList());
In my UI, I have a table containing all accounts, and I am using the cashflow list property to show the number of cashflows for each account, which works fine.
The accounts table also provides checkboxes to select or unselect specific accounts. There's a property for this in my Account class as well:
// in Account class:
private final SimpleBooleanProperty selected = new SimpleBooleanProperty();
Now I want to add another table to the UI, which contains the cashflows, but only for the selected accounts, and preferably I want to implement this via data binding.
But I don't know how to achieve this. I quickly dismissed the idea of using directly the cashflows property of the Account class in some way, because I wouldn't even know where to start here.
So what I tried is define a separate observable list for the cashflows in my data model:
private final ObservableList<Cashflow> cashflowsOfSelectedAccounts = observableArrayList();
I know that I can define extractors for the account list that will notify observers when something changes. So for example, I could extend my account list to something like:
private final ObservableList<Account> accounts = observableArrayList(
account -> new Observable[]{
account.selectedProperty(),
account.cashflowsProperty().sizeProperty()});
This would trigger a notification to a listener on the accounts list on any of the following:
- an account is added or removed
- a cashflow is added to or removed from an account
- an account gets selected or unselected
But now I don't know how I can bring this together with my observable cashflow list, because I have two different data types here: Account, and Cashflow.
The only solution I can think of is to add a custom listener to the account list to react to all of the relevant events listed above, and maintain the cashflowsOfSelectedAccounts manually.
So here's my Question: Is it possible to sync the accounts list with the list of cashflows of selected accounts via data binding, or via some other way that I'm not aware of and that would be more elegant than manually maintaining the cashflow list with a custom listener on the accounts list?
Thanks!
Personally, I wouldn't try to overcomplicate the bindings too much beyond simple applications of built-in support for the high-level binding APIs. Once you add a few bindings things get complicated enough already.
Alternative 1
What I suggest you do is:
Create a filtered list of selected accounts.
Use the filtered list of selected accounts as the backing list for the second table.
As the second table is only to display cashflow data and not full account data, for column data, provide custom value factories to access the cashflow data in the account.
Making the second table a TreeTableView may make sense, that way it can group the cashflows by account.
This may or may not be a valid approach for your app.
Alternative 2
Alternately, also working off the filteredlist of accounts, add a list change listener to the filtered list, when it changes, update the content of a separate list of related cashflows which you use as the backing list for the cashflow table.
Handling your use cases.
Just add or remove from the account list.
An extractor on the account list and a list listener can be triggered when associated cashflows change to trigger an update to the cashflow list.
See the linked filtered list example, it is based on an extractor combined with a filtered list.
Alternative 3
Alternately, you could change your UI. For example, have separate edit and commit pages for cashflow data and account data with user button presses for committing or discarding changes. The commits update the backing database. Then, after committing, navigate back to the original page which just reads the new data from the source database again. That's generally how these things often work rather than a bunch of binding.
I realize that none of these options are what you are asking about and some of them probably do work you were trying to avoid through a different binding type, but, those are the ideas I came up with.
Example
FWIW, here is an example of Alternative 2, which relies on an account list, a filtered account list a separate cashflow list, and extractors and listeners to keep stuff in sync.
It won't be exactly what you want, but perhaps you can adapt it or learn something from it.
I would note that the cashflow list doesn't tie a given cashflow to a given account, so, if you want to do that, you might want to add additional functionality to support visual feedback for that association.
Initial state:
Select only a single account:
Remove an account and change the cashflow data for a given account: