React Class 组件中的同步 Meteor 调用

Synchronous Meteor Call in React Class Component

我已经对此进行了很长一段时间的试验,但一直无法完全弄清楚。我正在尝试在 React class 组件中进行同步 Meteor 方法调用,作为我正在做的注册表单的一部分。基本上,每次用户名输入字段更改 (onChange) 时,我都会不断检查用户名是否被占用,并更新视觉指示器。我知道这不是理想的设计,因为一旦你将它扩展,它是昂贵的数据库调用,所以我会根据需要改变 this/throttle 它,但那是在我降低这个基本功能之后。

这是我在服务器上的 Meteor 方法:

checkUsername({ username }) {
    var result = false;
    if (username != "") {
        if (Accounts.findUserByUsername(username)) {
            result = true;
        }
    }
    return result;
},

这是我在 React class 组件中的函数:

async checkUsername(username) {
    Session.set("usernameValid", true);
    var syncCall = await Meteor.call(
        "checkUsername",
        {
            username,
        }
    );
    // Shouldn't the line of code below run only AFTER the call above completes?
    if (syncCall === false) Session.set("usernameValid", false);
    return Session.get("usernameValid");
}

我知道这不需要异步,我也不需要等待。此外,我知道这段代码是错误的,因为 syncCall 总是未定义的。这是我检查用户名可用性的地方:

if (username === "") {
    this.state.username = false;
    this.setFeedback("username", "", false);
}
else if (this.checkUsername(username)) {
    // Username is valid and available
    console.log("Username is available!");
    this.state.username = username;
    this.setFeedback("username", "", true);
}
else {
    // Username is invalid and nonempty
    console.log("Username is invalid/unavailable!");
    this.state.username = false;
    this.setFeedback("username", "Username is invalid/unavailable :-(", false);
}

问题是 CheckUsername 函数 returns Meteor 调用完成之前,因此状态没有及时更新,我无法检查要么。

我该如何强制使其同步?我阅读了有关 Meteor.wrapAsync() 的信息,但我无法确定如何正确地合并它(或者即使它是正确的方法)。感谢任何帮助,谢谢!

客户端上的 Meteor 方法调用不 return Promise,因此您不能 await 它们。您需要提供回调函数:

checkUsername(username) {
    Session.set("usernameValid", true);
    Meteor.call("checkUsername", {username},
      (isTaken) => isTaken && Session.set("usernameValid", false));
}

但是对于 React,我不会使用会话变量。 React 通过 useState 对反应状态变量提供了自己的支持,并且集成得更好:

const [usernameValid, setUsernameValid] = useState(false);

...

checkUsername(username) {    
  Meteor.call("checkUsername", {username}, setUsernameValid);
}