/* * Ext GWT - Ext for GWT * Copyright(c) 2007-2009, Ext JS, LLC. * licensing@extjs.com * * http://extjs.com/license */ package com.extjs.gxt.ui.client.binding; import com.extjs.gxt.ui.client.data.ChangeEvent; import com.extjs.gxt.ui.client.data.ChangeEventSource; import com.extjs.gxt.ui.client.data.ChangeListener; import com.extjs.gxt.ui.client.data.Model; import com.extjs.gxt.ui.client.data.ModelData; import com.extjs.gxt.ui.client.data.PropertyChangeEvent; import com.extjs.gxt.ui.client.event.Events; import com.extjs.gxt.ui.client.event.FieldEvent; import com.extjs.gxt.ui.client.event.Listener; import com.extjs.gxt.ui.client.store.Record; import com.extjs.gxt.ui.client.store.Store; import com.extjs.gxt.ui.client.widget.form.Field; /** * A two-way binding between a ModelData and Field instance. The binding will be * 1-way when the bound model does not support change events. * * @see ModelData * @see Field */ @SuppressWarnings("unchecked") public class FieldBinding { protected Field field; protected ModelData model; protected String property; protected Store store; private Listener<FieldEvent> changeListener; private ChangeListener modelListener; private Converter convertor; /** * Creates a new binding instance. * * @param field the bound field for the binding */ public FieldBinding(Field field, String property) { this.field = field; this.property = property; changeListener = new Listener<FieldEvent>() { public void handleEvent(FieldEvent be) { onFieldChange(be); } }; modelListener = new ChangeListener() { public void modelChanged(ChangeEvent event) { if (event.getType() == ChangeEventSource.Update) { onModelChange((PropertyChangeEvent) event); } } }; } /** * Binds the model and field. * * @param model the model to be bound */ public void bind(ModelData model) { if (this.model != null) { unbind(); } this.model = model; field.addListener(Events.Change, changeListener); if (model instanceof Model) { ((Model) model).addChangeListener(modelListener); } updateField(); } /** * Returns the bindings converter. * * @return the converter */ public Converter getConvertor() { return convertor; } /** * Returns the bound field. * * @return the field */ public Field<Object> getField() { return field; } /** * Returns the bound model instance. * * @return the model */ public ModelData getModel() { return model; } /** * Returns the model's bound property name. * * @return the property name */ public String getProperty() { return property; } /** * Returns the binding's store. * * @return the store or null */ public Store getStore() { return store; } /** * Sets the converter which is used to translate data types when updating * either the field or model. * * @param convertor the converter */ public void setConvertor(Converter convertor) { this.convertor = convertor; } /** * Sets the store for the binding. When a store is specified edits are done * via Records instances obtained from the Store. * * @param store the store */ public void setStore(Store<? extends ModelData> store) { this.store = store; } /** * Unbinds the model and field by removing all listeners. */ public void unbind() { if (model != null) { if (model instanceof Model) { ((Model) model).removeChangeListener(modelListener); } model = null; } field.removeListener(Events.Change, changeListener); } /** * Updates the field's value with the model value. */ public void updateField() { Object val = onConvertModelValue(model.get(property)); field.setValue(val); } /** * Updates the model's value with the field value. */ public void updateModel() { Object val = onConvertFieldValue(field.getValue()); if (store != null) { Record r = store.getRecord(model); if (r != null) { r.setValid(property, field.isValid()); r.set(property, val); } } else { model.set(property, val); } } protected Object onConvertFieldValue(Object value) { if (convertor != null) { return convertor.convertFieldValue(value); } return value; } protected Object onConvertModelValue(Object value) { if (convertor != null) { return convertor.convertModelValue(value); } return value; } protected void onFieldChange(FieldEvent e) { updateModel(); } protected void onModelChange(PropertyChangeEvent event) { if (event.getName().equals(property)) { updateField(); } } }