有没有办法改变TableView的焦点可遍历行为?

问题描述 投票:1回答:1

我有一个简单的TableView,节点方向为Right-To-Left,setCellSelectionEnabled(true)。现在,当在表格上按下右方向键时,焦点可遍历行为在对面,所以左边的单元格将被选中。对于另一边也是如此;当按下左箭头键时,右侧的单元格将被选中。为了区域的目的,TableView的节点方向属性必须保持为RTL。那么如何解决这个问题呢?

下面是TableView的FXML代码。

<TableView fx:id="table" editable="true" layoutX="97.0" layoutY="170.0" nodeOrientation="RIGHT_TO_LEFT" prefHeight="400.0" prefWidth="816.0" AnchorPane.bottomAnchor="25.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" AnchorPane.topAnchor="121.0">
         <placeholder>
            <Label text="داده ای یافت نشد :(" />
         </placeholder>
         <columns>
            <TableColumn fx:id="numberColumn" maxWidth="120.0" minWidth="80.0" text="ردیف" />
            <TableColumn fx:id="nameColumn" maxWidth="400.0" minWidth="300.0" prefWidth="300.0" text="نام ماده اولیه" />
            <TableColumn fx:id="categoryColumn" maxWidth="300.0" minWidth="150.0" prefWidth="150.0" text="دسته بندی" />
            <TableColumn fx:id="priceColumn" maxWidth="300.0" minWidth="220.0" prefWidth="220.0" text="قیمت هر کیلوگرم(ریال)" />
            <TableColumn fx:id="unitColumn" maxWidth="200.0" minWidth="100.0" prefWidth="100.0" text="واحد" />
            <TableColumn fx:id="numberOfUsesColumn" maxWidth="120.0" minWidth="120.0" prefWidth="120.0" text="دفعات استفاده" />
            <TableColumn fx:id="deleteRowColumn" maxWidth="90.0" minWidth="90.0" prefWidth="90.0" resizable="false" text="حذف سطر" />
         </columns>
         <columnResizePolicy>
            <TableView fx:constant="CONSTRAINED_RESIZE_POLICY" />
         </columnResizePolicy>
</TableView>
java javafx tableview right-to-left
1个回答
1
投票

在表格导航中不尊重RTL是个问题。openjfx15中修复的bug。.

在这之前,我们可以通过黑客攻击(假设我们被允许去做脏事,即使用反射来访问TableViewSkin中的一个私有字段,使用内部的api和实现细节;)。

  • 获取表的皮肤(通常在表显示出来后才可以使用),并访问它的内部字段行为。
  • 从行为的输入映射中获取lef-tright keyMappings,并将其移除。
  • 使用它们各自的事件处理程序来交叉连接键到处理程序,并将它们添加到

在代码中,类似于(这里只针对leftright,需要对带修饰符的映射做同样的处理)。

protected void hackNavigation(TableView<?> table) {
    TableViewSkin<?> skin = (TableViewSkin<?>) table.getSkin();
    // access private field reflectively 
    // use your own favorite utility method :)
    TableViewBehavior<?> behavior = (TableViewBehavior<?>) 
            FXUtils.invokeGetFieldValue(TableViewSkin.class, skin, "behavior");
    // access mappings
    ObservableList<Mapping<?>> mappings = behavior.getInputMap().getMappings();
    // lookup the original mappings for left/right
    KeyBinding leftBinding = new KeyBinding(KeyCode.LEFT);
    KeyBinding rightBinding = new KeyBinding(KeyCode.RIGHT);
    KeyMapping leftMapping = getMapping(mappings, leftBinding); 
    KeyMapping rightMapping = getMapping(mappings, rightBinding);
    // remove the original mappings
    mappings.removeAll(leftMapping, rightMapping);
    // create new mappings with the opposite event handlers and add them
    KeyMapping replaceRight = new KeyMapping(rightBinding, leftMapping.getEventHandler());
    KeyMapping replaceLeft = new KeyMapping(leftBinding, rightMapping.getEventHandler());
    mappings.addAll(replaceRight, replaceLeft);
}

/**
 * Utility method to get hold of a KeyMapping for a binding. 
 * Note: only use if certain that it is contained, or guard against failure 
 */
protected KeyMapping getMapping(ObservableList<Mapping<?>> mappings, KeyBinding keyBinding) {
    Optional<KeyMapping> opt = mappings.stream()
            .filter(mapping -> mapping instanceof KeyMapping)
            .map(mapping -> (KeyMapping) mapping)
            .filter(keyMapping -> keyMapping.getMappingKey().equals(keyBinding))
            .findAny()
            ;
    return opt.get();
}

// useage
stage.show();
hackNavigation(myTable);
© www.soinside.com 2019 - 2024. All rights reserved.