javascript - 在提交表单之前等待异步调用完成
问题描述
这个有点棘手。
所以这是我的 jQuery 代码:
/**
* Updating websites
*/
$('.website').on('blur', function () {
var websiteId = this.id;
console.log(websiteId);
var website = this.value;
console.log(website);
fetch(`/api/edit/${websiteId}&${website}`).then(() => {
});
})
这是我的 HTML
<form method="post">
<% for(let i = 0; i < websites.length; i++){ let website = websites[i]; %>
<fieldset id="site<%= i %>">
<p style="color: <% if(errorMsgs[i] === 'Wrong website address'){ %> red;<% } else { %> green;<% } %>
padding-bottom: 0;
padding-top: 10px;
margin-bottom: 0;"
class="">
<%= errorMsgs[i] %>
</p>
<div class="">
<input class="website"
name="website<%= i %>"
id="<%= i %>"
value="<%= website %>"
type="text"/>
<a href="/websites/edit/<%= i %>"
class="btn btn-info hide">
Edit
</a>
<a href="/websites/delete/<%= i %>"
data-id="<%= i %>"
class="btn btn-danger confirmation removeBtn">
Remove
</a>
</div>
</fieldset>
<% } %>
<button type="submit"
class="btn btn-primary col-sm-offset-2"
value="Generate a report"
name="generateReport"
data-toggle="modal"
data-target="#exampleModal">
Generate a report
</button>
<!-- Save button created using Javascript. In case JS is disabled -->
</form>
这是我的 API
// GET: api/edit/_id - save updates
router.get('/edit/:_id&:url', (req, res, next) => {
Account.findById(req.user._id, (err, acc) => {
if (err) {
console.log(err);
} else {
acc.websites.set(req.params._id, req.params.url);
acc.save((err, webs) => {
if (err) {
console.log(err);
} else {
console.log('all good');
// res.redirect('/reports');
res.json(webs);
}
});
}
});
});
它是如何工作的:用户将值输入到输入字段中,模糊时,我对 API 进行 AJAX 调用,值保存在数据库中。这一切都有效。
不过有一个问题。如果用户点击提交按钮,则 fetch 没有时间运行。这是有道理的。
但我需要提交等到值更新。我正在考虑做类似的事情,将 a 添加preventDefault
到提交按钮,然后在保存后,删除preventDefault
.
但我不知道该怎么做,如果这是一个好主意,或者即使它有任何意义。
回顾一下:我需要表单提交等到 AJAX 调用完成。
这<% smth %>
只是EJS
模板系统。不应该影响它的运作方式。
解决方案
它看起来像fetch()
返回一个承诺。将这些承诺保存在共享变量中。在您的提交按钮代码中,使用Promise.all()
等待所有承诺解决。就像是:
const fetchPromises = [];
$('.website').on('blur', function () {
var websiteId = this.id;
console.log(websiteId);
var website = this.value;
console.log(website);
var fetchPromise = fetch(`/api/edit/${websiteId}&${website}`).then(() => {
});
fetchPromises.push(fetchPromise);
});
$('form').on('submit', function (e) {
e.preventDefault();
Promise.all(fetchPromises).then(() => this.submit());
});
您可能想要添加一些错误处理。如果您的任何fetch()
调用失败,这将阻止提交表单。理想情况下,失败的 AJAX 调用会导致向用户发送一些消息,一旦解除,就会从fetchPromises
数组中删除承诺,并且在解除之前,提交按钮应该被禁用(实际上是可见的)。
推荐阅读
- ansible - 如何在ansible playbook中调用加密密码?
- string - SAS为宏变量中的所有单词添加前缀
- python-3.x - Autoscaling 组中 EC2 实例的基于计划的启动/停止
- node.js - 如何在 Docker 容器中运行数据库服务?
- java - 我错过了什么?NumberFormatException 错误
- javascript - 必须在 for 循环中初始化正则表达式
- c# - 使用资产包在统一 webgl 中加载/卸载巨大的世界
- javascript - 将字符串转换为 json 对象
- ios - 突然 fcm 通知在某些 iOS 设备上不起作用
- youtube-api - Youtube API 视频列表返回的结果少于指示的结果