ckgl/node_modules/vant/es/sticky/Sticky.mjs
2024-12-21 13:52:42 +08:00

139 lines
4.2 KiB
JavaScript

import { ref, watch, computed, nextTick, reactive, defineComponent, createVNode as _createVNode } from "vue";
import { extend, isHidden, unitToPx, numericProp, windowWidth, windowHeight, getScrollTop, getZIndexStyle, makeStringProp, makeNumericProp, createNamespace } from "../utils/index.mjs";
import { useRect, useEventListener, useScrollParent } from "@vant/use";
import { useVisibilityChange } from "../composables/use-visibility-change.mjs";
const [name, bem] = createNamespace("sticky");
const stickyProps = {
zIndex: numericProp,
position: makeStringProp("top"),
container: Object,
offsetTop: makeNumericProp(0),
offsetBottom: makeNumericProp(0)
};
var stdin_default = defineComponent({
name,
props: stickyProps,
emits: ["scroll", "change"],
setup(props, {
emit,
slots
}) {
const root = ref();
const scrollParent = useScrollParent(root);
const state = reactive({
fixed: false,
width: 0,
// root width
height: 0,
// root height
transform: 0
});
const isReset = ref(false);
const offset = computed(() => unitToPx(props.position === "top" ? props.offsetTop : props.offsetBottom));
const rootStyle = computed(() => {
if (isReset.value) {
return;
}
const {
fixed,
height,
width
} = state;
if (fixed) {
return {
width: `${width}px`,
height: `${height}px`
};
}
});
const stickyStyle = computed(() => {
if (!state.fixed || isReset.value) {
return;
}
const style = extend(getZIndexStyle(props.zIndex), {
width: `${state.width}px`,
height: `${state.height}px`,
[props.position]: `${offset.value}px`
});
if (state.transform) {
style.transform = `translate3d(0, ${state.transform}px, 0)`;
}
return style;
});
const emitScroll = (scrollTop) => emit("scroll", {
scrollTop,
isFixed: state.fixed
});
const onScroll = () => {
if (!root.value || isHidden(root)) {
return;
}
const {
container,
position
} = props;
const rootRect = useRect(root);
const scrollTop = getScrollTop(window);
state.width = rootRect.width;
state.height = rootRect.height;
if (position === "top") {
if (container) {
const containerRect = useRect(container);
const difference = containerRect.bottom - offset.value - state.height;
state.fixed = offset.value > rootRect.top && containerRect.bottom > 0;
state.transform = difference < 0 ? difference : 0;
} else {
state.fixed = offset.value > rootRect.top;
}
} else {
const {
clientHeight
} = document.documentElement;
if (container) {
const containerRect = useRect(container);
const difference = clientHeight - containerRect.top - offset.value - state.height;
state.fixed = clientHeight - offset.value < rootRect.bottom && clientHeight > containerRect.top;
state.transform = difference < 0 ? -difference : 0;
} else {
state.fixed = clientHeight - offset.value < rootRect.bottom;
}
}
emitScroll(scrollTop);
};
watch(() => state.fixed, (value) => emit("change", value));
useEventListener("scroll", onScroll, {
target: scrollParent,
passive: true
});
useVisibilityChange(root, onScroll);
watch([windowWidth, windowHeight], () => {
if (!root.value || isHidden(root) || !state.fixed) {
return;
}
isReset.value = true;
nextTick(() => {
const rootRect = useRect(root);
state.width = rootRect.width;
state.height = rootRect.height;
isReset.value = false;
});
});
return () => {
var _a;
return _createVNode("div", {
"ref": root,
"style": rootStyle.value
}, [_createVNode("div", {
"class": bem({
fixed: state.fixed && !isReset.value
}),
"style": stickyStyle.value
}, [(_a = slots.default) == null ? void 0 : _a.call(slots)])]);
};
}
});
export {
stdin_default as default,
stickyProps
};