diff --git a/src/index-anchor/index.js b/src/index-anchor/index.tsx similarity index 75% rename from src/index-anchor/index.js rename to src/index-anchor/index.tsx index 144e7658b..365536c44 100644 --- a/src/index-anchor/index.js +++ b/src/index-anchor/index.tsx @@ -1,14 +1,13 @@ -import { ref, reactive, computed, onMounted } from 'vue'; +import { ref, reactive, computed, CSSProperties } from 'vue'; // Utils import { createNamespace } from '../utils'; import { BORDER_BOTTOM } from '../utils/constant'; -import { INDEX_BAR_KEY } from '../index-bar'; +import { INDEX_BAR_KEY, IndexBarProvide } from '../index-bar'; import { getScrollTop, getRootScrollTop } from '../utils/dom/scroll'; // Composition import { useRect, useParent } from '@vant/use'; -import { useHeight } from '../composables/use-height'; import { useExpose } from '../composables/use-expose'; const [createComponent, bem] = createNamespace('index-anchor'); @@ -28,7 +27,16 @@ export default createComponent({ }); const root = ref(); - const { parent } = useParent(INDEX_BAR_KEY); + const { parent } = useParent(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; @@ -42,11 +50,14 @@ export default createComponent({ width: state.width ? `${state.width}px` : null, transform: state.top ? `translate3d(0, ${state.top}px, 0)` : null, color: highlightColor, - }; + } as CSSProperties; } }); - const getRect = (scrollParent, scrollParentRect) => { + const getRect = ( + scrollParent: Window | Element, + scrollParentRect: { top: number } + ) => { const rootRect = useRect(root); state.rect.height = rootRect.height; @@ -60,10 +71,6 @@ export default createComponent({ return state.rect; }; - onMounted(() => { - state.rect.height = useHeight(root); - }); - useExpose({ state, getRect, @@ -75,7 +82,7 @@ export default createComponent({ return (
, default: genAlphabet, }, }, @@ -57,20 +66,22 @@ export default createComponent({ emits: ['select', 'change'], setup(props, { emit, slots }) { - const root = ref(); + const root = ref(); const activeAnchor = ref(); const touch = useTouch(); const scrollParent = useScrollParent(root); - const { children, linkChildren } = useChildren(INDEX_BAR_KEY); + const { children, linkChildren } = useChildren( + INDEX_BAR_KEY + ); linkChildren({ props }); const sidebarStyle = computed(() => { if (isDef(props.zIndex)) { return { - zIndex: 1 + props.zIndex, - }; + zIndex: +props.zIndex + 1, + } as CSSProperties; } }); @@ -78,12 +89,12 @@ export default createComponent({ if (props.highlightColor) { return { color: props.highlightColor, - }; + } as CSSProperties; } }); const getScrollerRect = () => { - if (scrollParent.value.getBoundingClientRect) { + if ('getBoundingClientRect' in scrollParent.value!) { return useRect(scrollParent); } return { @@ -92,7 +103,10 @@ export default createComponent({ }; }; - const getActiveAnchor = (scrollTop, rects) => { + const getActiveAnchor = ( + scrollTop: number, + rects: Array<{ top: number; height: number }> + ) => { for (let i = children.length - 1; i >= 0; i--) { const prevHeight = i > 0 ? rects[i - 1].height : 0; const reachTop = props.sticky ? prevHeight + props.stickyOffsetTop : 0; @@ -111,7 +125,7 @@ export default createComponent({ } const { sticky, indexList } = props; - const scrollTop = getScrollTop(scrollParent.value); + const scrollTop = getScrollTop(scrollParent.value!); const scrollParentRect = getScrollerRect(); const rects = children.map((item) => @@ -172,7 +186,7 @@ export default createComponent({ return ( {index} @@ -180,7 +194,7 @@ export default createComponent({ ); }); - const scrollTo = (index) => { + const scrollTo = (index: string) => { if (!index) { return; } @@ -198,29 +212,34 @@ export default createComponent({ } }; - const scrollToElement = (element) => { + const scrollToElement = (element: HTMLElement) => { const { index } = element.dataset; - scrollTo(index); + if (index) { + scrollTo(index); + } }; - const onClick = (event) => { - scrollToElement(event.target); + const onClickSidebar = (event: MouseEvent) => { + scrollToElement(event.target as HTMLElement); }; - let touchActiveIndex; - const onTouchMove = (event) => { + let touchActiveIndex: string; + + const onTouchMove = (event: TouchEvent) => { touch.move(event); if (touch.isVertical()) { preventDefault(event); const { clientX, clientY } = event.touches[0]; - const target = document.elementFromPoint(clientX, clientY); + const target = document.elementFromPoint( + clientX, + clientY + ) as HTMLElement; if (target) { const { index } = target.dataset; - /* istanbul ignore else */ - if (touchActiveIndex !== index) { + if (index && touchActiveIndex !== index) { touchActiveIndex = index; scrollToElement(target); } @@ -235,7 +254,7 @@ export default createComponent({