sync-query

Use the React higher-order component to store the React State in URL query (routing parameters)

Usage no npm install needed!

<script type="module">
  import syncQuery from 'https://cdn.skypack.dev/sync-query';
</script>

README

sync-query

❄️

使用 React 高阶组件,实现将 React state 存储在 URL query


English

Demo

^_^ 这是一个在 antd-design 里使用该库的实例

只需要 3 行代码!

// import
import { syncQueryHOC } from "sync-query";
// use syncQueryHOC
const MyComponentEnhance = syncQueryHOC(MyComponent, ['searchInput', 'pagination'], 'fetch');
<MyComponentEnhance></MyComponentEnhance>
//...

更多关于这个例子,请点击这里

一旦我们这样做了,就拥有了下面使用的功能。

基础功能

  • 自动同步 react state 到 url query (URLSearchParam)
  • 自动调用回调函数(比如网络请求等),当 react state 发生变化时.
  • 自动从 url query (URLSearchParam) ,初始化 react state
  • 零依赖,只有 2.8kb gzipped 大小 (查看JS体积优化过程

安装

yarn add sync-query

npm i --save sync-query

使用说明

Use TypeScript Decorator

import { SyncQueryFactory, syncQueryCb } from "sync-query";

@SyncQueryFactory(['searchInput', 'pagination']) // 监听到 searchInput 或 pagination 变化时同步到 URL query
export class MyComponent extends Component {
    this.state = {
        searchInput: 'hello world',
        pagination: {
        },
    }
    @syncQueryCb(['searchInput']) // 监听到 searchInput 变化时调用 fetch 函数
    fetch() {
        // network fetch...
    }
}

Use ES6 React HOC

import { syncQueryHOC } from "sync-query";

export class MyComponent extends Component {
    fetch() {
        // network fetch...
    }
}

export const MyComponentEnhance = 
    syncQueryHOC(
        MyComponent,
        ['searchInput', 'pagination'], // 监听到 searchInput 或 pagination 变化时同步到 URL query
        'fetch',
        {
            callbackDeps: ['searchInput'], // 监听到 searchInput 变化时调用 fetch 函数
            wait: 600, // 函数防抖,600ms
        }
    );

注意: SyncQueryFactory 装饰器工厂 和 syncQueryHOC 要放在离 MyComponent 最近的位置

其他说明

手动同步

该库会自动存储 state 到 url query,同时触发 callback 函数

关闭的方法是,在类装饰器配置中增加 disableAutoSync

手动同步的方法是 (this as SyncQueryHost).triggerSync),

triggerSync 该方法的调用,会同步state到路由参数

示例代码如下:

@SyncQueryFactory(
    ['pagination', 'searchInput'],
    null,
    {
        disableAutoSync: true
    }
)
class MyComponent extends Component {
    onHandlePageChange(current) {
        this.setState(
            {
                pagination: {
                    ...this.state.pagination,
                    current,
                },
            },
            this.fetchTable
        );
    }
    fetchTable() {
        (this as any).triggerSync();
        // fetch network...
    }
}

注意: (this as any).triggerSync() 要在 componentDidMount 和之后的声明周期才被初始化。

API

syncQueryHOC

接收一个 React 组件,返回带有同步 state 到路由参数功能的组件

syncQueryHOC(WrappedComponent, stateList: string[], callbackName?:string, config?:SyncQueryConfig): EnhanceComponent

  • WrappedComponent: 被装饰的原始组件
  • stateList: 传一个数组,state 中对应 key 的值会被监听
  • callbackName?: 监听到变化时,触发 effect 方法
  • config?: SyncQueryConfig
    type SyncQueryConfig = {
        wait: number,                           // 函数防抖的等待时间, 单位 ms
        callbackDeps?: string[],                // callbackDeps 存放 state key 的数组,监听到 state 中对应key 的 value 变化时,会调用 callback(网络请求等)
                                                // callbackDeps 没有传入时,默认监听的内容等于 stateList
        parser?: IQueryParser,                  // 解析器:用于将路由参数 query 解析到 state,默认是 JSON.parse
        stringify?: IQueryStringify,            // 生成器:用于生成 state 对应的 query 字符串,默认是 JSON.stringify
    }
    

SyncQueryFactory

SyncQueryFactory 装饰器工厂, 在 typescript 中使用

SyncQueryFactory(stateList: string[], callbackName?:string, config?:SyncQueryConfig)

注意 类装饰器工厂 SyncQueryFactory 和 方法装饰器 syncQueryCb 可以配合使用, 高阶组件 syncQueryHOC 与前两者尽量避免共用。

  • stateList: 传一个数组,state 中对应 key 的值会被监听
  • callbackName?: 监听到变化时,触发 effect 方法
  • config?: SyncQueryConfig
    type SyncQueryConfig = {
        wait: number,                           // 函数防抖的等待时间, 单位 ms
        callbackDeps?: string[],                // callbackDeps 存放 state key 的数组,监听到 state 中对应key 的 value 变化时,会调用 callback(网络请求等)
                                                // callbackDeps 没有传入时,默认监听的内容等于 stateList
        parser?: IQueryParser,                  // 解析器:用于将路由参数 query 解析到 state,默认是 JSON.parse
        stringify?: IQueryStringify,            // 生成器:用于生成 state 对应的 query 字符串,默认是 JSON.stringify
    }
    

代码实现如下:

function SyncQueryFactory(stateList: string[], callbackName?:string, config?:SyncQueryConfig) {
    return function(WrappedComponent) {
        return syncQueryHOC(WrappedComponent, stateList, callbackName, config);
    }
}

syncQueryCb

syncQueryCb 方法装饰器,与 SyncQueryFactory 配合使用

syncQueryCb(callbackDeps?:string[])

  • callbackDeps?: string[] callbackDeps 存放 state key 的数组,监听到 state 中对应key 的 value 变化时,会调用 callback(网络请求等)

使用示例:

import { SyncQueryFactory, syncQueryCb } from "sync-query";

@SyncQueryFactory(['searchInput', 'pagination']) // 监听到 searchInput 或 pagination 变化时同步到 URL query
export class MyComponent extends Component {

    @syncQueryCb(['searchInput']) // 监听到 searchInput 变化时调用 fetch 函数
    fetch() {
        // network fetch...
    }
}

License

MIT