state-in-url
Version:
Store state in URL as in object, types and structure are preserved, with TS validation. Same API as React.useState, wthout any hasssle or boilerplate. Next.js@14-15, react-router@6-7, and remix@2.
83 lines (82 loc) • 3.28 kB
TypeScript
import { type NavigateOptions } from "react-router-dom";
import { type JSONCompatible } from "../../utils";
/**
* Remix hook. Returns `urlState`, `setState`, and `setUrl` functions
*
* @param {JSONCompatible<T>} [defaultState] Fallback (default) values for state
* @param {Object} params - Object with other parameters
* @param {NavigateOptions} params.NavigateOptions See type from [`react-router-dom`](https://api.reactrouter.com/v7/interfaces/react_router.NavigateOptions.html)
* @param {boolean} params.replace replace URL or push, default `true`
* @param {boolean} params.useHistory use window.history for navigation, default `false`
* @param {boolean} params.preventScrollReset keep scroll position, default `true`
* @returns {Object} [result] State and callbacks
* @returns {Object} [result.state] - current state object
* @returns {Function} [result.setUrl] - function to update state and url
* @returns {Function} [result.setState] - function to update state only
* @returns {Function} [result.reset] - function to reset state and url to default
* * Example:
* ```ts
* export const form = { name: '', age: 0 };
* const { urlState, setState, setUrl } = useUrlState(form, { replace: false, preventScrollReset: false });
*
* setState({ name: 'test' });
* setUrl({ name: 'test' }, { replace: true });
* // similar to React.useState
* setUrl(curr => ({ ...curr, name: 'test' }), { replace: true });
* // reset state and url
* reset();
* reset({ replace: true });
* // same as setState(form) with setUrl(form)
* ```
*
* * Docs {@link https://github.com/asmyshlyaev177/state-in-url/tree/master/packages/urlstate/remix/useUrlState}
*/
export declare function useUrlState<T extends JSONCompatible>(defaultState: T, params?: Params): {
/**
* State object. Don't mutate directly, use `setState` or `setUrl`
*/
urlState: T;
/**
* * Example:
* ```ts
* setState({ name: 'test' });
* // or
* setState(curr => ({ ...curr, name: 'test' }) );
* ```
*
* * Docs {@link https://github.com/asmyshlyaev177/state-in-url/tree/master/packages/urlstate/remix/useUrlState#updatestate}
*/
setState: (value: Partial<T> | ((currState: T) => T)) => void;
/**
* * Example:
* ```ts
* setUrl({ name: 'test' });
* // or
* setUrl(curr => ({ ...curr, name: 'test' }), { replace: true } );
* // can pass optional React-Router `NavigateOptions`
* setUrl(curr => ({ ...curr, name: 'test' }), { preventScrollReset: false } );
* ```
*
* * Docs {@link https://github.com/asmyshlyaev177/state-in-url/tree/master/packages/urlstate/remix/useUrlState#updateurl}
*/
setUrl: (value?: Partial<T> | ((currState: T) => T), options?: Params) => void;
/**
* * Example:
* ```ts
* reset();
* // or
* reset({ replace: false, preventScrollReset: false })
* ```
*
* * Docs {@link https://github.com/asmyshlyaev177/state-in-url/tree/master/packages/urlstate/remix/useUrlState#reset}
*/
reset: (options?: NavigateOptions & {
[key: string]: unknown;
}) => void;
};
interface Params extends NavigateOptions {
useHistory?: boolean;
searchParams?: object;
replace?: boolean;
}
export {};