React Admin Panel Tutorial with Chakra UI and Strapi

Necati Özmen - Mar 6 '23 - - Dev Community

Image description

Author: Peter Osah

Introduction

In building data-intensive applications like React admin dashboards and in-house applications, among others, speed, customizability, and performance are of the essence as these applications are pivotal in the management of most businesses and software. This has led developers to opt for solutions like refine that provide these qualities out of the box.

refine is a React-based framework that helps to build data-intensive applications in due time. It does this by providing hooks, providers, and components that provide room for UI customizability, and data management, as well as easy integration with any custom backend APIs and popular backend services like Strapi, Hasura, etc.

In this React admin panel tutorial, we will be creating an admin application with refine that will be built with ChakraUI, an enterprise-level React component library, and Strapi, a popular backend service, as its backend provider.

Benefits of using refine

refine accelerates development by creating a higher abstraction of most functionalities such as routing, data providers, authentication, internationalization, layouts, asynchronous state management, and many others that developers would have to set up from scratch if building the application from the ground up.

With refine, you have the choice to build your application with a custom design (a headless approach) or a UI component library. refine includes integrations for some component libraries and design systems, like Material UI, Chakra UI, Ant Design, and Mantime.

It can also be integrated into any REST or GraphQL backend service and includes out-of-the-box support for popular BAAS (backend as a service) such as NestJs CRUD, Airtable, Strapi, Supabase, Altogic, and others.

A peculiarity of refine is that it provides a lot of out-of-the-box functionality for rapid development while still allowing for extreme customization. You can build a variety of applications with refine, and it isn't limited to major use cases like React admin panels, B2B applications, and dashboards. In this React admin tutorial, we will be illustrating how to build a CRUD app with refine.

Whether you're a beginner or an experienced developer, this React admin tutorial will walk you through the entire process, step by step, and show you how to integrate Strapi as your backend CMS to create an amazing React admin panel.

What is Strapi?

Strapi is a free and open-source headless Content Management System built on Node.js. It allows developers to easily create, manage, and consume content through a customizable API.
Strapi provides a user-friendly admin panel for content editors to manage content without having to write code, while also providing a flexible content modeling system that can be tailored to fit specific project needs.

Constituents of A refine Application

Before delving in to build the CRUD React admin tutorial app, we need to know the constitutents of an refine application.

The <Refine/> Component

The <Refine/> acts as the entrypoint of the application. it is a component that recieves the configurations passed into it as react props.

// src/App.tsx

import { Refine } from "@pankod/refine-core";
import dataProvider from "@pankod/refine-simple-rest";

import { PostList } from "pages/posts";

const API_URL = "https://api.fake-rest.refine.dev";

const App: React.FC = () => {
    return (
        <Refine
            dataProvider={dataProvider(API_URL)}
            resources={[
                {
                    name: "posts",
                    list: PostList,
                },
            ]}
        />
    );
};

export default App;
Enter fullscreen mode Exit fullscreen mode

There are a variety of configurations that can be passed into a refine application. they are:

  • DataProvider: a data provider enables a refine application connect to an external API or service. A dataProvider sends HTTP requests and receives responses via predefined methods.

Here is a sample on how to adding a dataProvider to the refine component.

// src/App.tsx

import { Refine } from "@pankod/refine-core";
import { authProvider, axiosInstance } from "./authProvider";
import { DataProvider } from "@pankod/refine-strapi-v4";

function App() {
 return (
   <Refine
     ...
     dataProvider={DataProvider(API_URL + `/api`, axiosInstance)}
   />
 );
}

export default App;
Enter fullscreen mode Exit fullscreen mode
  • Resources A resources is a fundamental component of a refine application. A resource acts as a bridge between the Data/API layer and the Document/Page Layer. A resource enables the application's pages to interact with the API.

  • In order to create a resource, we have to pass the resources property which will be an array of objects with each object specifying the pages route name and the basic operations the pages under that route name can perform which are the list(displaying records from an API or service), create(add or creating a record to an API or service), edit(modifying an existing record from an API or service), show(display a specific record from an API or service) operations. to the <Refine /> component.

 // src/App.tsx
import { Refine } from "@pankod/refine-core";
import { PostList, PostCreate, PostEdit, PostShow } from "pages/posts";

const API_URL = "https://api.fake-rest.refine.dev";

const App: React.FC = () => {
    return (
        <Refine
            ...
            resources={[
                {
                    name: "posts",
                    list: PostList,
                    create: PostCreate,
                    edit: PostEdit,
                    show: PostShow,
                },
            ]}
        />
    );
};

export default App;
Enter fullscreen mode Exit fullscreen mode

Bootstrapping the refine Application

To create our refine application for this article, we will use the create refine-app wizard.

create refine-app comes with built-in templates for the CRA, Next.js, and Remix environments, allowing you to setup a refine project quickly. It also provides an array of options for automatically configuring your UI framework, i18n, router, authentication, and data provider settings.

In order to use this, execute the following command:

npm create refine-app@latest <name of your application>
Enter fullscreen mode Exit fullscreen mode

You will be directed to the CLI wizard after running the command. To complete the CLI wizard, select the following options:

react admin tutorial

When these options are selected, the CLI will bootstrap a refine application with the Strapi-v4 provider.

Following installation, we will execute the following command:

npm run dev
Enter fullscreen mode Exit fullscreen mode

The refine application should be up and running after you run the command. To access it, go to http://localhost:3000.

react admin tutorial

 Easy way to bootstrap a refine application

refine.new enables you to quickly bootstrap a new refine application, an open-source React-based, headless UI library for creating enterprise applications, within your browser with features such as preview, tweaking, and instant download.

To get started, head on to refine.new , scroll down the page, and click on the Let's start button.



refine banner

Utilizing the Strapi v4 Provider in refine

On installation, since we selected the strapi-v4 data provider, refine adds a fake strapi API url to the data provider of the refine application

https://api.strapi-v4.refine.dev
Enter fullscreen mode Exit fullscreen mode

In this React admin tutorial, we will be using refine's fake strapi-v4 API url for demo purposes. If you have a strapi-v4 API url you wish to integrate into the application, you can always replace the fake strapi API with that.

Setting AuthProvider

The create-refine-app command generates a new refine application with a default AuthProvider. If you followed the steps above to create your application using create-react-app, you should already have the src/authProvider.ts file.

It's worth noting that the authProvider's login method is especially important. We'll be using email and password to sign in to our application, so make sure that the login method contains the code provided below.

// src/authProvider.ts

export const authProvider: AuthProvider = {
  //highlight-start
  login: async ({ email, password }) => {

    const { data, status } = await strapiAuthHelper.login(email, password);
    //highlight-end
    if (status === 200) {
      localStorage.setItem(TOKEN_KEY, data.jwt);

      // set header axios instance
      axiosInstance.defaults.headers.common = {
        Authorization: `Bearer ${data.jwt}`,
      };

      return Promise.resolve();
    }
    return Promise.reject();
  },
  ...
};
Enter fullscreen mode Exit fullscreen mode

Implementing CRUD operations on React admin panel

We will implement CRUD operations such as listing, creating, and editing records in our React admin tutorial application.

In this article, we describe our React admin tutorial app, which is one that can track the list of posts created, as well as have the functionality of viewing a post as well as updating and deleting a post, respectively.

Listing records

To display data on the UI, we must create a <PostList/> page.

To begin, we'll require an interface to interact with the data from the API endpoint.

If you don't already have one, we'll create one under the src folder called interfaces. Then, create a file called index.d.ts with the following code:

// src/interfaces/index.d.ts
export interface IPost {
    id: number;
    title: string;
    content: string;
    category: any;
    publishedAt: string
    createdAt: string;
}
Enter fullscreen mode Exit fullscreen mode

Next, we create a new folder named pages under the src folder. Under that folder, create a posts folder and add a list.tsx file with the following code:

// pages/posts/list.tsx

import React from "react";
import {
    IResourceComponentsProps,
    GetManyResponse,
    useMany,
} from "@pankod/refine-core";
import { useTable, ColumnDef, flexRender } from "@pankod/refine-react-table";
import {
    List,
    TableContainer,
    Table,
    Thead,
    Tr,
    Th,
    Tbody,
    Td,
    HStack,
    Button,
    IconButton,
    usePagination,
    Box,
    EditButton,
    ShowButton,
} from "@pankod/refine-chakra-ui";
import { IconChevronRight, IconChevronLeft } from "@tabler/icons";
import { IPost } from "interfaces";

export const PostList: React.FC<IResourceComponentsProps> = () => {
    const columns = React.useMemo<ColumnDef<IPost>[]>(
        () => [
            {
                id: "id",
                accessorKey: "id",
                header: "Id",
            },
            {
                id: "title",
                accessorKey: "title",
                header: "Title",
            },
            {
                id: "createdAt",
                accessorKey: "createdAt",
                header: "Created At",
                cell: function render({ getValue, table }) {
                    return (
                        <DateField value={getValue() as string} format="LLL" />
                    )
                },
            },
            {
                id: "actions",
                accessorKey: "id",
                header: "Actions",
                cell: function render({ getValue }) {
                    return (
                        <HStack>
                            <ShowButton
                                hideText
                                recordItemId={getValue() as string}
                            />
                            <EditButton
                                hideText
                                recordItemId={getValue() as string}
                            />
                        </HStack>
                    );
                },
            },
        ],
        [],
    );

    const {
        getHeaderGroups,
        getRowModel,
        setOptions,
        refineCore: {
            setCurrent,
            pageCount,
            current,
            tableQueryResult: { data: tableData },
        },
    } = useTable({
        columns,
    });


    setOptions((prev) => ({
        ...prev,
        meta: {
            ...prev.meta,
        },
    }));

    return (
        <List>
            <TableContainer whiteSpace="pre-line">
                <Table variant="simple">
                    <Thead>
                        {getHeaderGroups().map((headerGroup) => (
                            <Tr key={headerGroup.id}>
                                {headerGroup.headers.map((header) => (
                                    <Th key={header.id}>
                                        {!header.isPlaceholder &&
                                            flexRender(
                                                header.column.columnDef.header,
                                                header.getContext(),
                                            )}
                                    </Th>
                                ))}
                            </Tr>
                        ))}
                    </Thead>
                    <Tbody>
                        {getRowModel().rows.map((row) => (
                            <Tr key={row.id}>
                                {row.getVisibleCells().map((cell) => (
                                    <Td key={cell.id}>
                                        {flexRender(
                                            cell.column.columnDef.cell,
                                            cell.getContext(),
                                        )}
                                    </Td>
                                ))}
                            </Tr>
                        ))}
                    </Tbody>
                </Table>
            </TableContainer>
        </List>
    );
};
Enter fullscreen mode Exit fullscreen mode

In the code above, we use the useTable() hook from the @pankod/refine-react-table package to fetch records from Strapi. the useTable() maps the records into rows based on the columns variable definition.

After this, we can now add the component <PostList/> in the list.tsx file to our resource present in the App.tsx file

// src/App.tsx

import React from "react";
import { Refine } from "@pankod/refine-core";
import {
  notificationProvider,
  ChakraProvider,
  refineTheme,
  ReadyPage,
  ErrorComponent,
  Layout,
  AuthPage,
} from "@pankod/refine-chakra-ui";
import { DataProvider } from "@pankod/refine-strapi-v4";
import routerProvider from "@pankod/refine-react-router-v6";
import { authProvider, axiosInstance } from "./authProvider";
import { API_URL } from "./constants";
import { PostList} from './pages/posts';


function App() {
  return (
    <ChakraProvider theme={refineTheme}>
      <Refine
        ...
         // highlight-start
        LoginPage={AuthPage}
        resources={[
          {
            name: "posts",
            list: PostList,
          }
          // highlight-end
        ]}
      />
    </ChakraProvider>
  );
}

export default App;

Enter fullscreen mode Exit fullscreen mode

After adding the rescource, the application will now redirect to the URL specified by the name property.

It will prompt you to log in to the app. Since we are using refine's fake strapi-v4 API, Try these credentials:

Username: demo@refine.dev
Password: demodemo

Check that the URL is routed to /posts and that posts are present.

Handling Relationships

When retrieving recorda, relations are not populated. Hence, We will use the metaData option to populate the Strapi v4 API with relational data (category data in our case).

Refer to docs for handling relations with strapi

We will populate records with a category id field from the /posts endpoint with their respective titles. To automatically get category titles from the /categoriesendpoint for each record and display them on our table, we must use Strapi v4's populate feature.

We'll use the populate parameter to specify which fields should be filled in. we will assign an array to the populate parameter which will contain the map the values from the /categories endpoint to the records with a category id on the /posts endpoint. The populate parameter will be a property of the metaData property which is subsequently a property of the refineCoreProps object.

// pages/posts/list.tsx

const {
    getHeaderGroups,
    getRowModel,
    setOptions,
    refineCore: {
        setCurrent,
        pageCount,
        current,
        tableQueryResult: { data: tableData },
    },
} = useTable({
    columns,
    refineCoreProps: {
        metaData: {
            populate: ["category"],
        },
    },
});
Enter fullscreen mode Exit fullscreen mode

To show category field in table, we need to add new column to the PostList component.

// pages/posts/list.tsx

const columns = React.useMemo<GridColumns<IPost>>(
    () => [
        ...
        {
            id: "category",
            accessorKey: "category.title",
            header: "Category",
            cell: function render({ getValue, table }) {
                   return getValue();
                 }
        },
        ...
          ],
        [],
    );
Enter fullscreen mode Exit fullscreen mode

react admin tutorial

Pagination Of Listed Records

Next, we will add Pagination to our application. in order to achieve this, the usePagination hook provides certain functions that handle pagination. They are:

We will go to update the <PostList/> component with the code below:

// pages/posts/list.tsx

import React from "react";
import {
    IResourceComponentsProps,
    GetManyResponse,
    useMany,
} from "@pankod/refine-core";
import { useTable, ColumnDef, flexRender } from "@pankod/refine-react-table";
import {
    List,
    TableContainer,
    Table,
    Thead,
    Tr,
    Th,
    Tbody,
    Td,
    HStack,
    Button,
    IconButton,
    usePagination,
    Box,
    EditButton,
    ShowButton,
} from "@pankod/refine-chakra-ui";
import { IconChevronRight, IconChevronLeft } from "@tabler/icons";
import { IPost } from "interfaces";

export const PostList: React.FC<IResourceComponentsProps> = () => {
    const columns = React.useMemo<ColumnDef<IPost>[]>(
        () => [
            ...
        ],
        [],
    );


    const {
        getHeaderGroups,
        getRowModel,
        setOptions,
        refineCore: {
            setCurrent,
            pageCount,
            current,
            tableQueryResult: { data: tableData },
        },
    } = useTable({
        columns,
    });

    setOptions((prev) => ({
        ...prev,
        meta: {
            ...prev.meta,
        },
    }));

    return (
        <List>
            ...
            <Pagination
                current={current}
                pageCount={pageCount}
                setCurrent={setCurrent}
            />
        </List>
    );
};

type PaginationProps = {
    current: number;
    pageCount: number;
    setCurrent: (page: number) => void;
};

const Pagination: React.FC<PaginationProps> = ({
    current,
    pageCount,
    setCurrent,
}) => {
    const pagination = usePagination({
        current,
        pageCount,
    });

    return (
        <Box display="flex" justifyContent="flex-end">
            <HStack my="3" spacing="1">
                {pagination?.prev && (
                    <IconButton
                        aria-label="previous page"
                        onClick={() => setCurrent(current - 1)}
                        disabled={!pagination?.prev}
                        variant="outline"
                    >
                        <IconChevronLeft size="18" />
                    </IconButton>
                )}

                {pagination?.items.map((page) => {
                    if (typeof page === "string")
                        return <span key={page}>...</span>;

                    return (
                        <Button
                            key={page}
                            onClick={() => setCurrent(page)}
                            variant={page === current ? "solid" : "outline"}
                        >
                            {page}
                        </Button>
                    );
                })}
                {pagination?.next && (
                    <IconButton
                        aria-label="next page"
                        onClick={() => setCurrent(current + 1)}
                        variant="outline"
                    >
                        <IconChevronRight size="18" />
                    </IconButton>
                )}
            </HStack>
        </Box>
    );
};
Enter fullscreen mode Exit fullscreen mode

In the code above, we create a <Pagination/> components which accepts properties:

  • current: This property holds the current page number obtained from the useTable() hook,
  • pageCount: This property holds the total amount of pages present from the useTable() hook,
  • setCurrent(): This property sets the current page number to an value. it handles the navigation to either the next page or the previous page.

Viewing a record

To view a record, we will use the useShow() hook, which is included in the @pankod/refine-core package.
The hook useShow() allows you to retrieve the desired record. It uses the getOne method from the dataProvider passed to <Refine/> as the query function.

We'll add a show.tsx file in the posts folder under the pages folder next, with the following code:

Show code
// pages/posts/show.tsx

import { useOne, useShow } from "@pankod/refine-core";
import {
    Show,
    Heading,
    NumberField,
    TextField,
    DateField,
} from "@pankod/refine-chakra-ui";

export const PostShow = () => {
    const { queryResult } = useShow({
        metaData: {
            populate: ["category"],
        }
    });
    const { data, isLoading } = queryResult;

    const record = data?.data;

    const { data: categoryData, isLoading: categoryIsLoading } = useOne({
        resource: "categories",
        id: record?.category?.id || "",
        queryOptions: {
            enabled: !!record,
        },
    });

    return (
        <Show isLoading={isLoading}>
            <Heading as="h5" size="sm" mt={4}>
                Id
            </Heading>
            <NumberField value={record?.id ?? ""} />

            <Heading as="h5" size="sm" mt={4}>
                Title
            </Heading>
            <TextField value={record?.title} />

            <Heading as="h5" size="sm" mt={4}>
                Category
            </Heading>

            {categoryIsLoading ? (
                <>Loading...</>
            ) : (
                <>{categoryData?.data?.title}</>
            )}

            <Heading as="h5" size="sm" mt={4}>
                Content
            </Heading>
            <TextField value={record?.content} />

            <Heading as="h5" size="sm" mt={4}>
                Created At
            </Heading>
            <DateField value={record?.createdAt} />

            <Heading as="h5" size="sm" mt={4}>
                Published At
            </Heading>
            <DateField value={record?.publishedAt} />

            <Heading as="h5" size="sm" mt={4}>
                Locale
            </Heading>
           <TextField value={record?.locale} />
        </Show>
    );
};
Enter fullscreen mode Exit fullscreen mode

After this, we can now add the component <PostShow/> in the show.tsx file to our resource present in the App.tsx file

// src/App.tsx

...
import {
    PostList,
    //highlight-next-line
    PostShow 
} from './pages/posts';

function App() {
  return (
    <ChakraProvider theme={refineTheme}>
      <Refine
         ...
          {
            name: "posts",
            list: PostList,
            //highlight-next-line
            show: PostShow,
          }
        ]}
      />
    </ChakraProvider>
  );
}

export default App;
Enter fullscreen mode Exit fullscreen mode

react admin tutorial

Creating a record

To create a record, we will use the @pankod/refine-react-hook-form package.
It provides the useForm() hook, which includes form validation and handles our form submission request to Strapi.

Next, in the posts folder under the pages folder, we'll include a create.tsx file with the following code:

Show code
// pages/posts/create.tsx

import React from "react";
import {
    Create,
    FormControl,
    FormLabel,
    FormErrorMessage,
    Input,
    Textarea,
    Select,
} from "@pankod/refine-chakra-ui";
import { useSelect } from "@pankod/refine-core";
import { useForm } from "@pankod/refine-react-hook-form";

export const PostCreate = () => {
    const {
        refineCore: { formLoading, queryResult },
        saveButtonProps,
        register,
        resetField,
        formState: { errors },
    } = useForm();

    const postsData = queryResult?.data?.data;

    const { options: categoryOptions } = useSelect({
        resource: "categories",
        defaultValue: postsData?.category?.id,
    });

    React.useEffect(() => {
        resetField("category.id");
    }, [categoryOptions]);

    return (
        <Create isLoading={formLoading} saveButtonProps={saveButtonProps}>
            <FormControl mb="3" isInvalid={!!(errors as any)?.title}>
                <FormLabel>Title</FormLabel>
                <Input
                    type="text"
                    {...register("title", {
                        required: "This field is required",
                    })}
                />
                <FormErrorMessage>
                    {(errors as any)?.title?.message as string}
                </FormErrorMessage>
            </FormControl>

            <FormControl mb="3" isInvalid={!!(errors as any)?.title}>
                <FormLabel>Content</FormLabel>
                <Textarea
                    {...register("content", {
                        required: "This field is required",
                    })}
                />
                <FormErrorMessage>
                    {(errors as any)?.title?.message as string}
                </FormErrorMessage>
            </FormControl>

            <FormControl mb="3" isInvalid={!!errors?.category}>
                <FormLabel>Category</FormLabel>
                <Select
                    placeholder="Select category"
                    {...register("category.id", {
                        required: "This field is required",
                    })}
                >
                    {categoryOptions?.map((option) => (
                        <option value={option.value} key={option.value}>
                            {option.label}
                        </option>
                    ))}
                </Select>
                <FormErrorMessage>
                    {(errors as any)?.category?.id?.message as string}
                </FormErrorMessage>
            </FormControl>
        </Create>
    );
};
Enter fullscreen mode Exit fullscreen mode

In the code above, we used the useForm() hook to create records. This hook comes from the (@pankod/refine-react-hook-form) which is a refine adapter of the React Hook Form library. This library allows you to use the React Hook Form library with refine. More information about the useForm() hook can be obtained here.

we use methods provided by the useForm() hook like register() to validate the new post we will add into Strapi.
The hooks also provide an object like saveButtonProps which accepts the properties shown below:

{
    disabled: boolean;
    onClick: (e: React.BaseSyntheticEvent<object, any, any>) => void;
}
Enter fullscreen mode Exit fullscreen mode

These properties handles the submission of the form action, button loading, and disable states.

After this, we can now add the component <PostCreate/> in the create.tsx file to our resource present in the App.tsx file

// src/App.tsx

...
import {
     //highlight-next-line
     PostCreate
} from './pages/posts';

function App() {
  return (
    <ChakraProvider theme={refineTheme}>
      <Refine
        ...
        resources={[
          {
            name: "posts",
            list: PostList,
            show: PostShow,
            //highlight-next-line
            create: PostCreate,
          }
        ]}
      />
    </ChakraProvider>
  );
}

export default App;
Enter fullscreen mode Exit fullscreen mode

react admin tutorial

Modifying/Editing records.

For editing a record, we will add an edit.tsx file In the posts folder under the pages folder and add the code below to the file:

// pages/posts/edit.tsx

import React from "react";
import {
    Edit,
    FormControl,
    FormLabel,
    FormErrorMessage,
    Input,
    Select,
} from "@pankod/refine-chakra-ui";
import { useForm } from "@pankod/refine-react-hook-form";
import { useSelect } from "@pankod/refine-core";

export const PostEdit = () => {
    const {
        refineCore: { formLoading, queryResult },
        saveButtonProps,
        register,
        resetField,
        formState: { errors },
    } = useForm({
        refineCoreProps : {
            metaData: {
                populate: ["category"],
            }
        }
    });

    const postsData = queryResult?.data?.data;

    const { options: categoryOptions } = useSelect({
        resource: "categories",
        defaultValue: postsData?.category?.id,
    });

    React.useEffect(() => {
        resetField("category.id");
    }, [categoryOptions]);

    return (
        <Edit isLoading={formLoading} saveButtonProps={saveButtonProps}>
            <FormControl mb="3" isInvalid={!!(errors as any)?.id}>
                <FormLabel>Id</FormLabel>
                <Input
                    disabled
                    type="number"
                    {...register("id", {
                        required: "This field is required",
                    })}
                />
                <FormErrorMessage>
                    {(errors as any)?.id?.message as string}
                </FormErrorMessage>
            </FormControl>

            <FormControl mb="3" isInvalid={!!(errors as any)?.title}>
                <FormLabel>Title</FormLabel>
                <Input
                    type="text"
                    {...register("title", {
                        required: "This field is required",
                    })}
                />
                <FormErrorMessage>
                    {(errors as any)?.title?.message as string}
                </FormErrorMessage>
            </FormControl>

            <FormControl mb="3" isInvalid={!!(errors as any)?.content}>
                <FormLabel>Content</FormLabel>
                <Input
                    type="text"
                    {...register("content", {
                        required: "This field is required",
                    })}
                />
                <FormErrorMessage>
                    {(errors as any)?.content?.message as string}
                </FormErrorMessage>
            </FormControl>

            <FormControl mb="3" isInvalid={!!errors?.category}>
                <FormLabel>Category</FormLabel>
                <Select
                    placeholder="Select category"
                    {...register("category.id", {
                        required: "This field is required",
                    })}
                >
                    {categoryOptions?.map((option) => (
                        <option value={option.value} key={option.value}>
                            {option.label}
                        </option>
                    ))}
                </Select>
                <FormErrorMessage>
                    {(errors as any)?.category?.id?.message as string}
                </FormErrorMessage>
            </FormControl>

            <FormControl mb="3" isInvalid={!!(errors as any)?.createdAt}>
                <FormLabel>Created At</FormLabel>
                <Input
                    disabled
                    {...register("createdAt", {
                        required: "This field is required",
                    })}
                />
                <FormErrorMessage>
                    {(errors as any)?.createdAt?.message as string}
                </FormErrorMessage>
            </FormControl>

            <FormControl mb="3" isInvalid={!!(errors as any)?.updatedAt}>
                <FormLabel>Updated At</FormLabel>
                <Input
                    disabled
                    {...register("updatedAt", {
                        required: "This field is required",
                    })}
                />
                <FormErrorMessage>
                    {(errors as any)?.updatedAt?.message as string}
                </FormErrorMessage>
            </FormControl>

            <FormControl mb="3" isInvalid={!!(errors as any)?.publishedAt}>
                <FormLabel>Published At</FormLabel>
                <Input
                    disabled
                    {...register("publishedAt", {
                        required: "This field is required",
                    })}
                />
                <FormErrorMessage>
                    {(errors as any)?.publishedAt?.message as string}
                </FormErrorMessage>
            </FormControl>

            <FormControl mb="3" isInvalid={!!(errors as any)?.locale}>
                <FormLabel>Locale</FormLabel>
                <Input
                    disabled
                    type="text"
                    {...register("locale", {
                        required: "This field is required",
                    })}
                />
                <FormErrorMessage>
                    {(errors as any)?.locale?.message as string}
                </FormErrorMessage>
            </FormControl>
        </Edit>
    );
};
Enter fullscreen mode Exit fullscreen mode

In the code above, just like in the <PostCreate/> component, we use methods provided by the useForm() hook like register() to validate the new post we will add into Strapi.
The hooks also provide an object like saveButtonProps which handles the submission of the form action, button loading, and disable states

After this, we can now add the component <PostEdit/> in the edit.tsx file to our resource present in the App.tsx file

// src/App.tsx

import {
     ...
     // highlight-next-line
     PostEdit 
} from "./pages/posts";


function App() {
  return (
    <ChakraProvider theme={refineTheme}>
      <Refine
        ...
        resources={[
          {
            name: "posts",
            list: PostList,
            show: PostShow,
            create: PostCreate,
            // highlight-next-line
            edit: PostEdit,
          }
        ]}
      />
    </ChakraProvider>
  );
}

export default App;
Enter fullscreen mode Exit fullscreen mode

react admin tutorial

Deleting post record

To delete a record, We'll add a Delete button so we'll need to update our <PostList/>component to include the button to the actions row. Add the highlighted lines below to the existing list component.

// pages/posts/list.tsx

...
export const PostList: React.FC = () => {
 const { show, edit, create } = useNavigation();
 const { mutate } = useDelete();

 const columns = React.useMemo<ColumnDef<IPost>[]>(
   () => [
    ...,
    // highlight-start
      {
                id: "actions",
                accessorKey: "id",
                header: "Actions",
                cell: function render({ getValue }) {
                    return (
                        <HStack>
                            ...
                             <DeleteButton
                                hideText
                                recordItemId={getValue() as string}
                            />
                        </HStack>
                    );
                },
        },
    // highlight-end
   [],
 );
};
Enter fullscreen mode Exit fullscreen mode

Another way to include a delete button on the <PostEdit> page. To display the delete button on the edit page and the view page, the canDelete prop must be passed to the resource object.

// src/App.tsx

...
function App() {
  return (
    <ChakraProvider theme={refineTheme}>
      <Refine
        ...
        resources={[
          {
            ...
            // highlight-next-line
            canDelete: true,
          }
        ]}
      />
    </ChakraProvider>
  );
}

export default App;
Enter fullscreen mode Exit fullscreen mode

react admin tutorial

Implement sorting of listed records

Since the @pankod/refine-react-table package is based on the Tanstack Table package, we can add sorting and filtering features to our table.

Let's make a <ColumnSorter/> component for our table header. This component will be in charge of changing the table's sorting state.
we create a new folder named components under the src folder. Under that folder, add a ColumnSorter.tsx file with the following code:

// components/ColumnSorter.tsx

import { IconButton } from "@chakra-ui/react";
import { IconChevronDown, IconSelector } from "@tabler/icons";
import type { Column } from "@pankod/refine-react-table";

export const ColumnSorter: React.FC<{ column: Column<any, any> }> = ({
    column,
}) => {
    if (!column.getCanSort()) {
        return null;
    }

    const sorted = column.getIsSorted();

    return (
        <IconButton
            aria-label="Sort"
            size="xs"
            onClick={column.getToggleSortingHandler()}
        >
            {sorted ? (
                <IconChevronDown size={18} />
            ) : (
                <IconSelector size={18} />
            )}
        </IconButton>
    );
};
Enter fullscreen mode Exit fullscreen mode

In the code above, the <ColumnSorter/> is a simple component that displays a button.
The column.getToggleSortingHandler() method is called when the user clicks on the button. This method will alter the table's sorting state.

Also, we use the column.getCanSort() method to determine if the column is sortable. If it is, We will render the <ColumnSorter/> component and vice-versa.

Once the column is sorted, the IconChevronDown icon will be rendered. Otherwise, the IconSelector icon will be rendered.

Next, we import the <ColumnSorter/> component to the list.tsx file located at the src/pages/posts/ directory and add the component to the <Th/> of the table as a child as shown below:

// pages/posts/list.tsx

...
import { ColumnSorter } from "components/ColumnSorter";

<Thead>
{getHeaderGroups().map((headerGroup) => (
    <Tr key={headerGroup.id}>
        {headerGroup.headers.map((header) => (
            <Th key={header.id}>
                {
            !header.isPlaceholder 
            && flexRender(header.column.columnDef.header,
                                                 header.getContext(),
        )}
            <ColumnSorter column={header.column} />

            </Th>
        ))}
    </Tr>
    ))}
</Thead>
Enter fullscreen mode Exit fullscreen mode

You can set the enableSorting property of the column to false in the column definition to disable sorting for the actions column, as shown below:

// pages/posts/list.tsx

{
    id: "actions",
    accessorKey: "id",
    header: "Actions",
    enableSorting: false,
    ...
},
Enter fullscreen mode Exit fullscreen mode

Implement filters on listed records

Firstly, Let's make a <ColumnFilter/> component for our table header. This component will handle the filtering of listed records.
we create a new folder named components under the src folder. Under that folder, add a ColumnFilter.tsx file with the following code:

// components/ColumnFilter.tsx

import React, { useState } from "react";
import {
    Input,
    Menu,
    IconButton,
    MenuButton,
    MenuList,
    VStack,
    HStack,
} from "@pankod/refine-chakra-ui";
import { IconFilter, IconX, IconCheck } from "@tabler/icons";
import type { Column } from "@pankod/refine-react-table";

export const ColumnFilter: React.FC<{ column: Column<any, any>; }> = ({
    column,
}) => {

    const [state, setState] = useState(null as null | { value: any; });

    if (!column.getCanFilter()) {
        return null;
    }

    const open = () =>
        setState({
            value: column.getFilterValue(),
        });

    const close = () => setState(null);

    const change = (value: any) => setState({ value });

    const clear = () => {
        column.setFilterValue(undefined);
        close();
    };

    const save = () => {
        if (!state) return;
        column.setFilterValue(state.value);
        close();
    };

    const renderFilterElement = () => {
        const FilterComponent = (column.columnDef?.meta as any)?.filterElement;

        if (!FilterComponent && !!state) {
            return (
                <Input
                    borderRadius="md"
                    size="sm"
                    autoComplete="off"
                    value={state.value}
                    onChange={(e) => change(e.target.value)}
                />
            );
        }

        return (
            <FilterComponent
                value={state?.value}
                onChange={(e: any) => change(e.target.value)}
            />
        );
    };

    return (
        <Menu isOpen={!!state} onClose={close}>
            <MenuButton
                onClick={open}
                as={IconButton}
                aria-label="Options"
                icon={<IconFilter size="16" />}
                variant="ghost"
                size="xs"
            />
            <MenuList p="2">
                {!!state && (
                    <VStack align="flex-start">
                        {renderFilterElement()}
                        <HStack spacing="1">
                            <IconButton
                                aria-label="Clear"
                                size="sm"
                                colorScheme="red"
                                onClick={clear}
                            >
                                <IconX size={18} />
                            </IconButton>
                            <IconButton
                                aria-label="Save"
                                size="sm"
                                onClick={save}
                                colorScheme="green"
                            >
                                <IconCheck size={18} />
                            </IconButton>
                        </HStack>
                    </VStack>
                )}
            </MenuList>
        </Menu>
    );
};
Enter fullscreen mode Exit fullscreen mode

In the code above, <ColumnFilter/> is a button. on clicking the button, a menu will be opened. In the menu exists a filter element of the column that is being rendered. The filter element is an <Input/> component in which you can add a value which will be used to subsequently filter the records based on that value.

Next, we Import the <ColumnFilter/> component to the list.tsx file located at the src/pages/posts/ directory and add the component to the <Th/> of the table just like we did the <ColumnSorter/> component.

// pages/posts/list.tsx

import { ColumnFilter } from "components/ColumnFilter";

....

<Thead>
{getHeaderGroups().map((headerGroup) => (
    <Tr key={headerGroup.id}>
        {headerGroup.headers.map((header) => (
            <Th key={header.id}>
                {
            !header.isPlaceholder 
            && flexRender(header.column.columnDef.header,
                                                 header.getContext(),
        )}
            <ColumnSorter column={header.column} />
            <ColumnFilter column={header.column} />
            </Th>
        ))}
    </Tr>
    ))}
</Thead>
Enter fullscreen mode Exit fullscreen mode

We can change the filter operator for columns to "contains". The filterOperator is set to "eq" by default. As a result, we changed the filterOperator for specific columns to "contains" as shown below:

// pages/posts/list.tsx

{
    id: "id",
    accessorKey: "id",
    header: "Id",
    meta: {
        filterOperator: "contains",
    },
},
{
    id: "title",
    accessorKey: "title",
    header: "Title",
    meta: {
        filterOperator: "contains",
    },
},
{
    id: "category",
    accessorKey: "category.title",
    header: "Category",
    meta: {
        filterOperator: "contains",
    },
    ...
},
{
    id: "createdAt",
    accessorKey: "createdAt",
    header: "Created At",
    ...
},
Enter fullscreen mode Exit fullscreen mode

We can also disable filtering for the actions column by setting the column's enableColumnFilter property to false in the column definition, as shown below:

// pages/posts/list.tsx
{
    id: "actions",
    accessorKey: "id",
    header: "Actions",
    enableColumnFilter: false,
    ...
},
Enter fullscreen mode Exit fullscreen mode

react admin tutorial

Implementing mutation mode.

Mutation modes controls how side-effects or actions like deletion, creation, updating, etc are applied. refine provides three mutation modes namely:

  • pessimistic: In this mode, UI updates are delayed until the server confirms the mutation.
  • undoable: In this mode, UI updates are immediately applied, but the mutation can be reversed.
  • optimistic: UI updates are immediately updated before confirmed by server.

We will implement undoable mutation mode. The mutation is applied locally, and redirection and UI updates are performed as soon as the mutation is successful. It then Waits for a configurable amount of time before applying the mutation.

During the timeout, the mutation from the notification can be cancelled using an undo button and the UI will revert the changes to its previous state.

To enable mutation mode, we will set the mutationMode property in the Refine component options property.

// src/App.tsx

...
function App() {
  return (
    <ChakraProvider theme={refineTheme}>
      <Refine
        ...
        options={{ mutationMode: "undoable" }}
      />
    </ChakraProvider>
  );
}
export default App;
Enter fullscreen mode Exit fullscreen mode

react admin tutorial

Conclusion

In this article, we covered how to create a React admin panel with refine. We also learnt:

  • How to bootstrap a refine application.
  • Connecting a Strapiv4 data provider to a refine application.
  • implemented CRUD operations on a refine application.

If you're looking to build a robust and dynamic admin panel for your web application, then this React admin tutorial is just what you need.

This article demonstrates the flexibility that comes with using refine to create web applications. refine is a fantastic tool that accelerates development by abstracting many time-consuming tasks, giving the developer more time to focus on the application's core business logic.

It also comes with detailed documentation that will get you started. Visit here to access the documentation.

. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .