view org.eclipse.jface.databinding/src/org/eclipse/jface/internal/databinding/swt/TextObservableValue.d @ 78:0a55d2d5a946

Added file for databinding
author Frank Benoit <benoit@tionex.de>
date Tue, 14 Apr 2009 11:35:29 +0200
parents
children 6be48cf9f95c
line wrap: on
line source

/*******************************************************************************
 * Copyright (c) 2005, 2008 IBM Corporation and others.
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v1.0
 * which accompanies this distribution, and is available at
 * http://www.eclipse.org/legal/epl-v10.html
 *
 * Contributors:
 *     IBM Corporation - initial API and implementation
 *     Brad Reynolds (bug 135446)
 *     Brad Reynolds - bug 164653
 *******************************************************************************/
module org.eclipse.jface.internal.databinding.swt.TextObservableValue;

import java.lang.all;

import org.eclipse.core.databinding.observable.Diffs;
import org.eclipse.core.databinding.observable.IObservable;
import org.eclipse.core.databinding.observable.Realm;
import org.eclipse.jface.databinding.swt.SWTObservables;
import org.eclipse.jface.internal.databinding.provisional.swt.AbstractSWTVetoableValue;
import org.eclipse.swt.SWT;
import org.eclipse.swt.events.VerifyEvent;
import org.eclipse.swt.events.VerifyListener;
import org.eclipse.swt.widgets.Event;
import org.eclipse.swt.widgets.Listener;
import org.eclipse.swt.widgets.Text;

/**
 * {@link IObservable} implementation that wraps a {@link Text} widget. The time
 * at which listeners should be notified about changes to the text is specified
 * on construction.
 * 
 * <dl>
 * <dt>Events:</dt>
 * <dd> If the update event type (specified on construction) is
 * <code>SWT.Modify</code> a value change event will be fired on every key
 * stroke. If the update event type is <code>SWT.FocusOut</code> a value
 * change event will be fired on focus out. When in either mode if the user is
 * entering text and presses [Escape] the value will be reverted back to the
 * last value set using doSetValue(). Regardless of the update event type a
 * value changing event will fire on verify to enable vetoing of changes.</dd>
 * </dl>
 * 
 * @since 1.0
 */
public class TextObservableValue : AbstractSWTVetoableValue {

    /**
     * {@link Text} widget that this is being observed.
     */
    private final Text text;

    /**
     * Flag to track when the model is updating the widget. When
     * <code>true</code> the handlers for the SWT events should not process
     * the event as this would cause an infinite loop.
     */
    private bool updating = false;

    /**
     * SWT event that on firing this observable will fire change events to its
     * listeners.
     */
    private final int updateEventType;

    /**
     * Valid types for the {@link #updateEventType}.
     */
    private static const int[] validUpdateEventTypes = [ SWT.Modify,
            SWT.FocusOut, SWT.None ];

    /**
     * Previous value of the Text.
     */
    private String oldValue;

    private Listener updateListener = new class() Listener {
        public void handleEvent(Event event) {
            if (!updating) {
                String newValue = text.getText();

                if (!newValue.equals(oldValue)) {
                    fireValueChange(Diffs.createValueDiff(oldValue, newValue));                 
                    oldValue = newValue;
                }
            }
        }
    };

    private VerifyListener verifyListener;

    /**
     * Constructs a new instance bound to the given <code>text</code> widget
     * and configured to fire change events to its listeners at the time of the
     * <code>updateEventType</code>.
     * 
     * @param text
     * @param updateEventType
     *            SWT event constant as to what SWT event to update the model in
     *            response to. Appropriate values are: <code>SWT.Modify</code>,
     *            <code>SWT.FocusOut</code>, <code>SWT.None</code>.
     * @throws IllegalArgumentException
     *             if <code>updateEventType</code> is an incorrect type.
     */
    public this(Text text, int updateEventType) {
        this(SWTObservables.getRealm(text.getDisplay()), text, updateEventType);
    }
    
    /**
     * Constructs a new instance.
     * 
     * @param realm can not be <code>null</code>
     * @param text
     * @param updateEventType
     */
    public this(Realm realm, Text text, int updateEventType) {
        super(realm, text);
        
        bool eventValid = false;
        for (int i = 0; !eventValid && i < validUpdateEventTypes.length; i++) {
            eventValid = (updateEventType is validUpdateEventTypes[i]);
        }
        if (!eventValid) {
            throw new IllegalArgumentException(
                    "UpdateEventType [" + updateEventType + "] is not supported."); //$NON-NLS-1$//$NON-NLS-2$
        }
        this.text = text;
        this.updateEventType = updateEventType;
        if (updateEventType !is SWT.None) {
            text.addListener(updateEventType, updateListener);
        }
        
        oldValue = text.getText();
        
        verifyListener = new class() VerifyListener {
            public void verifyText(VerifyEvent e) {
                if (!updating) {
                    String currentText = text
                            .getText();
                    String newText = currentText.substring(0, e.start) + e.text
                            + currentText.substring(e.end);
                    if (!fireValueChanging(Diffs.createValueDiff(currentText,
                            newText))) {
                        e.doit = false;
                    }
                }
            }
        };
        text.addVerifyListener(verifyListener);
    }

    /**
     * Sets the bound {@link Text Text's} text to the passed <code>value</code>.
     * 
     * @param value
     *            new value, String expected
     * @see org.eclipse.core.databinding.observable.value.AbstractVetoableValue#doSetApprovedValue(java.lang.Object)
     * @throws ClassCastException
     *             if the value is anything other than a String
     */
    protected void doSetApprovedValue(Object value) {
        try {
            updating = true;
            text.setText(value is null ? "" : value.toString()); //$NON-NLS-1$
            oldValue = text.getText();
        } finally {
            updating = false;
        }
    }

    /**
     * Returns the current value of the {@link Text}.
     * 
     * @see org.eclipse.core.databinding.observable.value.AbstractVetoableValue#doGetValue()
     */
    public Object doGetValue() {
        return oldValue = text.getText();
    }

    /**
     * Returns the type of the value from {@link #doGetValue()}, i.e.
     * String.class
     * 
     * @see org.eclipse.core.databinding.observable.value.IObservableValue#getValueType()
     */
    public Object getValueType() {
        return String.classinfo;
    }

    public void dispose() {
        if (!text.isDisposed()) {
            if (updateEventType !is SWT.None) {
                text.removeListener(updateEventType, updateListener);
            }
            text.removeVerifyListener(verifyListener);
        }
        super.dispose();
    }
}