AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / coding / Perguntas / 79566179
Accepted
Arturos
Arturos
Asked: 2025-04-10 16:57:28 +0800 CST2025-04-10 16:57:28 +0800 CST 2025-04-10 16:57:28 +0800 CST

expansão "limitada" de um grupo de visualização de tabela em árvore quando a tecla Alt é mantida pressionada

  • 772

Meu objetivo é adicionar um recurso ao existente TreeTableViewonde clicar no nó de divulgação de um grupo com Alta tecla pressionada deve expandir apenas o grupo associado, mas com todos os seus filhos (subgrupos) recolhidos, independentemente de terem sido expandidos anteriormente.

Sem que a Alttecla seja mantida pressionada, o nó de divulgação deve apenas acionar o comportamento "padrão".

Por exemplo, vamos supor que o usuário tenha a seguinte hierarquia de nós:

ROOT
-> GROUP_1
----> GROUP_1_1
--------> CHILD_1_1_1
--------> CHILD_1_1_2
--------> CHILD_1_1_3
----> GROUP_1_2
--------> CHILD_1_2_1
--------> CHILD_1_2_2
-> GROUP_2
-> GROUP_3
-> ...

Quando o usuário mantém pressionada a Alttecla e clica no nó de divulgação ao lado ROOTpara expandi-lo, espera-se que o sistema mostre. É essencial que os grupos GROUP_1_1e GROUP_1_2sejam recolhidos.

ROOT
-> GROUP_1
-> GROUP_2
-> GROUP_3
-> ...

A diferença entre esse comportamento e o padrão é que se o usuário mantiver a Alttecla pressionada e clicar no nó de divulgação ao lado, ROOTo sistema deverá garantir que tudo dentro dele seja recolhido no final.

Estou procurando funções como CellBehaviorBase#doSelecte TreeTableCellBehavior#handleDisclosureNode, mas até agora não consegui encontrar nenhuma dica de como conseguir o que quero.

Exemplo mínimo para meus testes. O objetivo é ter a possibilidade de expandir o grupo raiz ("Carros") com todos os subgrupos recolhidos quando ALTpressionado:

import javafx.application.Application;
import javafx.beans.property.SimpleStringProperty;
import javafx.beans.property.StringProperty;
import javafx.scene.Scene;
import javafx.scene.control.TreeItem;
import javafx.scene.control.TreeTableColumn;
import javafx.scene.control.TreeTableView;
import javafx.scene.layout.Pane;
import javafx.scene.layout.VBox;
import javafx.stage.Stage;

public class HelloApplication extends Application {
    @Override
    public void start(Stage stage) {
        Pane rootNode = new VBox();
        Scene scene = new Scene(rootNode, 400, 300);
        stage.setTitle("TreeTableView Demo");
        stage.setScene(scene);

        TreeTableView<Car> treeTableView = new TreeTableView<>();

        TreeTableColumn<Car, String> treeTableColumn1 = new TreeTableColumn<>("Brand");
        treeTableColumn1.setPrefWidth(120);
        TreeTableColumn<Car, String> treeTableColumn2 = new TreeTableColumn<>("Model");
        treeTableColumn2.setPrefWidth(120);

        treeTableColumn1.setCellValueFactory(param -> param.getValue().getValue().brandProperty());
        treeTableColumn2.setCellValueFactory(param -> param.getValue().getValue().modelProperty());

        treeTableView.getColumns().add(treeTableColumn1);
        treeTableView.getColumns().add(treeTableColumn2);

        TreeItem<Car> mercedes1 = new TreeItem<>(new Car("Mercedes", "SL500"));
        TreeItem<Car> mercedes2 = new TreeItem<>(new Car("Mercedes", "SL500 AMG"));
        TreeItem<Car> mercedes3 = new TreeItem<>(new Car("Mercedes", "CLA 200"));

        TreeItem<Car> mercedes = new TreeItem<>(new Car("Mercedes", "..."));
        mercedes.getChildren().add(mercedes1);
        mercedes.getChildren().add(mercedes2);

        TreeItem<Car> audi1 = new TreeItem<>(new Car("Audi", "A1"));
        TreeItem<Car> audi2 = new TreeItem<>(new Car("Audi", "A5"));
        TreeItem<Car> audi3 = new TreeItem<>(new Car("Audi", "A7"));

        TreeItem<Car> audi = new TreeItem<>(new Car("Audi", "..."));
        audi.getChildren().add(audi1);
        audi.getChildren().add(audi2);
        audi.getChildren().add(audi3);

        TreeItem<Car> cars = new TreeItem<>(new Car("Cars", "..."));
        cars.getChildren().add(audi);
        cars.getChildren().add(mercedes);

        treeTableView.setRoot(cars);

        rootNode.getChildren().add(treeTableView);

        stage.show();
    }

    public static void main(String[] args) {
        launch();
    }
}

class Car {
    private StringProperty brand = new SimpleStringProperty();
    private StringProperty model = new SimpleStringProperty();

    public Car(String brand, String model) {
        this.brand.set(brand);
        this.model.set(model);
    }

    public String getBrand() {
        return brand.get();
    }

    public StringProperty brandProperty() {
        return brand;
    }

    public void setBrand(String brand) {
        this.brand.set(brand);
    }

    public String getModel() {
        return model.get();
    }

    public StringProperty modelProperty() {
        return model;
    }

    public void setModel(String model) {
        this.model.set(model);
    }
}
java
  • 2 2 respostas
  • 83 Views

2 respostas

  • Voted
  1. Best Answer
    Sai Dandem
    2025-04-11T14:14:45+08:002025-04-11T14:14:45+08:00

    Sua ideia de lidar com Behaviora lógica é válida. Mas, infelizmente, isso não ajuda. Como o comportamento da célula é estritamente limitado pela camada de célula, será muito difícil controlar a lógica que ocorre internamente antes de tentar assumir o controle (mesmo com a implementação personalizada de Célula, Pele e Comportamento).

    Dito isso, você pode procurar outras maneiras de ajustar isso. Uma delas que me vem à mente é implementar um despachador de eventos personalizado e desviar o processamento de eventos quando as condições desejadas forem atendidas.

    A ideia geral desta abordagem é:

    • Implemente um despachador de eventos personalizado na célula da tabela e rastreie seus eventos.
    • Depois de atingir a condição desejada de mouse pressionado com alt pressionado, você verifica se o evento ocorreu no nó de divulgação.
    • Se isso ocorrer no nó de divulgação, desvie/suprima o fluxo de eventos e execute sua lógica personalizada (reduzindo os itens filhos e expandindo o item atual)

    Abaixo está o código da célula personalizada conforme a abordagem acima.

    class CustomTreeTableCell<T, S> extends TreeTableCell<T, S> {
    
        public CustomTreeTableCell() {
            // Keep the reference of original dispatcher
            final EventDispatcher orig = getEventDispatcher();
    
            // Set a new event dispatcher
            setEventDispatcher((event, tail) -> {
                // If the event is mouse pressed and is alt down, then do this logic otherwise proceed with usual orig dispatch.
                if (event instanceof MouseEvent me && me.getEventType() == MouseEvent.MOUSE_PRESSED && me.isAltDown()) {
                    Node disclosureNode = getTableRow().getDisclosureNode();
                    // Check if disclosure node exists and visible
                    if (disclosureNode != null && disclosureNode.isVisible()) {
                        // Check if the mouse press happened on the node by checking the mouse point on node bounds.
                        if (disclosureNode.localToScene(disclosureNode.getLayoutBounds()).contains(me.getSceneX(), me.getSceneY())) {
                            // When you reach this point, you do your logic. ie. collapse all its child items and expand this item.
                            final TreeItem<T> treeItem = getTableRow().getTreeItem();
                            treeItem.getChildren().forEach(this::collapse);
                            treeItem.setExpanded(true);
    
                            // Ensure to consume the event by returning null;
                            return null;
                        }
                    }
                }
    
                // If it is not your desired event, let the original dispatcher do its stuff.
                return orig.dispatchEvent(event, tail);
            });
        }
    
        /**
         * Recursive method to collapse all child items and provided item.
         *
         * @param item tree item
         */
        private void collapse(TreeItem<?> item) {
            item.setExpanded(false);
            item.getChildren().forEach(this::collapse);
        }
    
        @Override
        protected void updateItem(S item, boolean empty) {
            super.updateItem(item, empty);
            setText(empty || item == null ? "" : item.toString());
        }
    }
    

    Observação: a abordagem de verificar o ponteiro do mouse nos limites do nó de divulgação não é algo novo. A lógica interna do método TreeTableCellBehaviortambém faz isso handleDisclosureNode.

    Abaixo está a demonstração completa com seu exemplo:

    insira a descrição da imagem aqui

    import javafx.application.Application;
    import javafx.beans.property.SimpleStringProperty;
    import javafx.beans.property.StringProperty;
    import javafx.event.EventDispatcher;
    import javafx.scene.Node;
    import javafx.scene.Scene;
    import javafx.scene.control.TreeItem;
    import javafx.scene.control.TreeTableCell;
    import javafx.scene.control.TreeTableColumn;
    import javafx.scene.control.TreeTableView;
    import javafx.scene.input.MouseEvent;
    import javafx.scene.layout.Pane;
    import javafx.scene.layout.VBox;
    import javafx.stage.Stage;
    
    public class TreeTableView_Demo extends Application {
        @Override
        public void start(Stage stage) {
            Pane rootNode = new VBox();
            Scene scene = new Scene(rootNode, 400, 300);
            stage.setTitle("TreeTableView Demo");
            stage.setScene(scene);
    
            TreeTableView<Car> treeTableView = new TreeTableView<>();
    
            TreeTableColumn<Car, String> treeTableColumn1 = new TreeTableColumn<>("Brand");
            treeTableColumn1.setCellFactory(p -> new CustomTreeTableCell<>());
            treeTableColumn1.setPrefWidth(120);
    
            TreeTableColumn<Car, String> treeTableColumn2 = new TreeTableColumn<>("Model");
            treeTableColumn2.setPrefWidth(120);
    
            treeTableColumn1.setCellValueFactory(param -> param.getValue().getValue().brandProperty());
            treeTableColumn2.setCellValueFactory(param -> param.getValue().getValue().modelProperty());
    
            treeTableView.getColumns().add(treeTableColumn1);
            treeTableView.getColumns().add(treeTableColumn2);
    
            TreeItem<Car> mercedes1 = new TreeItem<>(new Car("Mercedes", "SL500"));
            TreeItem<Car> mercedes2 = new TreeItem<>(new Car("Mercedes", "SL500 AMG"));
            TreeItem<Car> mercedes3 = new TreeItem<>(new Car("Mercedes", "CLA 200"));
    
            TreeItem<Car> mercedes = new TreeItem<>(new Car("Mercedes", "..."));
            mercedes.getChildren().add(mercedes1);
            mercedes.getChildren().add(mercedes2);
    
            TreeItem<Car> audi1 = new TreeItem<>(new Car("Audi", "A1"));
            TreeItem<Car> audi2 = new TreeItem<>(new Car("Audi", "A5"));
            TreeItem<Car> audi3 = new TreeItem<>(new Car("Audi", "A7"));
    
            TreeItem<Car> audi = new TreeItem<>(new Car("Audi", "..."));
            audi.getChildren().add(audi1);
            audi.getChildren().add(audi2);
            audi.getChildren().add(audi3);
    
            TreeItem<Car> cars = new TreeItem<>(new Car("Cars", "..."));
            cars.getChildren().add(audi);
            cars.getChildren().add(mercedes);
    
            treeTableView.setRoot(cars);
    
            rootNode.getChildren().add(treeTableView);
    
            stage.show();
        }
    
        public static void main(String[] args) {
            launch();
        }
    
        class CustomTreeTableCell<T, S> extends TreeTableCell<T, S> {
    
            public CustomTreeTableCell() {
                // Keep the reference of original dispatcher
                final EventDispatcher orig = getEventDispatcher();
                // Set a new event dispatcher
                setEventDispatcher((event, tail) -> {
                    // If the event is mouse pressed and is alt down, then do this logic otherwise proceed with usual orig dispatch.
                    if (event instanceof MouseEvent me && me.getEventType() == MouseEvent.MOUSE_PRESSED && me.isAltDown()) {
                        Node disclosureNode = getTableRow().getDisclosureNode();
                        // Check if disclosure node exists and visible
                        if (disclosureNode != null && disclosureNode.isVisible()) {
                            // Check if the mouse press happened on the node by checking the mouse point on node bounds.
                            if (disclosureNode.localToScene(disclosureNode.getLayoutBounds()).contains(me.getSceneX(), me.getSceneY())) {
                                // When you reach this point, you do your logic. ie. collapse all its child items and expand this item.
                                final TreeItem<T> treeItem = getTableRow().getTreeItem();
                                treeItem.getChildren().forEach(this::collapse);
                                treeItem.setExpanded(true);
    
                                // Ensure to consume the event by returning null;
                                return null;
                            }
                        }
                    }
    
                    // If it is not your desired event, let the original dispatcher do its stuff.
                    return orig.dispatchEvent(event, tail);
                });
            }
    
            /**
             * Recursive method to collapse all child items and provided item.
             *
             * @param item tree item
             */
            private void collapse(TreeItem<?> item) {
                item.setExpanded(false);
                item.getChildren().forEach(this::collapse);
            }
    
            @Override
            protected void updateItem(S item, boolean empty) {
                super.updateItem(item, empty);
                setText(empty || item == null ? "" : item.toString());
            }
        }
    }
    
    class Car {
        private StringProperty brand = new SimpleStringProperty();
        private StringProperty model = new SimpleStringProperty();
    
        public Car(String brand, String model) {
            this.brand.set(brand);
            this.model.set(model);
        }
    
        public String getBrand() {
            return brand.get();
        }
    
        public StringProperty brandProperty() {
            return brand;
        }
    
        public void setBrand(String brand) {
            this.brand.set(brand);
        }
    
        public String getModel() {
            return model.get();
        }
    
        public StringProperty modelProperty() {
            return model;
        }
    
        public void setModel(String model) {
            this.model.set(model);
        }
    }
    
    • 3
  2. VGR
    2025-04-12T06:01:20+08:002025-04-12T06:01:20+08:00

    Isso é um pouco complicado, mas... você pode ouvir a tecla Alt sendo pressionada e usá-la quando ocorrer um evento de expansão de ramificação :

    import javafx.application.Application;
    import javafx.beans.property.SimpleStringProperty;
    import javafx.beans.property.StringProperty;
    import javafx.scene.Scene;
    import javafx.scene.Node;
    import javafx.scene.Parent;
    import javafx.scene.control.TreeItem;
    import javafx.scene.control.TreeTableColumn;
    import javafx.scene.control.TreeTableRow;
    import javafx.scene.control.TreeTableView;
    import javafx.scene.layout.Pane;
    import javafx.scene.layout.VBox;
    import javafx.stage.Stage;
    import javafx.scene.input.KeyEvent;
    import javafx.scene.input.KeyCode;
    
    public class TreeTablePartialExpansionDemo extends Application {
        static class Car {
            private StringProperty brand = new SimpleStringProperty();
            private StringProperty model = new SimpleStringProperty();
    
            public Car(String brand, String model) {
                this.brand.set(brand);
                this.model.set(model);
            }
    
            public String getBrand() {
                return brand.get();
            }
    
            public StringProperty brandProperty() {
                return brand;
            }
    
            public void setBrand(String brand) {
                this.brand.set(brand);
            }
    
            public String getModel() {
                return model.get();
            }
    
            public StringProperty modelProperty() {
                return model;
            }
    
            public void setModel(String model) {
                this.model.set(model);
            }
        }
    
        private boolean altDown;
    
        @Override
        public void start(Stage stage) {
            Pane rootNode = new VBox();
            Scene scene = new Scene(rootNode, 400, 300);
            stage.setTitle("TreeTableView Demo");
            stage.setScene(scene);
    
            TreeTableView<Car> treeTableView = new TreeTableView<>();
    
            TreeTableColumn<Car, String> treeTableColumn1 = new TreeTableColumn<>("Brand");
            treeTableColumn1.setPrefWidth(120);
            TreeTableColumn<Car, String> treeTableColumn2 = new TreeTableColumn<>("Model");
            treeTableColumn2.setPrefWidth(120);
    
            treeTableColumn1.setCellValueFactory(param -> param.getValue().getValue().brandProperty());
            treeTableColumn2.setCellValueFactory(param -> param.getValue().getValue().modelProperty());
    
            treeTableView.getColumns().add(treeTableColumn1);
            treeTableView.getColumns().add(treeTableColumn2);
    
            TreeItem<Car> mercedes1 = new TreeItem<>(new Car("Mercedes", "SL500"));
            TreeItem<Car> mercedes2 = new TreeItem<>(new Car("Mercedes", "SL500 AMG"));
            TreeItem<Car> mercedes3 = new TreeItem<>(new Car("Mercedes", "CLA 200"));
    
            TreeItem<Car> mercedes = new TreeItem<>(new Car("Mercedes", "..."));
            mercedes.getChildren().add(mercedes1);
            mercedes.getChildren().add(mercedes2);
    
            TreeItem<Car> audi1 = new TreeItem<>(new Car("Audi", "A1"));
            TreeItem<Car> audi2 = new TreeItem<>(new Car("Audi", "A5"));
            TreeItem<Car> audi3 = new TreeItem<>(new Car("Audi", "A7"));
    
            TreeItem<Car> audi = new TreeItem<>(new Car("Audi", "..."));
            audi.getChildren().add(audi1);
            audi.getChildren().add(audi2);
            audi.getChildren().add(audi3);
    
            TreeItem<Car> cars = new TreeItem<>(new Car("Cars", "..."));
            cars.getChildren().add(audi);
            cars.getChildren().add(mercedes);
    
            treeTableView.setRoot(cars);
            rootNode.getChildren().add(treeTableView);
    
            treeTableView.getScene().addEventFilter(KeyEvent.KEY_PRESSED, e -> {
                if (e.getCode() == KeyCode.ALT) {
                    altDown = true;
                }
            });
            treeTableView.getScene().addEventFilter(KeyEvent.KEY_RELEASED, e -> {
                if (e.getCode() == KeyCode.ALT) {
                    altDown = false;
                }
            });
            treeTableView.getScene().getWindow().focusedProperty().subscribe(
                focused -> altDown &= focused);
            treeTableView.getRoot().addEventFilter(TreeItem.branchExpandedEvent(),
                e -> {
                    if (altDown) {
                        e.getTreeItem().getChildren().forEach(
                            item -> item.setExpanded(false));
                    }
                });
    
            stage.show();
        }
    
        public static void main(String[] args) {
            launch();
        }
    }
    

    Adicionei um private boolean altDown;campo, e adicionei isso logo antes de mostrar o Palco:

    treeTableView.getScene().addEventFilter(KeyEvent.KEY_PRESSED, e -> {
        if (e.getCode() == KeyCode.ALT) {
            altDown = true;
        }
    });
    treeTableView.getScene().addEventFilter(KeyEvent.KEY_RELEASED, e -> {
        if (e.getCode() == KeyCode.ALT) {
            altDown = false;
        }
    });
    treeTableView.getScene().getWindow().focusedProperty().subscribe(
        focused -> altDown &= focused);
    treeTableView.getRoot().addEventFilter(TreeItem.branchExpandedEvent(),
        e -> {
            if (altDown) {
                e.getTreeItem().getChildren().forEach(
                    item -> item.setExpanded(false));
            }
        });
    

    As três primeiras instruções rastreiam se a tecla Alt está pressionada. A última instrução intercepta eventos de expansão e, se altDownfor verdadeira, colapsa os filhos da raiz.

    • 1

relate perguntas

  • Lock Condition.notify está lançando java.lang.IllegalMonitorStateException

  • Resposta de microsserviço Muitos para Um não aparece no carteiro

  • Validação personalizada do SpringBoot Bean

  • Os soquetes Java são FIFO?

  • Por que não é possível / desencorajado definir um lado do servidor de tempo limite de solicitação?

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    Reformatar números, inserindo separadores em posições fixas

    • 6 respostas
  • Marko Smith

    Por que os conceitos do C++20 causam erros de restrição cíclica, enquanto o SFINAE antigo não?

    • 2 respostas
  • Marko Smith

    Problema com extensão desinstalada automaticamente do VScode (tema Material)

    • 2 respostas
  • Marko Smith

    Vue 3: Erro na criação "Identificador esperado, mas encontrado 'import'" [duplicado]

    • 1 respostas
  • Marko Smith

    Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores?

    • 1 respostas
  • Marko Smith

    Como faço para corrigir um erro MODULE_NOT_FOUND para um módulo que não importei manualmente?

    • 6 respostas
  • Marko Smith

    `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso?

    • 3 respostas
  • Marko Smith

    Um programa vazio que não faz nada em C++ precisa de um heap de 204 KB, mas não em C

    • 1 respostas
  • Marko Smith

    PowerBI atualmente quebrado com BigQuery: problema de driver Simba com atualização do Windows

    • 2 respostas
  • Marko Smith

    AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos

    • 1 respostas
  • Martin Hope
    Fantastic Mr Fox Somente o tipo copiável não é aceito na implementação std::vector do MSVC 2025-04-23 06:40:49 +0800 CST
  • Martin Hope
    Howard Hinnant Encontre o próximo dia da semana usando o cronógrafo 2025-04-21 08:30:25 +0800 CST
  • Martin Hope
    Fedor O inicializador de membro do construtor pode incluir a inicialização de outro membro? 2025-04-15 01:01:44 +0800 CST
  • Martin Hope
    Petr Filipský Por que os conceitos do C++20 causam erros de restrição cíclica, enquanto o SFINAE antigo não? 2025-03-23 21:39:40 +0800 CST
  • Martin Hope
    Catskul O C++20 mudou para permitir a conversão de `type(&)[N]` de matriz de limites conhecidos para `type(&)[]` de matriz de limites desconhecidos? 2025-03-04 06:57:53 +0800 CST
  • Martin Hope
    Stefan Pochmann Como/por que {2,3,10} e {x,3,10} com x=2 são ordenados de forma diferente? 2025-01-13 23:24:07 +0800 CST
  • Martin Hope
    Chad Feller O ponto e vírgula agora é opcional em condicionais bash com [[ .. ]] na versão 5.2? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench Por que um traço duplo (--) faz com que esta cláusula MariaDB seja avaliada como verdadeira? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng Por que `dict(id=1, **{'id': 2})` às vezes gera `KeyError: 'id'` em vez de um TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos 2024-03-20 03:12:31 +0800 CST

Hot tag

python javascript c++ c# java typescript sql reactjs html

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve