This guide covers the creation of a Penpot plugin. Penpot offers two ways to kickstart your development:
1. Using a Template:
- **Typescript template**: Using the <atarget="_blank"href="https://github.com/penpot/penpot-plugin-starter-template">Penpot Plugin Starter Template</a>: A basic template with the required files for quickstarting your plugin. This template uses Typescript and Vite.
- **Framework templates**: These templates already have everything you need to start <atarget="_blank"href="https://github.com/penpot/plugin-examples">developing a plugin using a JavaScript framework. </a>
<pclass="advice">
In case you'll use any of these templates, you can skip to <ahref="#2.7.-step-7.-load-the-plugin-in-penpot">step 2.7</a>
</p>
2. Creating a plugin from scratch using a major framework.
Follow to the next section to understand how to bootstrap a new plugin using one of the three major JavaScript frameworks.
## 2.1. Step 1. Create a project
Create your own app with the framework of your choice. See examples for each framework <atarget="_blank"href="https://github.com/penpot/plugin-examples"> here </a>
There are two libraries that can help you with your plugin's development. They are <codeclass="language-js">@penpot/plugin-styles</code> and <codeclass="language-js">@penpot/plugin-types</code>.
### Plugin styles
<codeclass="language-js">@penpot/plugin-styles</code> contains styles to help build the UI for Penpot plugins. To check the styles go to <atarget="_blank"href="https://penpot-plugins-styles.pages.dev/">Plugin styles</a>.
```bash
npm install @penpot/plugin-styles
```
You can add the styles to your global CSS file.
```css
@import "@penpot/plugin-styles/styles.css";
```
### Plugin types
<codeclass="language-js">@penpot/plugin-types</code> contains the typings for the Penpot Plugin API.
```bash
npm install @penpot/plugin-types
```
If you're using typescript, don't forget to add <codeclass="language-js">@penpot/plugin-types</code> to your typings in your <codeclass="language-js">tsconfig.json</code>.
A plugin file is needed to interact with Penpot and its API. You can use either javascript or typescript and it can be placed wherever you like. It normally goes alongside the main files inside the <codeclass="language-js">src/</code> folder. We highly recommend labeling your creation as <codeclass="language-js">plugin.js</code> or <codeclass="language-js">plugin.ts</code>, depending upon your preferred language.
You can start with something like this:
```ts
penpot.ui.open("Plugin name", "", {
width: 500,
height: 600,
});
```
The sizing values are optional. By default, the plugin will open with a size of 285x540 pixels.
## 2.4. Step 4. Connect API and plugin interface
To enable interaction between your plugin and the Penpot API, you'll need to implement message-based communication using JavaScript events. This communication occurs between the main Penpot application and your plugin, which runs in an iframe. The <codeclass="language-js">window</code> object facilitates this communication by sending and receiving messages between the two.
### Sending messages from Penpot to your plugin
To send a message from the Penpot API to your plugin interface, use the following command in <codeclass="language-js">plugin.ts</code>:
```js
penpot.ui.sendMessage(message);
```
Here, <codeclass="language-js">message</code> can be any data or instruction you want to pass to your plugin. This message is dispatched from Penpot and is received by your plugin's iframe.
### Receiving Messages in Your Plugin Interface
Your plugin can capture incoming messages from Penpot using the <codeclass="language-js">window</code> object's <codeclass="language-js">message</code> event. To do this, set up an event listener in your plugin like this:
```js
window.addEventListener("message", (event) => {
// Handle the incoming message
console.log(event.data);
});
```
The<codeclass="language-js">event.data</code> object contains the message sent from Penpot. You can use this data to update your plugin's interface or trigger specific actions within your plugin.
### Two-Way Communication
This setup allows for two-way communication between Penpot and your plugin. Penpot can send messages to your plugin, and your plugin can respond or send messages back to Penpot using the same<codeclass="language-js">postMessage</code> API. For example:
```js
// Sending a message back to Penpot from your plugin
-<codeclass="language-js">responseMessage</code> is the data you want to send back to Penpot.
-<codeclass="language-js">targetOrigin</code> should be the origin of the Penpot application to ensure messages are only sent to the intended recipient. You can use<codeclass="language-js">'*'</code> to allow all.
### Summary
By using these message-based events, any data retrieved through the Penpot API can be communicated to and from your plugin interface seamlessly.
For more detailed information, refer to the [Penpot Plugins API Documentation](https://penpot-plugins-api-doc.pages.dev/).
## 2.5. Step 5. Build the plugin file
<divclass="advice">
<p>This step is only for local serving.
For a detailed guide about building and deploying you can check the documentation at <atarget="_blank"href="/plugins/deployment/">Deployment</a></p>
<p>You can skip this step if working exclusively with JavaScript by simply moving <codeclass="language-text">plugin.js</code> to your <codeclass="language-text">public/</code> directory.</p>
</div>
If you wish to run your plugin locally and test it live you need to make your plugin file reachable. Right now, your <codeclass="language-js">plugin.ts</code> file is somewhere in the <codeclass="language-js">src/</code> folder, and you can't access it through <codeclass="language-js">http:\/\/localhost:XXXX/plugin.js</code>.
You can achieve this through multiple solutions, but we offer two simple ways of doing so. Of course, you can come up with your own.
#### Vite
If you're using Vite you can simply edit the configuration file and add the build to your <codeclass="language-js">vite.config.ts</code>.
Keep in mind that you'll need to build again your plugin file if you modify it mid-serve.
## 2.6. Step 6. Configure the manifest file
Now that everything is in place you need a <codeclass="language-js">manifest.json</code> file to provide Penpot with your plugin data. Remember to make it reachable by placing it in the <codeclass="language-js">public/</code> folder.
```json
{
"name": "Plugin name",
"description": "Plugin description",
"code": "/plugin.js",
"icon": "/icon.png",
"permissions": [
"content:read",
"content:write",
"library:read",
"library:write",
"user:read",
"comment:read",
"comment:write",
"allow:downloads"
]
}
```
### Icon
The plugin icon must be an image file. All image formats are valid, so you can use whichever format works best for your needs. Although there is no specific size requirement, it is recommended that the icon be 56x56 pixels in order to ensure its optimal appearance across all devices.
### Types of permissions
-<codeclass="language-js">content:read</code>: Allows reading of content-related data. Grants read access to all endpoints and operations dealing with content. Typical use cases: viewing shapes, pages, or other design elements in a project; accessing the properties and settings of content within the application.
-<codeclass="language-js">content:write</code>: Allows writing or modifying content-related data. Grants write access to all endpoints and operations dealing with content modifications, except those marked as read-only. Typical use cases: adding, updating, or deleting shapes and elements in a design; uploading media or other assets to the project.
-<codeclass="language-js">user:read</code>: Allows reading of user-related data. Grants read access to all endpoints and operations dealing with user data. Typical use cases: viewing user profiles and their associated information or listing active users in a particular context or project.
-<codeclass="language-js">library:read</code>: Allows reading of library-related data and assets. Grants read access to all endpoints and operations dealing with the library context. Typical use cases: accessing shared design elements and components from a library or viewing the details and properties of library assets.
-<codeclass="language-js">library:write</code>: Allows writing or modifying library-related data and assets. Grants write access to all endpoints and operations dealing with library modifications. Typical use cases: adding new components or assets to the library or updating or removing existing library elements.
-<codeclass="language-js">comment:read</code>: Allows reading of comment-related data. Grants read access to all endpoints and operations dealing with comments.
Typical use cases: viewing comments on pages; accessing feedback or annotations provided by collaborators in the project.
-<codeclass="language-js">comment:write</code>: Allows writing or modifying comment-related data. Grants write access to all endpoints and operations dealing with creating, replying, or deleting comments.
Typical use cases: adding new comments to pages; deleting existing comments; replying to comments within the project's context.
-<codeclass="language-js">allow:downloads</code>: Allows downloading of the project file. Grants access to endpoints and operations that enable the downloading of the entire project file.
Typical use cases: downloading the full project file for backup or sharing.
_Note: Write permissions automatically includes its corresponding read permission (e.g.,<codeclass="language-js">content:write</code> includes <codeclass="language-js">content:read</code>) because reading is required to perform write or modification actions._
## 2.7. Step 7. Load the Plugin in Penpot
<pclass="advice"><b>Serving an application:</b> This refers to making your application accessible over a network, typically for testing or development purposes. <br><br>When using a tool like <ahref="https://www.npmjs.com/package/live-server"target="_blank">live-server</a>, a local web server is created on your machine, which serves your application files over HTTP. Most modern frameworks offer their own methods for serving applications, and there are build tools like Vite and Webpack that can handle this process as well. </p>
**You don't need to deploy your plugin just to test it**. Locally serving your plugin is compatible with <codeclass="language-js">https:\/\/penpot.app/</code>. However, be mindful of potential CORS (Cross-Origin Resource Sharing) issues. To avoid these, ensure your plugin includes the appropriate cross-origin headers. (Find more info about this at the <atarget="_blank"href="/plugins/deployment/">Deployment step</a>)
Serving your plugin will generate a URL that looks something like <codeclass="language-js">http:\/\/localhost:XXXX</code>, where <codeclass="language-js">XXXX</code> represents the port number on which the plugin is served. Ensure that both <codeclass="language-js">http:\/\/localhost:XXXX/manifest.json</code> and <codeclass="language-js">http:\/\/localhost:XXXX/plugin.js</code> are accessible. If these files are inside a specific folder, the URL should be adjusted accordingly (e.g., <codeclass="language-js">http:\/\/localhost:XXXX/folder/manifest.json</code>).
Once your plugin is served you are ready to load it into Penpot. You can use the shortcut <codeclass="language-js">Ctrl + Alt + P</code> to open the Plugin Manager modal. In this modal, provide the URL to your plugin's manifest file (e.g., <codeclass="language-js">http:\/\/localhost:XXXX/manifest.json</code>) for installation. If everything is set up correctly, the plugin will be installed, and you can launch it whenever needed.
You can also open the Plugin manager modal via:
- Menu
<figure>
<videotitle="Open plugin manager from penpot menu"muted=""playsinline=""controls=""width="100%"poster="/img/plugins/plugins-menu.png"height="auto">