有ListChangeListener.Change.wasUpdated()
用于检测通过ObservableList.observableArrayList(Callback<E,Observable[]> extractor)
创建的UserId
元素内的更改。
如何检索导致更改被触发的确切项目?
也许这个问题不够清楚,我会给出一个例子。
ObservableList
class Foo {
private StringProperty name = new SimpleStringProperty();
public final StringProperty nameProperty() { return name; }
public final String getName() { return name.get(); }
public final void setName(String n) { name.set(n); }
public Foo(String fooName) { setName(fooName); }
}
// Creates an ObservableList with an extractor
ObservableList<Foo> fooList = FXCollections.observableArrayList(foo -> new Observable[] { foo.nameProperty() });
Foo fooA = new Foo("Hello");
Foo fooB = new Foo("Kitty");
fooList.add(fooA);
fooList.add(fooB);
fooList.addListener(new ListChangeListener<Foo>() {
public void onChanged(Change<Foo> c) {
while (c.next()) {
if (c.wasUpdated()) {
// One or more of the elements in list has/have an internal change, but I have no idea which element(s)!
}
}
}
});
fooB.setName("Mickey");
会触发fooB.setName()
的更改,ListChangeListener
条件会返回wasUpdated()
。但是,我无法知道在监听器中发生了变化true
。
这看似微不足道,但我有一个地图应用程序,其中列表存储地图必须渲染的内容。当其中一个项目改变其位置(即纬度/经度)时,我需要在地图上重新绘制。如果我不知道哪个项目改变了位置,我将不得不重新绘制我已经拥有的所有内容。
答案 0 :(得分:4)
您可以获取已更改的项目的索引,从而为您提供一些有用的信息:
import javafx.beans.Observable;
import javafx.beans.property.SimpleStringProperty;
import javafx.beans.property.StringProperty;
import javafx.collections.FXCollections;
import javafx.collections.ListChangeListener.Change;
import javafx.collections.ObservableList;
public class ListUpdateTest {
public static void main(String[] args) {
ObservableList<Foo> fooList = FXCollections.observableArrayList(foo -> new Observable[] { foo.nameProperty() });
Foo fooA = new Foo("Hello");
Foo fooB = new Foo("Kitty");
fooList.add(fooA);
fooList.add(fooB);
fooList.addListener((Change<? extends Foo> c) -> {
while (c.next()) {
if (c.wasUpdated()) {
int start = c.getFrom() ;
int end = c.getTo() ;
for (int i = start ; i < end ; i++) {
System.out.println("Element at position "+i+" was updated to: " +c.getList().get(i).getName() );
}
}
}
});
fooB.setName("Mickey");
}
public static class Foo {
private StringProperty name = new SimpleStringProperty();
public final StringProperty nameProperty() { return name; }
public final String getName() { return name.get(); }
public final void setName(String n) { name.set(n); }
public Foo(String fooName) { setName(fooName); }
}
}
请注意,您无法从列表更改事件中确定在这些列表元素中更改的实际属性(因此,如果提取器指向两个或更多属性,则无法找到哪些属性已更改),以及没有办法获得以前的价值。不过,这对您的用例来说已经足够了。
答案 1 :(得分:1)
直接来自docs。
典型用法是观察ObservableList上的更改,以便挂钩或取消挂钩(或添加或删除侦听器),或者为了在ObservableList中的每个元素上保持一些不变量。执行此操作的常见代码模式如下所示:
ObservableList theList = ...;
theList.addListener(new ListChangeListener<Item>() {
public void onChanged(Change<tem> c) {
while (c.next()) {
if (c.wasPermutated()) {
for (int i = c.getFrom(); i < c.getTo(); ++i) {
//permutate
}
} else if (c.wasUpdated()) {
//update item
} else {
for (Item remitem : c.getRemoved()) {
remitem.remove(Outer.this);
}
for (Item additem : c.getAddedSubList()) {
additem.add(Outer.this);
}
}
}
}
});
}