package scenegraphdemo;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;
import javafx.beans.property.ObjectProperty;
import javafx.beans.property.ObjectPropertyBase;
import javafx.beans.property.Property;
import javafx.beans.property.StringProperty;
import javafx.beans.value.ChangeListener;
import javafx.beans.value.ObservableValue;
import javafx.beans.value.WritableValue;
/**
* An object binding object that creates object bindings based on property names
* specified as strings. This is a convenience class to avoid having observable
* values directly dependent on the underlying bean. The actual bean the
* properties connect can be changed dynamically using the channel or
* setBean
. This allows you to setup binding when you don't have
* the bean to bind to or the bean to bind to will change as in the detail part
* of a master-detail screen. Generally, the return values from
* getProperty()
should be used for binding not the factory itself.
* The object should be a domain object bean with bean methods to get or set the
* value using java bean naming conventions OR it should use javafx property
* definition conventions. The factory will automatically listen for standard
* java bean changes using PropertyChangeListener
approaches unless
* you indicate to not to.
*
*
* When the bean itself changes, the properties fire to indicate that their
* values may have changed and the observing object should update itself. This
* object allows you to integrate your javafx/POJOs into the binding
* infrastructure without being dependent on the actual underlying bean. If you
* are familiar with jgoodies BindingAdapter
this is a similar
* class.
*
*
* This only handles reading bean properties. Need to add set() logic. * *
* TODO: Make this work better. Many corner cases to cover. *
* TODO: Instead of just method calls on POJOs, also check for PCL. Note that
* javafx properties won't work easily they are always tied to the underlying
* object so I would have to write another CaptiveObjectProperty object that is
* essentially a delegator.
*/
public class BeanPropertyFactory
* The difference between this and BeanPropertyListiner
to include the default
* behavior.
*
* @return
*/
protected PropertyChangeListener createBeanPropertyListener() {
return new BeanPropertyListener();
}
/**
* Add a listener only if the PCL methods exist. This listens for property
* changes on the bean's property not changes in the actually bean held by
* the bean channel.
*/
protected void addBeanPCLListener(Object obj) {
if(!isListenForBeanPCLEvents())
return;
if (obj != null) {
if (BeanPropertyUtils.hasPCL(obj.getClass())) {
if (beanPropertyListener == null)
beanPropertyListener = createBeanPropertyListener();
BeanPropertyUtils.addPCL(obj, beanPropertyListener, null);
}
}
}
/**
* Remove a listener only if the PCL methods exist.
*
* @see #attachBeanPCLListener
*/
protected void removeBeanPCLListener(Object obj) {
if (obj != null) {
if (BeanPropertyUtils.hasPCL(obj.getClass())) {
if (beanPropertyListener == null)
beanPropertyListener = createBeanPropertyListener();
BeanPropertyUtils.removePCL(obj, beanPropertyListener, null);
}
}
}
/**
* Invalidate the properties in the property cache. Then changed the bean in
* the bean channel. Then fire a value changed event.
*
* @param bean
*/
public void setBean(T bean) {
invalidateProperties();
if (getChannel() instanceof WritableValue) {
((WritableValue) getChannel()).setValue(bean);
} else {
throw new IllegalArgumentException(
"Could not set bean value into a non-writable bean channel");
}
fireValueChangedEvent();
}
/**
* Called to indicate that the underlying bean changed.
*/
protected void invalidateProperties() {
for (CaptiveObjectProperty p : properties.values()) {
p.invalidate();
}
}
public T getBean() {
return getChannel().getValue();
}
/**
* Lazily get the method representing the property.
*
* @author Mr. Java
*
* @param getBean()
being changed.
*/
public void invalidate() {
m = null;
fireValueChangedEvent();
}
/**
* This is used to externally signal that the property has changed. It
* is quite possible that this object does not detect those changes and
* hence, an external object (in all cases the BeanPropertyFactory) will
* signal when a change occurs. Property change detection is centralized
* in the factory for efficiency reasons.
*/
public void propertyChanged() {
fireValueChangedEvent();
}
protected Property getJavaFXPropety(String propertyName) {
if (factory.getBean() == null)
return null;
String methodName = getName() + "Property";
try {
Method mtmp = factory.getBean().getClass()
.getMethod(methodName, new Class>[0]);
enhancedProperty = (Property) mtmp.invoke(factory.getBean(),
new Object[0]);
return enhancedProperty;
} catch (Exception e) {
// e.printStackTrace();
// silently fail here
}
return null;
}
/**
* Sets the method, either an enhanced property or the POJO method via
* reflection.
*
* @return
*/
protected void getMethod() {
if (factory == null || factory.getBean() == null)
return;
enhancedProperty = getJavaFXPropety(getName());
if (enhancedProperty != null)
return;
// Look for standard pojo method.
m = getPOJOReadMethod(getName());
}
protected Method getPOJOReadMethod(String propertyName) {
if (factory.getBean() == null)
return null;
// Look for standard pojo method.
String methodName = "get"
+ Character.toUpperCase(getName().charAt(0));
if (getName().length() > 1)
methodName += getName().substring(1);
try {
Method mtmp = factory.getBean().getClass()
.getMethod(methodName, new Class>[0]);
return mtmp;
} catch (Exception e) {
// silently fail here
// e.printStackTrace();
}
return null;
}
}
/**
* Obtain a property using the binding path. The binding path should be
* simple property names separated by a dot. The bean has to specified
* because the return value is a property that can be used directly for
* binding. The first property specified in the binding path should be a
* property on the bean.
*
* Bindings.select()
is not
* much other that it uses the bean factory machinery and the path can be
* specified as a string. Of course, the bean can be a plain pojo.
*
* @param bindingPath
* @return
*/
public static Property propertyFromBindingPath(Object bean,
String bindingPath) {
if (bindingPath == null || bindingPath.isEmpty())
return null;
BeanPropertyFactory lastFactory = null;
Property lastProperty = null;
String[] parts = bindingPath.split("\\.");
if (parts.length > 0) {
for (int i = 0; i < parts.length; i++) {
if (parts[i].length() <= 0 || parts[i].isEmpty()) {
throw new IllegalArgumentException("Binding path part " + i
+ " has no length");
}
BeanPropertyFactory newFactory;
if (i == 0)
newFactory = new BeanPropertyFactory(bean);
else
newFactory = new BeanPropertyFactory(
(WritableValue) lastProperty);
lastProperty = newFactory.getProperty(parts[i].trim());
lastFactory = newFactory;
}
}
return lastProperty;
}
/**
* Alot like Bindings.select
but also handles pojos.
*
* @param bean
* @param path
* @return
*/
public static Property propertyFromBindingPath(Object bean, String... path) {
String tmp = "";
for (int i = 0; i < path.length; i++) {
tmp += path[i];
if (i <= path.length - 1)
tmp += ".";
}
return propertyFromBindingPath(bean, tmp);
}
}