<AutoSaveIndicator />
Refine's forms provide a built-in auto-save feature. This allows you to automatically save the form when the user makes changes to the form which can be useful for forms that are long or complex and the user may not want to lose their progress.
The <AutoSaveIndicator />
component is a utility component that can be used to show a visual indicator to the user about the auto-save status of the form.
Refine's core
useForm
hook does not automatically trigger the auto-save feature. You need to manually trigger theonFinishAutoSave
function returned from theuseForm
hook to trigger the auto-save feature.Extended implementations of Refine's
useForm
such as;@refinedev/antd
'suseForm
,@refinedev/react-hook-form
'suseForm
and@refinedev/mantine
'suseForm
automatically trigger the auto-save feature when a form value changes.The
<AutoSaveIndicator />
component is only designed to display a visual feedback to the user about the auto-save status of the form. It does not contain any logic to trigger the auto-save feature.To learn more about the auto-save feature check out Auto Save section in Forms guide
Usage
Usage is as simple as spreading the autoSaveProps
object returned from the useForm
hook into the <AutoSaveIndicator />
component. It will automatically determine the auto-save status and display the appropriate indicator.
import { AutoSaveIndicator, useForm } from "@refinedev/core";
const EditPage = () => {
const { autoSaveProps } = useForm({
autoSave: {
enabled: true,
},
});
console.log(autoSaveProps);
/*
{
status: "success", // "loading" | "error" | "idle" | "success"
error: null, // HttpError | null
data: { ... }, // UpdateResponse | undefined,
}
*/
return (
<div>
{/* We'll pass the autoSaveProps from useForm's response to the <AutoSaveIndicator /> component. */}
<AutoSaveIndicator {...autoSaveProps} />
<form
// ...
>
{/* ... */}
</form>
</div>
);
};
Example below shows the <AutoSaveIndicator />
component in action.
Code Example
Dependencies: @refinedev/core@latest,@refinedev/simple-rest@latest,@refinedev/react-router@latest,react-router@^7.0.2
Content: import React from "react"; import { Refine } from "@refinedev/core"; import dataProvider from "@refinedev/simple-rest"; import { BrowserRouter, Route, Routes, Navigate, Link, Outlet } from "react-router"; import routerProvider from "@refinedev/react-router"; import "./style.css"; import { Edit } from "./edit.tsx"; export default function App() { return ( <BrowserRouter> <Refine routerProvider={routerProvider} dataProvider={dataProvider("https://api.fake-rest.refine.dev")} resources={[ { name: "products", edit: "/products/edit/:id", } ]} > <Routes> <Route index element={<Navigate to="/products/edit/123" />} /> <Route path="/products" element={<Outlet />}> <Route path="edit/:id" element={<Edit />} /> </Route> </Routes> </Refine> </BrowserRouter> ); }
Content: html { margin: 0; padding: 0; } body { margin: 0; padding: 12px; } * { box-sizing: border-box; } body { font-family: sans-serif; } form button { display: block; // width: 100%; margin-bottom: 6px; } .page { padding: 12px; display: flex; flex-direction: column; gap: 12px; } .page form { display: flex; flex-direction: column; gap: 12px; width: 100%; max-width: 400px; } .page form label { display: flex; align-items: center; } .page form label input, .page form label textarea, .page form label select { flex: 1; } .page form label span { width: 90px; } .auto-save-wrapper { padding: 6px 0; font-weight: 600; }
Content: import React from "react"; import { useForm, useSelect, AutoSaveIndicator, HttpError, BaseKey } from "@refinedev/core"; export const Edit: React.FC = () => { const { query, isLoading, onFinish, autoSaveProps, onFinishAutoSave } = useForm< IProduct, HttpError, FormValues >({ autoSave: { enabled: true, interval: 1000, }, }); const { options: categorySelectOptions } = useSelect({ resource: "categories", }); const defaultValues = query?.data?.data; return ( <div className="page"> <div className="auto-save-wrapper"> <AutoSaveIndicator {...autoSaveProps} /> </div> <form onChange={(event) => { const formData = new FormData(event.currentTarget); onFinishAutoSave(transformValues(Object.fromEntries(formData.entries()) as RawFormValues)); }} onSubmit={(event) => { event.preventDefault(); const formData = new FormData(event.currentTarget); onFinish(transformValues(Object.fromEntries(formData.entries()) as RawFormValues)); }} > <label htmlFor="name"> <span>Name</span> <input name="name" placeholder="Name" defaultValue={defaultValues?.name} /> </label> <label htmlFor="description"> <span>Description</span> <textarea name="description" placeholder="Description" defaultValue={defaultValues?.description} /> </label> <label htmlFor="material"> <span>Material</span> <input name="material" placeholder="Material" defaultValue={defaultValues?.material} /> </label> <label htmlFor="category"> <span>Category</span> <select name="category" defaultValue={defaultValues?.category?.id}> {categorySelectOptions.map((option) => ( <option key={option.value} value={option.value}> {option.label} </option> ))} </select> </label> <button type="submit">Submit</button> </form> </div> ); }; const transformValues = (values: RawFormValues): FormValues => { return { ...values, category: values.category ? { id: values.category } : undefined, }; }; interface IProduct { id: BaseKey; name: string; material: string; description: string; category: { id: BaseKey; name: string }; } interface FormValues { name?: string; material?: string; description?: string; category?: { id: BaseKey }; } interface RawFormValues extends FormValues { category?: BaseKey; }
Customizing the indicator
The <AutoSaveIndicator />
component accepts an elements
prop which can be used to customize the indicator for each status.
import { AutoSaveIndicator, useForm } from "@refinedev/core";
const EditPage = () => {
const { autoSaveProps } = useForm({
autoSave: {
enabled: true,
},
});
return (
<div>
<AutoSaveIndicator
{...autoSaveProps}
elements={{
loading: <span>saving...</span>,
error: <span>auto save error.</span>,
idle: <span>waiting for changes.</span>,
success: <span>saved.</span>,
}}
/>
{/* ... */}
</div>
);
};
API Reference
Properties
Property | Type | Description |
---|---|---|
data |
| The data returned by the update request. |
error |
| The error returned by the update request. |
status ﹡ |
| The status of the update request. |
elements |
| The elements to display for each status. |