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

org.openide.nodes.IndexedCustomizer Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */

package org.openide.nodes;

import java.awt.Component;
import java.awt.Container;
import java.awt.Dimension;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.awt.Insets;
import java.awt.Point;
import java.awt.datatransfer.DataFlavor;
import java.awt.datatransfer.Transferable;
import java.awt.datatransfer.UnsupportedFlavorException;
import java.awt.dnd.Autoscroll;
import java.awt.dnd.DnDConstants;
import java.awt.dnd.DragGestureEvent;
import java.awt.dnd.DragGestureListener;
import java.awt.dnd.DragSource;
import java.awt.dnd.DragSourceDragEvent;
import java.awt.dnd.DragSourceDropEvent;
import java.awt.dnd.DragSourceEvent;
import java.awt.dnd.DragSourceListener;
import java.awt.dnd.DropTarget;
import java.awt.dnd.DropTargetDragEvent;
import java.awt.dnd.DropTargetDropEvent;
import java.awt.dnd.DropTargetEvent;
import java.awt.dnd.DropTargetListener;
import java.awt.dnd.InvalidDnDOperationException;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.beans.Customizer;
import java.beans.PropertyChangeListener;
import java.io.IOException;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.swing.JButton;
import javax.swing.JComponent;
import javax.swing.JDialog;
import javax.swing.JLabel;
import javax.swing.JList;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JViewport;
import javax.swing.SwingUtilities;
import javax.swing.UIManager;
import javax.swing.border.Border;
import javax.swing.border.LineBorder;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import javax.swing.event.ListSelectionEvent;
import javax.swing.event.ListSelectionListener;
import org.openide.util.NbBundle;
import org.openide.util.datatransfer.ExTransferable;

/** A dialog for reordering nodes. This dialog can reorder
* nodes for all implementors of the {@link Index} cookie.
* The dialog can invoke reorder actions on a given Index
* implementation immediatelly, or these actions can be accumulated
* and invoked at once, when the dialog is closed.
*
* 

This class is final only for performance reasons. * * @author Jan Jancura, Ian Formanek, Dafe Simonek * @deprecated Better to use {@link Index.Support#showIndexedCustomizer} which behaves better * with the window system. */ @Deprecated public final class IndexedCustomizer extends JDialog implements Customizer { // initializations ................................................................................ static final long serialVersionUID = -8731362267771694641L; // variables ..................................................................................... /** The actual JList control */ private JList control; /** Buttons */ private JButton buttonUp; /** Buttons */ private JButton buttonDown; /** Buttons */ private JButton buttonClose; /** index to sort */ private Index index; private Node[] nodes; /** Whether or not change the order immediatelly */ private boolean immediateReorder = true; /** Permutation array, which stores moves in case when * immediateReorder property is false */ private int[] permutation; /** Listener to the changes in the nodes */ private ChangeListener nodeChangesL; /** Construct a new customizer. */ public IndexedCustomizer() { this(null, true); } /** Construct a dummy customizer. * Might not actually be used as a JDialog, however its GUI * layout and logic will be used. * Cf. #9323. * @param c a container on which to draw the GUI * @param closeButton if true, add a Close button and other dialog logic, else no */ IndexedCustomizer(Container p, boolean closeButton) { super(TMUtil.mainWindow(), true); GridBagConstraints constraints; if (closeButton) { setDefaultCloseOperation(javax.swing.JDialog.DISPOSE_ON_CLOSE); // attach cancel also to Escape key getRootPane().registerKeyboardAction( new java.awt.event.ActionListener() { public void actionPerformed(java.awt.event.ActionEvent evt) { setVisible(false); dispose(); } }, javax.swing.KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_ESCAPE, 0, true), javax.swing.JComponent.WHEN_IN_FOCUSED_WINDOW ); setTitle(Node.getString("LAB_order")); } // closeButton if (p == null) { p = getContentPane(); } p.setLayout(new GridBagLayout()); JLabel l = new JLabel(Node.getString("LAB_listOrder")); l.setDisplayedMnemonic(Node.getString("LAB_listOrder_Mnemonic").charAt(0)); constraints = new GridBagConstraints(); constraints.gridx = 0; constraints.gridy = 0; constraints.anchor = GridBagConstraints.WEST; constraints.insets = new Insets(12, 12, 2, 12); p.add(l, constraints); control = new AutoscrollJList(); l.setLabelFor(control); control.addListSelectionListener( new ListSelectionListener() { public void valueChanged(ListSelectionEvent e) { if (control.isSelectionEmpty()) { buttonUp.setEnabled(false); buttonDown.setEnabled(false); } else { int i = control.getSelectedIndex(); if (i > 0) { //PENDING - jeste testovat, jestli jsou OrderedCookie.Child buttonUp.setEnabled(true); } else { buttonUp.setEnabled(false); } if (i < (nodes.length - 1)) { buttonDown.setEnabled(true); } else { buttonDown.setEnabled(false); } } } } ); control.setCellRenderer(new IndexedListCellRenderer()); control.setVisibleRowCount(15); control.setSelectionMode(javax.swing.ListSelectionModel.SINGLE_SELECTION); // list has to be scrolling constraints = new GridBagConstraints(); constraints.gridx = 0; constraints.gridy = 1; constraints.fill = GridBagConstraints.BOTH; constraints.weightx = 1.0; constraints.weighty = 1.0; constraints.insets = new Insets(0, 12, 11, 11); p.add(new JScrollPane(control), constraints); JPanel bb = new JPanel(); if (closeButton) { buttonClose = new JButton(Node.getString("Button_close")); buttonClose.setMnemonic(Node.getString("Button_close_Mnemonic").charAt(0)); } buttonUp = new JButton(Node.getString("Button_up")); buttonUp.setMnemonic(Node.getString("Button_up_Mnemonic").charAt(0)); buttonDown = new JButton(Node.getString("Button_down")); buttonDown.setMnemonic(Node.getString("Button_down_Mnemonic").charAt(0)); bb.setLayout(new GridBagLayout()); constraints = new GridBagConstraints(); constraints.gridx = 0; constraints.gridy = 0; constraints.anchor = GridBagConstraints.NORTH; constraints.fill = GridBagConstraints.HORIZONTAL; constraints.insets = new Insets(0, 0, 5, 11); bb.add(buttonUp, constraints); constraints = new GridBagConstraints(); constraints.gridx = 0; constraints.gridy = 1; constraints.anchor = GridBagConstraints.NORTH; constraints.fill = GridBagConstraints.HORIZONTAL; constraints.weighty = 1.0; constraints.insets = new Insets(0, 0, 0, 11); bb.add(buttonDown, constraints); if (closeButton) { constraints = new GridBagConstraints(); constraints.gridx = 0; constraints.gridy = 2; constraints.anchor = GridBagConstraints.SOUTH; constraints.fill = GridBagConstraints.HORIZONTAL; constraints.insets = new Insets(0, 0, 11, 11); bb.add(buttonClose, constraints); } buttonUp.addActionListener( new ActionListener() { public void actionPerformed(ActionEvent e) { int i = control.getSelectedIndex(); moveUp(i); updateList(); control.setSelectedIndex(i - 1); control.ensureIndexIsVisible(i - 1); control.repaint(); } } ); buttonDown.addActionListener( new ActionListener() { public void actionPerformed(ActionEvent e) { int i = control.getSelectedIndex(); moveDown(i); updateList(); control.setSelectedIndex(i + 1); control.ensureIndexIsVisible(i + 1); control.repaint(); } } ); if (closeButton) { buttonClose.addActionListener( new ActionListener() { public void actionPerformed(ActionEvent e) { doClose(); dispose(); } } ); } buttonUp.setEnabled(false); buttonDown.setEnabled(false); constraints = new GridBagConstraints(); constraints.gridx = 1; constraints.gridy = 1; constraints.fill = GridBagConstraints.VERTICAL; p.add(bb, constraints); // disable drag support, as DnD crashes all environment // under current implementation of VMs on unixes, ans causes // some deadlocks on windows... //dragSupport = new IndexedDragSource(control); //dropSupport = new IndexedDropTarget(this, dragSupport); if (closeButton) { pack(); setBounds(org.openide.util.Utilities.findCenterBounds(getSize())); buttonClose.requestFocus(); // to get shortcuts to work buttonClose.getAccessibleContext().setAccessibleDescription(Node.getString("ACSD_Button_close")); } buttonUp.getAccessibleContext().setAccessibleDescription(Node.getString("ACSD_Button_up")); buttonDown.getAccessibleContext().setAccessibleDescription(Node.getString("ACSD_Button_down")); control.getAccessibleContext().setAccessibleDescription(Node.getString("ACSD_ListOrder")); p.getAccessibleContext().setAccessibleDescription(Node.getString("ACSD_IndexedCustomizer")); getAccessibleContext().setAccessibleDescription(Node.getString("ACSD_IndexedCustomizer")); } /** Simulate behavior of the Close button, minus actual dialog disposal. * Might do the same as setImmediateReorder(true), but I am not sure. */ void doClose() { if ((!immediateReorder) && (index != null) && (permutation != null)) { int[] realPerm = new int[permutation.length]; for (int i = 0; i < realPerm.length; i++) { realPerm[permutation[i]] = i; //System.out.println (i + "-->" + permutation[i]); // NOI18N } index.reorder(realPerm); } } // other methods ................................................................................ /** Called when an explored context changes and the list needs to be * recreated. */ private void updateList() { if (index == null) { return; } Node[] localNodes = index.getNodes(); //System.out.println ("Nodes taken, size: " + localNodes.length); // NOI18N // obtain nodes with help from permutation array, if // conditions met if (!immediateReorder) { getPermutation(); int origLength = permutation.length; int newLength = localNodes.length; if (origLength < newLength) { // some nodes added, we must synchronize the permutation nodes = new Node[newLength]; int[] newPerm = new int[newLength]; System.arraycopy(newPerm, 0, permutation, 0, origLength); for (int i = 0; i < newLength; i++) { if (i < origLength) { nodes[i] = localNodes[permutation[i]]; } else { // added nodes.... nodes[i] = localNodes[i]; newPerm[i] = i; } } permutation = newPerm; } else if (origLength > newLength) { // some nodes removed, we must re-initialize the permutation nodes = new Node[newLength]; permutation = new int[newLength]; for (int i = 0; i < newLength; i++) { nodes[i] = localNodes[i]; permutation[i] = i; } } else { // node count is the same, only permute the nodes nodes = new Node[newLength]; for (int i = 0; i < newLength; i++) nodes[i] = localNodes[permutation[i]]; } } else { nodes = localNodes.clone(); } control.setListData(nodes); if ((nodes.length > 0) && (control.getSelectedIndex() == -1)) { control.setSelectedIndex(0); } } public Dimension getPreferredSize() { return new Dimension(300, super.getPreferredSize().height); } /** Will reorders be reflected immediately? * @return true if so */ public boolean isImmediateReorder() { return immediateReorder; } /** Set whether reorders will take effect immediately. * @param immediateReorder true if so */ public void setImmediateReorder(boolean immediateReorder) { if (this.immediateReorder == immediateReorder) { return; } this.immediateReorder = immediateReorder; if (immediateReorder) { if (permutation != null) { index.reorder(permutation); permutation = null; updateList(); } } } // implementation of Customizer ............................................................ /** Set the nodes to reorder. * @param bean must implement {@link Index} * @throws IllegalArgumentException if not */ public void setObject(Object bean) { if (bean instanceof Index) { index = (Index) bean; // add weak listener to the Index nodeChangesL = new ChangeListener() { public void stateChanged(ChangeEvent ev) { SwingUtilities.invokeLater( new Runnable() { public void run() { updateList(); } } ); } }; updateList(); control.invalidate(); validate(); index.addChangeListener(org.openide.util.WeakListeners.change(nodeChangesL, index)); } else { throw new IllegalArgumentException(); } } // I don't change any property... public void addPropertyChangeListener(PropertyChangeListener listener) { } public void removePropertyChangeListener(PropertyChangeListener listener) { } /** Moves up. Performs differently according to * immediateReorder property value. */ private void moveUp(final int position) { if (index == null) { return; } if (immediateReorder) { index.moveUp(position); } else { getPermutation(); int temp = permutation[position]; permutation[position] = permutation[position - 1]; permutation[position - 1] = temp; } } /** Moves down. Performs differently according to * immediateReorder property value. */ private void moveDown(final int position) { if (index == null) { return; } if (immediateReorder) { index.moveDown(position); } else { getPermutation(); int temp = permutation[position]; permutation[position] = permutation[position + 1]; permutation[position + 1] = temp; } } /** Safe getter for permutation. * Initializes permutation to identical permutation if it is null.
* index variable must not be null when this method called */ private int[] getPermutation() { if (permutation == null) { if (nodes == null) { nodes = index.getNodes().clone(); } permutation = new int[nodes.length]; for (int i = 0; i < nodes.length; permutation[i] = i++) ; } return permutation; } /** Permute the list as given permutation dictates. * Sets selection to the given selected index. * Called from dropSupport as result of succesfull DnD operation. */ void performReorder(int[] perm, int selected) { if (immediateReorder) { index.reorder(perm); } else { // merge current and reversed given permutation // (reverse given permutation first) int[] reversed = new int[perm.length]; for (int i = 0; i < reversed.length; i++) reversed[perm[i]] = i; int[] orig = getPermutation(); permutation = new int[orig.length]; for (int i = 0; i < orig.length; i++) { permutation[i] = orig[reversed[i]]; // System.out.println(permutation[i] + " ----> " + i); // NOI18N } } updateList(); control.setSelectedIndex(selected); control.repaint(); } /** Implementation of drag functionality in * reorder dialog. */ private static final class IndexedDragSource implements DragGestureListener, DragSourceListener { /** Asociated JList component where the drag will * take place */ JList comp; /** User gesture that initiated the drag */ DragGestureEvent dge; /** Out data flavor used to transfer the index */ DataFlavor myFlavor; /** Creates drag source with asociated list where drag * will take place. * Also creates the default gesture and asociates this with * given component */ IndexedDragSource(JList comp) { this.comp = comp; // initialize gesture DragSource ds = DragSource.getDefaultDragSource(); ds.createDefaultDragGestureRecognizer(comp, DnDConstants.ACTION_MOVE, this); } /** Initiating the drag */ public void dragGestureRecognized(DragGestureEvent dge) { // check allowed actions if ((dge.getDragAction() & DnDConstants.ACTION_MOVE) == 0) { return; } // prepare transferable and start the drag int index = comp.locationToIndex(dge.getDragOrigin()); // no index, then no dragging... if (index < 0) { return; } // System.out.println("Starting drag..."); // NOI18N // create our flavor for transferring the index myFlavor = new DataFlavor( String.class, NbBundle.getBundle(IndexedCustomizer.class).getString("IndexedFlavor") ); try { dge.startDrag(DragSource.DefaultMoveDrop, new IndexTransferable(myFlavor, index), this); // remember the gesture this.dge = dge; } catch (InvalidDnDOperationException exc) { Logger.getLogger(IndexedCustomizer.class.getName()).log(Level.WARNING, null, exc); // PENDING notify user - cannot start the drag } } public void dragEnter(DragSourceDragEvent dsde) { } public void dragOver(DragSourceDragEvent dsde) { } public void dropActionChanged(DragSourceDragEvent dsde) { } public void dragExit(DragSourceEvent dse) { } public void dragDropEnd(DragSourceDropEvent dsde) { } /** Utility accessor */ DragGestureEvent getDragGestureEvent() { return dge; } } // end of IndexedDragSource /** Implementation of drop functionality in * reorder dialog. */ private static final class IndexedDropTarget implements DropTargetListener { /** Asociated JList component for dropping */ JList comp; /** Cell renderer which renders the list items */ IndexedListCellRenderer cellRenderer; /** Indexed dialog */ IndexedCustomizer dialog; /** Drag source support instance */ IndexedDragSource ids; /** last index dragged over */ int lastIndex = -1; /** Creates the instance, makes given component active for * drop operation. */ IndexedDropTarget(IndexedCustomizer dialog, IndexedDragSource ids) { this.dialog = dialog; this.comp = dialog.control; this.cellRenderer = (IndexedListCellRenderer) this.comp.getCellRenderer(); this.ids = ids; new DropTarget(comp, DnDConstants.ACTION_MOVE, this, true); } /** User is starting to drag over us */ public void dragEnter(DropTargetDragEvent dtde) { if (!checkConditions(dtde)) { dtde.rejectDrag(); } else { lastIndex = comp.locationToIndex(dtde.getLocation()); cellRenderer.draggingEnter(lastIndex, ids.getDragGestureEvent().getDragOrigin(), dtde.getLocation()); comp.repaint(comp.getCellBounds(lastIndex, lastIndex)); } } /** User drag over us */ public void dragOver(DropTargetDragEvent dtde) { if (!checkConditions(dtde)) { dtde.rejectDrag(); if (lastIndex >= 0) { cellRenderer.draggingExit(); comp.repaint(comp.getCellBounds(lastIndex, lastIndex)); lastIndex = -1; } } else { dtde.acceptDrag(DnDConstants.ACTION_MOVE); int index = comp.locationToIndex(dtde.getLocation()); if (lastIndex == index) { cellRenderer.draggingOver(index, ids.getDragGestureEvent().getDragOrigin(), dtde.getLocation()); } else { if (lastIndex < 0) { lastIndex = index; } cellRenderer.draggingExit(); cellRenderer.draggingEnter(index, ids.getDragGestureEvent().getDragOrigin(), dtde.getLocation()); comp.repaint(comp.getCellBounds(lastIndex, index)); lastIndex = index; } } } public void dropActionChanged(DropTargetDragEvent dtde) { } /** User exits the dragging */ public void dragExit(DropTargetEvent dte) { if (lastIndex >= 0) { cellRenderer.draggingExit(); comp.repaint(comp.getCellBounds(lastIndex, lastIndex)); } } /** Takes given index transferable and reorders * the items as appropriate (and if possible) */ public void drop(DropTargetDropEvent dtde) { // reject all but local moves if ((DnDConstants.ACTION_MOVE != dtde.getDropAction()) || !dtde.isLocalTransfer()) { dtde.rejectDrop(); } int target = comp.locationToIndex(dtde.getLocation()); if (target < 0) { dtde.rejectDrop(); return; } Transferable t = dtde.getTransferable(); // System.out.println("Dropping..."); // NOI18N dtde.acceptDrop(DnDConstants.ACTION_MOVE); try { int source = Integer.parseInt((String) t.getTransferData(ids.myFlavor)); if (source != target) { performReorder(source, target); dtde.dropComplete(true); } else { dtde.dropComplete(false); } } catch (IOException exc) { dtde.dropComplete(false); } catch (UnsupportedFlavorException exc) { dtde.dropComplete(false); } catch (NumberFormatException exc) { dtde.dropComplete(false); } } /** Actually performs the reordering which results from * succesfull drag-drop operation. * @param source */ void performReorder(int source, int target) { int[] myPerm = new int[comp.getModel().getSize()]; // positions will change only between source and target // indexes, the rest remains the same for (int i = 0; i < Math.min(source, target); i++) myPerm[i] = i; for (int i = Math.max(source, target) + 1; i < myPerm.length; i++) myPerm[i] = i; // reorder the rest myPerm[source] = target; if (source > target) { // dragging was up the list for (int i = target; i < source; i++) myPerm[i] = i + 1; } else { // dragging was down the list for (int i = source + 1; i < (target + 1); i++) myPerm[i] = i - 1; } // and finally perform the reordering dialog.performReorder(myPerm, target); } /** @return True if conditions to continue with DnD * operation were satisfied */ boolean checkConditions(DropTargetDragEvent dtde) { int index = comp.locationToIndex(dtde.getLocation()); return (DnDConstants.ACTION_MOVE == dtde.getDropAction()) && (index >= 0); } } // end of IndexedDropTarget /** This class takes responsibility of presenting the * asociated index as transferable object. */ private static final class IndexTransferable extends ExTransferable.Single { /** Index to transfer */ int index; /** Creates transferable of given index */ IndexTransferable(DataFlavor flavor, int index) { super(flavor); this.index = index; } /* Returns string representation of index */ protected Object getData() throws IOException, UnsupportedFlavorException { return String.valueOf(index); } } // end of IndexTransferable /** Implements drag and drop visual feedback * support for node list cell rendeder. */ private static final class IndexedListCellRenderer implements javax.swing.ListCellRenderer { static final long serialVersionUID = -5526451942677242944L; protected static Border hasFocusBorder; static { hasFocusBorder = new LineBorder(UIManager.getColor("List.focusCellHighlight")); // NOI18N } /** delegate to use for rendering. Usually NodeRenderer, but if run * without explorer, then it uses DefaultListCellRenderer */ private javax.swing.ListCellRenderer delegate = TMUtil.findListCellRenderer(); /** Index of currently drag under cell in parent list */ int dragIndex; /** Creates new renderer */ IndexedListCellRenderer() { dragIndex = -1; } /** DnD operation enters, update visual * presentation to the drag under state */ public void draggingEnter(int index, Point startingLoc, Point currentLoc) { // System.out.println("Entering index: " + index); // NOI18N this.dragIndex = index; } /** DnD operation dragging over. */ public void draggingOver(int index, Point startingLoc, Point currentLoc) { } /** DnD operation exits, reset visual state * back to the normal */ public void draggingExit() { dragIndex = -1; } public Component getListCellRendererComponent( JList list, Object value, int index, boolean isSelected, boolean cellHasFocus ) { JComponent result = (JComponent) delegate.getListCellRendererComponent( list, value, index, isSelected, cellHasFocus ); if (index == dragIndex) { // System.out.println("Drawing...."); // NOI18N result.setBorder(hasFocusBorder); } return result; } } // end of IndexedListCellRenderer /** Implements autoscrolling support for JList. * However, JList must be contained in some JViewport. */ private static class AutoscrollJList extends JList implements Autoscroll { static final long serialVersionUID = 5495776972406885734L; /** Autoscroll insets */ Insets scrollInsets; /** Insets for the autoscroll method to decide * whether really perform or not */ Insets realInsets; /** Viewport we are in */ JViewport viewport; AutoscrollJList() { } /** notify the Component to autoscroll */ public void autoscroll(Point cursorLoc) { JViewport viewport = getViewport(); Point viewPos = viewport.getViewPosition(); int viewHeight = viewport.getExtentSize().height; if ((cursorLoc.y - viewPos.y) <= realInsets.top) { // scroll up viewport.setViewPosition(new Point(viewPos.x, Math.max(viewPos.y - realInsets.top, 0))); } else if (((viewPos.y + viewHeight) - cursorLoc.y) <= realInsets.bottom) { // scroll down viewport.setViewPosition( new Point(viewPos.x, Math.min(viewPos.y + realInsets.bottom, this.getHeight() - viewHeight)) ); } } /** @return the Insets describing the autoscrolling * region or border relative to the geometry of the * implementing Component. */ public Insets getAutoscrollInsets() { if (scrollInsets == null) { int height = this.getHeight(); scrollInsets = new Insets(height, 0, height, 0); // compute also autoscroll insets for viewport //Rectangle rect = getViewport().getViewRect(); realInsets = new Insets(15, 0, 15, 0); } return scrollInsets; } /** Asociates given viewport with this list. * (Viewport is usually parent containing this component) */ JViewport getViewport() { if (viewport == null) { Component comp = this; while (!(comp instanceof JViewport) && (comp != null)) { comp = comp.getParent(); } viewport = (JViewport) comp; } return viewport; } } // end of AutoscrollJViewport }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy