0
Fork 0
mirror of https://github.com/penpot/penpot-plugins.git synced 2025-01-21 06:02:34 -05:00

feat: plugin libraries

This commit is contained in:
Juanfran 2024-03-27 13:57:02 +01:00
parent 03c055eac6
commit 070bb62328
41 changed files with 2508 additions and 298 deletions

35
.eslintrc.base.json Normal file
View file

@ -0,0 +1,35 @@
{
"root": true,
"ignorePatterns": ["**/*"],
"plugins": ["@nx"],
"overrides": [
{
"files": ["*.ts", "*.tsx", "*.js", "*.jsx"],
"rules": {
"@nx/enforce-module-boundaries": [
"error",
{
"enforceBuildableLibDependency": true,
"allow": [],
"depConstraints": [
{
"sourceTag": "*",
"onlyDependOnLibsWithTags": ["*"]
}
]
}
]
}
},
{
"files": ["*.ts", "*.tsx"],
"extends": ["plugin:@nx/typescript"],
"rules": {}
},
{
"files": ["*.js", "*.jsx"],
"extends": ["plugin:@nx/javascript"],
"rules": {}
}
]
}

View file

@ -1,7 +1,5 @@
{ {
"root": true,
"ignorePatterns": ["**/*"], "ignorePatterns": ["**/*"],
"plugins": ["@nx"],
"overrides": [ "overrides": [
{ {
"files": ["*.ts", "*.tsx", "*.js", "*.jsx"], "files": ["*.ts", "*.tsx", "*.js", "*.jsx"],
@ -14,15 +12,27 @@
"depConstraints": [ "depConstraints": [
{ {
"sourceTag": "type:plugin", "sourceTag": "type:plugin",
"onlyDependOnLibsWithTags": ["type:util", "type:ui", "type:feature"] "onlyDependOnLibsWithTags": [
"type:util",
"type:ui",
"type:feature"
]
}, },
{ {
"sourceTag": "type:app", "sourceTag": "type:app",
"onlyDependOnLibsWithTags": [ "type:util", "type:ui", "type:feature"] "onlyDependOnLibsWithTags": [
"type:util",
"type:ui",
"type:feature"
]
}, },
{ {
"sourceTag": "type:feature", "sourceTag": "type:feature",
"onlyDependOnLibsWithTags": ["type:feature", "type:ui", "type:util"] "onlyDependOnLibsWithTags": [
"type:feature",
"type:ui",
"type:util"
]
}, },
{ {
"sourceTag": "type:ui", "sourceTag": "type:ui",
@ -42,7 +52,6 @@
"extends": [ "extends": [
"plugin:@typescript-eslint/recommended", "plugin:@typescript-eslint/recommended",
"plugin:@typescript-eslint/recommended-requiring-type-checking", "plugin:@typescript-eslint/recommended-requiring-type-checking",
"plugin:@nx/typescript",
"plugin:deprecation/recommended", "plugin:deprecation/recommended",
"prettier" "prettier"
], ],
@ -68,12 +77,10 @@
"extends": [ "extends": [
"plugin:@typescript-eslint/recommended", "plugin:@typescript-eslint/recommended",
"plugin:@typescript-eslint/recommended-requiring-type-checking", "plugin:@typescript-eslint/recommended-requiring-type-checking",
"plugin:@nx/typescript",
"plugin:deprecation/recommended", "plugin:deprecation/recommended",
"prettier" "prettier"
], ],
"rules": { "rules": {
"@typescript-eslint/no-unused-vars": ["error"], "@typescript-eslint/no-unused-vars": ["error"],
"no-multiple-empty-lines": [ "no-multiple-empty-lines": [
2, 2,
@ -99,8 +106,8 @@
}, },
{ {
"files": ["*.js", "*.jsx"], "files": ["*.js", "*.jsx"],
"extends": ["plugin:@nx/javascript"],
"rules": {} "rules": {}
} }
] ],
"extends": ["./.eslintrc.base.json"]
} }

25
.verdaccio/config.yml Normal file
View file

@ -0,0 +1,25 @@
# a list of other known repositories we can talk to
uplinks:
npmjs:
url: https://registry.npmjs.org/
maxage: 60m
packages:
'**':
# give all users (including non-authenticated users) full access
# because it is a local registry
access: $all
publish: $all
unpublish: $all
# if package is not available locally, proxy requests to npm registry
proxy: npmjs
# log settings
logs:
type: stdout
format: pretty
level: warn
publish:
allow_offline: true # set offline to true to allow publish offline

View file

@ -1,5 +1,5 @@
{ {
"extends": ["../../.eslintrc.json"], "extends": ["../../.eslintrc.base.json", "../../.eslintrc.json"],
"ignorePatterns": ["!**/*", "vite.config.ts"], "ignorePatterns": ["!**/*", "vite.config.ts"],
"parser": "@typescript-eslint/parser", "parser": "@typescript-eslint/parser",
"parserOptions": { "parserOptions": {

View file

@ -5,5 +5,5 @@
"types": ["node"] "types": ["node"]
}, },
"exclude": ["src/**/*.spec.ts", "src/**/*.test.ts"], "exclude": ["src/**/*.spec.ts", "src/**/*.test.ts"],
"include": ["src/**/*.ts", "../../libs/plugins-runtime/src/lib/index.d.ts"] "include": ["src/**/*.ts", "../../libs/plugin-types/index.d.ts"]
} }

View file

@ -1,5 +1,5 @@
{ {
"extends": ["../../.eslintrc.json"], "extends": ["../../.eslintrc.base.json", "../../.eslintrc.json"],
"ignorePatterns": ["!**/*", "vite.config.ts"], "ignorePatterns": ["!**/*", "vite.config.ts"],
"parser": "@typescript-eslint/parser", "parser": "@typescript-eslint/parser",
"parserOptions": { "parserOptions": {

View file

@ -5,5 +5,5 @@
"types": ["node"] "types": ["node"]
}, },
"exclude": ["src/**/*.spec.ts", "src/**/*.test.ts"], "exclude": ["src/**/*.spec.ts", "src/**/*.test.ts"],
"include": ["src/**/*.ts", "../../libs/plugins-runtime/src/lib/index.d.ts"] "include": ["src/**/*.ts", "../../libs/plugin-types/index.d.ts"]
} }

View file

@ -1,5 +1,5 @@
{ {
"extends": ["../../.eslintrc.json"], "extends": ["../../.eslintrc.base.json", "../../.eslintrc.json"],
"ignorePatterns": ["!**/*", "vite.config.ts"], "ignorePatterns": ["!**/*", "vite.config.ts"],
"parser": "@typescript-eslint/parser", "parser": "@typescript-eslint/parser",
"parserOptions": { "parserOptions": {

View file

@ -1,5 +1,5 @@
{ {
"extends": ["../../.eslintrc.json"], "extends": ["../../.eslintrc.base.json", "../../.eslintrc.json"],
"ignorePatterns": ["!**/*"], "ignorePatterns": ["!**/*"],
"parser": "@typescript-eslint/parser", "parser": "@typescript-eslint/parser",
"parserOptions": { "parserOptions": {

View file

@ -1,24 +1,32 @@
# Create plugin # Creating a Plugin
To create the basic scaffolding run the following command. Remember to replace `example-plugin` with your own. This guide walks you through the steps to create a plugin for our platform. You'll start by setting up the basic structure, configuring necessary files, and then running a local server to preview your plugin. Let's dive in.
``` ### Step 1: Initialize the Plugin
First, you need to create the scaffolding for your plugin. Use the following command, replacing `example-plugin` with the name of your plugin:
```sh
npx nx g @nx/web:application example-plugin --directory=apps/example-plugin npx nx g @nx/web:application example-plugin --directory=apps/example-plugin
``` ```
Create a `manifest.json` in `/public`. ### Step 2: Configure the Manifest
Next, create a `manifest.json` file inside the `/public` directory. This file is crucial as it defines key properties of your plugin, including permissions and the entry point script.
```json ```json
{ {
"name": "Example plugin", "name": "Example Plugin",
"code": "http://localhost:4201/plugin.js", "code": "http://localhost:4201/plugin.js",
"permissions": ["page:read", "file:read", "selection:read"] "permissions": ["page:read", "file:read", "selection:read"]
} }
``` ```
Add to the example `vite.config.ts` ### Step 3: Update Vite Configuration
```json Now, add the following configuration to your `vite.config.ts` to specify the entry points for the build process:
```typescript
build: { build: {
rollupOptions: { rollupOptions: {
input: { input: {
@ -32,24 +40,39 @@ build: {
} }
``` ```
Add to `tsconfig.app.json` ### Step 4: Modify TypeScript Configuration
Update your `tsconfig.app.json` to include the necessary TypeScript files for your plugin:
```json ```json
"include": ["src/**/*.ts", "../../libs/plugins-runtime/src/lib/index.d.ts"] {
"include": ["src/**/*.ts", "../../libs/plugin-types/index.d.ts"]
}
``` ```
Then, run the static server ### Step 5: Run a Static Server
``` To preview your plugin, start a static server by running:
```sh
npx nx run example-plugin:build --watch & npx nx run example-plugin:preview npx nx run example-plugin:build --watch & npx nx run example-plugin:preview
``` ```
Finally, go to penpot and load the plugin. Run the command in the console devtools from your browser. ### Step 6: Load the Plugin in Penpot
```ts Finally, to load your plugin into Penpot, execute the following command in the browser's console devtools:
```typescript
ɵloadPlugin({ manifest: 'http://localhost:4201/manifest.json' }); ɵloadPlugin({ manifest: 'http://localhost:4201/manifest.json' });
``` ```
### More about plugin development ### Learn More About Plugin Development
Check the [plugin usage](docs/plugin-usage.md) and the [create API](docs/create-api.md) documentation. For more detailed information on plugin development, check out our guides:
- [Plugin Usage Documentation](docs/plugin-usage.md)
- [Create API Documentation](docs/create-api.md)
### Using a Starter Template
If you prefer to kickstart your plugin development, consider using the [Penpot Plugin Starter Template](https://github.com/penpot/penpot-plugin-starter-template). It's a template designed to streamline the creation process for Penpot plugins.

85
docs/publish-package.md Normal file
View file

@ -0,0 +1,85 @@
# Publishing Packages
## Introduction
This guide details the process of publishing `plugin-types` and `plugins-styles` packages, which are essential for plugin development. To facilitate testing and distribution, we leverage npm for publishing and Verdaccio for setting up a local registry. Below is a walkthrough for publishing these packages, setting up a local registry, and managing releases.
## Setting Up a Local Registry with Verdaccio
Setting up a local registry is for testing plugins in isolation from the monorepo. We utilize Verdaccio, a npm proxy registry, for this purpose.
**Launch the Registry**: Initiate the Verdaccio registry by executing the command:
```shell
npm run registry
```
## Publishing Libraries
Publishing packages allows you to distribute your libraries to other developers and environments. Follow the steps below for both automated and manual publishing processes.
### Automated Publishing:
To publish the libraries automatically, use the command:
```shell
npm run publish -- --version 0.1.0 --tag 0.1.0 --registry http://localhost:4873
```
### Manual Publishing:
For manual publication, navigate to the library directory and execute:
```shell
npm publish --registry http://localhost:4873
```
### Independent Publishing:
To publish libraries independently, specify the package name along with version and tag:
```shell
npx nx publish plugin-types -- --version 0.1.0 --tag 0.1.0
npx nx publish plugins-styles -- --version 0.1.0 --tag 0.1.0
```
### Installing Libraries:
When installing the library, ensure to specify the registry:
```shell
npm i --registry http://localhost:4873
```
**Note**: For direct npm publication, omit the `--registry` flag.
## Managing Releases
### Generating a Release:
For regular releases, execute:
```shell
npx nx release
```
For the initial release, use the `--first-release` flag:
```shell
npx nx release --first-release
```
Refer to the [Nx Release Documentation](https://nx.dev/recipes/nx-release/publish-in-ci-cd) for detailed information.
## Important Reminders
Ensure to update the [penpot-plugin-starter-template](https://github.com/penpot/penpot-plugin-starter-template) with every release to provide developers with the latest configuration and features.
## Relevant Files and Scripts
- **Verdaccio Configuration**: `./project.json`
- **CSS Build Script**: `./tools/scripts/build-css.mjs`
- **Types Build Script**: `./tools/scripts/build-types.mjs`
- **Publish Script**: `./tools/scripts/publish.mjs`
This improved documentation aims to streamline the package publishing process, making it more accessible and understandable for developers.

View file

@ -0,0 +1,25 @@
{
"extends": ["../../.eslintrc.base.json"],
"ignorePatterns": ["!**/*"],
"overrides": [
{
"files": ["*.ts", "*.tsx", "*.js", "*.jsx"],
"rules": {}
},
{
"files": ["*.ts", "*.tsx"],
"rules": {}
},
{
"files": ["*.js", "*.jsx"],
"rules": {}
},
{
"files": ["*.json"],
"parser": "jsonc-eslint-parser",
"rules": {
"@nx/dependency-checks": "error"
}
}
]
}

View file

@ -0,0 +1,21 @@
# Penpot plugin-types
This repository contains the typings for the Penpot Plugin API.
### Getting started
Install the package:
```bash
npm install @penpot/plugin-types
```
Configure `tsconfig.json`
```json
"typeRoots": [
"./node_modules/@types",
"./node_modules/@penpot"
],
"types": ["plugin-types"],
```

View file

@ -1,26 +1,24 @@
/* eslint-disable @typescript-eslint/no-explicit-any */ export interface PageState {
interface Page {
name: string; name: string;
id: string; id: string;
} }
interface File { export interface FileState {
name: string; name: string;
id: string; id: string;
revn: number; revn: number;
} }
interface EventsMap { export interface EventsMap {
pagechange: Page; pagechange: PageState;
filechange: File; filechange: FileState;
selectionchange: string[]; selectionchange: string[];
themechange: Theme; themechange: Theme;
} }
type Theme = 'light' | 'dark'; export type Theme = 'light' | 'dark';
interface Penpot { export interface Penpot {
ui: { ui: {
open: ( open: (
name: string, name: string,
@ -30,7 +28,7 @@ interface Penpot {
sendMessage: (message: unknown) => void; sendMessage: (message: unknown) => void;
onMessage: <T>(callback: (message: T) => void) => void; onMessage: <T>(callback: (message: T) => void) => void;
}; };
log: (...data: any[]) => void; log: (...data: unknown[]) => void;
setTimeout: (callback: () => void, time: number) => void; setTimeout: (callback: () => void, time: number) => void;
closePlugin: () => void; closePlugin: () => void;
on: <T extends keyof EventsMap>( on: <T extends keyof EventsMap>(
@ -41,13 +39,13 @@ interface Penpot {
type: T, type: T,
callback: (event: EventsMap[T]) => void callback: (event: EventsMap[T]) => void
) => void; ) => void;
getFileState: () => File | null; getFileState: () => FileState | null;
getPageState: () => Page | null; getPageState: () => PageState | null;
getSelection: () => string[]; getSelection: () => string[];
getTheme: () => Theme; getTheme: () => Theme;
fetch: typeof fetch; fetch: typeof fetch;
} }
declare namespace globalThis { declare global {
const penpot: Penpot; const penpot: Penpot;
} }

View file

@ -0,0 +1,5 @@
{
"name": "@penpot/plugin-types",
"version": "0.1.0",
"typings": "./index.d.ts"
}

View file

@ -0,0 +1,19 @@
{
"name": "plugin-types",
"$schema": "../../node_modules/nx/schemas/project-schema.json",
"sourceRoot": "libs/plugin-types",
"projectType": "library",
"targets": {
"build": {
"command": "node tools/scripts/build-types.mjs",
"options": {
"outputPath": "dist/plugin-types"
}
},
"publish": {
"command": "node tools/scripts/publish.mjs plugin-types {args.ver} {args.tag}",
"dependsOn": ["build"]
}
},
"tags": ["type:ui"]
}

View file

@ -0,0 +1,19 @@
{
"compilerOptions": {
"module": "node16",
"lib": [
"es6",
"dom"
],
"noImplicitAny": true,
"noImplicitThis": true,
"strictNullChecks": true,
"strictFunctionTypes": true,
"types": [],
"noEmit": true,
"forceConsistentCasingInFileNames": true
},
"files": [
"index.d.ts"
]
}

View file

@ -1,5 +1,5 @@
{ {
"extends": ["../../.eslintrc.json"], "extends": ["../../.eslintrc.base.json", "../../.eslintrc.json"],
"ignorePatterns": ["!**/*", "vite.config.ts"], "ignorePatterns": ["!**/*", "vite.config.ts"],
"parser": "@typescript-eslint/parser", "parser": "@typescript-eslint/parser",
"parserOptions": { "parserOptions": {

View file

@ -1,6 +1,6 @@
{ {
"name": "plugins-data-parser", "name": "plugins-data-parser",
"version": "0.0.1", "version": "0.1.0",
"dependencies": {}, "dependencies": {},
"type": "commonjs", "type": "commonjs",
"main": "./src/index.js", "main": "./src/index.js",

View file

@ -1,5 +1,5 @@
{ {
"extends": ["../../.eslintrc.json"], "extends": ["../../.eslintrc.base.json", "../../.eslintrc.json"],
"ignorePatterns": ["!**/*"], "ignorePatterns": ["!**/*"],
"parser": "@typescript-eslint/parser", "parser": "@typescript-eslint/parser",
"parserOptions": { "parserOptions": {

View file

@ -1,11 +1,12 @@
{ {
"name": "plugins-runtime", "name": "plugins-runtime",
"version": "0.0.1", "version": "0.1.0",
"devDependencies": { "devDependencies": {
"happy-dom": "^13.6.2" "happy-dom": "^13.6.2"
}, },
"dependencies": { "dependencies": {
"plugins-data-parser": "^0.0.1", "@penpot/plugin-types": "^0.1.0",
"plugins-data-parser": "^0.1.0",
"vitest": "1.2.2", "vitest": "1.2.2",
"ses": "^1.1.0", "ses": "^1.1.0",
"zod": "^3.22.4" "zod": "^3.22.4"

View file

@ -1,6 +1,13 @@
import { setModalTheme } from '../create-modal'; import { setModalTheme } from '../create-modal';
import { Manifest, Permissions } from '../models/manifest.model'; import { Manifest, Permissions } from '../models/manifest.model';
import { OpenUIOptions } from '../models/open-ui-options.model'; import { OpenUIOptions } from '../models/open-ui-options.model';
import type {
Penpot,
EventsMap,
FileState,
PageState,
Theme,
} from '@penpot/plugin-types';
import openUIApi from './openUI.api'; import openUIApi from './openUI.api';
import z from 'zod'; import z from 'zod';
@ -17,8 +24,8 @@ export let uiMessagesCallbacks: Callback<unknown>[] = [];
let modal: HTMLElement | null = null; let modal: HTMLElement | null = null;
let pageState: Page | null = null; let pageState: PageState | null = null;
let fileState: File | null = null; let fileState: FileState | null = null;
let selection: string[] = []; let selection: string[] = [];
let themeState: Theme = 'dark'; let themeState: Theme = 'dark';
@ -38,13 +45,13 @@ export function triggerEvent(
listeners.forEach((listener) => listener(message)); listeners.forEach((listener) => listener(message));
} }
export function setPageState(page: Page) { export function setPageState(page: PageState) {
pageState = page; pageState = page;
triggerEvent('pagechange', page); triggerEvent('pagechange', page);
} }
export function setFileState(file: File) { export function setFileState(file: FileState) {
fileState = file; fileState = file;
triggerEvent('filechange', file); triggerEvent('filechange', file);

View file

@ -9,6 +9,7 @@ import {
uiMessagesCallbacks, uiMessagesCallbacks,
} from './index.js'; } from './index.js';
import openUIApi from './openUI.api.js'; import openUIApi from './openUI.api.js';
import { FileState } from '@penpot/plugin-types';
vi.mock('./openUI.api', () => { vi.mock('./openUI.api', () => {
return { return {
@ -220,7 +221,7 @@ describe('Plugin api', () => {
name: 'test', name: 'test',
id: '123', id: '123',
revn: 0, revn: 0,
} as File; } as FileState;
setFileState(exampleFile); setFileState(exampleFile);

View file

@ -1,5 +1,7 @@
import { OpenUIOptions } from './models/open-ui-options.model'; import { OpenUIOptions } from './models/open-ui-options.model';
import type { Theme } from '@penpot/plugin-types';
export function setModalTheme(modal: HTMLElement, theme: Theme) { export function setModalTheme(modal: HTMLElement, theme: Theme) {
modal.setAttribute('data-theme', theme); modal.setAttribute('data-theme', theme);
} }

View file

@ -1,5 +1,5 @@
{ {
"extends": ["../../.eslintrc.json"], "extends": ["../../.eslintrc.base.json", "../../.eslintrc.json"],
"ignorePatterns": ["!**/*", "vite.config.ts"], "ignorePatterns": ["!**/*", "vite.config.ts"],
"parser": "@typescript-eslint/parser", "parser": "@typescript-eslint/parser",
"parserOptions": { "parserOptions": {

View file

@ -1,11 +1,15 @@
# plugins-styles # Penpot plugin-styles
This library was generated with [Nx](https://nx.dev). This plugin contains a CSS file to help build the UI for Penpot plugins.
## Building ### Getting started
Run `nx build plugins-styles` to build the library. Install the package:
## Running unit tests ```bash
npm install @penpot/plugin-styles
```
Run `nx test plugins-styles` to execute the unit tests via [Jest](https://jestjs.io). ```css
@import '@penpot/plugin-styles/styles.css';
```

View file

@ -1,8 +1,5 @@
{ {
"name": "plugins-styles", "name": "@penpot/plugin-styles",
"version": "0.0.1", "version": "0.1.0",
"dependencies": {}, "dependencies": {}
"main": "./index.js",
"module": "./index.mjs",
"typings": "./index.d.ts"
} }

View file

@ -5,12 +5,15 @@
"projectType": "library", "projectType": "library",
"targets": { "targets": {
"build": { "build": {
"executor": "@nx/vite:build", "command": "node tools/scripts/build-css.mjs",
"outputs": ["{options.outputPath}"],
"options": { "options": {
"outputPath": "dist/plugins-styles" "outputPath": "dist/plugins-styles"
} }
}, },
"publish": {
"command": "node tools/scripts/publish.mjs plugins-styles {args.ver} {args.tag}",
"dependsOn": ["build"]
},
"lint": { "lint": {
"executor": "@nx/eslint:lint", "executor": "@nx/eslint:lint",
"outputs": ["{options.outputFile}"] "outputs": ["{options.outputFile}"]

View file

@ -1,3 +0,0 @@
export function pluginsCssLib(): string {
return 'plugins-styles';
}

View file

@ -1,22 +0,0 @@
{
"extends": "../../tsconfig.base.json",
"compilerOptions": {
"module": "commonjs",
"forceConsistentCasingInFileNames": true,
"strict": true,
"noImplicitOverride": true,
"noPropertyAccessFromIndexSignature": true,
"noImplicitReturns": true,
"noFallthroughCasesInSwitch": true
},
"files": [],
"include": [],
"references": [
{
"path": "./tsconfig.lib.json"
},
{
"path": "./tsconfig.spec.json"
}
]
}

View file

@ -1,10 +0,0 @@
{
"extends": "./tsconfig.json",
"compilerOptions": {
"outDir": "../../dist/out-tsc",
"declaration": true,
"types": ["node", "vite/client"]
},
"include": ["src/**/*.ts"],
"exclude": ["src/**/*.spec.ts", "src/**/*.test.ts"]
}

View file

@ -1,13 +0,0 @@
{
"extends": "./tsconfig.json",
"compilerOptions": {
"outDir": "../dist/out-tsc",
"module": "commonjs",
"types": ["node"]
},
"include": [
"src/**/*.test.ts",
"src/**/*.spec.ts",
"src/**/*.d.ts"
]
}

View file

@ -1,47 +0,0 @@
/// <reference types='vitest' />
import { defineConfig } from 'vite';
import dts from 'vite-plugin-dts';
import * as path from 'path';
import { nxViteTsPaths } from '@nx/vite/plugins/nx-tsconfig-paths.plugin';
export default defineConfig({
root: __dirname,
cacheDir: '../node_modules/.vite/plugins-styles',
plugins: [
nxViteTsPaths(),
dts({
entryRoot: 'src',
tsConfigFilePath: path.join(__dirname, 'tsconfig.lib.json'),
skipDiagnostics: true,
}),
],
// Uncomment this if you are using workers.
// worker: {
// plugins: [ nxViteTsPaths() ],
// },
// Configuration for building your library.
// See: https://vitejs.dev/guide/build.html#library-mode
build: {
outDir: '../dist/plugins-styles',
reportCompressedSize: true,
commonjsOptions: {
transformMixedEsModules: true,
},
lib: {
// Could also be a dictionary or array of multiple entry points.
entry: 'src/index.ts',
name: 'plugins-styles',
fileName: 'index',
// Change this to the formats you want to support.
// Don't forget to update your package.json as well.
formats: ['es', 'cjs'],
},
rollupOptions: {
// External packages that should not be bundled into your library.
external: [],
},
},
});

2170
package-lock.json generated

File diff suppressed because it is too large Load diff

View file

@ -1,6 +1,6 @@
{ {
"name": "penpot-plugins", "name": "penpot-plugins",
"version": "0.0.0", "version": "0.1.0",
"license": "MIT", "license": "MIT",
"scripts": { "scripts": {
"start": "npx nx run plugins-runtime:build --watch & npx nx run plugins-runtime:preview", "start": "npx nx run plugins-runtime:build --watch & npx nx run plugins-runtime:preview",
@ -11,6 +11,8 @@
"lint": "nx run-many --all --target=lint --parallel", "lint": "nx run-many --all --target=lint --parallel",
"lint:affected": "npx nx affected --target=lint", "lint:affected": "npx nx affected --target=lint",
"test": "npx nx test plugins-runtime", "test": "npx nx test plugins-runtime",
"publish": "nx run-many -t publish -p plugins-styles plugin-types --parallel=false --",
"registry": "nx local-registry",
"prepare": "husky" "prepare": "husky"
}, },
"private": true, "private": true,
@ -37,6 +39,7 @@
"eslint": "~8.48.0", "eslint": "~8.48.0",
"eslint-config-prettier": "^9.0.0", "eslint-config-prettier": "^9.0.0",
"eslint-plugin-deprecation": "^2.0.0", "eslint-plugin-deprecation": "^2.0.0",
"fs-extra": "^11.2.0",
"happy-dom": "^13.6.2", "happy-dom": "^13.6.2",
"husky": "^9.0.10", "husky": "^9.0.10",
"jsdom": "~22.1.0", "jsdom": "~22.1.0",
@ -45,6 +48,7 @@
"swc-loader": "0.1.15", "swc-loader": "0.1.15",
"ts-node": "10.9.1", "ts-node": "10.9.1",
"typescript": "~5.2.2", "typescript": "~5.2.2",
"verdaccio": "^5.0.4",
"vite": "^5.0.0", "vite": "^5.0.0",
"vite-plugin-dts": "~2.3.0", "vite-plugin-dts": "~2.3.0",
"vitest": "1.2.2" "vitest": "1.2.2"
@ -63,5 +67,8 @@
"tslib": "^2.3.0", "tslib": "^2.3.0",
"uuid": "^9.0.1", "uuid": "^9.0.1",
"zod": "^3.22.4" "zod": "^3.22.4"
},
"nx": {
"includedScripts": []
} }
} }

15
project.json Normal file
View file

@ -0,0 +1,15 @@
{
"name": "penpot-plugins",
"$schema": "node_modules/nx/schemas/project-schema.json",
"targets": {
"local-registry": {
"executor": "@nx/js:verdaccio",
"options": {
"port": 4873,
"config": ".verdaccio/config.yml",
"storage": "tmp/local-registry/storage",
"clear": false
}
}
}
}

View file

@ -0,0 +1,23 @@
import esbuild from 'esbuild';
import { copy } from 'fs-extra';
const source = 'libs/plugins-styles';
const dist = 'dist/plugins-styles';
const handleErr = (err) => {
console.error(err);
process.exit(1);
};
esbuild.build({
entryPoints: [`${source}/src/lib/styles.css`],
bundle: true,
outfile: `${dist}/styles.css`,
minify: true,
loader: {
'.svg': 'text'
}
}).catch(handleErr);
copy(`${source}/package.json`, `${dist}/package.json`).catch(handleErr);
copy(`${source}/README.md`, `${dist}/README.md`).catch(handleErr);

View file

@ -0,0 +1,13 @@
import { copy } from 'fs-extra';
const source = 'libs/plugin-types';
const dist = 'dist/plugin-types';
const handleErr = (err) => {
console.error(err);
process.exit(1);
};
copy(`${source}/package.json`, `${dist}/package.json`).catch(handleErr);
copy(`${source}/README.md`, `${dist}/README.md`).catch(handleErr);
copy(`${source}/index.d.ts`, `${dist}/index.d.ts`).catch(handleErr);

65
tools/scripts/publish.mjs Normal file
View file

@ -0,0 +1,65 @@
/**
* This is a minimal script to publish your package to "npm".
* This is meant to be used as-is or customize as you see fit.
*
* This script is executed on "dist/path/to/library" as "cwd" by default.
*
* You might need to authenticate with NPM before running this script.
*/
import { execSync } from 'child_process';
import { readFileSync, writeFileSync } from 'fs';
import devkit from '@nx/devkit';
const { readCachedProjectGraph } = devkit;
function invariant(condition, message) {
if (!condition) {
console.error(message);
process.exit(1);
}
}
// Executing publish script: node path/to/publish.mjs {name} --version {version} --tag {tag}
// Default "tag" to "next" so we won't publish the "latest" tag by accident.
const [, , name, version, tag = 'next', registry] = process.argv;
// A simple SemVer validation to validate the version
const validVersion = /^\d+\.\d+\.\d+(-\w+\.\d+)?/;
invariant(
version && validVersion.test(version),
`No version provided or version did not match Semantic Versioning, expected: #.#.#-tag.# or #.#.#, got ${version}.`
);
const graph = readCachedProjectGraph();
const project = graph.nodes[name];
invariant(
project,
`Could not find project "${name}" in the workspace. Is the project.json configured correctly?`
);
const outputPath = project.data?.targets?.build?.options?.outputPath;
invariant(
outputPath,
`Could not find "build.options.outputPath" of project "${name}". Is project.json configured correctly?`
);
process.chdir(outputPath);
// Updating the version in "package.json" before publishing
try {
const json = JSON.parse(readFileSync(`package.json`).toString());
json.version = version;
writeFileSync(`package.json`, JSON.stringify(json, null, 2));
} catch (e) {
console.error(`Error reading package.json file from library build output.`);
}
// Execute "npm publish" to publish
let command = `npm publish --access public --tag ${tag}`;
if (registry) {
command += ` --registry ${registry}`;
}
execSync(command);

View file

@ -15,6 +15,7 @@
"skipDefaultLibCheck": true, "skipDefaultLibCheck": true,
"baseUrl": ".", "baseUrl": ".",
"paths": { "paths": {
"@penpot/plugin-types": ["libs/plugin-types/index.d.ts"],
"plugins-parser": ["libs/plugins-data-parser/src/index.ts"], "plugins-parser": ["libs/plugins-data-parser/src/index.ts"],
"plugins-runtime": ["libs/plugins-runtime/src/index.ts"], "plugins-runtime": ["libs/plugins-runtime/src/index.ts"],
"plugins-styles/*": ["libs/plugins-styles/src/*"] "plugins-styles/*": ["libs/plugins-styles/src/*"]