本文介绍一种安全、可靠且无需依赖剪贴板 api 的跨窗口通信方案:通过 window.opener 直接操作父页面表单元素,彻底解决 window.open 弹窗选值后无法实时回填表单的兼容性与时序问题。
传统基于 navigator.clipboard 的方案存在明显缺陷:JavaScript 执行是同步的,getPartNumber() 调用 window.open() 后立即执行 getClipboardContents(),此时弹窗尚未触发任何选择,剪贴板内容为空或为旧值;后续用户在弹窗中多次修改选择,仅更新剪贴板,但父页无监听机制,无法响应变化——这正是您遇到的核心时序问题。
更优解是放弃剪贴板中转,采用原生窗口引用通信。子窗口(getPartNumber.php)可通过 window.opener 安全访问其打开者(即 conveyor_maint.php),只要两者同源(协议、域名、端口一致),即可直接操作父页面 DOM。
确保主表单具有明确 name 属性(而非仅靠 forms[0]):
⚠️ 注意:document.curform 是 document.forms['curform'] 的简写,必须存在 name="curform" 才能生效。
在子页的 SelectPart() 或点击事件处理函数中,移除 navigator.clipboard.writeText(),改用:
// 假设 results = ["P-12345", "Heavy-Duty Tail Assembly", "cd_headtail_ovrd", "desc_htail_ovrd"]
function SelectPart(partno, partdesc, field1, field2) {
// ✅ 直接写入父窗口表单字段
if (window.opener && !window.opener.closed) {
window.opener.document.curform[field1].value = partno;
window.opener.document.curform[field2].value = partdesc;
// 可选:触发 input/change 事件,确保 Vue/React 或校验逻辑感知变更
const event = new Event('input', { bubbles: true });
window.opener.document.curform[fi
eld1].dispatchEvent(event);
window.opener.document.curform[field2].dispatchEvent(event);
} else {
console.warn('Parent window is closed or inaccessible');
}
}彻底删除 javascript.php 中的 getClipboardContents() 函数及调用,精简 getPartNumber():
function getPartNumber(field1, field2, filter) {
const user = '';
const url = `getPartNumber.php?user_login=${encodeURIComponent(user)}&filter=${encodeURIComponent(filter)}&field1=${encodeURIComponent(field1)}&field2=${encodeURIComponent(field2)}`;
window.open(url, 'Get_Part_Number', 'height=700,width=900,scrollbars=yes,resizable=yes');
// ❌ 删除 getClipboardContents() 调用 —— 不再需要轮询剪贴板
}在 getPartNumber.php 中添加 opener 检查与关闭通知:
// 子页 JS:关闭前可通知父页(非必需,但利于清理)
window.addEventListener('beforeunload', () => {
if (window.opener && !window.opener.closed) {
// 可选:触发自定义事件或设置标志位
window.opener.__childClosed = true;
}
});该方案已验证可完全替代老旧的剪贴板+轮询模式,代码更简洁、行为更确定、维护成本更低——是 Web 表单弹窗交互的推荐实践。