
Product
Introducing Socket Firewall Enterprise: Flexible, Configurable Protection for Modern Package Ecosystems
Socket Firewall Enterprise is now available with flexible deployment, configurable policies, and expanded language support.
@trycourier/react-editor
Advanced tools
The Courier Template editor is a package designed to facilitate the editing of templates in a multi-tenant environment. In platforms like Courier, where the application serves different users or clients (tenants) with unique configurations, this package makes it easy to edit and manage templates associated with specific tenants directly within your React application.
Embedded React Integration
Easily integrate the template editor directly into your existing React application.
Multi-Tenant Support
Seamlessly manage and edit templates for different tenants, each with their own branding.
Brand Editing
Customize branding elements such as logos, colors, and styles per tenant to ensure consistency with each client's identity.
Developer Friendly
Designed with intuitive APIs and flexible components to speed up development and integration.
npm install @trycourier/react-editor
or
yarn add @trycourier/react-editor
To use the Courier Editor, you'll need to authenticate your requests using a JWT token. For development environments, you can quickly get started with a client key, while JWT authentication is recommended for production deployments.
Scopes
The follow scopes are available for the editor:
tenants:read: allow the user to read all tenant datatenants:brand:read: allow the user to read brand data for all tenantstenants:notifications:read: allow the user to read all notification data for all tenantstenants:notifications:write: allow the user to write all notification data for all tenantstenant:${TENANT_ID}:read : allow the user to read tenant data for a specific tenanttenant:${TENANT_ID}:notification:read : allow the user to write notification data for a specific tenanttenant:${TENANT_ID}:notification:write : allow the user to write notification data for a specific tenanttenant:${TENANT_ID}:brand:read : allow the user to read brand data for a specific tenanttenant:${TENANT_ID}:brand:write: allow the user to write brand data for a specific tenantHere is a curl example with all the scopes needed that the SDK uses. Change the scopes to the scopes you need for your use case.
Example
This example will generate a JWT that has access to all tenants and tenant notifications
curl --request POST \
     --url https://api.courier.com/auth/issue-token \
     --header 'Accept: application/json' \
     --header 'Authorization: Bearer $YOUR_AUTH_KEY' \
     --header 'Content-Type: application/json' \
     --data \
 '{
    "scope": "user_id:$YOUR_USER_ID tenants:read tenants:notifications:read tenants:notifications:write",
    "expires_in": "$YOUR_NUMBER days"
  }'
Example
This example will generate a JWT that only has access to a specific tenant and it's notifications
curl --request POST \
     --url https://api.courier.com/auth/issue-token \
     --header 'Accept: application/json' \
     --header 'Authorization: Bearer $YOUR_AUTH_KEY' \
     --header 'Content-Type: application/json' \
     --data \
 '{
    "scope": "user_id:$YOUR_USER_ID tenant:tenant-123:read tenant:tenant-123:notifications:read tenants:tenant-123:notifications:write",
    "expires_in": "$YOUR_NUMBER days"
  }'
Example
This example will generate a JWT that gives a user access to change brand information for a specific tenant
curl --request POST \
     --url https://api.courier.com/auth/issue-token \
     --header 'Accept: application/json' \
     --header 'Authorization: Bearer $YOUR_AUTH_KEY' \
     --header 'Content-Type: application/json' \
     --data \
 '{
    "scope": "user_id:$YOUR_USER_ID tenant:tenant-123:read tenant:tenant-123:brand:read tenant:tenant-123:brand:write",
    "expires_in": "$YOUR_NUMBER days"
  }'
This basic setup provides a fully functional editor with default settings. The TemplateProvider component handles state management and provides necessary context to the Editor.
As template changes are made, they are automatically saved to Courier.
import { TemplateEditor, TemplateProvider } from '@trycourier/react-editor';
function App() {
  return (
    <TemplateProvider templateId="template-123" tenantId="tenant-123" token="jwt">
      <TemplateEditor />
    </TemplateProvider>
  );
}
By default, the Courier template editor exposes a publish button that the template editor can interact with after completing changes. To override this default publishing behavior, you can hide the publishing button and interact with the publishing action directly allowing you to tightly integrate with your application's workflow.
Note: useTemplateActions must be used inside of the <TemplateProvider /> context
import { TemplateEditor, TemplateProvider, useTemplateActions } from '@trycourier/react-editor';
function SaveButtonComponent() {
  const { publishTemplate } = useTemplateActions();
  
  const handlePublishTemplate = () => {
	  //... other publish logic
	  await publishTemplate();
  }
	return (
		<TemplateProvider templateId="template-123" tenantId="tenant-123" token="jwt">
      <TemplateEditor hidePublish />
			<button onClick={handlePublishTemplate}>Save Template</button>;
    </TemplateProvider>
  );
}
You can customize the look and feel through the theming API, which allows you to modify colors, and other visual elements via configuration.
import { Editor, TemplateProvider } from '@trycourier/editor';
function App() {
  return (
    <Editor
      theme={{
        background: '#ffffff',
        foreground: '#292929',
        muted: '#D9D9D9',
        mutedForeground: '#A3A3A3',
        popover: '#ffffff',
        popoverForeground: '#292929',
        border: '#DCDEE4',
        input: '#DCDEE4',
        card: '#FAF9F8',
        cardForeground: '#292929',
        primary: '#ffffff',
        primaryForeground: '#696F8C',
        secondary: '#F5F5F5',
        secondaryForeground: '#171717',
        accent: '#E5F3FF',
        accentForeground: '#1D4ED8',
        destructive: '#292929',
        destructiveForeground: '#FF3363',
        ring: '#80849D',
        radius: '6px',
      }}
    />
  );
}
By default, the Courier Editor auto-saves content. To disable this feature, configure the provider as follows
import { TemplateEditor, TemplateProvider, useTemplateActions } from '@trycourier/react-editor';
function SaveButtonComponent() {
  const { saveTemplate, publishTemplate } = useTemplateActions();
  
  const handleSaveTemplate = () => {
	  //... other publish logic
	  await saveTemplate(); // the template must be saved before publishing
	  await publishTemplate();
  }
	return (
		<TemplateProvider templateId="template-123" tenantId="tenant-123" token="jwt">
      <TemplateEditor autoSave={false} hidePublish />
			<button onClick={handleSaveTemplate}>Save Template</button>;
    </TemplateProvider>
  );
}
The Brand Editor component allows you to customize and manage a tenant's brand settings directly within your application. This specialized editor provides an interface for modifying brand colors, logos, and other visual elements that will be applied to your templates.
Similar to the basic Template Editor setup, implementing the Brand Editor requires minimal configuration. The BrandProvider component handles state management and provides the necessary context to the Brand Editor component.
import { BrandEditor, BrandProvider } from '@trycourier/editor';
function App() {
  return (
    <BrandProvider tenantId="tenant-123" token="jwt">
      <BrandEditor />
    </BrandProvider>
  );
}
Similar to the Template Editor, the Brand Editor also provides a publishing hook that allows you to customize the publishing behavior. You can use the useBrandActions hook to programmatically trigger brand updates and integrate them with your application's workflow.
Note: useBrandActions must be used inside of the <BrandProvider /> context
import { BrandEditor, BrandProvider, useBrandActions } from '@trycourier/editor';
function SaveBrandComponent() {
  const { publishBrand } = useBrandActions();
  
  const handlePublishBrand = () => {
	  //... other publish logic
	  await publishBrand();
  }
  return (
    <BrandProvider tenantId="tenant-123" token="jwt | client key">
      <BrandEditor hidePublish />
      <button onClick={handlePublishBrand}>Save Brand</button>
    </BrandProvider>
  );
}
The Properties section details the configuration options available for both the Editor and TemplateProvider components. These properties allow you to customize the behavior and functionality of the editor to match your specific requirements.
The Editor component is the core element that provides the template editing interface. If you are using the Template Editor with the Template provider, required properties will be provided.
| Property | Type | Default | Description | 
|---|---|---|---|
| autoSave | boolean | true | Enables automatic saving of changes to the template. When true, changes are automatically persisted. | 
| autoSaveDebounce | number | 200ms | Time in milliseconds to wait after changes before triggering an auto-save operation. Controls save frequency. | 
| brandEditor | boolean | false | When enabled, shows the brand editor interface alongside the template editor. Allows editing brand settings. | 
| brandProps | BrandEditorProps | Configuration options for the brand editor when enabled. Passed directly to the BrandEditor component. | |
| hidePublish | boolean | false | When true, hides the "Publish Changes" button in the editor interface. | 
| onChange | (value: ElementalContent) => void | Callback function that fires whenever the editor content changes, providing the updated ElementalContent structure. | |
| theme | ThemeObj | cssClass | |
| value | ElementalContent | Initial content for the editor in ElementalContent format. Used as the starting template when the editor loads. | |
| variables | Record<string, any | Custom variables available for template personalization. These can be referenced within the template content. | 
The TemplateProvider component wraps the Editor component and manages the template state, authentication, and data flow. It provides essential context and functionality needed for the editor to operate. Below are the key properties that can be configured when using the
| Property | Type | Required | Description | 
|---|---|---|---|
| templateId | string | Yes | Unique identifier for the template being edited. Required for loading and saving template data. | 
| tenantId | string | Yes | The tenant ID associated with the template. Used for brand theming and tenant-specific functionality. | 
| token | string | Yes | Authentication token (JWT or ClientKey) used to authorize API requests to Courier services. | 
The Brand Editor component accepts properties that allow you to customize its behavior and appearance. These properties provide control over the editing interface and functionality specific to brand management.
| Property | Type | Default | Description | 
|---|---|---|---|
| autoSave | boolean | true | Enables automatic saving of changes to the template. When true, changes are automatically persisted. | 
| autoSaveDebounce | number | 200ms | Time in milliseconds to wait after changes before triggering an auto-save operation. Controls save frequency. | 
| hidePublish | boolean | false | When true, hides the "Publish Changes" button in the editor interface. | 
| onChange | (value: BrandSettings) => void | Callback function that fires whenever brand settings are modified, providing the updated brand configuration values. | |
| theme | ThemeObj | cssClass | |
| value | BrandSettings | Initial brand settings values to populate the editor with, including colors, logo, social links, and header style preferences. | |
| variables | Record<string, any | Custom variables available for brand personalization. These can be referenced within the brand footer content. | 
The Brand Provider component is responsible for managing brand-related state and context in your application. It wraps the Brand Editor component and handles data flow, authentication, and state management for brand customization. Here are the key properties that can be configured:
| Property | Type | Required | Description | 
|---|---|---|---|
| tenantId | string | Yes | The unique identifier for the tenant whose brand settings are being edited. | 
| token | string | Yes | Authentication token (JWT or ClientKey) used to authorize brand-related API requests. | 
React Only
The Courier Editor is currently only compatible with React applications. This means you'll need to have a React-based frontend to integrate the editor into your application. Support for other frameworks like Vue.js or Angular may be considered in future releases based on community demand.
Minimal supported React version is 18.2.0.
Email channel support only
Currently, the editor only supports creating and editing email templates. Support for additional channels like SMS, push notifications, and direct messages is currently under development and will be added in the next month or so.
pnpm dev - Starts the development environmentpnpm build - Builds the packagepnpm test - Runs testspnpm lint - Runs lintingThis package can be published to npm using the provided scripts. The publishing process includes version bumping, building, testing, and pushing changes to git.
pnpm release - Bumps patch version and publishespnpm release:patch - Same as abovepnpm release:minor - Bumps minor version and publishespnpm release:major - Bumps major version and publishespnpm release:dry-run - Simulates the release process without publishingpnpm release:canary - Creates a canary release with a unique tagThe release script will:
Canary builds are pre-release versions that allow users to test the latest changes without affecting the stable release. Each canary build is published with a unique npm tag based on the timestamp when it was created.
Run the canary release command:
pnpm release:canary
This will:
canary-{timestamp}0.0.1-canary-1718392847To see all available canary builds:
npm dist-tag ls @trycourier/react-editor
To install a specific canary build:
# Example for a specific canary build
npm install @trycourier/react-editor@canary-1718392847
If you need more control over the publishing process:
pnpm buildpnpm testpnpm version patch|minor|majorpnpm publish --access publicNote: You need to have npm publishing rights for the @trycourier organization to publish this package.
FAQs
React-based rich text editor component
We found that @trycourier/react-editor demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 4 open source maintainers collaborating on the project.
Did you know?

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Product
Socket Firewall Enterprise is now available with flexible deployment, configurable policies, and expanded language support.

Security News
Open source dashboard CNAPulse tracks CVE Numbering Authorities’ publishing activity, highlighting trends and transparency across the CVE ecosystem.

Product
Detect malware, unsafe data flows, and license issues in GitHub Actions with Socket’s new workflow scanning support.