@virid/main 0.1.0 → 0.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +160 -1
- package/README.zh.md +138 -1
- package/package.json +3 -3
package/README.md
CHANGED
|
@@ -1 +1,160 @@
|
|
|
1
|
-
|
|
1
|
+
# @virid/main
|
|
2
|
+
|
|
3
|
+
**@virid/main** is the main process adapter for Electron applications. It provides automatic routing for `ToRendererMessage` and `FromRendererMessage`, along with the capability to receive and manage reporting from rendering processes.
|
|
4
|
+
|
|
5
|
+
## 🌟 Core Design Philosophy
|
|
6
|
+
|
|
7
|
+
In **@virid/main**, messages sent from the main process to a rendering process are abstracted as `ToRendererMessage`. These messages can be distributed directly via the dispatcher, traveling through IPC to the target rendering process to trigger the corresponding **System**.
|
|
8
|
+
|
|
9
|
+
Conversely, messages sent from a rendering process to the main process are abstracted as `FromRendererMessage`. Once a rendering process sends a message, the `FromRendererMessage` is automatically delivered to the dispatcher to trigger the relevant main process **System**.
|
|
10
|
+
|
|
11
|
+
- **Automatic Route Registration**: When a rendering process window opens, it automatically registers itself with the main process. All subsequent messages will carry the window's specific metadata (context).
|
|
12
|
+
- **Type Restoration**: After passing through the IPC channel, messages are restored into their original **Message** classes before re-entering `@virid/core`. This allows them to be recognized and distributed by the dispatcher, achieving **location transparency** across different processes.
|
|
13
|
+
- **Targeted Messaging & Broadcasting**: You can achieve point-to-point communication by using the specific IDs of rendering processes, or use `*` to broadcast messages to all windows.
|
|
14
|
+
|
|
15
|
+
## 🔌Enable plugins
|
|
16
|
+
|
|
17
|
+
```ts
|
|
18
|
+
import { createVirid } from '@virid/core'
|
|
19
|
+
import { MainPlugin } from '@virid/main'
|
|
20
|
+
import { app } from 'electron'
|
|
21
|
+
const virid = createVirid()
|
|
22
|
+
.use(MainPlugin, { electronApp: app })
|
|
23
|
+
```
|
|
24
|
+
|
|
25
|
+
## 🛠️ @virid/main Core API Overview
|
|
26
|
+
|
|
27
|
+
### ToRendererMessage
|
|
28
|
+
|
|
29
|
+
- **Function**: A specialized message base class designed for inheritance. All messages inheriting from `ToRendererMessage` will be dispatched to the rendering process.
|
|
30
|
+
- **Logic**: This message type requires two specific metadata tags:
|
|
31
|
+
- `__virid_target`: Specifies the destination (e.g., a specific `windowId`).
|
|
32
|
+
- `__virid_messageType`: Defines the message type it should be restored to upon reaching the rendering process.
|
|
33
|
+
|
|
34
|
+
- Example:
|
|
35
|
+
|
|
36
|
+
**In the Main Process:**
|
|
37
|
+
|
|
38
|
+
```ts
|
|
39
|
+
import { ToRendererMessage } from '@virid/main'
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* - __virid_target = 'renderer': Indicates the message is bound for the rendering process
|
|
43
|
+
* with the windowId 'renderer'.
|
|
44
|
+
* - __virid_messageType = 'file-dialog': Describes the type this message will transform
|
|
45
|
+
* into once it arrives in the rendering process.
|
|
46
|
+
*/
|
|
47
|
+
export class RenderDialogMessage extends ToRendererMessage {
|
|
48
|
+
__virid_target: string = 'renderer'
|
|
49
|
+
__virid_messageType: string = 'file-dialog'
|
|
50
|
+
|
|
51
|
+
constructor(public path: string) {
|
|
52
|
+
super()
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
```
|
|
56
|
+
|
|
57
|
+
In the **Rendering Process**, you can use the `@FromIpc` decorator in conjunction with `FromMainMessage` to handle the conversion. For more details, refer to `@virid/renderer`.
|
|
58
|
+
|
|
59
|
+
**In the Rendering Process:**
|
|
60
|
+
|
|
61
|
+
```ts
|
|
62
|
+
import { FromIpc, FromMainMessage } from '@virid/renderer'
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* @FromIpc('file-dialog') indicates that when the Main Process sends a ToRendererMessage
|
|
66
|
+
* where __virid_messageType is 'file-dialog', this message will be automatically
|
|
67
|
+
* delivered to the Virid dispatcher to trigger the corresponding rendering process System.
|
|
68
|
+
*/
|
|
69
|
+
@FromIpc('file-dialog')
|
|
70
|
+
class ChooseBgImageMessage extends FromMainMessage {
|
|
71
|
+
constructor(public path: string) {
|
|
72
|
+
super()
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
```
|
|
76
|
+
|
|
77
|
+
### FromRenderMessage / @FromRenderer()
|
|
78
|
+
|
|
79
|
+
- **Function**: A specialized message base class designed for inheritance. All messages inheriting from `FromRenderMessage` will be automatically converted and dispatched to the **Virid Dispatcher** in the Main Process when sent from a Rendering Process.
|
|
80
|
+
- **Logic**: The `@FromRenderer()` decorator accepts a string-based ID. This ID must match the `__virid_messageType` of the `ToMainMessage` dispatched from the Rendering Process.
|
|
81
|
+
|
|
82
|
+
#### Example:
|
|
83
|
+
|
|
84
|
+
**In the Main Process:**
|
|
85
|
+
|
|
86
|
+
```ts
|
|
87
|
+
import { FromRenderer, FromRenderMessage, ToRendererMessage } from '@virid/main'
|
|
88
|
+
|
|
89
|
+
// Message to be sent to the Rendering Process
|
|
90
|
+
export class RenderDialogMessage extends ToRendererMessage {
|
|
91
|
+
__virid_target: string = 'renderer'
|
|
92
|
+
__virid_messageType: string = 'file-dialog'
|
|
93
|
+
constructor(public path: string) {
|
|
94
|
+
super()
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
/**
|
|
99
|
+
* Converts incoming messages from the Rendering Process with
|
|
100
|
+
* __virid_messageType: 'open-dialog' into the Main Process's OpenDialogMessage.
|
|
101
|
+
*/
|
|
102
|
+
@FromRenderer('open-dialog')
|
|
103
|
+
export class OpenDialogMessage extends FromRenderMessage {
|
|
104
|
+
constructor(
|
|
105
|
+
public options: {
|
|
106
|
+
title?: string
|
|
107
|
+
filters?: Array<{ name: string; extensions: string[] }>
|
|
108
|
+
properties?: Array<'openFile' | 'openDirectory' | 'multiSelections' | 'showHiddenFiles'>
|
|
109
|
+
}
|
|
110
|
+
) {
|
|
111
|
+
super()
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
export class WindowSystem {
|
|
116
|
+
@System()
|
|
117
|
+
async openDialog(@Message(OpenDialogMessage) message: OpenDialogMessage) {
|
|
118
|
+
// Invoke native Electron dialog
|
|
119
|
+
const result = await dialog.showOpenDialog(message.senderWindow, message.options)
|
|
120
|
+
|
|
121
|
+
// If the user did not cancel and selected a file
|
|
122
|
+
if (!result.canceled && result.filePaths.length > 0) {
|
|
123
|
+
const selectedPath = result.filePaths[0]
|
|
124
|
+
return new RenderDialogMessage(selectedPath)
|
|
125
|
+
}
|
|
126
|
+
return
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
```
|
|
130
|
+
|
|
131
|
+
In the **Rendering Process**, you can use `@FromIpc` in conjunction with `ToMainMessage` to handle message conversion and dispatching. For more details, refer to `@virid/renderer`.
|
|
132
|
+
|
|
133
|
+
**In the Rendering Process:**
|
|
134
|
+
|
|
135
|
+
```ts
|
|
136
|
+
import { ToMainMessage, FromIpc, FromMainMessage } from '@virid/renderer'
|
|
137
|
+
|
|
138
|
+
// Handles the incoming file selection result
|
|
139
|
+
@FromIpc('file-dialog')
|
|
140
|
+
class ChooseBgImageMessage extends FromMainMessage {
|
|
141
|
+
constructor(public path: string) {
|
|
142
|
+
super()
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
// Initiates the request to open a dialog
|
|
147
|
+
class OpenDialogMessage extends ToMainMessage {
|
|
148
|
+
__virid_target: string = 'main'
|
|
149
|
+
__virid_messageType: string = 'open-dialog'
|
|
150
|
+
constructor(
|
|
151
|
+
public options: {
|
|
152
|
+
title?: string
|
|
153
|
+
filters?: Array<{ name: string; extensions: string[] }>
|
|
154
|
+
properties?: Array<'openFile' | 'openDirectory' | 'multiSelections' | 'showHiddenFiles'>
|
|
155
|
+
}
|
|
156
|
+
) {
|
|
157
|
+
super()
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
```
|
package/README.zh.md
CHANGED
|
@@ -1 +1,138 @@
|
|
|
1
|
-
|
|
1
|
+
# @virid/main
|
|
2
|
+
|
|
3
|
+
`@virid/main` 是electron应用的主适配器,提供`ToRendererMessage`和`FromRendererMessage`的自动路由功能与接受渲染进程报道功能。
|
|
4
|
+
|
|
5
|
+
## 🌟 核心设计理念
|
|
6
|
+
|
|
7
|
+
在 `@virid/main` 中,主进程朝渲染进程发送消息被抽象为`ToRendererMessage`,该消息可以直接通过调度器分发,消息将会通过IPC直接传送到渲染进程并触发对应的`System`。渲染进程朝主进程发送的消息被抽象为`FromRendererMessage`,一旦渲染进程发送了对应的消息,`FromRendererMessage`将会自动被投递到调度器触发相应的主进程`System`。
|
|
8
|
+
|
|
9
|
+
- **路由自动注册**:渲染进程窗口开启后,自动向主进程注册自身,以后所有的消息都将携带自身的窗口信息。
|
|
10
|
+
- **类型恢复**:消息在通过IPC通道后,恢复成真正的Message类重新进入`@virid/core`并被调度器识别并分发。实现不同进程的位置无关性。
|
|
11
|
+
- **消息定向与广播**:可以通过其他渲染进程的ID来实现定向通信,或者使用“*”来广播消息。
|
|
12
|
+
|
|
13
|
+
## 🔌启用插件
|
|
14
|
+
|
|
15
|
+
```ts
|
|
16
|
+
import { createVirid } from '@virid/core'
|
|
17
|
+
import { MainPlugin } from '@virid/main'
|
|
18
|
+
import { app } from 'electron'
|
|
19
|
+
const virid = createVirid()
|
|
20
|
+
.use(MainPlugin, { electronApp: app })
|
|
21
|
+
```
|
|
22
|
+
|
|
23
|
+
## 🛠️ @virid/main 核心 API 概览
|
|
24
|
+
|
|
25
|
+
### `ToRendererMessage`
|
|
26
|
+
|
|
27
|
+
- **功能**:一个特殊的消息基类,可被继承。所有继承自`ToRendererMessage`的Message将被发往渲染进程
|
|
28
|
+
- **逻辑**:该消息类型需要两个特殊标记,`__virid_target`标记了目的地,`__virid_messageType`标记了在目的地应该被还原为的Message类型
|
|
29
|
+
- **示例**:
|
|
30
|
+
|
|
31
|
+
```ts
|
|
32
|
+
//在主进程
|
|
33
|
+
import { ToRendererMessage } from '@virid/main'
|
|
34
|
+
// __virid_target=‘renderer’,说明消息需要发往windowId为renderer的渲染进程
|
|
35
|
+
// __virid_messageType: string = 'file-dialog',描述了在渲染进程,该消息将会重新变为的类型
|
|
36
|
+
export class RenderDialogMessage extends ToRendererMessage {
|
|
37
|
+
__virid_target: string = 'renderer'
|
|
38
|
+
__virid_messageType: string = 'file-dialog'
|
|
39
|
+
constructor(public path: string) {
|
|
40
|
+
super()
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
```
|
|
44
|
+
|
|
45
|
+
在渲染进程,可以通过`@FromIpc`结合`FromMainMessage`来实现对应的转换。详情见`@virid/renderer` 。
|
|
46
|
+
|
|
47
|
+
```ts
|
|
48
|
+
//在渲染进程
|
|
49
|
+
import { FromIpc, FromMainMessage } from '@virid/main'
|
|
50
|
+
//@FromIpc('file-dialog')表明当主进程发送一个ToRenderMessage,而且
|
|
51
|
+
// __virid_target: string = 'renderer'
|
|
52
|
+
// __virid_messageType: string = 'file-dialog'
|
|
53
|
+
// 时,这个消息将被自动投递到virid调度中心触发相应的渲染进程System
|
|
54
|
+
@FromIpc('file-dialog')
|
|
55
|
+
class ChooseBgImageMessage extends FromMainMessage {
|
|
56
|
+
constructor(public path: string) {
|
|
57
|
+
super()
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
```
|
|
61
|
+
|
|
62
|
+
### `FromRenderMessage/@FromRenderer()`
|
|
63
|
+
|
|
64
|
+
- **功能**:一个特殊的消息基类,可被继承。所有继承自`FromRenderMessage`的Message将在渲染进程发送消息时自动转换并投递到主进程的`virid`调度中心
|
|
65
|
+
- **逻辑**:`@FromRenderer()`接受一个字符串类型的ID,该ID与渲染进程中发送的`ToMainMessage`的ID应该相同。
|
|
66
|
+
- **示例**:
|
|
67
|
+
|
|
68
|
+
```ts
|
|
69
|
+
//在主进程
|
|
70
|
+
import { FromRenderer, FromRenderMessage } from '@virid/renderer'
|
|
71
|
+
// 发往渲染进程的消息
|
|
72
|
+
export class RenderDialogMessage extends ToRendererMessage {
|
|
73
|
+
__virid_target: string = 'renderer'
|
|
74
|
+
__virid_messageType: string = 'file-dialog'
|
|
75
|
+
constructor(public path: string) {
|
|
76
|
+
super()
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
// 将来自渲染进程且 __virid_messageType: string = 'file-dialog'的消息转换成主进程的OpenDialogMessage
|
|
80
|
+
@FromRenderer('open-dialog')
|
|
81
|
+
export class OpenDialogMessage extends FromRenderMessage {
|
|
82
|
+
constructor(
|
|
83
|
+
public options: {
|
|
84
|
+
title?: string
|
|
85
|
+
filters?: Array<{ name: string; extensions: string[] }>
|
|
86
|
+
properties?: Array<'openFile' | 'openDirectory' | 'multiSelections' | 'showHiddenFiles'>
|
|
87
|
+
}
|
|
88
|
+
) {
|
|
89
|
+
super()
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
export class WindowSystem {
|
|
94
|
+
@System()
|
|
95
|
+
async openDialog(@Message(OpenDialogMessage) message: OpenDialogMessage) {
|
|
96
|
+
// 调用原生对话框
|
|
97
|
+
const result = await dialog.showOpenDialog(message.senderWindow, message.options)
|
|
98
|
+
// 如果用户没有取消,并且确实选择了文件
|
|
99
|
+
if (!result.canceled && result.filePaths.length > 0) {
|
|
100
|
+
const selectedPath = result.filePaths[0]
|
|
101
|
+
return new RenderDialogMessage(selectedPath)
|
|
102
|
+
}
|
|
103
|
+
return
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
```
|
|
107
|
+
|
|
108
|
+
在渲染进程,可以通过`@FromIpc`结合`ToMainMessage`来实现对应的转换和消息的发送。详情见`@virid/renderer` 。
|
|
109
|
+
|
|
110
|
+
```ts
|
|
111
|
+
// 在渲染进程
|
|
112
|
+
import { ToMainMessage,FromIpc } from '@virid/renderer'
|
|
113
|
+
// 打开文件选择框
|
|
114
|
+
@FromIpc('file-dialog')
|
|
115
|
+
class ChooseBgImageMessage extends FromMainMessage {
|
|
116
|
+
constructor(public path: string) {
|
|
117
|
+
super()
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
//打开文件选择框
|
|
121
|
+
class OpenDialogMessage extends ToMainMessage {
|
|
122
|
+
__virid_target: string = 'main'
|
|
123
|
+
__virid_messageType: string = 'open-dialog'
|
|
124
|
+
constructor(
|
|
125
|
+
public options: {
|
|
126
|
+
title?: string
|
|
127
|
+
filters?: Array<{ name: string; extensions: string[] }>
|
|
128
|
+
properties?: Array<'openFile' | 'openDirectory' | 'multiSelections' | 'showHiddenFiles'>
|
|
129
|
+
}
|
|
130
|
+
) {
|
|
131
|
+
super()
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
```
|
|
136
|
+
|
|
137
|
+
|
|
138
|
+
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@virid/main",
|
|
3
3
|
"type": "module",
|
|
4
|
-
"version": "0.1.
|
|
4
|
+
"version": "0.1.1",
|
|
5
5
|
"description": "Electron main process adapter for Virid, responsible for sending, receiving, and broadcasting rendering process messages",
|
|
6
6
|
"author": "Ailrid",
|
|
7
7
|
"license": "Apache 2.0",
|
|
@@ -13,7 +13,7 @@
|
|
|
13
13
|
],
|
|
14
14
|
"repository": {
|
|
15
15
|
"type": "git",
|
|
16
|
-
"url": "git+https://github.com/
|
|
16
|
+
"url": "git+https://github.com/Ailrid/virid.git"
|
|
17
17
|
},
|
|
18
18
|
"main": "./dist/index.cjs",
|
|
19
19
|
"module": "./dist/index.js",
|
|
@@ -35,7 +35,7 @@
|
|
|
35
35
|
"dependencies": {},
|
|
36
36
|
"peerDependencies": {
|
|
37
37
|
"electron": "^30.0.0",
|
|
38
|
-
"@virid/core": "0.1.
|
|
38
|
+
"@virid/core": "0.1.1"
|
|
39
39
|
},
|
|
40
40
|
"scripts": {
|
|
41
41
|
"build": "tsup --config tsup.config.ts",
|