-
Notifications
You must be signed in to change notification settings - Fork 16
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
fix(react-utils): fixed useSticky #369
base: main
Are you sure you want to change the base?
Changes from 1 commit
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 |
---|---|---|
@@ -1,45 +1,139 @@ | ||
import {RefObject, useState} from 'react'; | ||
|
||
import {useEffectOnce, useLatest} from 'react-use'; | ||
|
||
const events: ReadonlySet<keyof WindowEventMap> = new Set<keyof WindowEventMap>([ | ||
'resize', | ||
'scroll', | ||
'touchstart', | ||
'touchmove', | ||
'touchend', | ||
'pageshow', | ||
'load', | ||
'orientationchange', | ||
]); | ||
|
||
export function useSticky<T extends HTMLElement>(elemRef: RefObject<T>) { | ||
import {RefObject, useEffect, useState} from 'react'; | ||
|
||
import throttle from 'lodash/throttle'; | ||
|
||
/** | ||
* Finds the closest parent element with overflow: auto or overflow: scroll. | ||
* | ||
* This function traverses up the DOM tree starting from the given element, | ||
* checking each parent element for overflow: auto or overflow: scroll styles. | ||
* If such an element is found, it is returned as the scroll container. | ||
* If no matching element is found, the global window object is returned. | ||
* | ||
* @param {HTMLElement | null} element - The starting element from which to begin the search. | ||
* @returns {HTMLElement | Window} - The first parent with overflow: auto/scroll, or window. | ||
*/ | ||
const findScrollContainer = (element: HTMLElement | null): HTMLElement | Window => { | ||
let currentElement = element; | ||
while (currentElement) { | ||
const overflow = window.getComputedStyle(currentElement).overflow; | ||
if (overflow === 'auto' || overflow === 'scroll') { | ||
return currentElement; | ||
} | ||
currentElement = currentElement.parentElement; | ||
Comment on lines
+19
to
+22
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. or |
||
} | ||
return window; | ||
}; | ||
|
||
/** | ||
* Finds the value of a CSS variable starting from the specified element. | ||
* If not found locally, it will fallback to the global :root. | ||
* | ||
* @param {HTMLElement | null} element - The starting element to search for the CSS variable. | ||
* @param {string} variableName - The name of the CSS variable to search for. | ||
* @returns {number} - The value of the CSS variable or 0 if not found. | ||
*/ | ||
const findCssVariableValue = (element: HTMLElement | null, variableName: string): number => { | ||
let currentElement = element; | ||
while (currentElement) { | ||
const value = getComputedStyle(currentElement).getPropertyValue(variableName); | ||
if (value) { | ||
return parseFloat(value); | ||
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. If the value is |
||
} | ||
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. If you need to calculate numeric value of css-var, you need to use something like this function, because css-vars can store a css expression like |
||
currentElement = currentElement.parentElement; | ||
} | ||
// Fallback to global :root if not found locally | ||
return ( | ||
parseFloat(getComputedStyle(document.documentElement).getPropertyValue(variableName)) || 0 | ||
); | ||
}; | ||
Comment on lines
+37
to
+48
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. It's can be shorter, like this:
Because |
||
|
||
interface UseStickyOptions { | ||
/** | ||
* Throttle delay in milliseconds for the scroll event handler. | ||
* This controls how frequently the scroll event is processed. | ||
* Lower values make the scroll handler more responsive but can increase CPU usage. | ||
* | ||
* Default is 100ms. | ||
*/ | ||
throttleDelay?: number; | ||
|
||
/** | ||
* An optional name of a CSS variable that defines an offset value for the sticky element. | ||
* The variable is needed to set an offset if there are other sticky or | ||
* fixed elements on the page. | ||
* | ||
* Default is '--g-md-sticky-offset-compensate'. | ||
*/ | ||
offsetCssVariable?: string; | ||
|
||
/** | ||
* An optional element reference to scope the search for the CSS variable. | ||
* If not provided, the search defaults to the global :root. | ||
*/ | ||
cssVariableScope?: HTMLElement | null; | ||
} | ||
|
||
/** | ||
* useSticky hook | ||
* | ||
* This hook determines whether an element should be in a sticky state based on its position | ||
* relative to the scroll container and a custom offset. | ||
* The offset is calculated based on the CSS variable `--g-md-toolbar-sticky-offset`. | ||
* | ||
* @param {RefObject<T>} elemRef - A reference to the DOM element for which sticky state is being managed. | ||
* @param {UseStickyOptions} options - Options for configuring the hook's behavior. | ||
* @returns {boolean} - A boolean indicating whether the element is currently sticky. | ||
*/ | ||
export function useSticky<T extends HTMLElement>( | ||
elemRef: RefObject<T>, | ||
{ | ||
throttleDelay = 100, | ||
offsetCssVariable = '--g-md-sticky-offset-compensate', | ||
cssVariableScope = null, | ||
}: UseStickyOptions = {}, | ||
) { | ||
const [sticky, setSticky] = useState(false); | ||
const stickyRef = useLatest(sticky); | ||
const [initialOffset, setInitialOffset] = useState<number | null>(null); | ||
|
||
useEffect(() => { | ||
const scrollContainer = findScrollContainer(elemRef.current); | ||
|
||
useEffectOnce(() => { | ||
observe(); | ||
if (elemRef.current) { | ||
if (initialOffset === null) { | ||
// Determine the scope element for the CSS variable search | ||
const scopeElement = cssVariableScope || document.documentElement; | ||
const stickyOffsetCompensate = findCssVariableValue( | ||
scopeElement, | ||
offsetCssVariable, | ||
); | ||
|
||
for (const eventName of events) { | ||
window.addEventListener(eventName, observe, true); | ||
setInitialOffset( | ||
elemRef.current.getBoundingClientRect().top - stickyOffsetCompensate, | ||
); | ||
} | ||
} | ||
|
||
return () => { | ||
for (const eventName of events) { | ||
window.removeEventListener(eventName, observe, true); | ||
// Throttled scroll handler | ||
const handleScroll = throttle(() => { | ||
if (initialOffset !== null) { | ||
const scrollY = | ||
scrollContainer === window | ||
? window.scrollY | ||
: (scrollContainer as HTMLElement).scrollTop; | ||
const newSticky = (initialOffset ?? 0) <= scrollY; | ||
|
||
setSticky(newSticky); | ||
} | ||
}; | ||
}, throttleDelay); | ||
|
||
function observe() { | ||
if (!elemRef.current) return; | ||
const refPageOffset = elemRef.current.getBoundingClientRect().top; | ||
const stickyOffset = parseInt(getComputedStyle(elemRef.current).top, 10); | ||
const stickyActive = refPageOffset <= stickyOffset; | ||
scrollContainer.addEventListener('scroll', handleScroll); | ||
|
||
if (stickyActive && !stickyRef.current) setSticky(true); | ||
else if (!stickyActive && stickyRef.current) setSticky(false); | ||
} | ||
}); | ||
return () => { | ||
scrollContainer.removeEventListener('scroll', handleScroll); | ||
handleScroll.cancel(); // Cancel the throttled function | ||
}; | ||
}, [elemRef, initialOffset, offsetCssVariable, throttleDelay, cssVariableScope]); | ||
|
||
return sticky; | ||
} |
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.
plz, use
throttle
fromsrc/lodash.ts