【问题标题】:Java FX Send Function to Method as ParameterJava FX 将函数作为参数发送到方法
【发布时间】:2019-05-14 00:11:23
【问题描述】:

更新 2019.05.14 美国东部标准时间下午 4:54 - 好的 - 这是说明我的问题的代码 - 可能我花了太长时间才得到这个,而且可能太长了,但我还是 Java 新手。无论如何 - 它可以工作,提出表单,并且无法让第二个按钮看到并对“事件”做出反应。我想我现在可以“提高”事件了 - 至少它在事件中击中了代码。但是,它仍在运行类代码,而不是通过 setOnFormStateChange 传入的“自定义”代码????我不确定出了什么问题。

所有进口

import java.util.ArrayList;
import java.util.Date;
import javafx.application.Application;
import javafx.beans.property.ObjectProperty;
import javafx.beans.property.SimpleObjectProperty;
import javafx.event.Event;
import javafx.event.EventHandler;
import javafx.event.EventType;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.Label;
import javafx.scene.layout.AnchorPane;
import javafx.stage.Stage;

MyApp.java

public class MyApp extends Application {

    public static void main(String[] args) {
        launch(args); // this method will hang here until main form is closed!
    }

    @Override
    public void start(Stage stage) throws Exception {
        Form myForm = new Form();
        myForm.ShowForm();
    }

}

Form.java

class Form {

    private boolean modified;
    private ArrayList<FormStateChangeListener> registry = new ArrayList<>();

    public void setModified(boolean m) {
        modified = m;
        this.throwStateChange(m);
    }

    public void throwStateChange(boolean m) {
        for (FormStateChangeListener o : registry) {
          //  o.onFormStateChange(m);         
            FormStateChangeEvent.fireEvent(o,m);
        }
    }

    public void ShowForm() {

        Label lbl = new Label("NORMAL STATE");
        lbl.setLayoutX(50);
        lbl.setLayoutY(20);

        Btn myBtn1 = new Btn(this);
        myBtn1.setLayoutX(100);
        myBtn1.setLayoutY(100);
        myBtn1.setPrefWidth(200);

        myBtn1.setText("Press To Change State");
        myBtn1.setOnAction(e -> {
            lbl.setText("CHANGED STATE:" + new Date().toString());
            this.setModified(true);
        });

        Btn myBtn2 = new Btn(this);
        myBtn2.setLayoutX(100);
        myBtn2.setLayoutY(200);
        myBtn2.setPrefWidth(200);
        myBtn2.setText("And This Should React");

        myBtn2.setOnFormStateChange(e -> {
            myBtn2.setText("I REACTED!");
        });


        Stage stage = new Stage();
        AnchorPane root = new AnchorPane();
        Scene scene = new Scene(root, 430, 400);
       // root.getChildren().add(lbl);
       // root.getChildren().add(myBtn1);
       // root.getChildren().add(myBtn2);
        root.getChildren().addAll(lbl,myBtn1,myBtn2);
        stage.setScene(scene);
        stage.showAndWait();
    }

    void registerForEvent(FormStateChangeListener t) {
        registry.add(t);

    }

}

FormStateChangeListener.java

interface FormStateChangeListener {
    public void onFormStateChange(boolean mod);
}

FormState.java

enum FormState {
    NORMAL, MODIFIED, NEW
}

Btn.java

class Btn extends Button implements FormStateChangeListener {

    private final ObjectProperty<EventHandler<? super FormStateChangeEvent>> onFormStateChange
            = new SimpleObjectProperty<EventHandler<? super FormStateChangeEvent>>(this, "onFormStateChange") {

        @Override
        protected void invalidated() {
            setEventHandler(FormStateChangeEvent.FORM_STATE_CHANGE, get());
        }
    };

    public Btn(Form f) { // constructor
        f.registerForEvent(this); // register for the event
    }

    public final void setOnFormStateChange(EventHandler<? super FormStateChangeEvent> handler) {
        onFormStateChange.set(handler);
    }

    public final EventHandler<? super FormStateChangeEvent> getOnFormStateChange() {
        return onFormStateChange.get();
    }

    public final ObjectProperty<EventHandler<? super FormStateChangeEvent>> onFormStateChangeProperty() {
        return onFormStateChange;
    }

    public void onFormStateChange(boolean mod) {
        //in reality nothing would be here, but is just for testing
       System.out.println("Code from class.");
    }
;

}

FormStateChangeEvent.java

class FormStateChangeEvent extends Event {

    public static final EventType<FormStateChangeEvent> ANY = new EventType<>(Event.ANY, "MY_EVENT");
    public static final EventType<FormStateChangeEvent> FORM_STATE_CHANGE = new EventType<>(ANY, "FORM_STATE_CHANGE");

    static void fireEvent(FormStateChangeListener o, boolean mod) {
        //throw new UnsupportedOperationException("Not supported yet."); 
        o.onFormStateChange(mod);
    }

    public FormStateChangeEvent(EventType<? extends FormStateChangeEvent> eventType) {
        super(eventType);
    }
}

【问题讨论】:

  • 是什么阻止您使用普通的二传手? (或者如果你想提供它,使用除此之外的属性?)如果发生了应该触发事件的事情,你调用事件处理程序的handle 方法,如果属性是非空的。跨度>
  • 相关例子或者函数参数可以看herehere

标签: java javafx lambda


【解决方案1】:

setOnMouseClicked 等方法接受一个javafx.event.EventHandler,这是一个功能接口。换句话说,它有一个抽象方法,这意味着它可以用作 lambda 表达式或方法引用的目标。如果您的目标是拥有自己的 Event 类和相应的 onXXX 属性,那么您的 setOnXXX 方法必须像所有其他方法一样接受 EventHandler

但是,如果您希望在正常事件分派期间调用EventHandler,仅设置属性是不够的。您必须向事件调度程序注册它,这可以通过受保护的Node.setEventHandler(EventType,EventHandler) 方法完成。

例如,假设以下是您的事件类:

import javafx.event.Event;
import javafx.event.EventType;

public class MyEvent extends Event {

    public static final EventType<MyEvent> ANY = new EventType<>(Event.ANY, "MY_EVENT");
    public static final EventType<MyEvent> AWESOME_THING = new EventType<>(ANY, "AWESOME_THING");

    public MyEvent(EventType<? extends MyEvent> eventType) {
        super(eventType);
    }

}

那么您的自定义节点将如下所示:

import javafx.beans.property.ObjectProperty;
import javafx.beans.property.SimpleObjectProperty;
import javafx.event.EventHandler;
import javafx.scene.control.Button;

public class MyButton extends Button {

    private final ObjectProperty<EventHandler<? super MyEvent>> onAwesomeThing 
            = new SimpleObjectProperty<>(this, "onAwesomeThing") {
            @Override protected void invalidated() {
                setEventHandler(MyEvent.AWESOME_THING, get());
            }
    };

    public final void setOnAwesomeThing(EventHandler<? super MyEvent> handler) {
        onAwesomeThing.set(handler);
    }

    public final EventHandler<? super MyEvent> getOnAwesomeThing() {
        return onAwesomeThing.get();
    }

    public final ObjectProperty<EventHandler<? super MyEvent>> onAwesomeThingProperty() {
        return onAwesomeThing;
    }

}

通常情况下,除了“常规”事件类型(例如MyEvent.ANY)之外,您会为与您的事件关联的每个 EventType 拥有一个属性。

现在您可以调用myBtn.fireEvent(new MyEvent(MyEvent.AWESOME_THING)),您注册的EventHandler 将被调用。您也可以通过addEventFilteraddEventHandler 注册处理程序。

【讨论】:

  • 谢谢@Slaw!我觉得这是一个非常好的答案,可能是“正确的”,但是在这方面太新了,我需要花一点时间来理解它,以便将它转移到我的特定情况。谢谢!
  • 好的 - 尝试将其复制进去以使其正常工作并使用它来帮助我理解它。遇到编译问题:`````` = new SimpleObjectProperty(this, "onAwesomeThing") ````` 告诉我“无法推断 SimpleObjectProperty 的类型参数,然后在源 8 中不支持- 使用源 9? - 请告知
  • 在 Java 8 中,当实现一个通用匿名类时,您不能使用菱形运算符。只需将其更改为new SimpleObjectProperty&lt;EventHandler&lt;? super MyEvent&gt;&gt;(this, "onAwesomeThing") { ... };。 Java 9 中改进了类型推断系统,这使得在匿名类中重新声明泛型参数变得不必要(谢天谢地)。
  • 好的,谢谢 - 这有效(编译),我可以在我的按钮实例上对一些愚蠢的测试代码执行 btn.setOnMyEvent。但是,请原谅我的初学者,当我调用 "btn.onMyEvent" 时,它正在运行类代码,而不是我在实例时放入的“自定义”代码。也许我不理解“调用”该处理程序的正确方法?感谢您的耐心和帮助!
  • 能否编辑您的问题以提供minimal reproducible example
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2017-09-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多