我已经对此进行了很长一段时间的试验,但一直无法完全弄清楚。我正在尝试在 React 类组件中进行同步 Meteor 方法调用,作为我正在做的注册表单的一部分。基本上,每次用户名输入字段更改 (onChange) 时,我都会不断检查用户名是否被占用,并更新视觉指示器。我知道这不是理想的设计,因为一旦你将它扩展,它是昂贵的数据库调用,所以我会根据需要改变它/限制它,但那是在我降低这个基本功能之后。
这是我在服务器上的 Meteor 方法:
checkUsername({ username }) {
var result = false;
if (username != "") {
if (Accounts.findUserByUsername(username)) {
result = true;
}
}
return result;
},
这是我在 React 类组件中的函数:
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 函数在 Meteor 调用完成之前返回,因此状态没有及时更新,我也无法检查。
我如何着手强制这是同步的?我阅读了有关 Meteor.wrapAsync()
的内容,但我无法确定如何正确地合并它(或者即使它是正确的方法)。感谢您的帮助,谢谢!
最佳答案
客户端上的 Meteor 方法调用不会返回 Promise,因此您不能 await
它们。你需要提供一个回调函数:
checkUsername(username) {
Session.set("usernameValid", true);
Meteor.call("checkUsername", {username},
(isTaken) => isTaken && Session.set("usernameValid", false));
}
但是对于 React,我不会使用 session 变量。 React 通过 useState
自身支持响应式状态变量,并且集成得更好:
const [usernameValid, setUsernameValid] = useState(false);
...
checkUsername(username) {
Meteor.call("checkUsername", {username}, setUsernameValid);
}
关于javascript - React类组件中的同步Meteor调用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65414860/