Builds an editor that copies data from the domain back and forth
/* * Copyright (c) 2002-2005 JGoodies Karsten Lentzsch. 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 JGoodies Karsten Lentzsch 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 com.jgoodies.binding.tutorial.basics; import java.awt.event.ActionEvent; import javax.swing.*; import javax.swing.event.ChangeEvent; import javax.swing.event.ChangeListener; import javax.swing.text.JTextComponent; import com.jgoodies.binding.tutorial.Album; import com.jgoodies.binding.tutorial.TutorialUtils; import com.jgoodies.forms.builder.PanelBuilder; import com.jgoodies.forms.factories.ButtonBarFactory; import com.jgoodies.forms.layout.CellConstraints; import com.jgoodies.forms.layout.FormLayout; /** * Builds an editor that copies data from the domain back and forth. * This approach is known as the "copying" approach or "push/pull".<p> * * The event handling used to enable and disable the composer text field * is invoked by a ChangeListener that hooks into the classical check box. * Note that this lacks the domain logic, where the composer is set to * <code>null</code> if the classical property is set to false. * This logic is deferred until the component values are written to the * edited Album via <code>#updateModel</code> when OK is pressed. * * @author Karsten Lentzsch * @version $Revision: 1.6 $ */ public class EditorCopyingExample { /** * Refers to the Album that is to be edited by this example editor. */ private final Album editedAlbum; private JTextComponent titleField; private JTextComponent artistField; private JCheckBox classicalBox; private JTextComponent composerField; private JButton okButton; private JButton cancelButton; private JButton resetButton; // Launching ************************************************************** public static void main(String[] args) { try { UIManager.setLookAndFeel("com.jgoodies.looks.plastic.PlasticXPLookAndFeel"); } catch (Exception e) { // Likely PlasticXP is not in the class path; ignore. } JFrame frame = new JFrame(); frame.setTitle("Binding Tutorial :: Editor (Copying)"); frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE); EditorCopyingExample example = new EditorCopyingExample(); JComponent panel = example.build(); example.updateView(); frame.getContentPane().add(panel); frame.pack(); TutorialUtils.locateOnScreenCenter(frame); frame.setVisible(true); } // Instance Creation ****************************************************** /** * Constructs an editor for an example Album. */ public EditorCopyingExample() { this(Album.ALBUM1); } /** * Constructs an editor for an Album to be edited. * * @param album the Album to be edited */ public EditorCopyingExample(Album album) { this.editedAlbum = album; } // Initialization ********************************************************* /** * Creates and intializes the UI components. */ private void initComponents() { titleField = new JTextField(); artistField = new JTextField(); classicalBox = new JCheckBox("Classical"); composerField = new JTextField(); okButton = new JButton(new OKAction()); cancelButton = new JButton(new CancelAction()); resetButton = new JButton(new ResetAction()); updateComposerField(); } /** * Observes the classical check box to update the composer field's * enablement and contents. For demonstration purposes a listener * is registered that writes all changes to the console. */ private void initEventHandling() { classicalBox.addChangeListener(new ClassicalChangeHandler()); // Report changes in all bound Album properties. editedAlbum.addPropertyChangeListener( TutorialUtils.createDebugPropertyChangeListener()); } // Copying Data Back and Forth ******************************************** /** * Reads the property values from the edited Album * and sets them in this editor's components. */ private void updateView() { titleField.setText(editedAlbum.getTitle()); artistField.setText(editedAlbum.getArtist()); classicalBox.setSelected(editedAlbum.isClassical()); composerField.setText(editedAlbum.getComposer()); } /** * Reads the values from this editor's components * and set the associated Album properties. */ private void updateModel() { editedAlbum.setTitle(titleField.getText()); editedAlbum.setArtist(artistField.getText()); editedAlbum.setClassical(classicalBox.isSelected()); editedAlbum.setComposer(composerField.getText()); } // Building *************************************************************** /** * Builds and returns the editor panel. * * @return the built panel */ public JComponent build() { initComponents(); initEventHandling(); FormLayout layout = new FormLayout( "right:pref, 3dlu, 150dlu:grow", "p, 3dlu, p, 3dlu, p, 3dlu, p, 9dlu, p"); PanelBuilder builder = new PanelBuilder(layout); builder.setDefaultDialogBorder(); CellConstraints cc = new CellConstraints(); builder.addLabel("Title", cc.xy (1, 1)); builder.add(titleField, cc.xy (3, 1)); builder.addLabel("Artist", cc.xy (1, 3)); builder.add(artistField, cc.xy (3, 3)); builder.add(classicalBox, cc.xy (3, 5)); builder.addLabel("Composer", cc.xy (1, 7)); builder.add(composerField, cc.xy (3, 7)); builder.add(buildButtonBar(), cc.xyw(1, 9, 3)); return builder.getPanel(); } private JComponent buildButtonBar() { return ButtonBarFactory.buildRightAlignedBar( okButton, cancelButton, resetButton); } // Event Handling ********************************************************* /** * Updates the composer field's enablement and contents. * Sets the enablement according to the selection state * of the classical check box. If the composer is not enabled, * we copy the domain logic and clear the composer field's text. */ private void updateComposerField() { boolean composerEnabled = classicalBox.isSelected(); composerField.setEnabled(composerEnabled); if (!composerEnabled) { composerField.setText(""); } } private class ClassicalChangeHandler implements ChangeListener { /** * The selection state of the classical check box has changed. * Updates the enablement and contents of the composer field. */ public void stateChanged(ChangeEvent evt) { updateComposerField(); } } // Actions **************************************************************** private final class OKAction extends AbstractAction { private OKAction() { super("OK"); } public void actionPerformed(ActionEvent e) { updateModel(); System.out.println(editedAlbum); System.exit(0); } } private final class CancelAction extends AbstractAction { private CancelAction() { super("Cancel"); } public void actionPerformed(ActionEvent e) { // Just ignore the current content. System.out.println(editedAlbum); System.exit(0); } } private final class ResetAction extends AbstractAction { private ResetAction() { super("Reset"); } public void actionPerformed(ActionEvent e) { updateView(); System.out.println(editedAlbum); } } }