/* * Copyright (c) 2011, 2014, Oracle and/or its affiliates. All rights reserved. * ORACLE PROPRIETARY/CONFIDENTIAL. Use is subject to license terms. * * * * * * * * * * * * * * * * * * * * */ package com.sun.javafx.binding; import javafx.beans.Observable; import javafx.beans.WeakListener; import javafx.beans.property.*; import javafx.beans.value.ChangeListener; import javafx.beans.value.ObservableValue; import javafx.util.StringConverter; import java.lang.ref.WeakReference; import java.text.Format; import java.text.ParseException; public abstract class BidirectionalBinding implements ChangeListener, WeakListener { private static void checkParameters(Object property1, Object property2) { if ((property1 == null) || (property2 == null)) { throw new NullPointerException("Both properties must be specified."); } if (property1 == property2) { throw new IllegalArgumentException("Cannot bind property to itself"); } } public static BidirectionalBinding bind(Property property1, Property property2) { checkParameters(property1, property2); final BidirectionalBinding binding = ((property1 instanceof DoubleProperty) && (property2 instanceof DoubleProperty)) ? new BidirectionalDoubleBinding((DoubleProperty) property1, (DoubleProperty) property2) : ((property1 instanceof FloatProperty) && (property2 instanceof FloatProperty)) ? new BidirectionalFloatBinding((FloatProperty) property1, (FloatProperty) property2) : ((property1 instanceof IntegerProperty) && (property2 instanceof IntegerProperty)) ? new BidirectionalIntegerBinding((IntegerProperty) property1, (IntegerProperty) property2) : ((property1 instanceof LongProperty) && (property2 instanceof LongProperty)) ? new BidirectionalLongBinding((LongProperty) property1, (LongProperty) property2) : ((property1 instanceof BooleanProperty) && (property2 instanceof BooleanProperty)) ? new BidirectionalBooleanBinding((BooleanProperty) property1, (BooleanProperty) property2) : new TypedGenericBidirectionalBinding(property1, property2); property1.setValue(property2.getValue()); property1.addListener(binding); property2.addListener(binding); return binding; } public static Object bind(Property stringProperty, Property otherProperty, Format format) { checkParameters(stringProperty, otherProperty); if (format == null) { throw new NullPointerException("Format cannot be null"); } final StringConversionBidirectionalBinding binding = new StringFormatBidirectionalBinding(stringProperty, otherProperty, format); stringProperty.setValue(format.format(otherProperty.getValue())); stringProperty.addListener(binding); otherProperty.addListener(binding); return binding; } public static Object bind(Property stringProperty, Property otherProperty, StringConverter converter) { checkParameters(stringProperty, otherProperty); if (converter == null) { throw new NullPointerException("Converter cannot be null"); } final StringConversionBidirectionalBinding binding = new StringConverterBidirectionalBinding(stringProperty, otherProperty, converter); stringProperty.setValue(converter.toString(otherProperty.getValue())); stringProperty.addListener(binding); otherProperty.addListener(binding); return binding; } public static void unbind(Property property1, Property property2) { checkParameters(property1, property2); final BidirectionalBinding binding = new UntypedGenericBidirectionalBinding(property1, property2); property1.removeListener(binding); property2.removeListener(binding); } public static void unbind(Object property1, Object property2) { checkParameters(property1, property2); final BidirectionalBinding binding = new UntypedGenericBidirectionalBinding(property1, property2); if (property1 instanceof ObservableValue) { ((ObservableValue) property1).removeListener(binding); } if (property2 instanceof Observable) { ((ObservableValue) property2).removeListener(binding); } } public static BidirectionalBinding bindNumber(Property property1, IntegerProperty property2) { return bindNumber(property1, (Property)property2); } public static BidirectionalBinding bindNumber(Property property1, LongProperty property2) { return bindNumber(property1, (Property)property2); } public static BidirectionalBinding bindNumber(Property property1, FloatProperty property2) { return bindNumber(property1, (Property)property2); } public static BidirectionalBinding bindNumber(Property property1, DoubleProperty property2) { return bindNumber(property1, (Property)property2); } public static BidirectionalBinding bindNumber(IntegerProperty property1, Property property2) { return bindNumberObject(property1, property2); } public static BidirectionalBinding bindNumber(LongProperty property1, Property property2) { return bindNumberObject(property1, property2); } public static BidirectionalBinding bindNumber(FloatProperty property1, Property property2) { return bindNumberObject(property1, property2); } public static BidirectionalBinding bindNumber(DoubleProperty property1, Property property2) { return bindNumberObject(property1, property2); } private static BidirectionalBinding bindNumberObject(Property property1, Property property2) { checkParameters(property1, property2); final BidirectionalBinding binding = new TypedNumberBidirectionalBinding(property2, property1); property1.setValue(property2.getValue()); property1.addListener(binding); property2.addListener(binding); return binding; } private static BidirectionalBinding bindNumber(Property property1, Property property2) { checkParameters(property1, property2); final BidirectionalBinding binding = new TypedNumberBidirectionalBinding(property1, property2); property1.setValue((T)property2.getValue()); property1.addListener(binding); property2.addListener(binding); return binding; } public static void unbindNumber(Property property1, Property property2) { checkParameters(property1, property2); final BidirectionalBinding binding = new UntypedGenericBidirectionalBinding(property1, property2); if (property1 instanceof ObservableValue) { ((ObservableValue) property1).removeListener(binding); } if (property2 instanceof Observable) { ((ObservableValue) property2).removeListener(binding); } } private final int cachedHashCode; private BidirectionalBinding(Object property1, Object property2) { cachedHashCode = property1.hashCode() * property2.hashCode(); } protected abstract Object getProperty1(); protected abstract Object getProperty2(); @Override public int hashCode() { return cachedHashCode; } @Override public boolean wasGarbageCollected() { return (getProperty1() == null) || (getProperty2() == null); } @Override public boolean equals(Object obj) { if (this == obj) { return true; } final Object propertyA1 = getProperty1(); final Object propertyA2 = getProperty2(); if ((propertyA1 == null) || (propertyA2 == null)) { return false; } if (obj instanceof BidirectionalBinding) { final BidirectionalBinding otherBinding = (BidirectionalBinding) obj; final Object propertyB1 = otherBinding.getProperty1(); final Object propertyB2 = otherBinding.getProperty2(); if ((propertyB1 == null) || (propertyB2 == null)) { return false; } if (propertyA1 == propertyB1 && propertyA2 == propertyB2) { return true; } if (propertyA1 == propertyB2 && propertyA2 == propertyB1) { return true; } } return false; } private static class BidirectionalBooleanBinding extends BidirectionalBinding { private final WeakReference propertyRef1; private final WeakReference propertyRef2; private boolean updating = false; private BidirectionalBooleanBinding(BooleanProperty property1, BooleanProperty property2) { super(property1, property2); propertyRef1 = new WeakReference(property1); propertyRef2 = new WeakReference(property2); } @Override protected Property getProperty1() { return propertyRef1.get(); } @Override protected Property getProperty2() { return propertyRef2.get(); } @Override public void changed(ObservableValue sourceProperty, Boolean oldValue, Boolean newValue) { if (!updating) { final BooleanProperty property1 = propertyRef1.get(); final BooleanProperty property2 = propertyRef2.get(); if ((property1 == null) || (property2 == null)) { if (property1 != null) { property1.removeListener(this); } if (property2 != null) { property2.removeListener(this); } } else { try { updating = true; if (property1 == sourceProperty) { property2.set(newValue); } else { property1.set(newValue); } } catch (RuntimeException e) { try { if (property1 == sourceProperty) { property1.set(oldValue); } else { property2.set(oldValue); } } catch (Exception e2) { e2.addSuppressed(e); unbind(property1, property2); throw new RuntimeException( "Bidirectional binding failed together with an attempt" + " to restore the source property to the previous value." + " Removing the bidirectional binding from properties " + property1 + " and " + property2, e2); } throw new RuntimeException( "Bidirectional binding failed, setting to the previous value", e); } finally { updating = false; } } } } } private static class BidirectionalDoubleBinding extends BidirectionalBinding { private final WeakReference propertyRef1; private final WeakReference propertyRef2; private boolean updating = false; private BidirectionalDoubleBinding(DoubleProperty property1, DoubleProperty property2) { super(property1, property2); propertyRef1 = new WeakReference(property1); propertyRef2 = new WeakReference(property2); } @Override protected Property getProperty1() { return propertyRef1.get(); } @Override protected Property getProperty2() { return propertyRef2.get(); } @Override public void changed(ObservableValue sourceProperty, Number oldValue, Number newValue) { if (!updating) { final DoubleProperty property1 = propertyRef1.get(); final DoubleProperty property2 = propertyRef2.get(); if ((property1 == null) || (property2 == null)) { if (property1 != null) { property1.removeListener(this); } if (property2 != null) { property2.removeListener(this); } } else { try { updating = true; if (property1 == sourceProperty) { property2.set(newValue.doubleValue()); } else { property1.set(newValue.doubleValue()); } } catch (RuntimeException e) { try { if (property1 == sourceProperty) { property1.set(oldValue.doubleValue()); } else { property2.set(oldValue.doubleValue()); } } catch (Exception e2) { e2.addSuppressed(e); unbind(property1, property2); throw new RuntimeException( "Bidirectional binding failed together with an attempt" + " to restore the source property to the previous value." + " Removing the bidirectional binding from properties " + property1 + " and " + property2, e2); } throw new RuntimeException( "Bidirectional binding failed, setting to the previous value", e); } finally { updating = false; } } } } } private static class BidirectionalFloatBinding extends BidirectionalBinding { private final WeakReference propertyRef1; private final WeakReference propertyRef2; private boolean updating = false; private BidirectionalFloatBinding(FloatProperty property1, FloatProperty property2) { super(property1, property2); propertyRef1 = new WeakReference(property1); propertyRef2 = new WeakReference(property2); } @Override protected Property getProperty1() { return propertyRef1.get(); } @Override protected Property getProperty2() { return propertyRef2.get(); } @Override public void changed(ObservableValue sourceProperty, Number oldValue, Number newValue) { if (!updating) { final FloatProperty property1 = propertyRef1.get(); final FloatProperty property2 = propertyRef2.get(); if ((property1 == null) || (property2 == null)) { if (property1 != null) { property1.removeListener(this); } if (property2 != null) { property2.removeListener(this); } } else { try { updating = true; if (property1 == sourceProperty) { property2.set(newValue.floatValue()); } else { property1.set(newValue.floatValue()); } } catch (RuntimeException e) { try { if (property1 == sourceProperty) { property1.set(oldValue.floatValue()); } else { property2.set(oldValue.floatValue()); } } catch (Exception e2) { e2.addSuppressed(e); unbind(property1, property2); throw new RuntimeException( "Bidirectional binding failed together with an attempt" + " to restore the source property to the previous value." + " Removing the bidirectional binding from properties " + property1 + " and " + property2, e2); } throw new RuntimeException( "Bidirectional binding failed, setting to the previous value", e); } finally { updating = false; } } } } } private static class BidirectionalIntegerBinding extends BidirectionalBinding{ private final WeakReference propertyRef1; private final WeakReference propertyRef2; private boolean updating = false; private BidirectionalIntegerBinding(IntegerProperty property1, IntegerProperty property2) { super(property1, property2); propertyRef1 = new WeakReference(property1); propertyRef2 = new WeakReference(property2); } @Override protected Property getProperty1() { return propertyRef1.get(); } @Override protected Property getProperty2() { return propertyRef2.get(); } @Override public void changed(ObservableValue sourceProperty, Number oldValue, Number newValue) { if (!updating) { final IntegerProperty property1 = propertyRef1.get(); final IntegerProperty property2 = propertyRef2.get(); if ((property1 == null) || (property2 == null)) { if (property1 != null) { property1.removeListener(this); } if (property2 != null) { property2.removeListener(this); } } else { try { updating = true; if (property1 == sourceProperty) { property2.set(newValue.intValue()); } else { property1.set(newValue.intValue()); } } catch (RuntimeException e) { try { if (property1 == sourceProperty) { property1.set(oldValue.intValue()); } else { property2.set(oldValue.intValue()); } } catch (Exception e2) { e2.addSuppressed(e); unbind(property1, property2); throw new RuntimeException( "Bidirectional binding failed together with an attempt" + " to restore the source property to the previous value." + " Removing the bidirectional binding from properties " + property1 + " and " + property2, e2); } throw new RuntimeException( "Bidirectional binding failed, setting to the previous value", e); } finally { updating = false; } } } } } private static class BidirectionalLongBinding extends BidirectionalBinding { private final WeakReference propertyRef1; private final WeakReference propertyRef2; private boolean updating = false; private BidirectionalLongBinding(LongProperty property1, LongProperty property2) { super(property1, property2); propertyRef1 = new WeakReference(property1); propertyRef2 = new WeakReference(property2); } @Override protected Property getProperty1() { return propertyRef1.get(); } @Override protected Property getProperty2() { return propertyRef2.get(); } @Override public void changed(ObservableValue sourceProperty, Number oldValue, Number newValue) { if (!updating) { final LongProperty property1 = propertyRef1.get(); final LongProperty property2 = propertyRef2.get(); if ((property1 == null) || (property2 == null)) { if (property1 != null) { property1.removeListener(this); } if (property2 != null) { property2.removeListener(this); } } else { try { updating = true; if (property1 == sourceProperty) { property2.set(newValue.longValue()); } else { property1.set(newValue.longValue()); } } catch (RuntimeException e) { try { if (property1 == sourceProperty) { property1.set(oldValue.longValue()); } else { property2.set(oldValue.longValue()); } } catch (Exception e2) { e2.addSuppressed(e); unbind(property1, property2); throw new RuntimeException( "Bidirectional binding failed together with an attempt" + " to restore the source property to the previous value." + " Removing the bidirectional binding from properties " + property1 + " and " + property2, e2); } throw new RuntimeException( "Bidirectional binding failed, setting to the previous value", e); } finally { updating = false; } } } } } private static class TypedGenericBidirectionalBinding extends BidirectionalBinding { private final WeakReference> propertyRef1; private final WeakReference> propertyRef2; private boolean updating = false; private TypedGenericBidirectionalBinding(Property property1, Property property2) { super(property1, property2); propertyRef1 = new WeakReference>(property1); propertyRef2 = new WeakReference>(property2); } @Override protected Property getProperty1() { return propertyRef1.get(); } @Override protected Property getProperty2() { return propertyRef2.get(); } @Override public void changed(ObservableValue sourceProperty, T oldValue, T newValue) { if (!updating) { final Property property1 = propertyRef1.get(); final Property property2 = propertyRef2.get(); if ((property1 == null) || (property2 == null)) { if (property1 != null) { property1.removeListener(this); } if (property2 != null) { property2.removeListener(this); } } else { try { updating = true; if (property1 == sourceProperty) { property2.setValue(newValue); } else { property1.setValue(newValue); } } catch (RuntimeException e) { try { if (property1 == sourceProperty) { property1.setValue(oldValue); } else { property2.setValue(oldValue); } } catch (Exception e2) { e2.addSuppressed(e); unbind(property1, property2); throw new RuntimeException( "Bidirectional binding failed together with an attempt" + " to restore the source property to the previous value." + " Removing the bidirectional binding from properties " + property1 + " and " + property2, e2); } throw new RuntimeException( "Bidirectional binding failed, setting to the previous value", e); } finally { updating = false; } } } } } private static class TypedNumberBidirectionalBinding extends BidirectionalBinding { private final WeakReference> propertyRef1; private final WeakReference> propertyRef2; private boolean updating = false; private TypedNumberBidirectionalBinding(Property property1, Property property2) { super(property1, property2); propertyRef1 = new WeakReference>(property1); propertyRef2 = new WeakReference>(property2); } @Override protected Property getProperty1() { return propertyRef1.get(); } @Override protected Property getProperty2() { return propertyRef2.get(); } @Override public void changed(ObservableValue sourceProperty, Number oldValue, Number newValue) { if (!updating) { final Property property1 = propertyRef1.get(); final Property property2 = propertyRef2.get(); if ((property1 == null) || (property2 == null)) { if (property1 != null) { property1.removeListener(this); } if (property2 != null) { property2.removeListener(this); } } else { try { updating = true; if (property1 == sourceProperty) { property2.setValue(newValue); } else { property1.setValue((T)newValue); } } catch (RuntimeException e) { try { if (property1 == sourceProperty) { property1.setValue((T)oldValue); } else { property2.setValue(oldValue); } } catch (Exception e2) { e2.addSuppressed(e); unbind(property1, property2); throw new RuntimeException( "Bidirectional binding failed together with an attempt" + " to restore the source property to the previous value." + " Removing the bidirectional binding from properties " + property1 + " and " + property2, e2); } throw new RuntimeException( "Bidirectional binding failed, setting to the previous value", e); } finally { updating = false; } } } } } private static class UntypedGenericBidirectionalBinding extends BidirectionalBinding { private final Object property1; private final Object property2; public UntypedGenericBidirectionalBinding(Object property1, Object property2) { super(property1, property2); this.property1 = property1; this.property2 = property2; } @Override protected Object getProperty1() { return property1; } @Override protected Object getProperty2() { return property2; } @Override public void changed(ObservableValue sourceProperty, Object oldValue, Object newValue) { throw new RuntimeException("Should not reach here"); } } public abstract static class StringConversionBidirectionalBinding extends BidirectionalBinding { private final WeakReference> stringPropertyRef; private final WeakReference> otherPropertyRef; private boolean updating; public StringConversionBidirectionalBinding(Property stringProperty, Property otherProperty) { super(stringProperty, otherProperty); stringPropertyRef = new WeakReference>(stringProperty); otherPropertyRef = new WeakReference>(otherProperty); } protected abstract String toString(T value); protected abstract T fromString(String value) throws ParseException; @Override protected Object getProperty1() { return stringPropertyRef.get(); } @Override protected Object getProperty2() { return otherPropertyRef.get(); } @Override public void changed(ObservableValue observable, Object oldValue, Object newValue) { if (!updating) { final Property property1 = stringPropertyRef.get(); final Property property2 = otherPropertyRef.get(); if ((property1 == null) || (property2 == null)) { if (property1 != null) { property1.removeListener(this); } if (property2 != null) { property2.removeListener(this); } } else { try { updating = true; if (property1 == observable) { try { property2.setValue(fromString(property1.getValue())); } catch (Exception e) { Logging.getLogger().warning("Exception while parsing String in bidirectional binding", e); property2.setValue(null); } } else { try { property1.setValue(toString(property2.getValue())); } catch (Exception e) { Logging.getLogger().warning("Exception while converting Object to String in bidirectional binding", e); property1.setValue(""); } } } finally { updating = false; } } } } } private static class StringFormatBidirectionalBinding extends StringConversionBidirectionalBinding { private final Format format; @SuppressWarnings("unchecked") public StringFormatBidirectionalBinding(Property stringProperty, Property otherProperty, Format format) { super(stringProperty, otherProperty); this.format = format; } @Override protected String toString(Object value) { return format.format(value); } @Override protected Object fromString(String value) throws ParseException { return format.parseObject(value); } } private static class StringConverterBidirectionalBinding extends StringConversionBidirectionalBinding { private final StringConverter converter; public StringConverterBidirectionalBinding(Property stringProperty, Property otherProperty, StringConverter converter) { super(stringProperty, otherProperty); this.converter = converter; } @Override protected String toString(T value) { return converter.toString(value); } @Override protected T fromString(String value) throws ParseException { return converter.fromString(value); } } }