Introduction
Refine provides an integration package for Ant Design framework. This package provides a set of ready to use components and hooks that connects Refine with Ant Design components. While Refine's integration offers a set of components and hooks, it is not a replacement for the Ant Design package, you will be able to use all the features of Ant Design in the same way you would use it in a regular React application. Refine's integration only provides components and hooks for an easier usage of Ant Design components in combination with Refine's features and functionalities.
Code Example
// file: /App.tsx import React from "react"; import { Refine, Authenticated } from "@refinedev/core"; import dataProvider from "@refinedev/simple-rest"; import routerProvider, { NavigateToResource } from "@refinedev/react-router"; import { BrowserRouter, Route, Routes, Outlet, Navigate } from "react-router"; import { ErrorComponent, RefineThemes, ThemedLayoutV2, useNotificationProvider, AuthPage } from "@refinedev/antd"; import { App as AntdApp, ConfigProvider } from "antd"; import authProvider from "./auth-provider"; import "@refinedev/antd/dist/reset.css"; import { ProductList } from "./pages/products/list"; import { ProductShow } from "./pages/products/show"; import { ProductEdit } from "./pages/products/edit"; import { ProductCreate } from "./pages/products/create"; export default function App() { return ( <BrowserRouter> <ConfigProvider theme={RefineThemes.Blue}> <AntdApp> <Refine routerProvider={routerProvider} dataProvider={dataProvider("https://api.fake-rest.refine.dev")} authProvider={authProvider} notificationProvider={useNotificationProvider} resources={[ { name: "products", list: "/products", show: "/products/:id", edit: "/products/:id/edit", create: "/products/create" } ]} options={{ syncWithLocation: true }} > <Routes> <Route element={<Authenticated fallback={<Navigate to="/login" />}><Outlet /></Authenticated>}> <Route element={ <ThemedLayoutV2> <Outlet /> </ThemedLayoutV2> } > <Route path="/products" element={<Outlet />}> <Route index element={<ProductList />} /> <Route path="create" element={<ProductCreate />} /> <Route path=":id" element={<ProductShow />} /> <Route path=":id/edit" element={<ProductEdit />} /> </Route> <Route path="*" element={<ErrorComponent />} /> </Route> </Route> <Route element={<Authenticated fallback={<Outlet />}><NavigateToResource resource="products" /></Authenticated>}> <Route path="/login" element={( <AuthPage type="login" formProps={{ initialValues: { email: "demo@refine.dev", password: "demodemo", }, }} /> )} /> <Route path="/register" element={<AuthPage type="register" />} /> <Route path="/forgot-password" element={<AuthPage type="forgotPassword" />} /> <Route path="/reset-password" element={<AuthPage type="resetPassword" />} /> <Route path="*" element={<ErrorComponent />} /> </Route> </Routes> </Refine> </AntdApp> </ConfigProvider> </BrowserRouter> ); };
// file: /pages/products/list.tsx import { List, ShowButton, EditButton, useTable } from "@refinedev/antd"; import { BaseRecord } from "@refinedev/core"; import { Space, Table } from "antd"; import React from "react"; export const ProductList = () => { const { tableProps } = useTable(); return ( <List> <Table {...tableProps} rowKey="id"> <Table.Column dataIndex="id" title="Id" /> <Table.Column dataIndex="name" title="Name" /> <Table.Column dataIndex="price" title="Price" /> <Table.Column title="Actions" dataIndex="actions" render={(_, record: BaseRecord) => ( <Space> <ShowButton hideText size="small" recordItemId={record.id} /> <EditButton hideText size="small" recordItemId={record.id} /> </Space> )} /> </Table> </List> ); };
// file: /pages/products/show.tsx import { MarkdownField, NumberField, Show, TextField } from "@refinedev/antd"; import { useShow } from "@refinedev/core"; import { Typography } from "antd"; import React from "react"; const { Title } = Typography; export const ProductShow = () => { const { queryResult } = useShow(); const { data, isLoading } = queryResult; const record = data?.data; return ( <Show isLoading={isLoading}> <Title level={5}>Id</Title> <NumberField value={record?.id ?? ""} /> <Title level={5}>Name</Title> <TextField value={record?.name} /> <Title level={5}>Material</Title> <TextField value={record?.material} /> <Title level={5}>Description</Title> <MarkdownField value={record?.description} /> <Title level={5}>Price</Title> <NumberField value={record?.price ?? ""} /> </Show> ); };
// file: /pages/products/edit.tsx import React from "react"; import { Typography, Form, Input, InputNumber } from "antd"; import { Edit, useForm } from "@refinedev/antd"; const { Title } = Typography; const { TextArea } = Input; export const ProductEdit: React.FC = () => { const { formProps, saveButtonProps } = useForm(); return ( <Edit saveButtonProps={saveButtonProps}> <Form {...formProps} layout="vertical"> <Form.Item label="Name" name="name" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Material" name="material" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Description" name="description" rules={[ { required: true, }, ]} > <TextArea rows={4} /> </Form.Item> <Form.Item label="Price" name="price" rules={[ { required: true, }, ]} > <InputNumber /> </Form.Item> </Form> </Edit> ); };
// file: /pages/products/create.tsx import React from "react"; import { Typography, Form, Input, InputNumber } from "antd"; import { Create, useForm } from "@refinedev/antd"; const { Title } = Typography; const { TextArea } = Input; export const ProductCreate = () => { const { formProps, saveButtonProps } = useForm(); return ( <Create saveButtonProps={saveButtonProps}> <Form {...formProps} layout="vertical"> <Form.Item label="Name" name="name" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Material" name="material" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Description" name="description" rules={[ { required: true, }, ]} > <TextArea rows={4} /> </Form.Item> <Form.Item label="Price" name="price" rules={[ { required: true, }, ]} > <InputNumber /> </Form.Item> </Form> </Create> ); };
Installation
Installing the package is as simple as just by running the following command without any additional configuration:
- npm
- pnpm
- yarn
npm i @refinedev/antd antd
pnpm add @refinedev/antd antd
yarn add @refinedev/antd antd
Usage
We'll wrap our app with the <ConfigProvider />
to make sure we have the theme available for our app, then we'll use the layout components to wrap them around our routes. Check out the examples below to see how to use Refine's Ant Design integration.
- React Router
- Next.js
- Remix
React Router
Code Example
// file: /App.tsx import { Refine, Authenticated } from "@refinedev/core"; import dataProvider from "@refinedev/simple-rest"; import routerProvider, { NavigateToResource } from "@refinedev/react-router"; import { BrowserRouter, Route, Routes, Outlet, Navigate } from "react-router"; import { ErrorComponent, RefineThemes, ThemedLayoutV2, useNotificationProvider, AuthPage } from "@refinedev/antd"; import { App as AntdApp, ConfigProvider } from "antd"; import authProvider from "./auth-provider"; import "@refinedev/antd/dist/reset.css"; import { ProductList, ProductShow, ProductEdit, ProductCreate } from "./pages/products"; export default function App() { return ( <BrowserRouter> <ConfigProvider theme={RefineThemes.Blue}> <AntdApp> <Refine routerProvider={routerProvider} dataProvider={dataProvider("https://api.fake-rest.refine.dev")} authProvider={authProvider} notificationProvider={useNotificationProvider} resources={[ { name: "products", list: "/products", show: "/products/:id", edit: "/products/:id/edit", create: "/products/create" } ]} options={{ syncWithLocation: true }} > <Routes> <Route element={<Authenticated fallback={<Navigate to="/login" />}><Outlet /></Authenticated>}> <Route element={ <ThemedLayoutV2> <Outlet /> </ThemedLayoutV2> } > <Route path="/products" element={<Outlet />}> <Route index element={<ProductList />} /> <Route path="create" element={<ProductCreate />} /> <Route path=":id" element={<ProductShow />} /> <Route path=":id/edit" element={<ProductEdit />} /> </Route> <Route path="*" element={<ErrorComponent />} /> </Route> </Route> <Route element={<Authenticated fallback={<Outlet />}><NavigateToResource resource="products" /></Authenticated>}> <Route path="/login" element={<AuthPage type="login" />} /> <Route path="/register" element={<AuthPage type="register" />} /> <Route path="/forgot-password" element={<AuthPage type="forgotPassword" />} /> <Route path="/reset-password" element={<AuthPage type="resetPassword" />} /> <Route path="*" element={<ErrorComponent />} /> </Route> </Routes> </Refine> </AntdApp> </ConfigProvider> </BrowserRouter> ); };
// file: /pages/products/index.tsx export * from "./list"; export * from "./show"; export * from "./edit"; export * from "./create";
// file: /pages/products/list.tsx import React from "react"; import { List, ShowButton, EditButton, useTable } from "@refinedev/antd"; import { Space, Table } from "antd"; export const ProductList = () => { const { tableProps } = useTable(); return ( <List> <Table {...tableProps} rowKey="id"> <Table.Column dataIndex="id" title="Id" /> <Table.Column dataIndex="name" title="Name" /> <Table.Column dataIndex="price" title="Price" /> <Table.Column title="Actions" dataIndex="actions" render={(_, record: BaseRecord) => ( <Space> <ShowButton hideText size="small" recordItemId={record.id} /> <EditButton hideText size="small" recordItemId={record.id} /> </Space> )} /> </Table> </List> ); };
// file: /pages/products/show.tsx import React from "react"; import { useShow } from "@refinedev/core"; import { MarkdownField, NumberField, Show, TextField } from "@refinedev/antd"; import { Typography } from "antd"; const { Title } = Typography; export const ProductShow = () => { const { queryResult } = useShow(); const { data, isLoading } = queryResult; const record = data?.data; return ( <Show isLoading={isLoading}> <Title level={5}>Id</Title> <NumberField value={record?.id ?? ""} /> <Title level={5}>Name</Title> <TextField value={record?.name} /> <Title level={5}>Material</Title> <TextField value={record?.material} /> <Title level={5}>Description</Title> <MarkdownField value={record?.description} /> <Title level={5}>Price</Title> <NumberField value={record?.price ?? ""} /> </Show> ); };
// file: /pages/products/edit.tsx import React from "react"; import { Typography, Form, Input, InputNumber } from "antd"; import { Edit, useForm } from "@refinedev/antd"; const { Title } = Typography; const { TextArea } = Input; export const ProductEdit = () => { const { formProps, saveButtonProps, formLoading } = useForm(); return ( <Edit saveButtonProps={saveButtonProps} isLoading={formLoading}> <Form {...formProps} layout="vertical"> <Form.Item label="Name" name="name" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Material" name="material" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Description" name="description" rules={[ { required: true, }, ]} > <TextArea rows={4} /> </Form.Item> <Form.Item label="Price" name="price" rules={[ { required: true, }, ]} > <InputNumber /> </Form.Item> </Form> </Edit> ); };
// file: /pages/products/create.tsx import React from "react"; import { Typography, Form, Input, InputNumber } from "antd"; import { Create, useForm } from "@refinedev/antd"; const { Title } = Typography; const { TextArea } = Input; export const ProductCreate = () => { const { formProps, saveButtonProps, formLoading } = useForm(); return ( <Create saveButtonProps={saveButtonProps} isLoading={formLoading}> <Form {...formProps} layout="vertical"> <Form.Item label="Name" name="name" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Material" name="material" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Description" name="description" rules={[ { required: true, }, ]} > <TextArea rows={4} /> </Form.Item> <Form.Item label="Price" name="price" rules={[ { required: true, }, ]} > <InputNumber /> </Form.Item> </Form> </Create> ); };
Next.js
Code Example
// file: /pages/_app.tsx import React from "react"; import { Refine } from "@refinedev/core"; import routerProvider from "@refinedev/nextjs-router/pages"; import dataProvider from "@refinedev/simple-rest"; import type { AppProps } from "next/app"; import { RefineThemes, ThemedLayoutV2, useNotificationProvider } from "@refinedev/antd"; import { App as AntdApp, ConfigProvider } from "antd"; import "@refinedev/antd/dist/reset.css"; import authProvider from "../src/auth-provider"; export type ExtendedNextPage = NextPage & { noLayout?: boolean; }; type ExtendedAppProps = AppProps & { Component: ExtendedNextPage; }; function App({ Component, pageProps }: ExtendedAppProps) { const renderComponent = () => { if (Component.noLayout) { return <Component {...pageProps} />; } return ( <ThemedLayoutV2> <Component {...pageProps} /> </ThemedLayoutV2> ); } return ( <ConfigProvider theme={RefineThemes.Blue}> <AntdApp> <Refine routerProvider={routerProvider} dataProvider={dataProvider("https://api.fake-rest.refine.dev")} notificationProvider={useNotificationProvider} authProvider={authProvider} resources={[ { name: "products", list: "/products", show: "/products/:id", edit: "/products/:id/edit", create: "/products/create" }, ]} options={{ syncWithLocation: true }} > {renderComponent()} </Refine> </AntdApp> </ConfigProvider> ); } export default App;
// file: /pages/products/index.tsx import React from "react"; import { List, ShowButton, EditButton, useTable } from "@refinedev/antd"; import { Space, Table } from "antd"; import authProvider from "../../src/auth-provider"; export default function ProductList() { const { tableProps } = useTable(); return ( <List> <Table {...tableProps} rowKey="id"> <Table.Column dataIndex="id" title="Id" /> <Table.Column dataIndex="name" title="Name" /> <Table.Column dataIndex="price" title="Price" /> <Table.Column title="Actions" dataIndex="actions" render={(_, record: BaseRecord) => ( <Space> <ShowButton hideText size="small" recordItemId={record.id} /> <EditButton hideText size="small" recordItemId={record.id} /> </Space> )} /> </Table> </List> ); }; /** * Same check can also be done via `<Authenticated />` component. * But we're using a server-side check for a better UX. */ export const getServerSideProps = async () => { const { authenticated } = await authProvider.check(); if (!authenticated) { return { redirect: { destination: "/login", permanent: false, }, }; } return { props: {}, }; }
// file: /pages/products/[id].tsx import React from "react"; import { useShow } from "@refinedev/core"; import { MarkdownField, NumberField, Show, TextField } from "@refinedev/antd"; import { Typography } from "antd"; import authProvider from "../../src/auth-provider"; const { Title } = Typography; export default function ProductShow() { const { queryResult } = useShow(); const { data, isLoading } = queryResult; const record = data?.data; return ( <Show isLoading={isLoading}> <Title level={5}>Id</Title> <NumberField value={record?.id ?? ""} /> <Title level={5}>Name</Title> <TextField value={record?.name} /> <Title level={5}>Material</Title> <TextField value={record?.material} /> <Title level={5}>Description</Title> <MarkdownField value={record?.description} /> <Title level={5}>Price</Title> <NumberField value={record?.price ?? ""} /> </Show> ); }; /** * Same check can also be done via `<Authenticated />` component. * But we're using a server-side check for a better UX. */ export const getServerSideProps = async () => { const { authenticated } = await authProvider.check(); if (!authenticated) { return { redirect: { destination: "/login", permanent: false, }, }; } return { props: {}, }; }
// file: /pages/products/[id]/edit.tsx import React from "react"; import { Typography, Form, Input, InputNumber } from "antd"; import { Edit, useForm } from "@refinedev/antd"; import authProvider from "../../../src/auth-provider"; const { Title } = Typography; const { TextArea } = Input; export default function ProductEdit() { const { formProps, saveButtonProps, formLoading } = useForm(); return ( <Edit saveButtonProps={saveButtonProps} isLoading={formLoading}> <Form {...formProps} layout="vertical"> <Form.Item label="Name" name="name" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Material" name="material" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Description" name="description" rules={[ { required: true, }, ]} > <TextArea rows={4} /> </Form.Item> <Form.Item label="Price" name="price" rules={[ { required: true, }, ]} > <InputNumber /> </Form.Item> </Form> </Edit> ); }; /** * Same check can also be done via `<Authenticated />` component. * But we're using a server-side check for a better UX. */ export const getServerSideProps = async () => { const { authenticated } = await authProvider.check(); if (!authenticated) { return { redirect: { destination: "/login", permanent: false, }, }; } return { props: {}, }; }
// file: /pages/products/create.tsx import React from "react"; import { Typography, Form, Input, InputNumber } from "antd"; import { Create, useForm } from "@refinedev/antd"; import authProvider from "../../src/auth-provider"; const { Title } = Typography; const { TextArea } = Input; export default function ProductCreate() { const { formProps, saveButtonProps, formLoading } = useForm(); return ( <Create saveButtonProps={saveButtonProps} isLoading={formLoading}> <Form {...formProps} layout="vertical"> <Form.Item label="Name" name="name" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Material" name="material" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Description" name="description" rules={[ { required: true, }, ]} > <TextArea rows={4} /> </Form.Item> <Form.Item label="Price" name="price" rules={[ { required: true, }, ]} > <InputNumber /> </Form.Item> </Form> </Create> ); }; /** * Same check can also be done via `<Authenticated />` component. * But we're using a server-side check for a better UX. */ export const getServerSideProps = async () => { const { authenticated } = await authProvider.check(); if (!authenticated) { return { redirect: { destination: "/login", permanent: false, }, }; } return { props: {}, }; }
// file: /pages/login.tsx import React from "react"; import { AuthPage } from "@refinedev/antd"; import authProvider from "../src/auth-provider"; import type { ExtendedNextPage } from "./_app"; const Login: ExtendedNextPage = () => { return <AuthPage type="login" />; }; Login.noLayout = true; export default Login; /** * Same check can also be done via `<Authenticated />` component. * But we're using a server-side check for a better UX. */ export const getServerSideProps = async () => { const { authenticated } = await authProvider.check(); if (authenticated) { return { redirect: { destination: "/products", permanent: false, }, }; } return { props: {}, };
Remix
Code Example
// file: /app/root.tsx import React from "react"; import { Links, LiveReload, Meta, Outlet, Scripts, ScrollRestoration, } from "@remix-run/react"; import { Refine } from "@refinedev/core"; import routerProvider from "@refinedev/remix-router"; import dataProvider from "@refinedev/simple-rest"; import { useNotificationProvider, RefineThemes } from "@refinedev/antd"; import { ConfigProvider, App as AntdApp } from "antd"; import resetStyle from "@refinedev/antd/dist/reset.css"; import authProvider from "./auth-provider"; export default function App() { return ( <html lang="en"> <head> <Meta /> <Links /> </head> <body> <ConfigProvider theme={RefineThemes.Blue}> <AntdApp> <Refine routerProvider={routerProvider} dataProvider={dataProvider("https://api.fake-rest.refine.dev")} authProvider={authProvider} notificationProvider={useNotificationProvider} resources={[ { name: "products", list: "/products", show: "/products/:id", edit: "/products/:id/edit", create: "/products/create", }, ]} options={{ syncWithLocation: true }} > <Outlet /> </Refine> </AntdApp> </ConfigProvider> <ScrollRestoration /> <Scripts /> <LiveReload /> </body> </html> ); }
// file: /app/routes/_protected.tsx import { ThemedLayoutV2 } from "@refinedev/antd"; import { Outlet } from "@remix-run/react"; import { LoaderFunctionArgs, redirect } from "@remix-run/node"; import authProvider from "../auth-provider"; export default function AuthenticatedLayout() { // `<ThemedLayoutV2>` is only applied to the authenticated users return ( <ThemedLayoutV2> <Outlet /> </ThemedLayoutV2> ); } /** * We're checking if the current session is authenticated. * If not, we're redirecting the user to the login page. * This is applied for all routes that are nested under this layout (_protected). */ export async function loader({ request }: LoaderFunctionArgs) { const { authenticated, redirectTo } = await authProvider.check(request); if (!authenticated) { throw redirect(redirectTo ?? "/login"); } return {}; }
// file: /app/routes/_protected.products._index.tsx import React from "react"; import { List, ShowButton, EditButton, useTable } from "@refinedev/antd"; import { Space, Table } from "antd"; export default function ProductList() { const { tableProps } = useTable(); return ( <List> <Table {...tableProps} rowKey="id"> <Table.Column dataIndex="id" title="Id" /> <Table.Column dataIndex="name" title="Name" /> <Table.Column dataIndex="price" title="Price" /> <Table.Column title="Actions" dataIndex="actions" render={(_, record: BaseRecord) => ( <Space> <ShowButton hideText size="small" recordItemId={record.id} /> <EditButton hideText size="small" recordItemId={record.id} /> </Space> )} /> </Table> </List> ); };
// file: /app/routes/_protected.products.$id.tsx import React from "react"; import { useShow } from "@refinedev/core"; import { MarkdownField, NumberField, Show, TextField } from "@refinedev/antd"; import { Typography } from "antd"; const { Title } = Typography; export default function ProductShow() { const { queryResult } = useShow(); const { data, isLoading } = queryResult; const record = data?.data; return ( <Show isLoading={isLoading}> <Title level={5}>Id</Title> <NumberField value={record?.id ?? ""} /> <Title level={5}>Name</Title> <TextField value={record?.name} /> <Title level={5}>Material</Title> <TextField value={record?.material} /> <Title level={5}>Description</Title> <MarkdownField value={record?.description} /> <Title level={5}>Price</Title> <NumberField value={record?.price ?? ""} /> </Show> ); };
// file: /app/routes/_protected.products.$id.edit.tsx import React from "react"; import { Typography, Form, Input, InputNumber } from "antd"; import { Edit, useForm } from "@refinedev/antd"; const { Title } = Typography; const { TextArea } = Input; export default function ProductEdit() { const { formProps, saveButtonProps, formLoading } = useForm(); return ( <Edit saveButtonProps={saveButtonProps} isLoading={formLoading}> <Form {...formProps} layout="vertical"> <Form.Item label="Name" name="name" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Material" name="material" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Description" name="description" rules={[ { required: true, }, ]} > <TextArea rows={4} /> </Form.Item> <Form.Item label="Price" name="price" rules={[ { required: true, }, ]} > <InputNumber /> </Form.Item> </Form> </Edit> ); };
// file: /app/routes/_protected.products.create.tsx import React from "react"; import { Typography, Form, Input, InputNumber } from "antd"; import { Create, useForm } from "@refinedev/antd"; const { Title } = Typography; const { TextArea } = Input; export default function ProductCreate() { const { formProps, saveButtonProps, formLoading } = useForm(); return ( <Create saveButtonProps={saveButtonProps} isLoading={formLoading}> <Form {...formProps} layout="vertical"> <Form.Item label="Name" name="name" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Material" name="material" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Description" name="description" rules={[ { required: true, }, ]} > <TextArea rows={4} /> </Form.Item> <Form.Item label="Price" name="price" rules={[ { required: true, }, ]} > <InputNumber /> </Form.Item> </Form> </Create> ); };
// file: /app/routes/_auth.tsx import { Outlet } from "@remix-run/react"; import { LoaderFunctionArgs, redirect } from "@remix-run/node"; import { authProvider } from "~/authProvider"; export default function AuthLayout() { // no layout is applied for the auth routes return <Outlet />; } /** * If the current session is authenticated, we're redirecting the user to the home page. * Alternatively, we could also use the `Authenticated` component inside the `AuthLayout` to handle the redirect. * But, server-side redirects are more performant. */ export async function loader({ request }: LoaderFunctionArgs) { const { authenticated, redirectTo } = await authProvider.check(request); if (authenticated) { throw redirect(redirectTo ?? "/"); } return {}; }
// file: /app/routes/_auth.login.tsx import { AuthPage } from "@refinedev/antd"; export default function LoginPage() { return <AuthPage type="login" />; }
Tables
Refine provides a seamless integration with the <Table />
component of Ant Design from pagination to sorting and filtering via the useTable
hook exported from the @refinedev/antd
package. This hook is an extension of the @refinedev/core
's useTable
and provides a set of additional features and transformations to make it work with Ant Design's <Table />
component without any additional configuration.
import { useTable } from "@refinedev/antd";
import { Table } from "antd";
export const ProductList = () => {
const { tableProps } = useTable<IProduct>();
// `tableProps` contains the necessary props to be passed
// to the `<Table />` component of Ant Design
// by transforming the values to fit the Ant Design's API.
return (
<Table {...tableProps} rowKey="id">
<Table.Column dataIndex="id" title="ID" />
<Table.Column dataIndex="name" title="Name" />
<Table.Column dataIndex="price" title="Price" />
</Table>
);
};
interface IProduct {
id: string;
name: string;
price: number;
description: string;
}
@refinedev/antd
package also provides a <FilterDropdown />
component to be used in the filter popover of the <Table />
component. This component makes it easy to apply filters from the Ant Design UI without any additional configuration.
Forms
Refine provides a seamless integration with the <Form />
component of Ant Design from validation to submission via the useForm
hook exported from the @refinedev/antd
package. This hook is an extension of the @refinedev/core
's useForm
and provides a set of additional features and transformations to make it work with Ant Design's <Form />
component.
import { useForm, SaveButton } from "@refinedev/antd";
import { Form, Input, InputNumber } from "antd";
export const ProductCreate = () => {
const { formProps, saveButtonProps } = useForm<IProduct>();
// `formProps` contains the necessary props to be passed
// to the `<Form />` component of Ant Design
// by transforming the values to fit the Ant Design's API.
return (
<Form {...formProps} layout="vertical">
<Form.Item
label="Name"
name="name"
rules={[{ required: true }]}
>
<Input />
</Form.Item>
<Form.Item
label="Price"
name="price"
rules={[{ required: true }]}
>
<InputNumber />
</Form.Item>
<Form.Item
label="Description"
name="description"
rules={[{ required: true }]}
>
<Input.TextArea rows={4} />
</Form.Item>
<SaveButton {...saveButtonProps}>
</Form>
)
}
interface IProduct {
id: string;
name: string;
price: number;
description: string;
}
@refinedev/antd
also offers hooks to implement different types of forms such as useDrawerForm
, useModalForm
and useStepsForm
hooks. Additionally useSelect
, useCheckboxGroup
and useRadioGroup
hooks are also provided to make it easier to implement form fields with relational data. These hooks leverage the useSelect
hook from the @refinedev/core
package.
Notifications
Ant Design has its own notification system which works seamlessly with its UI elements. Refine also provides a seamless integration with Ant Design's notification system and show notifications for related actions and events. This integration is provided by the useNotificationProvider
hook exported from the @refinedev/antd
package which can be directly used in the notificationProvider
prop of the <Refine />
component.
import { Refine } from "@refinedev/core";
import { useNotificationProvider, RefineThemes } from "@refinedev/antd";
import { App as AntdApp } from "antd";
const App = () => {
return (
<ConfigProvider theme={RefineThemes.Green}>
<AntdApp>
<Refine notificationProvider={useNotificationProvider}>
{/* ... */}
</Refine>
</AntdApp>
</ConfigProvider>
);
};
TIP
If you have any configurations in the Ant Design's theme, you should wrap your app with the <App />
component to make sure the notifications are also receiving the current theme configuration.
Predefined Components and Views
Layouts, Menus and Breadcrumbs
Refine provides Layout components that can be used to implement a layout for the application. These components are crafted using Ant Design's components and includes Refine's features and functionalities such as navigation menus, headers, authentication, authorization and more.
- React Router
- Next.js
- Remix
React Router
Code Example
// file: /App.tsx import React from "react"; import { Refine, Authenticated } from "@refinedev/core"; import dataProvider from "@refinedev/simple-rest"; import routerProvider from "@refinedev/react-router"; import { BrowserRouter, Route, Routes, Outlet } from "react-router"; import { ErrorComponent, RefineThemes, ThemedLayoutV2, useNotificationProvider } from "@refinedev/antd"; import { App as AntdApp, ConfigProvider } from "antd"; import "@refinedev/antd/dist/reset.css"; import authProvider from "./auth-provider"; import { ProductList } from "./pages/products/list"; export default function App() { return ( <BrowserRouter> <ConfigProvider theme={RefineThemes.Blue}> <AntdApp> <Refine routerProvider={routerProvider} dataProvider={dataProvider("https://api.fake-rest.refine.dev")} notificationProvider={useNotificationProvider} authProvider={authProvider} resources={[ { name: "products", list: "/products", } ]} > <Routes> <Route // The layout will wrap all the pages inside this route element={ <ThemedLayoutV2> <Outlet /> </ThemedLayoutV2> } > <Route path="/products" element={<ProductList />} /> <Route path="*" element={<ErrorComponent />} /> </Route> </Routes> </Refine> </AntdApp> </ConfigProvider> </BrowserRouter> ); };
Next.js
Code Example
// file: /pages/_app.tsx import React from "react"; import { Refine } from "@refinedev/core"; import routerProvider from "@refinedev/nextjs-router/pages"; import dataProvider from "@refinedev/simple-rest"; import type { AppProps } from "next/app"; import { RefineThemes, ThemedLayoutV2, useNotificationProvider } from "@refinedev/antd"; import { App as AntdApp, ConfigProvider } from "antd"; import "@refinedev/antd/dist/reset.css"; function App({ Component, pageProps }: AppProps) { return ( <ConfigProvider theme={RefineThemes.Blue}> <AntdApp> <Refine routerProvider={routerProvider} dataProvider={dataProvider("https://api.fake-rest.refine.dev")} notificationProvider={useNotificationProvider} resources={[ { name: "products", list: "/products", }, ]} > <ThemedLayoutV2> <Component {...pageProps} /> </ThemedLayoutV2> </Refine> </AntdApp> </ConfigProvider> ); } export default App;
Remix
Code Example
// file: /app/root.tsx import React from "react"; import { Links, LiveReload, Meta, Outlet, Scripts, ScrollRestoration, } from "@remix-run/react"; import { Refine } from "@refinedev/core"; import routerProvider from "@refinedev/remix-router"; import dataProvider from "@refinedev/simple-rest"; import { useNotificationProvider, RefineThemes } from "@refinedev/antd"; import { ConfigProvider, App as AntdApp } from "antd"; import resetStyle from "@refinedev/antd/dist/reset.css"; export default function App() { return ( <html lang="en"> <head> <Meta /> <Links /> </head> <body> <ConfigProvider theme={RefineThemes.Blue}> <AntdApp> <Refine routerProvider={routerProvider} dataProvider={dataProvider("https://api.fake-rest.refine.dev")} notificationProvider={useNotificationProvider} resources={[ { name: "products", list: "/products", }, ]} > <Outlet /> </Refine> </AntdApp> </ConfigProvider> <ScrollRestoration /> <Scripts /> <LiveReload /> </body> </html> ); }
// file: /app/routes/_layout.tsx import { ThemedLayoutV2 } from "@refinedev/antd"; import { Outlet } from "@remix-run/react"; import { LoaderFunctionArgs, redirect } from "@remix-run/node"; /** * Routes starting with `_layout` will have their children rendered inside the layout. */ export default function Layout() { return ( <ThemedLayoutV2> <Outlet /> </ThemedLayoutV2> ); }
<ThemedLayoutV2 />
component consists of a header, sider and a content area. The sider have a navigation menu items for the defined resources of Refine, if an authentication provider is present, it will also have a functional logout button. The header contains the app logo and name and also information about the current user if an authentication provider is present.
Additionally, Refine also provides a <Breadcrumb />
component that uses the Ant Design's component as a base and provide appropriate breadcrumbs for the current route. This component is used in the basic views provided by Refine's Ant Design package automatically.
Buttons
Refine's Ant Design integration offers variety of buttons that are built above the <Button />
component of Ant Design and includes many logical functionalities such as;
- Authorization checks
- Confirmation dialogs
- Loading states
- Invalidation
- Navigation
- Form actions
- Import/Export and more.
You can use buttons such as <EditButton />
or <ListButton />
etc. in your views to provide navigation for the related routes or <DeleteButton />
and <SaveButton />
etc. to perform related actions without having to worry about the authorization checks and other logical functionalities.
An example usage of the <EditButton />
component is as follows:
import { useTable, EditButton } from "@refinedev/antd";
import { Table } from "antd";
export const ProductList = () => {
const { tableProps } = useTable<IProduct>();
return (
<Table {...tableProps} rowKey="id">
<Table.Column dataIndex="id" title="ID" />
<Table.Column dataIndex="name" title="Name" />
<Table.Column dataIndex="price" title="Price" />
<Table.Column
title="Actions"
dataIndex="actions"
render={(_, record) => (
<EditButton hideText size="small" recordItemId={record.id} />
)}
/>
</Table>
);
};
interface IProduct {
id: string;
name: string;
price: number;
description: string;
}
The list of provided buttons are:
<CreateButton />
<EditButton />
<ListButton />
<ShowButton />
<CloneButton />
<DeleteButton />
<SaveButton />
<RefreshButton />
<ImportButton />
<ExportButton />
Many of these buttons are already used in the views provided by Refine's Ant Design integration. If you're using the basic view elements provided by Refine, you will have the appropriate buttons placed in your application out of the box.
Views
Views are designed as wrappers around the content of the pages in the application. They are designed to be used within the layouts and provide basic functionalities such as titles based on the resource, breadcrumbs, related actions and authorization checks. Refine's Ant Design integration uses components such as <Card />
and <Space />
to provide these views and provides customization options by passing related props to these components.
The list of provided views are:
Code Example
// file: /pages/products/list.tsx import { List, ShowButton, EditButton, useTable } from "@refinedev/antd"; import { Space, Table } from "antd"; import React from "react"; export const ProductList = () => { const { tableProps } = useTable(); return ( <List> <Table {...tableProps} rowKey="id"> <Table.Column dataIndex="id" title="Id" /> <Table.Column dataIndex="name" title="Name" /> <Table.Column dataIndex="price" title="Price" /> <Table.Column title="Actions" dataIndex="actions" render={(_, record: BaseRecord) => ( <Space> <ShowButton hideText size="small" recordItemId={record.id} /> <EditButton hideText size="small" recordItemId={record.id} /> </Space> )} /> </Table> </List> ); };
// file: /pages/products/show.tsx import { MarkdownField, NumberField, Show, TextField } from "@refinedev/antd"; import { useShow } from "@refinedev/core"; import { Typography } from "antd"; import React from "react"; const { Title } = Typography; export const ProductShow = () => { const { queryResult } = useShow(); const { data, isLoading } = queryResult; const record = data?.data; return ( <Show isLoading={isLoading}> <Title level={5}>Id</Title> <NumberField value={record?.id ?? ""} /> <Title level={5}>Name</Title> <TextField value={record?.name} /> <Title level={5}>Material</Title> <TextField value={record?.material} /> <Title level={5}>Description</Title> <MarkdownField value={record?.description} /> <Title level={5}>Price</Title> <NumberField value={record?.price ?? ""} /> </Show> ); };
// file: /pages/products/edit.tsx import React from "react"; import { Typography, Form, Input, InputNumber } from "antd"; import { Edit, useForm } from "@refinedev/antd"; const { Title } = Typography; const { TextArea } = Input; export const ProductEdit: React.FC = () => { const { formProps, saveButtonProps, formLoading } = useForm(); return ( <Edit saveButtonProps={saveButtonProps} isLoading={formLoading}> <Form {...formProps} layout="vertical"> <Form.Item label="Name" name="name" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Material" name="material" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Description" name="description" rules={[ { required: true, }, ]} > <TextArea rows={4} /> </Form.Item> <Form.Item label="Price" name="price" rules={[ { required: true, }, ]} > <InputNumber /> </Form.Item> </Form> </Edit> ); };
// file: /pages/products/create.tsx import React from "react"; import { Typography, Form, Input, InputNumber } from "antd"; import { Create, useForm } from "@refinedev/antd"; const { Title } = Typography; const { TextArea } = Input; export const ProductCreate = () => { const { formProps, saveButtonProps, formLoading } = useForm(); return ( <Create saveButtonProps={saveButtonProps} isLoading={formLoading}> <Form {...formProps} layout="vertical"> <Form.Item label="Name" name="name" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Material" name="material" rules={[ { required: true, }, ]} > <Input /> </Form.Item> <Form.Item label="Description" name="description" rules={[ { required: true, }, ]} > <TextArea rows={4} /> </Form.Item> <Form.Item label="Price" name="price" rules={[ { required: true, }, ]} > <InputNumber /> </Form.Item> </Form> </Create> ); };
Fields
Refine's Ant Design also provides field components to render values with appropriate design and format of Ant Design. These components are built on top of respective Ant Design components and also provide logic for formatting of the values. While these components might not always be suitable for your use case, they can be combined or extended to provide the desired functionality.
The list of provided field components are:
<BooleanField />
<DateField />
<EmailField />
<FileField />
<ImageField />
<MarkdownField />
<NumberField />
<TagField />
<TextField />
<UrlField />
import { useShow } from "@refinedev/core";
import { Show, TextField, NumberField } from "@refinedev/antd";
import { Typography } from "antd";
export const ProductShow = () => {
const { queryResult } = useShow<IProduct>();
const { data, isLoading } = queryResult;
const record = data?.data;
return (
<Show isLoading={isLoading}>
<Typography.Title level={5}>Id</Typography.Title>
<TextField value={record?.id} />
<Typography.Title level={5}>Title</Typography.Title>
<TextField value={record?.title} />
<Typography.Title level={5}>Price</Typography.Title>
<NumberField
value={record?.price}
options={{ style: "currency", currency: "USD" }}
/>
</Show>
);
};
interface IProduct {
id: string;
name: string;
price: number;
description: string;
}
Auth Pages
Auth pages are designed to be used as the pages of the authentication flow of the application. They offer an out of the box solution for the login, register, forgot password and reset password pages by leveraging the authentication hooks of Refine. Auth page components are built on top of basic Ant Design components such as <Form />
and <Card />
etc.
The list of types of auth pages that are available in the UI integrations are:
<AuthPage type="login" />
<AuthPage type="register" />
<AuthPage type="forgot-password" />
<AuthPage type="reset-password" />
An example usage of the <AuthPage />
component is as follows:
Code Example
// file: /pages/login.tsx import { AuthPage } from "@refinedev/antd"; export const LoginPage = () => { return ( <AuthPage type="login" formProps={{ initialValues: { email: "demo@refine.dev", password: "demodemo", }, }} /> ); };
// file: /pages/register.tsx import { AuthPage } from "@refinedev/antd"; export const RegisterPage = () => { return <AuthPage type="register" />; };
// file: /pages/forgot-password.tsx import { AuthPage } from "@refinedev/antd"; export const ForgotPasswordPage = () => { return <AuthPage type="forgotPassword" />; };
// file: /pages/reset-password.tsx import { AuthPage } from "@refinedev/antd"; export const ResetPasswordPage = () => { return <AuthPage type="resetPassword" />; };
Error Components
Refine's Ant Design integration also provides an <ErrorComponent />
component that you can use to render a 404 page in your app. While these components does not offer much functionality, they are provided as an easy way to render an error page with a consistent design language.
An example usage of the <ErrorComponent />
component is as follows:
import { ErrorComponent } from "@refinedev/antd";
const NotFoundPage = () => {
return <ErrorComponent />;
};
Theming
Since Refine offers application level components such as layout, sidebar and header and page level components for each action, it is important to have it working with the styling of Ant Design. All components and providers exported from the @refinedev/antd
package will use the current theme of Ant Design without any additional configuration.
Additionally, Refine also provides a set of carefully crafted themes for Ant Design which outputs a nice UI with Refine's components with light and dark theme support. These themes are exported as RefineThemes
object from the @refinedev/antd
package and can be used in <ConfigProvider />
component of Ant Design.
Code Example
// file: /theme-provider.tsx import { ConfigProvider, App } from "antd"; import { RefineThemes } from "@refinedev/antd"; export const ThemeProvider = ({ children }) => ( // Available themes: Blue, Purple, Magenta, Red, Orange, Yellow, Green // Change the line below to change the theme <ConfigProvider theme={RefineThemes.Magenta}> <App> {children} </App> </ConfigProvider> );
To learn more about the theme configuration of Ant Design, please refer to the official documentation.
Inferencer
You can automatically generate views for your resources using @refinedev/inferencer
. Inferencer exports the AntdListInferencer
, AntdShowInferencer
, AntdEditInferencer
, AntdCreateInferencer
components and finally the AntdInferencer
component, which combines all in one place.
To learn more about Inferencer, please refer to the Ant Design Inferencer docs.
Known Issues
Next.js Pages Router with version 14 and above gives the following error when using @ant-design
package:
Server Error
SyntaxError: Unexpected token 'export'
This error happened while generating the page. Any console logs will be displayed in the terminal window.
Call Stack
<unknown>
/Users/user/Desktop/refine/node_modules/ (ant-design/icons-svg/es/asn/AccountBookFilled.js (3)
You can find issue details from the official Ant Design repository: