diff --git a/packages/vant/src/sticky/Sticky.tsx b/packages/vant/src/sticky/Sticky.tsx index 47c2ab5a0..eab933d47 100644 --- a/packages/vant/src/sticky/Sticky.tsx +++ b/packages/vant/src/sticky/Sticky.tsx @@ -7,6 +7,7 @@ import { type PropType, type CSSProperties, type ExtractPropTypes, + nextTick, } from 'vue'; // Utils @@ -20,6 +21,8 @@ import { makeStringProp, makeNumericProp, createNamespace, + windowWidth, + windowHeight, } from '../utils'; // Composables @@ -56,12 +59,16 @@ export default defineComponent({ 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 { @@ -72,7 +79,7 @@ export default defineComponent({ }); const stickyStyle = computed(() => { - if (!state.fixed) { + if (!state.fixed || isReset.value) { return; } @@ -146,9 +153,25 @@ export default defineComponent({ }); 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 () => (
-
+
{slots.default?.()}
diff --git a/packages/vant/src/sticky/test/__snapshots__/index.spec.tsx.snap b/packages/vant/src/sticky/test/__snapshots__/index.spec.tsx.snap index 32d6a2b7c..f6c40491d 100644 --- a/packages/vant/src/sticky/test/__snapshots__/index.spec.tsx.snap +++ b/packages/vant/src/sticky/test/__snapshots__/index.spec.tsx.snap @@ -82,6 +82,34 @@ exports[`should sticky inside container when using container prop 2`] = `
`; +exports[`should sticky resize or orientationchange reset root height and width 1`] = ` +
+
+
+ Content +
+
+
+`; + +exports[`should sticky resize or orientationchange reset root height and width 2`] = ` +
+
+
+ Content +
+
+
+`; + exports[`should sticky to bottom after scrolling 1`] = `
{ restore(); }); + +test('should sticky resize or orientationchange reset root height and width', async () => { + const wrapper = mount({ + render() { + return ( + +
+ Content +
+
+ ); + }, + }); + + window.innerWidth = 375; + const mockStickyRect = jest + .spyOn(wrapper.element, 'getBoundingClientRect') + .mockReturnValue({ + top: -100, + bottom: -90, + width: window.innerWidth, + height: 20, + } as DOMRect); + + await mockScrollTop(100); + expect(wrapper.html()).toMatchSnapshot(); + + window.innerWidth = 677; + mockStickyRect.mockReturnValue({ + width: window.innerWidth, + height: 20, + } as DOMRect); + await trigger(window, 'resize'); + await flushPromises(); + expect(wrapper.html()).toMatchSnapshot(); + + mockStickyRect.mockRestore(); +});