Skip to content

Instantly share code, notes, and snippets.

@rpggio
Last active September 26, 2024 20:28
Show Gist options
  • Save rpggio/f88bfb9e8d53e51f36e185a167a76152 to your computer and use it in GitHub Desktop.
Save rpggio/f88bfb9e8d53e51f36e185a167a76152 to your computer and use it in GitHub Desktop.
React hook to persist MUI DataGrid column settings
/**
* If argument is a string, try to parse as JSON.
* Otherwise return null.
*/
export function parseOrNull(raw: unknown) {
if (!raw) return null
if (typeof raw === 'string') {
try {
return JSON.parse(raw)
} catch(e) {
console.warn(`Failed to parse: ${raw.substring(0, 50)}`)
return null
}
}
return null
}
import { GridApiPro } from '@mui/x-data-grid-pro/models/gridApiPro'
import { GridEventsStr } from '@mui/x-data-grid/models/events'
import { MutableRefObject, useEffect, useRef } from 'react'
import { parseOrNull } from './parseOrNull'
/**
* Listens to column-related changes on grid, and saves column settings to local storage.
* Restores settings upon reload.
* Docs: https://mui.com/x/react-data-grid/state/#save-and-restore-the-state
* SO: https://stackoverflow.com/a/75090454/207291
* Usage:
* `
* function MyGrid() {
* const apiRef = useGridApiRef()
* usePersistColumnSettings(apiRef, 'customers-grid')
* return <DataGrid apiRef={apiRef} />
* }
* `
* Note: to persist column visibility, you may need to add the following to grid props:
* `initialState={{ columns: { columnVisibilityModel: {} } }}`
*/
export function usePersistColumnSettings (
apiRef: MutableRefObject<GridApiPro>,
key: string
) {
const initialized = useRef(false)
const storageKey = `${key}_grid-state`
useEffect(() => {
const ref = apiRef.current
if (!ref?.subscribeEvent) return
// Restore state on first ref load
if (!initialized.current) {
initialized.current = true
const raw = localStorage.getItem(storageKey)
if (raw) {
const parsed = parseOrNull(raw)
if (parsed) {
try {
// console.debug(`Restoring grid state for ${key}`, parsed)
ref.restoreState(parsed)
}
catch (e) {
console.warn(`Failed to restore grid state`, e)
}
}
}
}
const subs: VoidFunction[] = []
const save = () => {
const state = ref.exportState()
if (state) {
// console.debug(`Storing grid state for ${key}`)
localStorage.setItem(storageKey, JSON.stringify(state))
}
}
const subscribe = <E extends GridEventsStr> (event: E) => {
subs.push(ref.subscribeEvent(event, save))
}
subscribe('columnResizeStop')
subscribe('columnOrderChange')
subscribe('pinnedColumnsChange')
subscribe('columnVisibilityModelChange')
return () => {
subs.forEach(unsub => {
unsub()
})
}
}, [apiRef])
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment