From d78438a899a599d936e893674f09f99dd8e75b29 Mon Sep 17 00:00:00 2001 From: neverland Date: Sun, 15 Nov 2020 10:43:12 +0800 Subject: [PATCH] fix(IndexBar): incorrect behavior inside popup (#7559) --- src/index-anchor/index.js | 21 +++++++++++++++++++-- src/index-bar/index.js | 22 ++++------------------ src/index-bar/test/index.spec.js | 14 ++------------ 3 files changed, 25 insertions(+), 32 deletions(-) diff --git a/src/index-anchor/index.js b/src/index-anchor/index.js index 2852203a8..f93ca0dc4 100644 --- a/src/index-anchor/index.js +++ b/src/index-anchor/index.js @@ -1,6 +1,7 @@ import { createNamespace } from '../utils'; import { ChildrenMixin } from '../mixins/relation'; import { BORDER_BOTTOM } from '../utils/constant'; +import { getScrollTop, getRootScrollTop } from '../utils/dom/scroll'; const [createComponent, bem] = createNamespace('index-anchor'); @@ -15,6 +16,7 @@ export default createComponent({ return { top: 0, left: null, + rect: { top: 0, height: 0 }, width: null, active: false, }; @@ -39,20 +41,35 @@ export default createComponent({ }, mounted() { - this.height = this.$el.offsetHeight; + const rect = this.$el.getBoundingClientRect(); + this.rect.height = rect.height; }, methods: { scrollIntoView() { this.$el.scrollIntoView(); }, + + getRect(scroller, scrollerRect) { + const el = this.$el; + const elRect = el.getBoundingClientRect(); + this.rect.height = elRect.height; + + if (scroller === window || scroller === document.body) { + this.rect.top = elRect.top + getRootScrollTop(); + } else { + this.rect.top = elRect.top + getScrollTop(scroller) - scrollerRect.top; + } + + return this.rect; + }, }, render() { const { sticky } = this; return ( -
+
({ - height: item.height, - top: this.getElementTop(item.$el, scrollerRect), - })); + const rects = this.children.map((item) => + item.getRect(this.scroller, scrollerRect) + ); const active = this.getActiveAnchorIndex(scrollTop, rects); @@ -131,7 +129,7 @@ export default createComponent({ } else if (index === active - 1) { const activeItemTop = rects[active].top - scrollTop; item.active = activeItemTop > 0; - item.top = activeItemTop + scrollerRect.top - item.height; + item.top = activeItemTop + scrollerRect.top - rects[index].height; } else { item.active = false; } @@ -150,18 +148,6 @@ export default createComponent({ }; }, - getElementTop(ele, scrollerRect) { - const { scroller } = this; - - if (scroller === window || scroller === document.body) { - return getElementTop(ele); - } - - const eleRect = ele.getBoundingClientRect(); - - return eleRect.top - scrollerRect.top + getScrollTop(scroller); - }, - getActiveAnchorIndex(scrollTop, rects) { for (let i = this.children.length - 1; i >= 0; i--) { const prevHeight = i > 0 ? rects[i - 1].height : 0; diff --git a/src/index-bar/test/index.spec.js b/src/index-bar/test/index.spec.js index 65390e9fb..9d557d693 100644 --- a/src/index-bar/test/index.spec.js +++ b/src/index-bar/test/index.spec.js @@ -1,13 +1,5 @@ import { mount, trigger, triggerDrag, mockScrollIntoView } from '../../../test'; -function mockOffsetHeight(offsetHeight) { - Object.defineProperty(HTMLElement.prototype, 'offsetHeight', { - get() { - return offsetHeight; - }, - }); -} - test('should allow to custom anchor text', () => { const wrapper = mount({ template: ` @@ -97,11 +89,10 @@ test('should update active anchor after page scroll', () => { const { index } = this.dataset; return { top: index ? index * 10 : 0, + height: 10, }; }; - mockOffsetHeight(10); - const wrapper = mount({ template: ` @@ -138,11 +129,10 @@ test('should emit change event when active index changed', () => { const { index } = this.dataset; return { top: index ? index * 10 : 0, + height: 10, }; }; - mockOffsetHeight(10); - const onChange = jest.fn(); mount({