添加FXML中定义的元素以循环列出

时间:2015-12-16 10:49:13

标签: java list javafx javafx-2

我有很多名为a1,a2,a3,...的对象 我需要将它们放入List中,以便使用它们更简单,我会以某种方式使用循环吗?

我的尝试是:

List<SomeObject> list = new LinkedList<SomeObject>();
for (int i=0; i<1000; i++){
    String varName = "a"+i;
    list.add((SomeObject) varName);
}

在这种情况下有人有建议吗? 在循环内创建变量不是解决方案,因为它们是.fxml文档的一部分。 或者给我一个如何使用循环创建它的建议,因为它在.fxml中创建行并行添加循环新对象。

更容易理解.fxml文件看起来像

  <SomeObject fx:id="a1" *other props* />
  <SomeObject fx:id="a2" *other props* />
  <SomeObject fx:id="a3" *other props* />
  <SomeObject fx:id="a4" *other props* />

非常感谢您的建议!

3 个答案:

答案 0 :(得分:5)

如果您有这么多项目,最好使用Java初始化它们,而不是使用FXML。例如,而不是:

<FlowPane fx:id="container" minWidth="..." minHeight="...">
    <Label fx:id="label1" text="Label 1"/>
    <Label fx:id="label2" text="Label 2"/>
    <Label fx:id="label3" text="Label 3"/>

    <!-- ... -->

    <Label fx:id="label1000" text="Label 1000"/>
</FlowPane>

和控制器

public class Controller {

    @FXML
    private FlowPane container ;
    @FXML
    private Label label1 ;
    @FXML
    private Label label2 ;
    // ...

    @FXML
    private Label label1000 ;

    // ...
}

我愿意

<FlowPane fx:id="container" minWidth="..." minHeight="...">
</FlowPane>

public class Controller {

    @FXML
    private FlowPane container ;

    private List<Label> labels ;

    public void initialize() {
        labels = new ArrayList<>();
        for (int i = 1; i <= 1000; i++) {
            Label label = new Label("Label "+i);
            labels.add(label);
            container.getChildren().add(label);
        }
    }
}

作为此想法的变体,请考虑定义自定义组件:

public class LabelFlow extends FlowPane {

    private List<Label> labels ;

    public LabelFlow(@NamedArg("numLabels") int numLabels) {
        labels = new ArrayList<>();
        for(int i = 1 ; i <= numLabels ; i++) {
            Label label = new Label("Label "+i);
            labels.add(label);
        }
        getChildren().addAll(labels);
    }

    public List<Label> getLabels() {
        return Collections.unmodifiableList(labels);
    }
}

现在使用FXML

<LabelFlow fx:id="labelFlow" numLabels="1000"/>

并在您的控制器中

public class Controller {
    @FXML
    private LabelFlow labelFlow ;

    public void initialize() {
        for (Label label : labelFlow.getLabels()) {
            // do whatever you need with label....
        }
    }
}

如果你想在Scene Builder中使用类似的自定义类,你需要跳过几个箍。见Adding a custom component to SceneBuilder 2.0

如果确实想要在FXML中定义所有这些控件,这将是一个维护噩梦,你可以使用反射来访问变量。我不建议这样做,不仅因为它很难维护,而且因为它本质上的反射容易出错(没有编译时检查)而且复杂。

但你可以做到

public class Controller {

    @FXML
    private FlowPane container ;
    @FXML
    private Label label1 ;
    @FXML
    private Label label2 ;
    // ...

    @FXML
    private Label label1000 ;

    private List<Label> labels ;

    public void initialize() throws Exception {
        labels = new ArrayList<>();
        for (int i = 1; i <= 1000; i++) {
            Field field = getClass().getDeclaredField("label"+i);
            boolean wasAccessible = field.isAccessible();
            field.setAccessible(true);
            Label label = (Label) field.get(this);
            field.setAccessible(wasAccessible);
            labels.add(label);
        }
    }
}

答案 1 :(得分:4)

您可能希望直接将对象放入fxml的列表中:

<fx:define>
     <FXCollections fx:id="theList" fx:factory="observableArrayList">
         <SomeObject someProperty="SomeValue 1" />
         <SomeObject someProperty="SomeValue 2" />
         <SomeObject someProperty="SomeValue 3" />
         <SomeObject someProperty="SomeValue 4" />
      </FXCollections>
</fx:define>

并在控制器中访问它:

@FXML
private ObservableList<SomeObject> theList;

@Override
public void initialize( URL url, ResourceBundle rb )
{
    // do whatever with the list
    System.out.println( "the list of some objects = " + theList);
}

如果您没有单独访问每个SomeObject,则可以删除fx:id个。有关fxml功能的更多信息,请参阅Introduction to FXML

答案 2 :(得分:2)

我个人更喜欢另一种可能性,但仅仅是为了完整性:

使用FXMLLoader.getNamespace()加载后,您还可以通过Object属性访问FXMLLoader创建的fx:id

实施例

namespace.fxml

<AnchorPane xmlns:fx="http://javafx.com/fxml/1">
    <children>
        <Text fx:id="node1" text="42"/>
    </children>
</AnchorPane>

载入

FXMLLoader loader = new FXMLLoader(getClass().getResource("namespace.fxml"));
loader.load();
System.out.println(((Text)loader.getNamespace().get("node1")).getText());

打印fxml中Text元素的文本(即42)。

当然,您无法在控制器中访问FXMLLoader实例,这意味着使用此方法您需要自己将信息传递给控制器​​,如果需要的话。