mirror of
https://github.com/youzan/vant.git
synced 2025-10-15 23:55:08 +00:00
98 lines
2.5 KiB
TypeScript
98 lines
2.5 KiB
TypeScript
import { ref, reactive, computed, CSSProperties } from 'vue';
|
|
|
|
// Utils
|
|
import { createNamespace } from '../utils';
|
|
import { BORDER_BOTTOM } from '../utils/constant';
|
|
import { INDEX_BAR_KEY, IndexBarProvide } from '../index-bar';
|
|
import { getScrollTop, getRootScrollTop } from '../utils/dom/scroll';
|
|
|
|
// Composition
|
|
import { useRect, useParent } from '@vant/use';
|
|
import { useExpose } from '../composables/use-expose';
|
|
|
|
const [createComponent, bem] = createNamespace('index-anchor');
|
|
|
|
export default createComponent({
|
|
props: {
|
|
index: [Number, String],
|
|
},
|
|
|
|
setup(props, { slots }) {
|
|
const state = reactive({
|
|
top: 0,
|
|
left: null,
|
|
rect: { top: 0, height: 0 },
|
|
width: null,
|
|
active: false,
|
|
});
|
|
|
|
const root = ref<HTMLElement>();
|
|
const { parent } = useParent<IndexBarProvide>(INDEX_BAR_KEY);
|
|
|
|
if (!parent) {
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
console.error(
|
|
'[Vant] <IndexAnchor> must be a child component of <IndexBar>.'
|
|
);
|
|
}
|
|
return;
|
|
}
|
|
|
|
const isSticky = () => state.active && parent.props.sticky;
|
|
|
|
const anchorStyle = computed(() => {
|
|
const { zIndex, highlightColor } = parent.props;
|
|
|
|
if (isSticky()) {
|
|
return {
|
|
zIndex: `${zIndex}`,
|
|
left: state.left ? `${state.left}px` : null,
|
|
width: state.width ? `${state.width}px` : null,
|
|
transform: state.top ? `translate3d(0, ${state.top}px, 0)` : null,
|
|
color: highlightColor,
|
|
} as CSSProperties;
|
|
}
|
|
});
|
|
|
|
const getRect = (
|
|
scrollParent: Window | Element,
|
|
scrollParentRect: { top: number }
|
|
) => {
|
|
const rootRect = useRect(root);
|
|
state.rect.height = rootRect.height;
|
|
|
|
if (scrollParent === window || scrollParent === document.body) {
|
|
state.rect.top = rootRect.top + getRootScrollTop();
|
|
} else {
|
|
state.rect.top =
|
|
rootRect.top + getScrollTop(scrollParent) - scrollParentRect.top;
|
|
}
|
|
|
|
return state.rect;
|
|
};
|
|
|
|
useExpose({
|
|
state,
|
|
getRect,
|
|
});
|
|
|
|
return () => {
|
|
const sticky = isSticky();
|
|
|
|
return (
|
|
<div
|
|
ref={root}
|
|
style={{ height: sticky ? `${state.rect.height}px` : undefined }}
|
|
>
|
|
<div
|
|
style={anchorStyle.value}
|
|
class={[bem({ sticky }), { [BORDER_BOTTOM]: sticky }]}
|
|
>
|
|
{slots.default ? slots.default() : props.index}
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
},
|
|
});
|