如何在另一个 class 中编辑 javafx ui?

How do I edit the javafx ui in another class?

AppRunner class

import java.io.IOException;
import java.util.logging.Level;
import java.util.logging.Logger;
import javafx.fxml.FXMLLoader;

public class AppRunner {

    public static void main(String[] args) {
        startUi.launchPanel(args);
    }

    public static void waitFiveSecToClear() {
        new Thread(
                new Runnable() {
            public void run() {
                try {
                    Thread.sleep(5000);
                } catch (InterruptedException ex) {
                    Logger.getLogger(AppRunner.class.getName()).log(Level.SEVERE, null, ex);
                }
                FXMLLoader loader = new FXMLLoader(getClass().getResource("listView.fxml"));
                try {
                    loader.load();
                } catch (IOException ex) {
                    Logger.getLogger(AppRunner.class.getName()).log(Level.SEVERE, null, ex);
                }
                System.out.println("clearing listView");
                ((ListViewController) loader.getController()).clearAll();
            }
        }).start();
    }
}

ListViewController class

package apprunner;

import java.net.URL;
import java.util.ResourceBundle;
import javafx.application.Platform;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.ListView;

/**
 * FXML Controller class
 *
 */
public class ListViewController implements Initializable {
        
    public ListViewController() {
        
    }
    
    @FXML
    private ListView<String> listView;

    /**
     * Initializes the controller class.
     */
    @Override
    public void initialize(URL url, ResourceBundle rb) {
        // TODO
        listView.getItems().add("clear all!");
        listView.getItems().add("clear all!");
        listView.getItems().add("clear all!");
        listView.getItems().add("clear all!");
        AppRunner.waitFiveSecToClear();
    }

    public void clearAll() {
        Platform.runLater(new Runnable() {
            @Override
            public void run() {
                listView.getItems().clear();
            }
        });

    }
}

startUi class

package apprunner;

/**
 *
 */
import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;

public class startUi extends Application {

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

    @Override
    public void start(Stage stage) throws Exception {
        Parent root = FXMLLoader.load(getClass().getResource("listView.fxml"));
        Scene scene = new Scene(root);
        stage.setScene(scene);
        stage.show();
    }
}

listView.fxml

<?xml version="1.0" encoding="UTF-8"?>

<?import javafx.scene.control.ListView?>
<?import javafx.scene.layout.AnchorPane?>


<AnchorPane id="AnchorPane" prefHeight="400.0" prefWidth="600.0" xmlns:fx="http://javafx.com/fxml/1" xmlns="http://javafx.com/javafx/17" fx:controller="apprunner.ListViewController">
   <children>
      <ListView fx:id="listView" layoutX="68.0" layoutY="39.0" prefHeight="200.0" prefWidth="200.0" />
   </children>
</AnchorPane>

目前没有错误。但是我的代码没有按照我的意愿进行。我希望 AppRunner class 中的 waitFiveSecToClear() 方法清除 javafx ui 中的 listView。然而,这并没有发生。我使用 ((ListViewController) loader.getController()).clearAll(); 试图清除似乎没有做任何事情的 listView。如果有人能帮我解决我的问题,那将是非常感谢!

我的问题的背景

我有一个高中顶点项目。我选择制作一个音乐互联网下载器。当我的程序正在下载音乐时,它会在没有线程的情况下冻结整个 javafx 应用程序,因此我在我的 waitFiveSecToClear() 方法中使用了一个线程和 Thread.sleep(5000) 来模拟它。我的程序有一个 listView,它显示了当前正在下载的所有音乐 url 的队列。我想要这样,当一个 url 下载完成时,listView 会更新,这必须在 requires Platform.runlater 的 Javafx 线程上完成。所以,我在 clearAll() 方法中使用 Platform.runLater 来模拟它。我不想在控制器中这样做,因为根据我了解到的情况,这会破坏模型、视图、控制器架构。我有一个 AppRunner class 因为我的教师程序示例有那个所以我只是复制它。

我按照@James_D 和@jewelsea 的建议使用任务和 ObservableList 解决了我的问题。我的模型中有一个任务,它会下载音乐并在下载完成后从 ObservableList 中删除音乐的 url。然后我的控制器中有一个侦听器,它会在 ObservableList 更改时更新 ListView。