@virid/renderer 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 +199 -1
- package/README.zh.md +169 -1
- package/package.json +3 -3
package/README.md
CHANGED
|
@@ -1 +1,199 @@
|
|
|
1
|
-
|
|
1
|
+
# @virid/renderer
|
|
2
|
+
|
|
3
|
+
**@virid/renderer** is a rendering process adapter for Electron applications. It provides automatic routing for `ToMainMessage` and `FromMainMessage`, along with main process reporting capabilities.
|
|
4
|
+
|
|
5
|
+
## 🌟 Core Design Philosophy
|
|
6
|
+
|
|
7
|
+
In **@virid/renderer**, messages sent from the rendering process to the main process are abstracted as `ToMainMessage`. These messages can be distributed directly via the dispatcher, traveling through IPC to the main process or other rendering processes to trigger the corresponding **System**.
|
|
8
|
+
|
|
9
|
+
Conversely, messages sent from the main process to the rendering process are abstracted as `FromMainMessage`. Once the main process sends a message, it is automatically delivered to the dispatcher to trigger the relevant rendering 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.
|
|
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 using the IDs of other rendering processes, or use `*` to broadcast messages.
|
|
14
|
+
|
|
15
|
+
------
|
|
16
|
+
|
|
17
|
+
## 🔌Enable plugins
|
|
18
|
+
|
|
19
|
+
```ts
|
|
20
|
+
import { createVirid } from '@virid/core'
|
|
21
|
+
import { RenderPlugin } from '@virid/renderer'
|
|
22
|
+
const app = createVirid()
|
|
23
|
+
//Each window needs to be assigned a unique windowId to report to the main process
|
|
24
|
+
app.use(RenderPlugin, {
|
|
25
|
+
windowId: 'renderer'
|
|
26
|
+
})
|
|
27
|
+
```
|
|
28
|
+
|
|
29
|
+
## 🛠️ @virid/renderer Core API Overview
|
|
30
|
+
|
|
31
|
+
### ToMainMessage
|
|
32
|
+
|
|
33
|
+
- **Function**: A specialized message base class designed for inheritance. All messages inheriting from `ToMainMessage` will be dispatched to other processes.
|
|
34
|
+
- **Logic**: This message type requires two specific metadata tags:
|
|
35
|
+
- `__virid_target`: Defines the destination.
|
|
36
|
+
- `__virid_messageType`: Defines the message type it should be restored to upon reaching the destination.
|
|
37
|
+
- Example
|
|
38
|
+
|
|
39
|
+
**In the Rendering Process:**
|
|
40
|
+
|
|
41
|
+
```ts
|
|
42
|
+
import { ToMainMessage } from '@virid/renderer'
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* - __virid_target = 'main': Indicates the message is bound for the Main Process.
|
|
46
|
+
* - If __virid_target = '*': The message will be broadcast to all rendering processes.
|
|
47
|
+
* - __virid_messageType: Describes the class type the message will transform back into
|
|
48
|
+
* once it arrives at the Main Process.
|
|
49
|
+
*/
|
|
50
|
+
|
|
51
|
+
export class CloseWindowMessage extends ToMainMessage {
|
|
52
|
+
__virid_target = 'main'
|
|
53
|
+
__virid_messageType: string = 'close-window'
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
export class MinimizeWindowMessage extends ToMainMessage {
|
|
57
|
+
__virid_target = 'main'
|
|
58
|
+
__virid_messageType: string = 'minimize-window'
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
export class MaximizeWindowMessage extends ToMainMessage {
|
|
62
|
+
__virid_target = 'main'
|
|
63
|
+
__virid_messageType: string = 'maximize-window'
|
|
64
|
+
}
|
|
65
|
+
```
|
|
66
|
+
|
|
67
|
+
In the **Main Process**, you can use the `@FromRenderer` decorator in conjunction with `FromRendererMessage` to handle the conversion. For more details, refer to `@virid/main`.
|
|
68
|
+
|
|
69
|
+
**In the Main Process:**
|
|
70
|
+
|
|
71
|
+
```ts
|
|
72
|
+
import { FromRendererMessage, FromRenderer } from '@virid/main'
|
|
73
|
+
|
|
74
|
+
// Each ID here maps to the __virid_messageType defined in the rendering process
|
|
75
|
+
@FromRenderer('close-window')
|
|
76
|
+
export class CloseWindowMessage extends FromRendererMessage {}
|
|
77
|
+
|
|
78
|
+
@FromRenderer('minimize-window')
|
|
79
|
+
export class MinimizeWindowMessage extends FromRendererMessage {}
|
|
80
|
+
|
|
81
|
+
@FromRenderer('maximize-window')
|
|
82
|
+
export class MaximizeWindowMessage extends FromRendererMessage {}
|
|
83
|
+
|
|
84
|
+
/**
|
|
85
|
+
* WindowSystem: Implements Minimize, Maximize, and Close functionality for all windows.
|
|
86
|
+
* senderWindow: Refers to the specific window instance that dispatched the message.
|
|
87
|
+
*/
|
|
88
|
+
export class WindowSystem {
|
|
89
|
+
@System()
|
|
90
|
+
closeWindow(@Message(CloseWindowMessage) message: CloseWindowMessage) {
|
|
91
|
+
message.senderWindow.close()
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
@System()
|
|
95
|
+
minimizeWindow(@Message(MinimizeWindowMessage) message: MinimizeWindowMessage) {
|
|
96
|
+
message.senderWindow.minimize()
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
@System()
|
|
100
|
+
maximizeWindow(@Message(MaximizeWindowMessage) message: MaximizeWindowMessage) {
|
|
101
|
+
if (message.senderWindow.isMaximized()) {
|
|
102
|
+
message.senderWindow.unmaximize()
|
|
103
|
+
} else {
|
|
104
|
+
message.senderWindow.maximize()
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
```
|
|
109
|
+
|
|
110
|
+
### FromMainMessage / @FromIpc()
|
|
111
|
+
|
|
112
|
+
- **Function**: A specialized message base class for inheritance. Any message inheriting from `FromMainMessage` is automatically converted and dispatched to the current process's **Virid Dispatcher** when sent from the Main Process.
|
|
113
|
+
- **Logic**: The `@FromIpc()` decorator accepts a string-based ID. This ID must match the `__virid_messageType` of the `ToRendererMessage` sent from the Main Process.
|
|
114
|
+
|
|
115
|
+
- Example:
|
|
116
|
+
|
|
117
|
+
**In the Rendering Process:**
|
|
118
|
+
|
|
119
|
+
```ts
|
|
120
|
+
import { FromIpc, FromMainMessage, ToMainMessage } from '@virid/renderer'
|
|
121
|
+
|
|
122
|
+
// This class handles the incoming file path from the Main Process
|
|
123
|
+
@FromIpc('file-dialog')
|
|
124
|
+
class ChooseBgImageMessage extends FromMainMessage {
|
|
125
|
+
constructor(public path: string) {
|
|
126
|
+
super()
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
// This class initiates the request to the Main Process
|
|
131
|
+
class OpenDialogMessage extends ToMainMessage {
|
|
132
|
+
__virid_target: string = 'main'
|
|
133
|
+
__virid_messageType: string = 'open-dialog'
|
|
134
|
+
|
|
135
|
+
constructor(
|
|
136
|
+
public options: {
|
|
137
|
+
title?: string
|
|
138
|
+
filters?: Array<{ name: string; extensions: string[] }>
|
|
139
|
+
properties?: Array<'openFile' | 'openDirectory' | 'multiSelections' | 'showHiddenFiles'>
|
|
140
|
+
}
|
|
141
|
+
) {
|
|
142
|
+
super()
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
```
|
|
146
|
+
|
|
147
|
+
In the **Main Process**, you use `@FromRenderer` and `FromRendererMessage` to handle the incoming request. For more details, refer to `@virid/main`.
|
|
148
|
+
|
|
149
|
+
**In the Main Process:**
|
|
150
|
+
|
|
151
|
+
```ts
|
|
152
|
+
import { FromRendererMessage, FromRenderer, ToRendererMessage } from '@virid/main'
|
|
153
|
+
|
|
154
|
+
/**
|
|
155
|
+
* @FromRenderer('open-dialog') ensures that when the Rendering Process
|
|
156
|
+
* dispatches an OpenDialogMessage, the Main Process's corresponding
|
|
157
|
+
* System will be triggered automatically.
|
|
158
|
+
*/
|
|
159
|
+
@FromRenderer('open-dialog')
|
|
160
|
+
export class OpenDialogMessage extends FromRendererMessage {
|
|
161
|
+
constructor(
|
|
162
|
+
public options: {
|
|
163
|
+
title?: string
|
|
164
|
+
filters?: Array<{ name: string; extensions: string[] }>
|
|
165
|
+
properties?: Array<'openFile' | 'openDirectory' | 'multiSelections' | 'showHiddenFiles'>
|
|
166
|
+
}
|
|
167
|
+
) {
|
|
168
|
+
super()
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
/**
|
|
173
|
+
* RenderDialogMessage is used to send the result back.
|
|
174
|
+
* - __virid_target = 'renderer' routes it back to the UI.
|
|
175
|
+
* - __virid_messageType = 'file-dialog' maps it to ChooseBgImageMessage in the renderer.
|
|
176
|
+
*/
|
|
177
|
+
export class RenderDialogMessage extends ToRendererMessage {
|
|
178
|
+
__virid_target: string = 'renderer'
|
|
179
|
+
__virid_messageType: string = 'file-dialog'
|
|
180
|
+
constructor(public path: string) {
|
|
181
|
+
super()
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
export class WindowSystem {
|
|
186
|
+
@System()
|
|
187
|
+
async openDialog(@Message(OpenDialogMessage) message: OpenDialogMessage) {
|
|
188
|
+
// Invoke the native Electron dialog
|
|
189
|
+
const result = await dialog.showOpenDialog(message.senderWindow, message.options)
|
|
190
|
+
|
|
191
|
+
// If the user didn't cancel and selected a file, return the result message
|
|
192
|
+
if (!result.canceled && result.filePaths.length > 0) {
|
|
193
|
+
const selectedPath = result.filePaths[0]
|
|
194
|
+
return new RenderDialogMessage(selectedPath)
|
|
195
|
+
}
|
|
196
|
+
return
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
```
|
package/README.zh.md
CHANGED
|
@@ -1 +1,169 @@
|
|
|
1
|
-
|
|
1
|
+
# @virid/renderer
|
|
2
|
+
|
|
3
|
+
`@virid/renderer` 是electron应用的渲染进程适配器,提供`ToMainMessage`和`FromMainMessage`的自动路由功能与主进程报道功能。
|
|
4
|
+
|
|
5
|
+
## 🌟 核心设计理念
|
|
6
|
+
|
|
7
|
+
在 `@virid/renderer` 中,渲染进程朝主进程发送消息被抽象为`ToMainMessage`,该消息可以直接通过调度器分发,消息将会通过IPC直接传送到主进程或其他渲染进程并触发对应的`System`。主进程朝渲染进程发送的消息被抽象为`FromMainMessage`,一旦主进程发送了对应的消息,`FromMainMessage`将会自动被投递到调度器触发相应的渲染进程`System`。
|
|
8
|
+
|
|
9
|
+
- **路由自动注册**:渲染进程窗口开启后,自动向主进程注册自身,以后所有的消息都将携带自身的窗口信息。
|
|
10
|
+
- **类型恢复**:消息在通过IPC通道后,恢复成真正的Message类重新进入`@virid/core`并被调度器识别并分发。实现不同进程的位置无关性。
|
|
11
|
+
- **消息定向与广播**:可以通过其他渲染进程的ID来实现定向通信,或者使用*来广播消息。
|
|
12
|
+
|
|
13
|
+
## 🔌启用插件
|
|
14
|
+
|
|
15
|
+
```ts
|
|
16
|
+
import { createVirid } from '@virid/core'
|
|
17
|
+
import { RenderPlugin } from '@virid/renderer'
|
|
18
|
+
const app = createVirid()
|
|
19
|
+
//需要给每个窗口指定一个唯一的windowId来向主进程报道
|
|
20
|
+
app.use(RenderPlugin, {
|
|
21
|
+
windowId: 'renderer'
|
|
22
|
+
})
|
|
23
|
+
```
|
|
24
|
+
|
|
25
|
+
## 🛠️ @virid/renderer 核心 API 概览
|
|
26
|
+
|
|
27
|
+
### `ToMainMessage`
|
|
28
|
+
|
|
29
|
+
- **功能**:一个特殊的消息基类,可被继承。所有继承自`ToMainMessage`的Message将被发往其他进程
|
|
30
|
+
- **逻辑**:该消息类型需要两个特殊标记,`__virid_target`标记了目的地,`__virid_messageType`标记了在目的地应该被还原为的Message类型
|
|
31
|
+
- **示例**:
|
|
32
|
+
|
|
33
|
+
```ts
|
|
34
|
+
//在渲染进程
|
|
35
|
+
import { ToMainMessage } from '@virid/renderer'
|
|
36
|
+
// __virid_target=‘main’,说明消息需要发往主进程
|
|
37
|
+
// 当指定__virid_target=‘*’时,消息将会对所有渲染进程广播
|
|
38
|
+
// __virid_messageType: string = 'close-window',描述了在主进程,该消息将会重新变为的类型
|
|
39
|
+
|
|
40
|
+
export class CloseWindowMessage extends ToMainMessage {
|
|
41
|
+
__virid_target = 'main'
|
|
42
|
+
__virid_messageType: string = 'close-window'
|
|
43
|
+
}
|
|
44
|
+
export class MinimizeWindowMessage extends ToMainMessage {
|
|
45
|
+
__virid_target = 'main'
|
|
46
|
+
__virid_messageType: string = 'minimize-window'
|
|
47
|
+
}
|
|
48
|
+
export class MaximizeWindowMessage extends ToMainMessage {
|
|
49
|
+
__virid_target = 'main'
|
|
50
|
+
__virid_messageType: string = 'maximize-window'
|
|
51
|
+
}
|
|
52
|
+
```
|
|
53
|
+
|
|
54
|
+
在主进程,可以通过`@FromRender`结合`FromRenderMessage`来实现对应的转换。详情见`@virid/main` 。
|
|
55
|
+
|
|
56
|
+
```ts
|
|
57
|
+
//在主进程
|
|
58
|
+
import { FromRenderMessage, FromRenderer, ToRenderMessage } from '@virid/main'
|
|
59
|
+
//这里的每个id都和渲染进程的消息对应
|
|
60
|
+
@FromRenderer('close-window')
|
|
61
|
+
export class CloseWindowMessage extends FromRendererMessage {}
|
|
62
|
+
@FromRenderer('minimize-window')
|
|
63
|
+
export class MinimizeWindowMessage extends FromRendererMessage {}
|
|
64
|
+
@FromRenderer('maximize-window')
|
|
65
|
+
export class MaximizeWindowMessage extends FromRendererMessage {}
|
|
66
|
+
|
|
67
|
+
// 通过这三个System,可以实现所有窗口的最小化、最大化、关闭功能
|
|
68
|
+
// senderWindow为发送消息的窗口本身
|
|
69
|
+
export class WindowSystem {
|
|
70
|
+
@System()
|
|
71
|
+
closeWindow(@Message(CloseWindowMessage) message: CloseWindowMessage) {
|
|
72
|
+
message.senderWindow.close()
|
|
73
|
+
}
|
|
74
|
+
@System()
|
|
75
|
+
minimizeWindow(@Message(MinimizeWindowMessage) message: MinimizeWindowMessage) {
|
|
76
|
+
message.senderWindow.minimize()
|
|
77
|
+
}
|
|
78
|
+
@System()
|
|
79
|
+
maximizeWindow(@Message(MaximizeWindowMessage) message: MaximizeWindowMessage) {
|
|
80
|
+
if (message.senderWindow.isMaximized()) {
|
|
81
|
+
message.senderWindow.unmaximize()
|
|
82
|
+
} else {
|
|
83
|
+
message.senderWindow.maximize()
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
```
|
|
88
|
+
|
|
89
|
+
### `FromMainMessage/@FromIpc()`
|
|
90
|
+
|
|
91
|
+
- **功能**:一个特殊的消息基类,可被继承。所有继承自`FromMainMessage`的Message将在主进程发送消息时自动转换并投递到当前进程的`virid`调度中心
|
|
92
|
+
- **逻辑**:`@FromIpc()`接受一个字符串类型的ID,该ID与主进程中发送的`ToRendererMessage`的ID应该相同。
|
|
93
|
+
- **示例**:
|
|
94
|
+
|
|
95
|
+
```ts
|
|
96
|
+
//在渲染进程
|
|
97
|
+
import { FromIpc, FromMainMessage, ToMainMessage } from '@virid/renderer'
|
|
98
|
+
|
|
99
|
+
@FromIpc('file-dialog')
|
|
100
|
+
class ChooseBgImageMessage extends FromMainMessage {
|
|
101
|
+
constructor(public path: string) {
|
|
102
|
+
super()
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
class OpenDialogMessage extends ToMainMessage {
|
|
107
|
+
__virid_target: string = 'main'
|
|
108
|
+
__virid_messageType: string = 'open-dialog'
|
|
109
|
+
constructor(
|
|
110
|
+
public options: {
|
|
111
|
+
title?: string
|
|
112
|
+
filters?: Array<{ name: string; extensions: string[] }>
|
|
113
|
+
properties?: Array<'openFile' | 'openDirectory' | 'multiSelections' | 'showHiddenFiles'>
|
|
114
|
+
}
|
|
115
|
+
) {
|
|
116
|
+
super()
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
```
|
|
120
|
+
|
|
121
|
+
在主进程,可以通过`@FromRender`结合`FromRenderMessage`来实现对应的转换。详情见`@virid/main` 。
|
|
122
|
+
|
|
123
|
+
```ts
|
|
124
|
+
//在主进程
|
|
125
|
+
import { FromRendererMessage, FromRenderer, ToRenderMessage } from '@virid/main'
|
|
126
|
+
|
|
127
|
+
// @FromRender('open-dialog')表明,当上面的渲染进程调用OpenDialogMessage.send(options)时
|
|
128
|
+
// 主进程的OpenDialogMessage将被自动投递,因此下面的openDialog System将被virid自动调用
|
|
129
|
+
// 当openDialog执行完毕,将返回一个RenderDialogMessage,该RenderDialogMessage标记了目的地与类型
|
|
130
|
+
// 其会转换为渲染进程的OpenDialogMessage并触发渲染进程的System或者Listener执行
|
|
131
|
+
|
|
132
|
+
@FromRenderer('open-dialog')
|
|
133
|
+
export class OpenDialogMessage extends FromRendererMessage {
|
|
134
|
+
constructor(
|
|
135
|
+
public options: {
|
|
136
|
+
title?: string
|
|
137
|
+
filters?: Array<{ name: string; extensions: string[] }>
|
|
138
|
+
properties?: Array<'openFile' | 'openDirectory' | 'multiSelections' | 'showHiddenFiles'>
|
|
139
|
+
}
|
|
140
|
+
) {
|
|
141
|
+
super()
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
export class RenderDialogMessage extends ToRendererMessage {
|
|
146
|
+
__virid_target: string = 'renderer'
|
|
147
|
+
__virid_messageType: string = 'file-dialog'
|
|
148
|
+
constructor(public path: string) {
|
|
149
|
+
super()
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
export class WindowSystem {
|
|
154
|
+
@System()
|
|
155
|
+
async openDialog(@Message(OpenDialogMessage) message: OpenDialogMessage) {
|
|
156
|
+
// 调用原生对话框
|
|
157
|
+
const result = await dialog.showOpenDialog(message.senderWindow, message.options)
|
|
158
|
+
// 如果用户没有取消,并且确实选择了文件
|
|
159
|
+
if (!result.canceled && result.filePaths.length > 0) {
|
|
160
|
+
const selectedPath = result.filePaths[0]
|
|
161
|
+
return new RenderDialogMessage(selectedPath)
|
|
162
|
+
}
|
|
163
|
+
return
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
```
|
|
167
|
+
|
|
168
|
+
|
|
169
|
+
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@virid/renderer",
|
|
3
3
|
"type": "module",
|
|
4
|
-
"version": "0.1.
|
|
4
|
+
"version": "0.1.1",
|
|
5
5
|
"description": "Electron renderer process adapter for virid, responsible for forwarding and receiving messages from the main process",
|
|
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",
|
|
@@ -34,7 +34,7 @@
|
|
|
34
34
|
},
|
|
35
35
|
"dependencies": {},
|
|
36
36
|
"peerDependencies": {
|
|
37
|
-
"@virid/core": "0.1.
|
|
37
|
+
"@virid/core": "0.1.1"
|
|
38
38
|
},
|
|
39
39
|
"scripts": {
|
|
40
40
|
"build": "tsup --config tsup.config.ts",
|