index.js 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. import { guid } from '@/uilts/help'
  2. // iframe 通信中心
  3. class IframeMessageCenter {
  4. constructor() {
  5. // 缓存事件队列
  6. this.messageCallbackMap = new Map();
  7. this.messageFailbackMap = new Map();
  8. this.timer = null
  9. this.init()
  10. }
  11. send({ info = {}, data = {}, callback, overTime, failback }) {
  12. if (callback) {
  13. info.messageId = `${info.actionType}-${guid()}` // 唯一的ID,用于标记回调函数
  14. }
  15. window.parent.postMessage({
  16. info,
  17. data
  18. }, '*');
  19. if (info.messageId && callback) {
  20. // 带回调callback 的message, 要求携带messageId,callback,failback等
  21. this.listen(info.messageId, callback)
  22. if (failback) {
  23. this.addFailback(info.messageId, overTime, failback)
  24. }
  25. }
  26. }
  27. listen(key, callback) {
  28. // 序列添加失败回调
  29. this.messageCallbackMap.set(key, { callback })
  30. }
  31. addFailback(key, overTime = 2000, failback) {
  32. // 序列添加失败回调
  33. this.messageFailbackMap.set(key, {
  34. time: new Date().getTime(),
  35. overTime,
  36. failback
  37. })
  38. this.checkTimer()
  39. }
  40. init() {
  41. window.addEventListener('message', (e) => {
  42. let { info = {}, data = {} } = e.data;
  43. let key, item
  44. // 如果有messageId && 是自己发送的
  45. if (info.messageId) {
  46. key = info.messageId
  47. item = this.messageCallbackMap.get(key)
  48. }
  49. if (info.actionType && !item) {
  50. key = info.actionType
  51. item = this.messageCallbackMap.get(key)
  52. }
  53. // 执行成功回调
  54. if (item) {
  55. // 序列删除 失败回调
  56. this.messageFailbackMap.delete(key)
  57. const callback = item.callback
  58. callback(data)
  59. // 序列删除 成功回调
  60. this.messageCallbackMap.delete(key)
  61. }
  62. })
  63. this.checkTimer()
  64. }
  65. checkTimer() {
  66. if (this.timer) {
  67. return
  68. }
  69. let key, value, now_time
  70. this.timer = setInterval(() => {
  71. if (this.messageFailbackMap.size == 0) {
  72. clearInterval(this.timer)
  73. this.timer = null
  74. }
  75. // 轮询查看有无超期的message信息
  76. now_time = new Date().getTime();
  77. for (let item of this.messageFailbackMap.entries()) {
  78. key = item[0] || ''
  79. value = item[1] || {}
  80. if (now_time - value.time > value.overTime) {
  81. const callback = value.failback
  82. callback && callback({
  83. error: 1,
  84. msg: "message 超时错误"
  85. })
  86. this.messageFailbackMap.delete(key)
  87. }
  88. }
  89. }, 1000)
  90. }
  91. }
  92. // messageCenter在每个iframe内实例化一次
  93. const iframeMessageCenter = new IframeMessageCenter();
  94. export default iframeMessageCenter;