mirror of
https://github.com/toeverything/AFFiNE.git
synced 2024-12-22 23:01:35 +03:00
feat(plugin): add vue example (#3592)
This commit is contained in:
parent
48350d7654
commit
7bf77b566d
@ -25,7 +25,8 @@
|
||||
"storage",
|
||||
"infra",
|
||||
"plugin-cli",
|
||||
"sdk"
|
||||
"sdk",
|
||||
"plugin"
|
||||
]
|
||||
]
|
||||
}
|
||||
|
@ -135,7 +135,14 @@ const pluginFetch = createFetch({});
|
||||
const timer = createTimers(abortController.signal);
|
||||
|
||||
const sharedGlobalThis = Object.assign(Object.create(null), timer, {
|
||||
Object: globalThis.Object,
|
||||
fetch: pluginFetch,
|
||||
Symbol: globalThis.Symbol,
|
||||
Error: globalThis.Error,
|
||||
TypeError: globalThis.TypeError,
|
||||
RangeError: globalThis.RangeError,
|
||||
console: globalThis.console,
|
||||
crypto: globalThis.crypto,
|
||||
});
|
||||
|
||||
const dynamicImportMap = new Map<
|
||||
@ -222,6 +229,9 @@ export const createOrGetGlobalThis = (
|
||||
if (sharedGlobalThis[key]) return sharedGlobalThis[key];
|
||||
const result = Reflect.get(window, key);
|
||||
if (typeof result === 'function') {
|
||||
if (result === ShadowRoot) {
|
||||
return result;
|
||||
}
|
||||
return function (...args: any[]) {
|
||||
permissionLogger.debug(
|
||||
`${pluginName} is calling window`,
|
||||
@ -262,15 +272,11 @@ export const createOrGetGlobalThis = (
|
||||
userAgent: navigator.userAgent,
|
||||
},
|
||||
|
||||
// safe to use for all plugins
|
||||
Error: globalThis.Error,
|
||||
TypeError: globalThis.TypeError,
|
||||
RangeError: globalThis.RangeError,
|
||||
console: globalThis.console,
|
||||
crypto: globalThis.crypto,
|
||||
MouseEvent: globalThis.MouseEvent,
|
||||
KeyboardEvent: globalThis.KeyboardEvent,
|
||||
CustomEvent: globalThis.CustomEvent,
|
||||
|
||||
// copilot uses these
|
||||
CustomEvent: globalThis.CustomEvent,
|
||||
Date: globalThis.Date,
|
||||
Math: globalThis.Math,
|
||||
URL: globalThis.URL,
|
||||
@ -284,6 +290,10 @@ export const createOrGetGlobalThis = (
|
||||
Blob: globalThis.Blob,
|
||||
ClipboardItem: globalThis.ClipboardItem,
|
||||
|
||||
// vue uses these
|
||||
Element: globalThis.Element,
|
||||
SVGElement: globalThis.SVGElement,
|
||||
|
||||
// fixme: use our own db api
|
||||
indexedDB: globalThis.indexedDB,
|
||||
IDBRequest: globalThis.IDBRequest,
|
||||
@ -299,6 +309,7 @@ export const createOrGetGlobalThis = (
|
||||
IDBVersionChangeEvent: globalThis.IDBVersionChangeEvent,
|
||||
}
|
||||
);
|
||||
pluginGlobalThis.global = pluginGlobalThis;
|
||||
globalThisMap.set(pluginName, pluginGlobalThis);
|
||||
return pluginGlobalThis;
|
||||
};
|
||||
|
@ -1,5 +1,7 @@
|
||||
import { DebugLogger } from '@affine/debug';
|
||||
import { registeredPluginAtom, rootStore } from '@toeverything/infra/atom';
|
||||
import { packageJsonOutputSchema } from '@toeverything/infra/type';
|
||||
import type { z } from 'zod';
|
||||
|
||||
import { evaluatePluginEntry, setupPluginCode } from './plugins/setup';
|
||||
|
||||
@ -8,6 +10,7 @@ const builtinPluginUrl = new Set([
|
||||
'/plugins/copilot',
|
||||
'/plugins/hello-world',
|
||||
'/plugins/image-preview',
|
||||
'/plugins/vue-hello-world',
|
||||
]);
|
||||
|
||||
const logger = new DebugLogger('register-plugins');
|
||||
@ -23,7 +26,10 @@ export const pluginRegisterPromise = Promise.all(
|
||||
[...builtinPluginUrl].map(url => {
|
||||
return fetch(`${url}/package.json`)
|
||||
.then(async res => {
|
||||
const packageJson = await res.json();
|
||||
const packageJson = (await res.json()) as z.infer<
|
||||
typeof packageJsonOutputSchema
|
||||
>;
|
||||
packageJsonOutputSchema.parse(packageJson);
|
||||
const {
|
||||
name: pluginName,
|
||||
affinePlugin: {
|
||||
@ -38,6 +44,12 @@ export const pluginRegisterPromise = Promise.all(
|
||||
if (!release && !runtimeConfig.enablePlugin) {
|
||||
return Promise.resolve();
|
||||
}
|
||||
if (
|
||||
release === 'development' &&
|
||||
process.env.NODE_ENV !== 'development'
|
||||
) {
|
||||
return Promise.resolve();
|
||||
}
|
||||
const baseURL = url;
|
||||
const entryURL = `${baseURL}/${core}`;
|
||||
rootStore.set(registeredPluginAtom, prev => [...prev, pluginName]);
|
||||
|
@ -65,7 +65,7 @@
|
||||
"@faker-js/faker": "^8.0.2",
|
||||
"@istanbuljs/schema": "^0.1.3",
|
||||
"@magic-works/i18n-codegen": "^0.5.0",
|
||||
"@nx/vite": "16.5.5",
|
||||
"@nx/vite": "16.6.0",
|
||||
"@perfsee/sdk": "^1.8.5",
|
||||
"@playwright/test": "^1.36.2",
|
||||
"@taplo/cli": "^0.5.2",
|
||||
@ -90,6 +90,7 @@
|
||||
"eslint-plugin-sonarjs": "^0.19.0",
|
||||
"eslint-plugin-unicorn": "^48.0.0",
|
||||
"eslint-plugin-unused-imports": "^3.0.0",
|
||||
"eslint-plugin-vue": "^9.16.1",
|
||||
"fake-indexeddb": "4.0.2",
|
||||
"happy-dom": "^10.8.0",
|
||||
"husky": "^8.0.3",
|
||||
@ -97,7 +98,7 @@
|
||||
"madge": "^6.1.0",
|
||||
"msw": "^1.2.3",
|
||||
"nanoid": "^4.0.2",
|
||||
"nx": "16.5.5",
|
||||
"nx": "16.6.0",
|
||||
"nx-cloud": "latest",
|
||||
"nyc": "^15.1.0",
|
||||
"prettier": "^3.0.0",
|
||||
|
@ -9,7 +9,7 @@ export const packageJsonInputSchema = z.object({
|
||||
version: z.string(),
|
||||
description: z.string(),
|
||||
affinePlugin: z.object({
|
||||
release: z.boolean(),
|
||||
release: z.union([z.boolean(), z.enum(['development'])]),
|
||||
entry: z.object({
|
||||
core: z.string(),
|
||||
server: z.string().optional(),
|
||||
@ -23,7 +23,7 @@ export const packageJsonOutputSchema = z.object({
|
||||
version: z.string(),
|
||||
description: z.string(),
|
||||
affinePlugin: z.object({
|
||||
release: z.boolean(),
|
||||
release: z.union([z.boolean(), z.enum(['development'])]),
|
||||
entry: z.object({
|
||||
core: z.string(),
|
||||
}),
|
||||
|
@ -14,6 +14,7 @@
|
||||
"@swc/core": "^1.3.72",
|
||||
"@toeverything/infra": "workspace:^",
|
||||
"@vanilla-extract/rollup-plugin": "^1.2.2",
|
||||
"@vitejs/plugin-vue": "^4.2.3",
|
||||
"rollup": "^3.27.0",
|
||||
"rollup-plugin-swc3": "^0.9.1",
|
||||
"ts-node": "^10.9.1"
|
||||
|
@ -11,6 +11,7 @@ import {
|
||||
} from '@toeverything/infra/type';
|
||||
import { vanillaExtractPlugin } from '@vanilla-extract/vite-plugin';
|
||||
import react from '@vitejs/plugin-react-swc';
|
||||
import vue from '@vitejs/plugin-vue';
|
||||
import { build, type PluginOption } from 'vite';
|
||||
import type { z } from 'zod';
|
||||
|
||||
@ -30,7 +31,19 @@ if (!plugin) {
|
||||
|
||||
const command = result.positionals[0];
|
||||
|
||||
const isWatch = command === 'dev';
|
||||
const isWatch = (() => {
|
||||
switch (command) {
|
||||
case 'dev': {
|
||||
return true;
|
||||
}
|
||||
case 'build': {
|
||||
return false;
|
||||
}
|
||||
default: {
|
||||
throw new Error('invalid command');
|
||||
}
|
||||
}
|
||||
})();
|
||||
|
||||
const external = [
|
||||
// built-in packages
|
||||
@ -131,6 +144,7 @@ await build({
|
||||
build: {
|
||||
watch: isWatch ? {} : undefined,
|
||||
minify: false,
|
||||
target: 'es2020',
|
||||
outDir: coreOutDir,
|
||||
emptyOutDir: true,
|
||||
lib: {
|
||||
@ -169,6 +183,7 @@ await build({
|
||||
},
|
||||
plugins: [
|
||||
vanillaExtractPlugin(),
|
||||
vue(),
|
||||
react(),
|
||||
{
|
||||
name: 'parse-bundle',
|
||||
|
@ -1,6 +1,13 @@
|
||||
{
|
||||
"name": "@affine/bookmark-plugin",
|
||||
"$schema": "../../node_modules/nx/schemas/project-schema.json",
|
||||
"namedInputs": {
|
||||
"default": [
|
||||
"{projectRoot}/**/*",
|
||||
"{workspaceRoot}/packages/plugin-cli/**/*",
|
||||
"sharedGlobals"
|
||||
]
|
||||
},
|
||||
"targets": {
|
||||
"build": {
|
||||
"executor": "nx:run-script",
|
||||
@ -8,7 +15,7 @@
|
||||
"script": "build"
|
||||
},
|
||||
"dependsOn": ["^build"],
|
||||
"inputs": ["{projectRoot}/**/*"],
|
||||
"inputs": ["default"],
|
||||
"outputs": [
|
||||
"{workspaceRoot}/apps/core/public/plugins/bookmark",
|
||||
"{workspaceRoot}/apps/electron/dist/plugins/bookmark"
|
||||
|
@ -1,7 +1,13 @@
|
||||
{
|
||||
"name": "@affine/copilot-plugin",
|
||||
"$schema": "../../node_modules/nx/schemas/project-schema.json",
|
||||
"tags": ["plugin"],
|
||||
"namedInputs": {
|
||||
"default": [
|
||||
"{projectRoot}/**/*",
|
||||
"{workspaceRoot}/packages/plugin-cli/src/**/*",
|
||||
"sharedGlobals"
|
||||
]
|
||||
},
|
||||
"targets": {
|
||||
"build": {
|
||||
"executor": "nx:run-script",
|
||||
@ -9,11 +15,12 @@
|
||||
"script": "build"
|
||||
},
|
||||
"dependsOn": ["^build"],
|
||||
"inputs": ["{projectRoot}/**/*"],
|
||||
"inputs": ["default"],
|
||||
"outputs": [
|
||||
"{workspaceRoot}/apps/core/public/plugins/copilot",
|
||||
"{workspaceRoot}/apps/electron/dist/plugins/copilot"
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
"tags": ["plugin"]
|
||||
}
|
||||
|
@ -1,7 +1,13 @@
|
||||
{
|
||||
"name": "@affine/hello-world-plugin",
|
||||
"$schema": "../../node_modules/nx/schemas/project-schema.json",
|
||||
"tags": ["plugin"],
|
||||
"namedInputs": {
|
||||
"default": [
|
||||
"{projectRoot}/**/*",
|
||||
"{workspaceRoot}/packages/plugin-cli/src/**/*",
|
||||
"sharedGlobals"
|
||||
]
|
||||
},
|
||||
"targets": {
|
||||
"build": {
|
||||
"executor": "nx:run-script",
|
||||
@ -9,11 +15,12 @@
|
||||
"script": "build"
|
||||
},
|
||||
"dependsOn": ["^build"],
|
||||
"inputs": ["{projectRoot}/**/*"],
|
||||
"inputs": ["default"],
|
||||
"outputs": [
|
||||
"{workspaceRoot}/apps/core/public/plugins/hello-world",
|
||||
"{workspaceRoot}/apps/electron/dist/plugins/hello-world"
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
"tags": ["plugin"]
|
||||
}
|
||||
|
@ -1,7 +1,13 @@
|
||||
{
|
||||
"name": "@affine/image-preview-plugin",
|
||||
"$schema": "../../node_modules/nx/schemas/project-schema.json",
|
||||
"tags": ["plugin"],
|
||||
"namedInputs": {
|
||||
"default": [
|
||||
"{projectRoot}/**/*",
|
||||
"{workspaceRoot}/packages/plugin-cli/src/**/*",
|
||||
"sharedGlobals"
|
||||
]
|
||||
},
|
||||
"targets": {
|
||||
"build": {
|
||||
"executor": "nx:run-script",
|
||||
@ -9,11 +15,12 @@
|
||||
"script": "build"
|
||||
},
|
||||
"dependsOn": ["^build"],
|
||||
"inputs": ["{projectRoot}/**/*"],
|
||||
"inputs": ["default"],
|
||||
"outputs": [
|
||||
"{workspaceRoot}/apps/core/public/plugins/image-preview",
|
||||
"{workspaceRoot}/apps/electron/dist/plugins/image-preview"
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
"tags": ["plugin"]
|
||||
}
|
||||
|
10
plugins/vue-hello-world/.eslintrc.json
Normal file
10
plugins/vue-hello-world/.eslintrc.json
Normal file
@ -0,0 +1,10 @@
|
||||
{
|
||||
"root": false,
|
||||
"parser": "vue-eslint-parser",
|
||||
"parserOptions": {
|
||||
"parser": "@typescript-eslint/parser",
|
||||
"sourceType": "module",
|
||||
"extraFileExtensions": [".vue"]
|
||||
},
|
||||
"extends": ["plugin:vue/vue3-recommended"]
|
||||
}
|
26
plugins/vue-hello-world/package.json
Normal file
26
plugins/vue-hello-world/package.json
Normal file
@ -0,0 +1,26 @@
|
||||
{
|
||||
"name": "@affine/vue-hello-world-plugin",
|
||||
"type": "module",
|
||||
"private": true,
|
||||
"description": "Vue hello world plugin",
|
||||
"version": "0.8.0-canary.11",
|
||||
"scripts": {
|
||||
"dev": "af dev",
|
||||
"build": "af build"
|
||||
},
|
||||
"affinePlugin": {
|
||||
"release": "development",
|
||||
"entry": {
|
||||
"core": "./src/index.ts"
|
||||
}
|
||||
},
|
||||
"dependencies": {
|
||||
"@affine/component": "workspace:*",
|
||||
"@affine/sdk": "workspace:*",
|
||||
"element-plus": "^2.3.9",
|
||||
"vue": "^3.3.4"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@affine/plugin-cli": "workspace:*"
|
||||
}
|
||||
}
|
26
plugins/vue-hello-world/project.json
Normal file
26
plugins/vue-hello-world/project.json
Normal file
@ -0,0 +1,26 @@
|
||||
{
|
||||
"name": "@affine/vue-hello-world-plugin",
|
||||
"$schema": "../../node_modules/nx/schemas/project-schema.json",
|
||||
"namedInputs": {
|
||||
"default": [
|
||||
"{projectRoot}/**/*",
|
||||
"{workspaceRoot}/packages/plugin-cli/src/**/*",
|
||||
"sharedGlobals"
|
||||
]
|
||||
},
|
||||
"targets": {
|
||||
"build": {
|
||||
"executor": "nx:run-script",
|
||||
"options": {
|
||||
"script": "build"
|
||||
},
|
||||
"dependsOn": ["^build"],
|
||||
"inputs": ["default"],
|
||||
"outputs": [
|
||||
"{workspaceRoot}/apps/core/public/plugins/vue-hello-world",
|
||||
"{workspaceRoot}/apps/electron/dist/plugins/vue-hello-world"
|
||||
]
|
||||
}
|
||||
},
|
||||
"tags": ["plugin"]
|
||||
}
|
13
plugins/vue-hello-world/src/app.vue
Normal file
13
plugins/vue-hello-world/src/app.vue
Normal file
@ -0,0 +1,13 @@
|
||||
<script setup lang="ts">
|
||||
import { ref } from 'vue';
|
||||
|
||||
const count = ref(0);
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<el-button @click="count++">
|
||||
{{ count }}
|
||||
</el-button>
|
||||
</template>
|
||||
|
||||
<style scoped></style>
|
5
plugins/vue-hello-world/src/global.d.ts
vendored
Normal file
5
plugins/vue-hello-world/src/global.d.ts
vendored
Normal file
@ -0,0 +1,5 @@
|
||||
declare module '*.vue' {
|
||||
import type { ComponentOptions } from 'vue';
|
||||
const component: ComponentOptions;
|
||||
export default component;
|
||||
}
|
18
plugins/vue-hello-world/src/index.ts
Normal file
18
plugins/vue-hello-world/src/index.ts
Normal file
@ -0,0 +1,18 @@
|
||||
import type { PluginContext } from '@affine/sdk/entry';
|
||||
import ElementPlus from 'element-plus';
|
||||
import { createApp } from 'vue';
|
||||
|
||||
import App from './app.vue';
|
||||
|
||||
export const entry = (context: PluginContext) => {
|
||||
context.register('headerItem', div => {
|
||||
const app = createApp(App);
|
||||
app.use(ElementPlus);
|
||||
app.mount(div, false, false);
|
||||
return () => {
|
||||
app.unmount();
|
||||
};
|
||||
});
|
||||
|
||||
return () => {};
|
||||
};
|
14
plugins/vue-hello-world/tsconfig.json
Normal file
14
plugins/vue-hello-world/tsconfig.json
Normal file
@ -0,0 +1,14 @@
|
||||
{
|
||||
"extends": "../../tsconfig.json",
|
||||
"include": ["./src"],
|
||||
"compilerOptions": {
|
||||
"noEmit": false,
|
||||
"outDir": "lib",
|
||||
"jsx": "preserve"
|
||||
},
|
||||
"references": [
|
||||
{
|
||||
"path": "../../packages/sdk"
|
||||
}
|
||||
]
|
||||
}
|
@ -44,5 +44,11 @@ test('plugin should exist', async ({ page }) => {
|
||||
description: expect.any(String),
|
||||
affinePlugin: expect.anything(),
|
||||
},
|
||||
{
|
||||
name: '@affine/vue-hello-world-plugin',
|
||||
version: expect.any(String),
|
||||
description: expect.any(String),
|
||||
affinePlugin: expect.anything(),
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
@ -130,6 +130,9 @@
|
||||
{
|
||||
"path": "./plugins/image-preview"
|
||||
},
|
||||
{
|
||||
"path": "./plugins/vue-hello-world"
|
||||
},
|
||||
// Packages
|
||||
{
|
||||
"path": "./packages/cli"
|
||||
|
Loading…
Reference in New Issue
Block a user