Skip to main content

How to Create an Admin Widgetbeta

beta

In this document, you will learn about what Admin widgets are and how you can create your own.

Overview

Admin Widgets are custom React components that developers create to be injected into predetermined injection zones across the Medusa Admin dashboard.

Widgets allow you to customize the admin dashboard by providing merchants with new features. For example, you can add a widget on the order details page that shows payment details retrieved from Stripe.

This guide explains the available injection zones and how to create an admin widget.


Injection Zones

Injection zones are areas in the admin that you can add widgets into. Widgets can only be added into these areas.

There are different types of injection zones, and the type affects where the Widget is injected. For the different domains such as productCopy to Clipboard, orderCopy to Clipboard, and customerCopy to Clipboard there are listCopy to Clipboard and detailsCopy to Clipboard zones.

Below is a full list of injection zones:

You can learn more about the additional props in the Props section.

Injection Zone NameDescriptionAdditional Props

order.list.beforeCopy to Clipboard

Added at the top of the orders list page

-

order.list.afterCopy to Clipboard

Added at the bottom of the order list page

-

order.details.beforeCopy to Clipboard

Added at the top of the order details page

Type OrderDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
order, // Order object
}

order.details.afterCopy to Clipboard

Added at the end of the order details page

Type OrderDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
order, // Order object
}

draft_order.list.beforeCopy to Clipboard

Added at the top of the draft orders list page

-

draft_order.list.afterCopy to Clipboard

Added at the bottom of the draft orders list page

-

draft_order.details.beforeCopy to Clipboard

Added at the top of the draft order details page

Type DraftOrderDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
draftOrder, // DraftOrder object
}

draft_order.details.afterCopy to Clipboard

Added at the bottom of the draft order details page

Type DraftOrderDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
draftOrder, // DraftOrder object
}

customer.list.beforeCopy to Clipboard

Added at the top of the customers list page

-

customer.list.afterCopy to Clipboard

Added at the bottom of the customers list page

-

customer.details.beforeCopy to Clipboard

Added at the top of the customer details page

Type CustomerDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
customer, // Customer object
}

customer.details.afterCopy to Clipboard

Added at the bottom of the customer details page

Type CustomerDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
customer, // Customer object
}

customer_group.list.beforeCopy to Clipboard

Added at the top of the customer groups list page

-

customer_group.list.afterCopy to Clipboard

Added at the bottom of the customer groups list page

-

customer_group.details.beforeCopy to Clipboard

Added at the top of the customer group details page

Type CustomerGroupDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
customerGroup, // CustomerGroup object
}

customer_group.details.afterCopy to Clipboard

Added at the bottom of the customer group details page

Type CustomerGroupDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
customerGroup, // CustomerGroup object
}

product.list.beforeCopy to Clipboard

Added at the top of the product list page

-

product.list.afterCopy to Clipboard

Added at the bottom of the products list page

-

product.details.beforeCopy to Clipboard

Added at the top of the product details page

Type ProductDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
product, // Product object
}

product.details.afterCopy to Clipboard

Added at the bottom of the product details page

Type ProductDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
product, // Product object
}

product_collection.list.beforeCopy to Clipboard

Added at the top of the product collections list page

-

product_collection.list.afterCopy to Clipboard

Added at the bottom of the product collections list page

-

product_collection.details.beforeCopy to Clipboard

Added at the top of the product collection details page

-

product_collection.details.afterCopy to Clipboard

Added at the bottom of the product collections list page

-

price_list.list.beforeCopy to Clipboard

Added at the top of the “price list” list page

-

price_list.list.afterCopy to Clipboard

Added at the bottom of the “price list” list page

-

price_list.details.beforeCopy to Clipboard

Added at the top of the “price list” details page

Type PriceListDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
priceList, // PriceList object
}

price_list.details.afterCopy to Clipboard

Added at the bottom of the “price list” details page

Type PriceListDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
priceList, // PriceList object
}

discount.list.beforeCopy to Clipboard

Added at the top of the discounts list page

-

discount.list.afterCopy to Clipboard

Added at the bottom of the discounts list page

-

discount.details.beforeCopy to Clipboard

Added at the top of the discounts details page

Type DiscountDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
discount, // Discount object
}

discount.details.afterCopy to Clipboard

Added at the bottom of the discount details page

Type DiscountDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
discount, // Discount object
}

gift_card.list.beforeCopy to Clipboard

Added at the top of the gift cards list page

-

gift_card.list.afterCopy to Clipboard

Added at the bottom of the gift cards list page

-

gift_card.details.beforeCopy to Clipboard

Added at the top of the gift card details page

Type GiftCardDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
giftCard, // Product object
}

gift_card.details.afterCopy to Clipboard

Added at the bottom of the gift card details page

Type GiftCardDetailsWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
giftCard, // Product object
}

custom_gift_card.beforeCopy to Clipboard

Added at the top of the custom gift card page

Type GiftCardCustomWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
giftCard, // GiftCard object
}

custom_gift_card.afterCopy to Clipboard

Added at the bottom of the custom gift card page

Type GiftCardCustomWidgetPropsCopy to Clipboard imported from @medusajs/adminCopy to Clipboard

{
giftCard, // GiftCard object
}

login.beforeCopy to Clipboard

Added before the login form

-

login.afterCopy to Clipboard

Added after the login form

-


Widget Requirements

A Widget must adhere to a set of criteria that determines if it is valid for injection. These are:

  1. All widget files must be placed in the folder /src/admin/widgetsCopy to Clipboard in your backend directory.
  2. A widget file must have a valid React component as its default export.
  3. A widget file must export a config object of type WidgetConfigCopy to Clipboard imported from @medusajs/adminCopy to Clipboard.

WidgetConfig

WidgetConfigCopy to Clipboard is used to determine the configurations of the widget, mainly the injection zones. It’s an object that accepts the property zoneCopy to Clipboard, which can be a single or an array of injection zone strings. For example:

{
zone: "product.details.after"
}
Report Incorrect CodeCopy to Clipboard

How to Create a Widget

In this section, you’ll learn how to create an admin widget.

Prerequisites

It’s assumed you already have a Medusa backend with the admin plugin installed before you move forward with this guide. If not, you can follow this documentation page to install a Medusa project.

Furthermore, Admin Widgets are currently available as a beta feature. So, you must install the betaCopy to Clipboard version of the @medusajs/adminCopy to Clipboard and @medusajs/medusaCopy to Clipboard packages:

yarn add @medusajs/admin@beta @medusajs/medusa@beta
Report Incorrect CodeCopy to Clipboard

(Optional) TypeScript Preparations

Since Widgets are React components, they should be written in .tsxCopy to Clipboard or .jsxCopy to Clipboard files. If you’re using Typescript, you need to make some adjustments to avoid Typescript errors in your Admin files.

This section provides recommended configurations to avoid any TypeScript errors.

These changes may already be available in your Medusa project. They're included here for reference purposes.

First, update your tsconfig.jsonCopy to Clipboard with the following configurations:

tsconfig.json
{
"compilerOptions": {
"target": "es2019",
"module": "commonjs",
"allowJs": true,
"checkJs": false,
"jsx": "react-jsx",
"declaration": true,
"outDir": "./dist",
"rootDir": "./src",
"experimentalDecorators": true,
"emitDecoratorMetadata": true,
"noEmit": false,
"strict": false,
"moduleResolution": "node",
"esModuleInterop": true,
"resolveJsonModule": true,
"skipLibCheck": true,
"forceConsistentCasingInFileNames": true
},
"include": ["src/"],
"exclude": [
"dist",
"build",
".cache",
"tests",
"**/*.spec.js",
"**/*.spec.ts",
"node_modules",
".eslintrc.js"
]
}
Report Incorrect CodeCopy to Clipboard

The important changes to note here are the inclusion of the field "jsx": "react-jsx"Copy to Clipboard and the addition of "build"Copy to Clipboard and “.cache”Copy to Clipboard to excludeCopy to Clipboard.

The addition of "jsx": "react-jsx"Copy to Clipboard specified how should TypeScript transform JSX, and excluding buildCopy to Clipboard and .cacheCopy to Clipboard ensures that TypeScript ignores build and development files.

Next, create the file tsconfig.server.jsonCopy to Clipboard with the following content:

tsconfig.server.json
{
"extends": "./tsconfig.json",
"compilerOptions": {
/* Emit a single file with source maps instead of having a separate file. */
"inlineSourceMap": true
},
"exclude": ["src/admin", "**/*.spec.js"]
}
Report Incorrect CodeCopy to Clipboard

This is the configuration that will be used to transpile your custom backend code, such as services or entities. The important part is that it excludes src/adminCopy to Clipboard as that is where your Admin code will live.

Finally, create the file tsconfig.admin.jsonCopy to Clipboard with the following content:

tsconfig.admin.json
{
"extends": "./tsconfig.json",
"compilerOptions": {
"module": "esnext"
},
"include": ["src/admin"],
"exclude": ["**/*.spec.js"]
}
Report Incorrect CodeCopy to Clipboard

This is the configuration that will be used when transpiling your admin code.

(Optional) Update Scripts in package.json

You can optionally update the following scripts in package.jsonCopy to Clipboard to make your development process easier:

package.json
{
// ...
"scripts": {
"clean": "cross-env ./node_modules/.bin/rimraf dist",
"build": "cross-env npm run clean && npm run build:server && npm run build:admin",
"build:server": "cross-env npm run clean && tsc -p tsconfig.server.json",
"build:admin": "cross-env medusa-admin build",
"watch": "cross-env tsc --watch",
"test": "cross-env jest",
"seed": "cross-env medusa seed -f ./data/seed.json",
"start": "cross-env npm run build && medusa start",
"start:custom": "cross-env npm run build && node --preserve-symlinks index.js",
"dev": "cross-env npm run build:server && medusa develop"
},
// ...
}
Report Incorrect CodeCopy to Clipboard

If you have autoRebuildCopy to Clipboard enabled in the options of @medusajs/adminCopy to Clipboard, you shouldn’t include npm run build:adminCopy to Clipboard in the buildCopy to Clipboard script. It will lead to the admin being built twice during development.

Create the Admin Widget

To create a new admin widget, start by creating the folder src/admin/widgetsCopy to Clipboard. This is where your widgets must be located, as explained in the Widgets Requirements section.

Then, create the file src/admin/widgets/product-widget.tsxCopy to Clipboard with the following content:

src/admin/widgets/product-widget.tsx
import type { WidgetConfig } from "@medusajs/admin"

const ProductWidget = () => {
return (
<div>
<h1>Product Widget</h1>
</div>
)
}

export const config: WidgetConfig = {
zone: "product.details.after",
}

export default ProductWidget
Report Incorrect CodeCopy to Clipboard

This file creates a React Component ProductWidgetCopy to Clipboard which renders an H1 header. This React Component is the default export in the file, which is one of the Widgets Requirements.

You also export the object configCopy to Clipboard of type WidgetConfigCopy to Clipboard, which is another widget requirement. It indicates that the widget must be injected in the product.details.afterCopy to Clipboard zone.

To test out your widget, run the following command in the root backend directory:

npx @medusajs/medusa-cli@latest develop
Report Incorrect CodeCopy to Clipboard

This command will build your backend and admin, then runs the backend.

Open localhost:7001Copy to Clipboard in your browser and log in. Then, go to the details page of any product. You should now see your widget at the bottom of the page.

Try making any changes to the component. The development server will hot-reload and your widget will be updated immediately.

Styling your Widget

Admin Widgets support Tailwind CSS out of the box.

For example, you can update the widget you created earlier to use Tailwind CSS classes:

src/admin/widgets/product-widget.tsx
import type { 
WidgetConfig,
} from "@medusajs/admin"

const ProductWidget = () => {
return (
<div
className="bg-white p-8 border border-gray-200 rounded-lg">
<h1>Product Widget</h1>
</div>
)
}

export const config: WidgetConfig = {
zone: "product.details.after",
}

export default ProductWidget
Report Incorrect CodeCopy to Clipboard

Widget Props

Every widget receives props of the type WidgetPropsCopy to Clipboard, which includes the notifyCopy to Clipboard prop. The notifyCopy to Clipboard prop is an object that includes the following attributes:

  • successCopy to Clipboard: a function that can be used to show a success message.
  • errorCopy to Clipboard: a function that can be used to show an error message.
  • warnCopy to Clipboard: a function that can be used to show a warning message.
  • infoCopy to Clipboard: a function that can be used to show an info message.

In addition, some injection zones provide additional props specific to the context of the page. For example, widgets in the product.details.afterCopy to Clipboard zone will also receive a productCopy to Clipboard prop, which is an object holding the data of the product being viewed.

You can learn about what additional props each injection zone may receive in the Injection Zone section.

For example, you can modify the widget you created to show the title of the product:

src/admin/widgets/product-widget.tsx
import type { 
WidgetConfig,
ProductDetailsWidgetProps,
} from "@medusajs/admin"

const ProductWidget = ({
product,
notify,
}: ProductDetailsWidgetProps) => {
return (
<div className="bg-white p-8 border border-gray-200 rounded-lg">
<h1>Product Widget {product.title}</h1>
<button
className="bg-black rounded p-1 text-white"
onClick={() => notify.success("success", "You clicked the button!")}
>
Click me
</button>
</div>
)
}

export const config: WidgetConfig = {
zone: "product.details.after",
}

export default ProductWidget
Report Incorrect CodeCopy to Clipboard

Routing Functionalities

If you want to navigate to other pages, link to other pages, or use other routing functionalities, you can use react-router-dom package.

react-router-domCopy to Clipboard is available as one of the @medusajs/adminCopy to Clipboard dependencies. You can also install it within your project using the following command:

yarn add react-router-dom
Report Incorrect CodeCopy to Clipboard

If you're installing it in a plugin with admin customizations, make sure to include it in peerDependenciesCopy to Clipboard.

For example:

src/admin/widgets/product-widget.tsx
import type { WidgetConfig } from "@medusajs/admin"
import { Link } from "react-router-dom"

const ProductWidget = () => {
return (
<div
className="bg-white p-8 border border-gray-200 rounded-lg">
<h1>Product Widget</h1>
<Link to={"/a/orders"}>
View Orders
</Link>
</div>
)
}

export const config: WidgetConfig = {
zone: "product.details.after",
}

export default ProductWidget
Report Incorrect CodeCopy to Clipboard

View react-router-dom’s documentation for other available components and hooks.

Querying and Mutating Data

You will most likely need to interact with the Medusa backend from your Widgets. To do so, you can utilize the Medusa React package. It contains a collection of queries and mutation built on @tanstack/react-queryCopy to Clipboard that lets you interact with the Medusa backend.

Make sure to also install the Medusa React package first if you’re intending to use it, as explained in the Medusa React guide.

For example, you can modify the widget you created to retrieve the tags of a product from the Medusa backend:

src/admin/widgets/product-widget.tsx
import type { ProductDetailsWidgetProps, WidgetConfig } from "@medusajs/admin"
import { useAdminProductTags } from "medusa-react"

const ProductWidget = ({ product }: ProductDetailsWidgetProps) => {
const { product_tags } = useAdminProductTags({
id: product.tags.map((tag) => tag.id),
limit: 10,
offset: 0,
})

return (
<div className="bg-white p-8 border border-gray-200 rounded-lg">
<h3 className="text-lg font-medium mb-4">Product Tags</h3>
<div className="flex flex-wrap">
{product_tags?.map((tag) => (
<span
key={tag.id}
className="bg-gray-100 text-gray-800 px-2 py-1 rounded-full text-xs mr-2 mb-2"
>
{tag.value}
</span>
))}
</div>
</div>
)
}

export const config: WidgetConfig = {
zone: "product.details.after",
}

export default ProductWidget
Report Incorrect CodeCopy to Clipboard

You can also use medusa-reactCopy to Clipboard to interact with custom endpoints using the createCustomAdminHooks utility function.


See Also

Was this page helpful?