| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- <template>
- <transition-group
- :class="prefixCls"
- v-bind="$attrs"
- ref="elRef"
- :name="transitionName"
- :tag="tag"
- mode="out-in"
- >
- <div key="component" v-if="isInit">
- <slot :loading="loading"></slot>
- </div>
- <div key="skeleton" v-else>
- <slot name="skeleton" v-if="$slots.skeleton"></slot>
- <Skeleton v-else />
- </div>
- </transition-group>
- </template>
- <script lang="ts">
- import type { PropType } from 'vue';
- import { defineComponent, reactive, onMounted, ref, toRef, toRefs } from 'vue';
- import { Skeleton } from 'ant-design-vue';
- import { useTimeoutFn } from '/@/hooks/core/useTimeout';
- import { useIntersectionObserver } from '/@/hooks/event/useIntersectionObserver';
- import { propTypes } from '/@/utils/propTypes';
- import { useDesign } from '/@/hooks/web/useDesign';
- interface State {
- isInit: boolean;
- loading: boolean;
- intersectionObserverInstance: IntersectionObserver | null;
- }
- export default defineComponent({
- name: 'LazyContainer',
- components: { Skeleton },
- inheritAttrs: false,
- props: {
- // Waiting time, if the time is specified, whether visible or not, it will be automatically loaded after the specified time
- timeout: propTypes.number,
- // The viewport where the component is located. If the component is scrolling in the page container, the viewport is the container
- viewport: {
- type: (typeof window !== 'undefined'
- ? window.HTMLElement
- : Object) as PropType<HTMLElement>,
- default: () => null,
- },
- // Preload threshold, css unit
- threshold: propTypes.string.def('0px'),
- // The scroll direction of the viewport, vertical represents the vertical direction, horizontal represents the horizontal direction
- direction: propTypes.oneOf(['vertical', 'horizontal']).def('vertical'),
- // The label name of the outer container that wraps the component
- tag: propTypes.string.def('div'),
- maxWaitingTime: propTypes.number.def(80),
- // transition name
- transitionName: propTypes.string.def('lazy-container'),
- },
- emits: ['init'],
- setup(props, { emit }) {
- const elRef = ref<any>(null);
- const state = reactive<State>({
- isInit: false,
- loading: false,
- intersectionObserverInstance: null,
- });
- const { prefixCls } = useDesign('lazy-container');
- onMounted(() => {
- immediateInit();
- initIntersectionObserver();
- });
- // If there is a set delay time, it will be executed immediately
- function immediateInit() {
- const { timeout } = props;
- timeout &&
- useTimeoutFn(() => {
- init();
- }, timeout);
- }
- function init() {
- state.loading = true;
- useTimeoutFn(() => {
- if (state.isInit) return;
- state.isInit = true;
- emit('init');
- }, props.maxWaitingTime || 80);
- }
- function initIntersectionObserver() {
- const { timeout, direction, threshold } = props;
- if (timeout) return;
- // According to the scrolling direction to construct the viewport margin, used to load in advance
- let rootMargin = '0px';
- switch (direction) {
- case 'vertical':
- rootMargin = `${threshold} 0px`;
- break;
- case 'horizontal':
- rootMargin = `0px ${threshold}`;
- break;
- }
- try {
- const { stop, observer } = useIntersectionObserver({
- rootMargin,
- target: toRef(elRef.value, '$el'),
- onIntersect: (entries: any[]) => {
- const isIntersecting = entries[0].isIntersecting || entries[0].intersectionRatio;
- if (isIntersecting) {
- init();
- if (observer) {
- stop();
- }
- }
- },
- root: toRef(props, 'viewport'),
- });
- } catch (e) {
- init();
- }
- }
- return {
- elRef,
- prefixCls,
- ...toRefs(state),
- };
- },
- });
- </script>
- <style lang="less">
- @prefix-cls: ~'@{namespace}-lazy-container';
- .@{prefix-cls} {
- width: 100%;
- height: 100%;
- }
- </style>
|