All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.pushingpixels.flamingo.api.common.CommandToggleButtonGroup Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2005-2010 Flamingo Kirill Grouchnikov. All Rights Reserved.
 *
 * Redistribution and use in source and binary forms, with or without 
 * modification, are permitted provided that the following conditions are met:
 * 
 *  o Redistributions of source code must retain the above copyright notice, 
 *    this list of conditions and the following disclaimer. 
 *     
 *  o Redistributions in binary form must reproduce the above copyright notice, 
 *    this list of conditions and the following disclaimer in the documentation 
 *    and/or other materials provided with the distribution. 
 *     
 *  o Neither the name of Flamingo Kirill Grouchnikov nor the names of 
 *    its contributors may be used to endorse or promote products derived 
 *    from this software without specific prior written permission. 
 *     
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, 
 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 
 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR 
 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 
 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 
 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; 
 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, 
 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE 
 * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, 
 * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
 */
package org.pushingpixels.flamingo.api.common;

import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeSupport;
import java.io.Serializable;
import java.util.*;

import javax.swing.ButtonGroup;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;

/**
 * Group of command toggle buttons. Unlike the {@link ButtonGroup}, this class
 * operates on buttons and not on button models.
 * 
 * @author Kirill Grouchnikov
 */
public class CommandToggleButtonGroup implements Serializable {
	/**
	 * Contains all group buttons.
	 */
	protected Vector buttons;

	/**
	 * Map of registered model change listeners.
	 */
	protected Map modelChangeListeners;

	/**
	 * Property change support to track the registered property change
	 * listeners.
	 */
	private PropertyChangeSupport changeSupport;

	/**
	 * Name of the property change event fired when the group selection is
	 * changed.
	 */
	public static final String SELECTED_PROPERTY = "selected";

	/**
	 * The currently selected button. Can be null.
	 */
	protected JCommandToggleButton selection;

	/**
	 * If false, the selection cannot be cleared. By default the
	 * button group allows clearing the selection in {@link #clearSelection()}
	 * or {@link #setSelected(JCommandToggleButton, boolean)} (passing the
	 * currently selected button and false).
	 */
	protected boolean allowsClearingSelection;

	/**
	 * Creates a new button group.
	 */
	public CommandToggleButtonGroup() {
		this.buttons = new Vector();
		this.modelChangeListeners = new HashMap();
		this.allowsClearingSelection = true;
	}

	/**
	 * Sets the new value for clearing selection. If true is
	 * passed, the selection can be cleared in {@link #clearSelection()} or
	 * {@link #setSelected(JCommandToggleButton, boolean)} (passing the
	 * currently selected button and false).
	 * 
	 * @param allowsClearingSelection
	 *            The new value for clearing selection.
	 */
	public void setAllowsClearingSelection(boolean allowsClearingSelection) {
		this.allowsClearingSelection = allowsClearingSelection;
	}

	/**
	 * Returns the current value for clearing selection. true is
	 * returned when selection can be cleared in {@link #clearSelection()} or
	 * {@link #setSelected(JCommandToggleButton, boolean)} (passing the
	 * currently selected button and false).
	 * 
	 * @return The current value for clearing selection.
	 */
	public boolean isAllowsClearingSelection() {
		return allowsClearingSelection;
	}

	/**
	 * Adds the specified button to the group. If the button is selected, and
	 * the group has a selected button, the newly added button is marked as
	 * unselected.
	 * 
	 * @param b
	 *            The button to be added.
	 */
	public void add(final JCommandToggleButton b) {
		if (b == null) {
			return;
		}
		buttons.addElement(b);

		boolean wasSelectionNull = (this.selection == null);
		if (b.getActionModel().isSelected()) {
			if (wasSelectionNull) {
				selection = b;
			} else {
				b.getActionModel().setSelected(false);
			}
		}
		ChangeListener cl = new ChangeListener() {
			boolean wasSelected = b.getActionModel().isSelected();

			@Override
			public void stateChanged(ChangeEvent e) {
				boolean isSelected = b.getActionModel().isSelected();
				if (wasSelected != isSelected)
					setSelected(b, isSelected);
				wasSelected = isSelected;
			}
		};

		b.getActionModel().addChangeListener(cl);
		this.modelChangeListeners.put(b, cl);

		if (wasSelectionNull) {
			this.firePropertyChange(SELECTED_PROPERTY, null, b);
		}
	}

	/**
	 * Removes the specified button from the group.
	 * 
	 * @param b
	 *            The button to be removed
	 */
	public void remove(JCommandToggleButton b) {
		if (b == null) {
			return;
		}
		buttons.removeElement(b);
		boolean wasSelected = (b == selection);
		if (wasSelected) {
			selection = null;
		}
		b.getActionModel().removeChangeListener(
				this.modelChangeListeners.get(b));
		this.modelChangeListeners.remove(b);

		if (wasSelected) {
			this.firePropertyChange(SELECTED_PROPERTY, b, null);
		}
	}

	/**
	 * Changes the selected status of the specified button.
	 * 
	 * @param button
	 *            Button.
	 * @param isSelected
	 *            Selection indication.
	 */
	public void setSelected(JCommandToggleButton button, boolean isSelected) {
		if (isSelected && button != null && button != selection) {
			JCommandToggleButton oldSelection = selection;
			selection = button;
			if (oldSelection != null) {
				oldSelection.getActionModel().setSelected(false);
			}
			button.getActionModel().setSelected(true);

			this.firePropertyChange(SELECTED_PROPERTY, oldSelection, button);
		}

		if (!isSelected && (button != null) && (button == selection)) {
			if (this.allowsClearingSelection) {
				selection = null;
				button.getActionModel().setSelected(false);

				this.firePropertyChange(SELECTED_PROPERTY, button, null);
			} else {
				// set the button back to selected
				button.getActionModel().setSelected(true);
			}
		}
	}

	/**
	 * Returns the selected button of this group.
	 * 
	 * @return The selected button of this group. The result can be
	 *         null.
	 */
	public JCommandToggleButton getSelected() {
		return this.selection;
	}

	/**
	 * Clears the selection of this button group.
	 */
	public void clearSelection() {
		if (this.allowsClearingSelection && (this.selection != null)) {
			this.selection.getActionModel().setSelected(false);
		}
	}

	/**
	 * Adds the specified property change listener on this button group.
	 * 
	 * @param listener
	 *            Listener to add.
	 */
	public synchronized void addPropertyChangeListener(
			PropertyChangeListener listener) {
		if (listener == null) {
			return;
		}
		if (changeSupport == null) {
			changeSupport = new PropertyChangeSupport(this);
		}
		changeSupport.addPropertyChangeListener(listener);
	}

	/**
	 * Removes the specified property change listener from this button group.
	 * 
	 * @param listener
	 *            Listener to remove.
	 */
	public synchronized void removePropertyChangeListener(
			PropertyChangeListener listener) {
		if (listener == null || changeSupport == null) {
			return;
		}
		changeSupport.removePropertyChangeListener(listener);
	}

	/**
	 * Fires a property change event on all registered listeners.
	 * 
	 * @param propertyName
	 *            Name of the changed property.
	 * @param oldValue
	 *            Old property value.
	 * @param newValue
	 *            New property value.
	 */
	protected void firePropertyChange(String propertyName, Object oldValue,
			Object newValue) {
		PropertyChangeSupport changeSupport = this.changeSupport;
		if (changeSupport == null || oldValue == newValue) {
			return;
		}
		changeSupport.firePropertyChange(propertyName, oldValue, newValue);
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy