You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
81 lines
2.1 KiB
81 lines
2.1 KiB
import useEffect from './useIsomorphicEffect';
|
|
import { useState } from 'react';
|
|
var matchersByWindow = new WeakMap();
|
|
|
|
var getMatcher = function getMatcher(query, targetWindow) {
|
|
if (!query || !targetWindow) return undefined;
|
|
var matchers = matchersByWindow.get(targetWindow) || new Map();
|
|
matchersByWindow.set(targetWindow, matchers);
|
|
var mql = matchers.get(query);
|
|
|
|
if (!mql) {
|
|
mql = targetWindow.matchMedia(query);
|
|
mql.refCount = 0;
|
|
matchers.set(mql.media, mql);
|
|
}
|
|
|
|
return mql;
|
|
};
|
|
/**
|
|
* Match a media query and get updates as the match changes. The media string is
|
|
* passed directly to `window.matchMedia` and run as a Layout Effect, so initial
|
|
* matches are returned before the browser has a chance to paint.
|
|
*
|
|
* ```tsx
|
|
* function Page() {
|
|
* const isWide = useMediaQuery('min-width: 1000px')
|
|
*
|
|
* return isWide ? "very wide" : 'not so wide'
|
|
* }
|
|
* ```
|
|
*
|
|
* Media query lists are also reused globally, hook calls for the same query
|
|
* will only create a matcher once under the hood.
|
|
*
|
|
* @param query A media query
|
|
* @param targetWindow The window to match against, uses the globally available one as a default.
|
|
*/
|
|
|
|
|
|
export default function useMediaQuery(query, targetWindow) {
|
|
if (targetWindow === void 0) {
|
|
targetWindow = typeof window === 'undefined' ? undefined : window;
|
|
}
|
|
|
|
var mql = getMatcher(query, targetWindow);
|
|
|
|
var _useState = useState(function () {
|
|
return mql ? mql.matches : false;
|
|
}),
|
|
matches = _useState[0],
|
|
setMatches = _useState[1];
|
|
|
|
useEffect(function () {
|
|
var mql = getMatcher(query, targetWindow);
|
|
|
|
if (!mql) {
|
|
return setMatches(false);
|
|
}
|
|
|
|
var matchers = matchersByWindow.get(targetWindow);
|
|
|
|
var handleChange = function handleChange() {
|
|
setMatches(mql.matches);
|
|
};
|
|
|
|
mql.refCount++;
|
|
mql.addListener(handleChange);
|
|
handleChange();
|
|
return function () {
|
|
mql.removeListener(handleChange);
|
|
mql.refCount--;
|
|
|
|
if (mql.refCount <= 0) {
|
|
matchers == null ? void 0 : matchers.delete(mql.media);
|
|
}
|
|
|
|
mql = undefined;
|
|
};
|
|
}, [query]);
|
|
return matches;
|
|
} |