/plugins/ProjectViewer/tags/projectviewer_1_0_2/projectviewer/tree/ProjectTreeSelectionListener.java

# · Java · 241 lines · 117 code · 41 blank · 83 comment · 22 complexity · 073d99d4b0e8036a863377d34a872da3 MD5 · raw file

  1. /*
  2. * This program is free software; you can redistribute it and/or
  3. * modify it under the terms of the GNU General Public License
  4. * as published by the Free Software Foundation; either version 2
  5. * of the License, or any later version.
  6. *
  7. * This program is distributed in the hope that it will be useful,
  8. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. * GNU General Public License for more details.
  11. *
  12. * You should have received a copy of the GNU General Public License
  13. * along with this program; if not, write to the Free Software
  14. * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  15. */
  16. package projectviewer.tree;
  17. import java.awt.event.*;
  18. import javax.swing.*;
  19. import javax.swing.event.*;
  20. import javax.swing.tree.*;
  21. import org.gjt.sp.util.Log;
  22. import projectviewer.*;
  23. /**
  24. * Listens to the project JTree and responds to file selections.
  25. *
  26. * @author <A HREF="mailto:burton@relativity.yi.org">Kevin A. Burton</A>
  27. * @version $Revision: 5738 $
  28. */
  29. public class ProjectTreeSelectionListener
  30. implements TreeSelectionListener, MouseListener, ChangeListener,
  31. TreeModelListener, Runnable
  32. {
  33. private ProjectViewer viewer;
  34. private Launcher launcher;
  35. private JTree currentTree;
  36. private TreePath selectionPath;
  37. private int lastClickButton;
  38. private long lastClickTime;
  39. private Object lastClickTarget;
  40. /**
  41. * Create a new <code>ProjectTreeSelectionListener
  42. */
  43. public ProjectTreeSelectionListener(ProjectViewer aViewer, Launcher aLauncher) {
  44. viewer = aViewer;
  45. launcher = aLauncher;
  46. lastClickTime = 0L;
  47. }
  48. // MouseListener interfaces
  49. /**
  50. * Determines when the user clicks on the JTree.
  51. */
  52. public void mouseClicked(MouseEvent evt) {
  53. if( isDoubleClick(evt) && isFileClicked( evt ) ) {
  54. ProjectFile file = viewer.getSelectedFile();
  55. if ( file.isOpened() ) {
  56. if ( isCurrentBuffer( file ) )
  57. launcher.closeFile( file );
  58. else
  59. launcher.showFile( file );
  60. } else {
  61. launcher.launchFile( file );
  62. }
  63. }
  64. }
  65. /**
  66. * Because IBM's JDK doesn't support <code>getClickCount()</code> for <code>JTree</code>
  67. * properly, we have to do this.
  68. */
  69. private boolean isDoubleClick(MouseEvent evt) {
  70. if (evt.getClickCount() == 2) return true;
  71. Object target = viewer.getCurrentTree()
  72. .getPathForLocation(evt.getX(), evt.getY()).getLastPathComponent();
  73. if (target == lastClickTarget &&
  74. target == viewer.getSelectedNode() &&
  75. lastClickButton == evt.getModifiers() &&
  76. (System.currentTimeMillis() - lastClickTime < 2000L))
  77. {
  78. lastClickTarget = null;
  79. return true;
  80. }
  81. lastClickButton = evt.getModifiers();
  82. lastClickTarget = target;
  83. lastClickTime = System.currentTimeMillis();
  84. return false;
  85. }
  86. public void mousePressed(MouseEvent evt) { }
  87. public void mouseReleased(MouseEvent evt) { }
  88. public void mouseEntered(MouseEvent evt) { }
  89. public void mouseExited(MouseEvent evt) { }
  90. // ChangeListener interfaces (JTabbedPane)
  91. /**
  92. * Listen to tab changes.
  93. */
  94. public void stateChanged( ChangeEvent evt ) {
  95. checkState();
  96. if ( currentTree != null ) getCurrentModel().removeTreeModelListener( this );
  97. currentTree = viewer.getCurrentTree();
  98. getCurrentModel().addTreeModelListener( this );
  99. }
  100. // TreeSelectionListener interfaces
  101. /**
  102. * Receive notification that the tree selection has changed.
  103. */
  104. public void valueChanged(TreeSelectionEvent e) {
  105. lastClickTarget = null;
  106. checkState();
  107. }
  108. // TreeModelListener interfaces
  109. /**
  110. * Invoked after a node (or a set of siblings) has changed in some way.
  111. */
  112. public void treeNodesChanged(TreeModelEvent e) {
  113. //Log.log( Log.DEBUG, this, "Tree Node Changed" );
  114. handleTreeModelEvent( e );
  115. }
  116. /**
  117. * Invoked after nodes have been inserted into the tree.
  118. */
  119. public void treeNodesInserted(TreeModelEvent e) {
  120. handleTreeModelEvent( e );
  121. }
  122. /**
  123. * Invoked after nodes have been removed from the tree.
  124. */
  125. public void treeNodesRemoved(TreeModelEvent e) { }
  126. /**
  127. * Invoked after the tree has drastically changed structure from a
  128. * given node down.
  129. */
  130. public void treeStructureChanged(TreeModelEvent e) { }
  131. /**
  132. * Call on the current tree to select the given node.
  133. */
  134. public void run() {
  135. currentTree.setSelectionPath( selectionPath );
  136. }
  137. /**
  138. * Returns <code>true</code> if a node is selected and the given
  139. * mouse event points to the specified node.
  140. */
  141. private boolean isFileClicked( MouseEvent evt ) {
  142. if ( !viewer.isFileSelected() ) return false;
  143. Object selectedNode = viewer.getSelectedNode();
  144. Object clickedNode = viewer.getCurrentTree()
  145. .getPathForLocation( evt.getX(), evt.getY() ).getLastPathComponent();
  146. return selectedNode.equals( clickedNode );
  147. }
  148. /**
  149. * Handle the given <code>TreeModelEvent</code>. This method will
  150. * find the first added/changed node and select it.
  151. */
  152. private void handleTreeModelEvent( TreeModelEvent evt ) {
  153. Object node = getChild( evt.getTreePath(), evt.getChildIndices()[0] );
  154. if ( !( node instanceof ProjectFile ) ) return;
  155. selectionPath = buildPathFrom( evt, node );
  156. SwingUtilities.invokeLater( this );
  157. }
  158. /**
  159. * Returns the node pointed to by the given path and index.
  160. */
  161. private Object getChild( TreePath path, int index ) {
  162. return getCurrentModel().getChild( path.getLastPathComponent(), index );
  163. }
  164. /**
  165. * Build a <code>TreePath</code> from the given <code>TreeModelEvent</code>
  166. * and a child.
  167. */
  168. private TreePath buildPathFrom( TreeModelEvent evt, Object child ) {
  169. return evt.getTreePath().pathByAddingChild( child );
  170. }
  171. /**
  172. * Returns the <code>TreeModel</code> of the current tree.
  173. */
  174. private TreeModel getCurrentModel() {
  175. return currentTree.getModel();
  176. }
  177. /**
  178. * Check the current node, setting the button/status states as
  179. * necessary.
  180. */
  181. private void checkState() {
  182. Object node = viewer.getSelectedNode();
  183. if (node == null) return;
  184. viewer.enableButtonsForNode( node );
  185. if ( node instanceof ProjectFile )
  186. viewer.setStatus( ( (ProjectFile) node ).getPath() );
  187. else if ( node instanceof Project)
  188. viewer.setStatus( node + " [" + ( (Project) node ).getRoot().getPath() + "]" );
  189. else if ( node instanceof ProjectDirectory)
  190. viewer.setStatus( ( (ProjectDirectory) node ).getPath() );
  191. else
  192. viewer.setStatus( "" );
  193. }
  194. /**
  195. * Returns <code>true</code> if the given file is the current buffer.
  196. */
  197. private boolean isCurrentBuffer( ProjectFile aFile ) {
  198. return aFile.pathEquals( viewer.getView().getBuffer().getPath() );
  199. }
  200. }