-
Notifications
You must be signed in to change notification settings - Fork 273
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat(anchor): [anchor] add top-offset props #2388
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
@@ -168,11 +168,11 @@ export const onItersectionObserver = | |||||||||||||||||||||||||||||||||||||
({ state, props, api, vm, emit }: Pick<IAnchorRenderlessParams, 'state' | 'props' | 'api' | 'vm' | 'emit'>) => | ||||||||||||||||||||||||||||||||||||||
() => { | ||||||||||||||||||||||||||||||||||||||
const { expandLink, scrollContainer } = state | ||||||||||||||||||||||||||||||||||||||
const { topOffset } = props | ||||||||||||||||||||||||||||||||||||||
state.currentLink && updateSkidPosition({ vm, state, emit }) | ||||||||||||||||||||||||||||||||||||||
const rootMargin = topOffset ? `${-topOffset}px 0px 0px 0px` : '' | ||||||||||||||||||||||||||||||||||||||
state.intersectionObserver = new IntersectionObserver( | ||||||||||||||||||||||||||||||||||||||
(entries) => { | ||||||||||||||||||||||||||||||||||||||
const { top } = scrollContainer.getBoundingClientRect() | ||||||||||||||||||||||||||||||||||||||
const scrollStartTop = top + state.offsetTop | ||||||||||||||||||||||||||||||||||||||
entries.forEach((item) => { | ||||||||||||||||||||||||||||||||||||||
const key = item.target.id | ||||||||||||||||||||||||||||||||||||||
state.observerLinks[key] = item | ||||||||||||||||||||||||||||||||||||||
|
@@ -194,11 +194,7 @@ export const onItersectionObserver = | |||||||||||||||||||||||||||||||||||||
for (let key in state.observerLinks) { | ||||||||||||||||||||||||||||||||||||||
if (Object.prototype.hasOwnProperty.call(state.observerLinks, key)) { | ||||||||||||||||||||||||||||||||||||||
const item = state.observerLinks[key] | ||||||||||||||||||||||||||||||||||||||
if ( | ||||||||||||||||||||||||||||||||||||||
item.isIntersecting && | ||||||||||||||||||||||||||||||||||||||
item.intersectionRatio >= 0 && | ||||||||||||||||||||||||||||||||||||||
item.target.getBoundingClientRect().top < scrollStartTop | ||||||||||||||||||||||||||||||||||||||
) { | ||||||||||||||||||||||||||||||||||||||
if (item.isIntersecting && item.intersectionRatio >= 0) { | ||||||||||||||||||||||||||||||||||||||
const link = `#${item.target.id}` | ||||||||||||||||||||||||||||||||||||||
if (!expandLink[link].children) { | ||||||||||||||||||||||||||||||||||||||
api.getCurrentAnchor(link) | ||||||||||||||||||||||||||||||||||||||
|
@@ -210,7 +206,7 @@ export const onItersectionObserver = | |||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||
}, | ||||||||||||||||||||||||||||||||||||||
{ root: scrollContainer, threshold: [0, 0.25, 0.5, 1] } | ||||||||||||||||||||||||||||||||||||||
{ root: scrollContainer, threshold: [0, 0.25, 0.5, 1], rootMargin } | ||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||
addObserver({ props, state }) | ||||||||||||||||||||||||||||||||||||||
|
@@ -232,7 +228,11 @@ export const linkClick = | |||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||
if (scrollContainer && scrollContainer !== document.body && !isChangeHash) { | ||||||||||||||||||||||||||||||||||||||
const linkEl = scrollContainer.querySelector(item.link) as HTMLElement | ||||||||||||||||||||||||||||||||||||||
const top = linkEl?.offsetTop - scrollContainer.offsetTop // 修复横向锚点无法滚动到顶部 | ||||||||||||||||||||||||||||||||||||||
const top = | ||||||||||||||||||||||||||||||||||||||
linkEl?.getBoundingClientRect().top - | ||||||||||||||||||||||||||||||||||||||
scrollContainer.getBoundingClientRect().top + | ||||||||||||||||||||||||||||||||||||||
scrollContainer.scrollTop - | ||||||||||||||||||||||||||||||||||||||
props.topOffset // 修复横向锚点无法滚动到顶部 | ||||||||||||||||||||||||||||||||||||||
Comment on lines
+231
to
+235
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Add null safety and consider cross-browser compatibility. The scroll position calculation looks correct but has the following concerns:
Add null safety and fallback: - const top =
- linkEl?.getBoundingClientRect().top -
- scrollContainer.getBoundingClientRect().top +
- scrollContainer.scrollTop -
- props.topOffset
- const param = { top, left: 0, behavior: 'smooth' } as ScrollToOptions
- scrollContainer?.scrollTo(param)
+ if (linkEl) {
+ const top =
+ linkEl.getBoundingClientRect().top -
+ scrollContainer.getBoundingClientRect().top +
+ scrollContainer.scrollTop -
+ (props.topOffset || 0)
+ try {
+ scrollContainer.scrollTo({ top, left: 0, behavior: 'smooth' })
+ } catch (e) {
+ // Fallback for browsers that don't support smooth scrolling
+ scrollContainer.scrollTop = top
+ }
+ } 📝 Committable suggestion
Suggested change
|
||||||||||||||||||||||||||||||||||||||
const param = { top, left: 0, behavior: 'smooth' } as ScrollToOptions | ||||||||||||||||||||||||||||||||||||||
scrollContainer?.scrollTo(param) | ||||||||||||||||||||||||||||||||||||||
scrollContainer?.addEventListener('scroll', api.handleScroll()) | ||||||||||||||||||||||||||||||||||||||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Add type checking and validation for topOffset.
The
topOffset
property is used without type checking or validation. Consider adding runtime checks to ensure it's a non-negative number.Add validation before using topOffset:
Also applies to: 173-173