2011-09-07 20 views
5

Mam przyciski JTree i 2, aby zaznaczyć i odznaczyć wszystkie węzły. Zrobiłem próbę takiego:JTree: Zaznaczanie wszystkich węzłów programowo

selectAll = new JButton("Select all"); 
selectAll.addActionListener(new ActionListener(){ 
     @Override 
     public void actionPerformed(ActionEvent e) { 
       int row = 0; 
       while (row < curvesTree.getRowCount()) 
       { 
        curvesTree.expandRow(row); 
        row++; 
       } 
      int entradasTree = curvesTree.getRowCount(); 
      for(int i=0; i<entradasTree; i++){ 
       TreePath path = curvesTree.getPathForRow(i); 
       curvesTree.setSelectionPath(path); 
      } 
     } 
    }); 

     unselectAll = new JButton("Unselect all"); 
     unselectAll.addActionListener(new ActionListener(){ 
      @Override 
      public void actionPerformed(ActionEvent e) { 
       curvesTree.clearSelection(); 
      } 
     }); 

Przycisk odznaczyć wydaje się działać, ale Zaznacz wszystko tylko rozszerza JTree i wybiera ostatni węzeł. Myślę, że za każdym razem, gdy program zostanie wybrany przez program, usuwam zaznaczenie poprzedniego.

JTree jest skonfigurowany tak:

curvesTree = new JTree(rootNode); 
curvesTree.setExpandsSelectedPaths(true); 
curvesTree.getSelectionModel().setSelectionMode(TreeSelectionModel. 
        DISCONTIGUOUS_TREE_SELECTION); 

Odpowiedz

6

odznaczyć Dzieje się tak dlatego ustawiasz nową ścieżkę wyboru zamiast dodawania. W pętli po rozwinięciu, zamiast robić

curvesTree.addSelectionPath(...) 

EDIT

czytanie API jest zawsze pouczające, nawet po latach ;-) Wystarczy znaleźć dużo Simper sposób, który pozostawia całą pracę do drzewa:

tree.setSelectionInterval(0, tree.getRowCount()); 
+1

znakomity chwyt, jak ty ... +1 – mKorbel

+0

Dzięki, czekałem dla tej metody pół rano :) –

0

tak jest to możliwe, na przykład:

import java.awt.Dimension; 
import javax.swing.*; 
import javax.swing.event.*; 
import javax.swing.tree.*; 

public class TreeWithMultiDiscontiguousSelections { 

    public static void main(String[] argv) { 
     JTree tree = new JTree(); 
     tree.getSelectionModel().setSelectionMode(TreeSelectionModel.DISCONTIGUOUS_TREE_SELECTION); 
     int treeSelectedRows[] = {3, 1}; 
     tree.setSelectionRows(treeSelectedRows); 
     TreeSelectionListener treeSelectionListener = new TreeSelectionListener() { 

      @Override 
      public void valueChanged(TreeSelectionEvent treeSelectionEvent) { 
       JTree treeSource = (JTree) treeSelectionEvent.getSource(); 
       System.out.println("Min: " + treeSource.getMinSelectionRow()); 
       System.out.println("Max: " + treeSource.getMaxSelectionRow()); 
       System.out.println("Lead: " + treeSource.getLeadSelectionRow()); 
       System.out.println("Row: " + treeSource.getSelectionRows()[0]); 
      } 
     }; 
     tree.addTreeSelectionListener(treeSelectionListener); 
     JFrame frame = new JFrame("JTree With Multi-Discontiguous selection"); 
     frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); 
     frame.add(new JScrollPane(tree)); 
     frame.setPreferredSize(new Dimension(380, 320)); 
     frame.setLocation(150, 150); 
     frame.pack(); 
     frame.setVisible(true); 
    } 

    private TreeWithMultiDiscontiguousSelections() { 
    } 
} 
+0

kod losowy fragment ponownie ;-) – kleopatra

0

Chciałbym dodać do odpowiedzi Kleopatry (w oparciu o moje własne rosnące bóle).

W moim szczególnym problemie, musiałem dodać element menu "Wybierz wszystkie dzieci" do menu wyskakującego węzła JTree. Dlatego to rozwiązanie dotyczy wszystkich dzieci wybranego węzła.

TreeNode selectedNode = tree.getSelectionPath().getLastPathComponent(); 
// Expand tree from selected node... 
List<TreePath> paths = new ArrayList<TreePath>(); 
determineTreePaths(selectedNode, paths); // Recursive method call... 

TreePath[] treePaths = new TreePath[paths.size()]; 
Iterator<TreePath> iter = paths.iterator(); 

for (int i = 0; iter.hasNext(); ++i) 
{ 
    treePaths[i] = iter.next(); 
} 

if (paths.size() > 0) 
{ 
    TreePath firstElement = paths.get(0); 
    setSelectionPath(firstElement); 
    scrollPathToVisible(firstElement); 
}  

determineTreePaths(selectedNode, paths) rekurencyjne wywołanie jest potrzebne, aby przechodzić przez drzewa od wybranego węzła w dół do węzłów liściowych. To rozwiązanie działa niezależnie od głębokości (zgodnie z moją najlepszą wiedzą). Nie mogę powiedzieć, że jest to najbardziej wydajne rozwiązanie. Każdy, kto ma lepsze rozwiązanie, możesz opublikować inne rozwiązanie lub edytować to.

Realizacja sposób jest następujący:

private void determineTreePaths(TreeNode currentNode, List<TreePath> paths) 
{ 
    paths.add(new TreePath(((DefaultTreeModel) getDefaultTreeModel()).getPathToRoot(currentNode)); 

    // Get all of my Children 
    Enumeration<?> children = currentNode.children(); 

    // iterate over my children 
    while (children.hasMoreElements()) 
    { 
     TreeNode child = (TreeNode) children.nextElement(); 
     determineTreePaths(child, paths); 
    } 
} 
Powiązane problemy