前端实现跳转子系统,但限制只能跳转一次

项目要求在主系统页面,点击子系统图标-后端返回子系统url-跳转到第三方子系统新窗口,但限制只能跳转一次。

1.记录子系统窗口window对象

于是一开始的实现方案是把子系统新窗口的window对象记录下来,每次点击图标前判断该window的closed属性,如果为false就是没关闭,就弹出"每个应用只能跳转一次,请到对应窗口进行操作"之类的提示语。

javascript 复制代码
handleClick(clientId) {
  // 先判断是否已打开窗口
  if (window[clientId] && !window[clientId].closed) {
    this.$message({ type: 'warning', message: '每个应用只能跳转一次,请切换到对应窗口进行操作' });
  } else {
    // 调用接口
    inter(clientId).then(res => {
      const childWindow = window.open(res.url);
    });
  }
}

但是发现一旦页面刷新,就能突破只能跳转一次的限制,原因是刷新后存储的window会清空,无法实现持久化储存,而window对象也无法存入localStorage、sessionStorage。

然后思考是否可以后端存储子系统窗口的状态,如果点击+跳转过,就不能再次跳转。

然而这种方案需要监听子窗口关闭事件,关闭后需要通知后端更新子窗口状态。但浏览器没有提供关闭事件,只有beforeunload或者unload这种事件,这俩事件除了关闭、刷新页面也可触发。因此放弃这种方案。

2.把需求改为永远跳转到同一个子窗口

如果限制是"只能跳转一次",那每次都跳转到同一个子窗口,其实也是满足需求的。

于是方案改为:先打开唯一名称但不设置url的子窗口,然后判断该窗口的url是否为"about:blank",如果是的话,说明该名称的窗口没有被打开过,则发起请求,然后把后端返回正确的url赋给子窗口的location.href。如果子窗口的url不是"about:blank",那说明已经打开过,这个动作正好focus到打开过的子窗口上。

大概的代码如下:

javascript 复制代码
handleClick(clientId) {
  // 先尝试打开子窗口,注意子窗口的名称需要唯一
  const childWindow = window.open("", childName + LOGIN_TIME);

  // 如果子窗口的url为about:blank,说明这是第一次打开
  if (childWindow.location.href === "about:blank") {
    // 发送请求,获得正确url
    inter(xxx).then(res => {
      // 也可以childWindow =  window.open(res.url, childName + LOGIN_TIME);
      // 如果子系统和门户系统不同源的话,用上面的方法
      childWindow.location.href = res.url;
    }).catch(err => {
      childWindow.close();
    });
}
}

这种方案遇到的新问题是需要将主窗口的一些数据传递给子窗口,比如子系统的token数据。

一开始使用BroadcastChannel来实现消息传递,但是实际操作发现总是未等子窗口准备好,主窗口就已经传递完毕了,所以pass。 最后使用的是localStorage这种笨方法。

相关推荐
军军君01几秒前
Three.js基础功能学习十三:太阳系实例上
前端·javascript·vue.js·学习·3d·前端框架·three
xiaoqi9221 小时前
React Native鸿蒙跨平台如何实现分类页面组件通过searchQuery状态变量管理搜索输入,实现了分类的实时过滤功能
javascript·react native·react.js·ecmascript·harmonyos
打小就很皮...1 小时前
Tesseract.js OCR 中文识别
前端·react.js·ocr
qq_177767371 小时前
React Native鸿蒙跨平台实现应用介绍页,实现了应用信息卡片展示、特色功能网格布局、权限/联系信息陈列、评分展示、模态框详情交互等通用场景
javascript·react native·react.js·ecmascript·交互·harmonyos
2603_949462101 小时前
Flutter for OpenHarmony社团管理App实战:预算管理实现
android·javascript·flutter
wuhen_n2 小时前
JavaScript内存管理与执行上下文
前端·javascript
Hi_kenyon2 小时前
理解vue中的ref
前端·javascript·vue.js
jin1233223 小时前
基于React Native鸿蒙跨平台地址管理是许多电商、外卖、物流等应用的重要功能模块,实现了地址的添加、编辑、删除和设置默认等功能
javascript·react native·react.js·ecmascript·harmonyos
2501_920931703 小时前
React Native鸿蒙跨平台医疗健康类的血压记录,包括收缩压、舒张压、心率、日期、时间、备注和状态
javascript·react native·react.js·ecmascript·harmonyos
落霞的思绪3 小时前
配置React和React-dom为CDN引入
前端·react.js·前端框架