Blame view

src/views/sys/iframe/index.vue 2.89 KB
陈文彬 authored
1
<template>
2
  <div :class="prefixCls" :style="getWrapStyle">
陈文彬 authored
3
    <Spin :spinning="loading" size="large" :style="getWrapStyle">
4
      <iframe :src="frameSrc" :class="`${prefixCls}__main`" ref="frameRef"></iframe>
陈文彬 authored
5
6
7
8
    </Spin>
  </div>
</template>
<script lang="ts">
9
10
  import type { CSSProperties } from 'vue';
  import { defineComponent, ref, unref, onMounted, nextTick, computed } from 'vue';
陈文彬 authored
11
12
13
  import { Spin } from 'ant-design-vue';

  import { getViewportOffset } from '/@/utils/domUtils';
14
15
  import { useWindowSizeFn } from '/@/hooks/event/useWindowSizeFn';
16
17
18
19
  import { propTypes } from '/@/utils/propTypes';
  import { useDesign } from '/@/hooks/web/useDesign';
陈文彬 authored
20
21
22
23
  export default defineComponent({
    name: 'IFrame',
    components: { Spin },
    props: {
24
      frameSrc: propTypes.string.def(''),
陈文彬 authored
25
26
    },
    setup() {
27
      const loading = ref(false);
陈文彬 authored
28
29
      const topRef = ref(50);
      const heightRef = ref(window.innerHeight);
30
31
32
33
34
35
36
37
38
39
40
41
      const frameRef = ref<HTMLFrameElement | null>(null);

      const { prefixCls } = useDesign('iframe-page');
      useWindowSizeFn(calcHeight, 150, { immediate: true });

      const getWrapStyle = computed(
        (): CSSProperties => {
          return {
            height: `${unref(heightRef)}px`,
          };
        }
      );
陈文彬 authored
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56

      function calcHeight() {
        const iframe = unref(frameRef);
        if (!iframe) {
          return;
        }
        let { top } = getViewportOffset(iframe);
        top += 20;
        topRef.value = top;
        heightRef.value = window.innerHeight - top;
        const clientHeight = document.documentElement.clientHeight - top;
        iframe.style.height = `${clientHeight}px`;
      }

      function hideLoading() {
57
        loading.value = false;
陈文彬 authored
58
59
60
61
62
63
        calcHeight();
      }

      function init() {
        nextTick(() => {
          const iframe = unref(frameRef);
64
65
66
67
68
          if (!iframe) return;

          const _frame = iframe as any;
          if (_frame.attachEvent) {
            _frame.attachEvent('onload', () => {
陈文彬 authored
69
70
71
72
73
74
75
76
77
78
              hideLoading();
            });
          } else {
            iframe.onload = () => {
              hideLoading();
            };
          }
        });
      }
      onMounted(() => {
79
        loading.value = true;
陈文彬 authored
80
81
        init();
      });
82
陈文彬 authored
83
      return {
84
85
        getWrapStyle,
        loading,
陈文彬 authored
86
        frameRef,
87
        prefixCls,
陈文彬 authored
88
89
90
91
92
      };
    },
  });
</script>
<style lang="less" scoped>
93
94
95
  @prefix-cls: ~'@{namespace}-iframe-page';

  .@{prefix-cls} {
陈文彬 authored
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
    .ant-spin-nested-loading {
      position: relative;
      height: 100%;

      .ant-spin-container {
        width: 100%;
        height: 100%;
        padding: 10px;
      }
    }

    &__mask {
      position: absolute;
      top: 0;
      left: 0;
      width: 100%;
      height: 100%;
    }

    &__main {
      width: 100%;
      height: 100%;
      overflow: hidden;
      background: #fff;
      border: 0;
      box-sizing: border-box;
    }
  }
</style>