element-plus
Version: 
A Component Library for Vue3.0
148 lines (142 loc) • 5.18 kB
JavaScript
import { defineComponent, ref, reactive, computed, watch, onMounted, onBeforeUnmount, openBlock, createBlock, createVNode, renderSlot } from 'vue';
import { getScrollContainer, on, off } from '../utils/dom';
import { addResizeListener, removeResizeListener } from '../utils/resize-event';
var script = defineComponent({
    name: 'ElAffix',
    props: {
        zIndex: {
            type: Number,
            default: 100,
        },
        target: {
            type: String,
            default: '',
        },
        offset: {
            type: Number,
            default: 0,
        },
        position: {
            type: String,
            default: 'top',
        },
    },
    emits: ['scroll', 'change'],
    setup(props, { emit }) {
        const target = ref(null);
        const root = ref(null);
        const scrollContainer = ref(null);
        const state = reactive({
            fixed: false,
            height: 0,
            width: 0,
            scrollTop: 0,
            clientHeight: 0,
            transform: 0,
        });
        const rootStyle = computed(() => {
            return {
                height: state.fixed ? `${state.height}px` : '',
                width: state.fixed ? `${state.width}px` : '',
            };
        });
        const affixStyle = computed(() => {
            if (!state.fixed) {
                return;
            }
            const offset = props.offset ? `${props.offset}px` : 0;
            const transform = state.transform ? `translateY(${state.transform}px)` : '';
            return {
                height: `${state.height}px`,
                width: `${state.width}px`,
                top: props.position === 'top' ? offset : '',
                bottom: props.position === 'bottom' ? offset : '',
                transform: transform,
                zIndex: props.zIndex,
            };
        });
        const updateState = () => {
            const rootRect = root.value.getBoundingClientRect();
            const targetRect = target.value.getBoundingClientRect();
            state.height = rootRect.height;
            state.width = rootRect.width;
            state.scrollTop = scrollContainer.value === window ? document.documentElement.scrollTop : scrollContainer.value.scrollTop;
            state.clientHeight = document.documentElement.clientHeight;
            if (props.position === 'top') {
                if (props.target) {
                    const difference = targetRect.bottom - props.offset - state.height;
                    state.fixed = props.offset > rootRect.top && targetRect.bottom > 0;
                    state.transform = difference < 0 ? difference : 0;
                }
                else {
                    state.fixed = props.offset > rootRect.top;
                }
            }
            else {
                if (props.target) {
                    const difference = state.clientHeight - targetRect.top - props.offset - state.height;
                    state.fixed = state.clientHeight - props.offset < rootRect.bottom && state.clientHeight > targetRect.top;
                    state.transform = difference < 0 ? -difference : 0;
                }
                else {
                    state.fixed = state.clientHeight - props.offset < rootRect.bottom;
                }
            }
        };
        const onScroll = () => {
            updateState();
            emit('scroll', {
                scrollTop: state.scrollTop,
                fixed: state.fixed,
            });
        };
        watch(() => state.fixed, () => {
            emit('change', state.fixed);
        });
        onMounted(() => {
            if (props.target) {
                target.value = document.querySelector(props.target);
                if (!target.value) {
                    throw new Error(`target is not existed: ${props.target}`);
                }
            }
            else {
                target.value = document.documentElement;
            }
            scrollContainer.value = getScrollContainer(root.value);
            on(scrollContainer.value, 'scroll', onScroll);
            addResizeListener(root.value, updateState);
        });
        onBeforeUnmount(() => {
            off(scrollContainer.value, 'scroll', onScroll);
            removeResizeListener(root.value, updateState);
        });
        return {
            root,
            state,
            rootStyle,
            affixStyle,
        };
    },
});
function render(_ctx, _cache, $props, $setup, $data, $options) {
  return (openBlock(), createBlock("div", {
    ref: "root",
    class: "el-affix",
    style: _ctx.rootStyle
  }, [
    createVNode("div", {
      class: {'el-affix--fixed': _ctx.state.fixed},
      style: _ctx.affixStyle
    }, [
      renderSlot(_ctx.$slots, "default")
    ], 6 /* CLASS, STYLE */)
  ], 4 /* STYLE */))
}
script.render = render;
script.__file = "packages/affix/src/index.vue";
script.install = (app) => {
    app.component(script.name, script);
};
const _Affix = script;
export default _Affix;