1   ////////////////////////////////////////////////////////////////////////////////
2   // checkstyle: Checks Java source code for adherence to a set of rules.
3   // Copyright (C) 2001-2019 the original author or authors.
4   //
5   // This library is free software; you can redistribute it and/or
6   // modify it under the terms of the GNU Lesser General Public
7   // License as published by the Free Software Foundation; either
8   // version 2.1 of the License, or (at your option) any later version.
9   //
10  // This library is distributed in the hope that it will be useful,
11  // but WITHOUT ANY WARRANTY; without even the implied warranty of
12  // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  // Lesser General Public License for more details.
14  //
15  // You should have received a copy of the GNU Lesser General Public
16  // License along with this library; if not, write to the Free Software
17  // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
18  ////////////////////////////////////////////////////////////////////////////////
19  
20  package com.puppycrawl.tools.checkstyle.gui;
21  
22  import javax.swing.JTree;
23  import javax.swing.SwingUtilities;
24  import javax.swing.event.TreeExpansionEvent;
25  import javax.swing.event.TreeExpansionListener;
26  import javax.swing.event.TreeModelEvent;
27  import javax.swing.event.TreeModelListener;
28  import javax.swing.table.AbstractTableModel;
29  import javax.swing.tree.TreePath;
30  
31  /**
32   * This is a wrapper class takes a TreeTableModel and implements
33   * the table model interface. The implementation is trivial, with
34   * all of the event dispatching support provided by the superclass:
35   * the AbstractTableModel.
36   *
37   * <a href=
38   * "https://docs.oracle.com/cd/E48246_01/apirefs.1111/e13403/oracle/ide/controls/TreeTableModel.html">
39   * Original&nbsp;Source&nbsp;Location</a>
40   *
41   */
42  public class TreeTableModelAdapter extends AbstractTableModel {
43  
44      private static final long serialVersionUID = 8269213416115369275L;
45  
46      /** JTree component. */
47      private final JTree tree;
48      /** Tree table model. */
49      private final transient ParseTreeTableModel treeTableModel;
50  
51      /**
52       * Initialise tree and treeTableModel class attributes.
53       * @param treeTableModel Tree table model.
54       * @param tree JTree component.
55       */
56      public TreeTableModelAdapter(ParseTreeTableModel treeTableModel, JTree tree) {
57          this.tree = tree;
58          this.treeTableModel = treeTableModel;
59  
60          tree.addTreeExpansionListener(new UpdatingTreeExpansionListener());
61  
62          // Install a TreeModelListener that can update the table when
63          // mTree changes. We use delayedFireTableDataChanged as we can
64          // not be guaranteed the mTree will have finished processing
65          // the event before us.
66          treeTableModel.addTreeModelListener(new UpdatingTreeModelListener());
67      }
68  
69      // Wrappers, implementing TableModel interface.
70  
71      @Override
72      public int getColumnCount() {
73          return treeTableModel.getColumnCount();
74      }
75  
76      @Override
77      public String getColumnName(int column) {
78          return treeTableModel.getColumnName(column);
79      }
80  
81      @Override
82      public Class<?> getColumnClass(int column) {
83          return treeTableModel.getColumnClass(column);
84      }
85  
86      @Override
87      public int getRowCount() {
88          return tree.getRowCount();
89      }
90  
91      @Override
92      public Object getValueAt(int row, int column) {
93          return treeTableModel.getValueAt(nodeForRow(row), column);
94      }
95  
96      @Override
97      public boolean isCellEditable(int row, int column) {
98          return treeTableModel.isCellEditable(column);
99      }
100 
101     /**
102      * Finds node for a given row.
103      * @param row Row for which to find a related node.
104      * @return Node for a given row.
105      */
106     private Object nodeForRow(int row) {
107         final TreePath treePath = tree.getPathForRow(row);
108         return treePath.getLastPathComponent();
109     }
110 
111     /**
112      * TreeExpansionListener that can update the table when tree changes.
113      */
114     private class UpdatingTreeExpansionListener implements TreeExpansionListener {
115 
116         // Don't use fireTableRowsInserted() here; the selection model
117         // would get updated twice.
118         @Override
119         public void treeExpanded(TreeExpansionEvent event) {
120             fireTableDataChanged();
121         }
122 
123         @Override
124         public void treeCollapsed(TreeExpansionEvent event) {
125             fireTableDataChanged();
126         }
127 
128     }
129 
130     /**
131      * TreeModelListener that can update the table when tree changes.
132      */
133     private class UpdatingTreeModelListener implements TreeModelListener {
134 
135         @Override
136         public void treeNodesChanged(TreeModelEvent event) {
137             delayedFireTableDataChanged();
138         }
139 
140         @Override
141         public void treeNodesInserted(TreeModelEvent event) {
142             delayedFireTableDataChanged();
143         }
144 
145         @Override
146         public void treeNodesRemoved(TreeModelEvent event) {
147             delayedFireTableDataChanged();
148         }
149 
150         @Override
151         public void treeStructureChanged(TreeModelEvent event) {
152             delayedFireTableDataChanged();
153         }
154 
155         /**
156          * Invokes fireTableDataChanged after all the pending events have been
157          * processed. SwingUtilities.invokeLater is used to handle this.
158          */
159         private void delayedFireTableDataChanged() {
160             SwingUtilities.invokeLater(TreeTableModelAdapter.this::fireTableDataChanged);
161         }
162 
163     }
164 
165 }