连接包

This commit is contained in:
王一之 2024-11-12 18:04:28 +08:00
parent d0d1793786
commit 93d41169e1
5 changed files with 99 additions and 58 deletions

View File

@ -19,7 +19,8 @@
"i18next": "^23.16.4",
"react": "^18.2.0",
"react-dom": "^18.2.0",
"react-i18next": "^15.1.0"
"react-i18next": "^15.1.0",
"uuid": "^11.0.3"
},
"devDependencies": {
"@eslint/compat": "^1.2.0",

View File

@ -1,18 +1,18 @@
import EventEmitter from 'eventemitter3';
import { Connect } from '.';
import EventEmitter from "eventemitter3";
import { IConnect, IServer } from ".";
export class ExtServer {
export class ExtServer implements IServer {
private EE: EventEmitter;
constructor() {
this.EE = new EventEmitter();
chrome.runtime.onConnect.addListener((port) => {
this.EE.emit('connect', new ExtConnect(port));
this.EE.emit("connect", port.name, new ExtConnect(port));
});
}
onConnect(callback: (con: Connect) => void) {
this.EE.on('connect', callback);
onConnect(callback: (eventName: string, con: IConnect) => void) {
this.EE.on("connect", callback);
}
}
@ -20,7 +20,7 @@ export function connect() {
return new ExtConnect(chrome.runtime.connect());
}
export class ExtConnect implements Connect {
export class ExtConnect implements IConnect {
private EE: EventEmitter;
private port: chrome.runtime.Port;
@ -28,10 +28,10 @@ export class ExtConnect implements Connect {
this.EE = new EventEmitter();
this.port = port;
port.onMessage.addListener((message) => {
this.EE.emit('message', message);
this.EE.emit("message", message);
});
port.onDisconnect.addListener(() => {
this.EE.emit('disconnect');
this.EE.emit("disconnect");
this.EE.removeAllListeners();
});
}
@ -41,11 +41,11 @@ export class ExtConnect implements Connect {
}
onMessage(callback: (message: unknown) => void) {
this.EE.on('message', callback);
this.EE.on("message", callback);
}
onDisconnect(callback: () => void) {
this.EE.on('disconnect', callback);
this.EE.on("disconnect", callback);
}
disconnect() {

View File

@ -1,11 +1,11 @@
import EventEmitter from 'eventemitter3';
import { v4 as uuidv4 } from 'uuid';
import EventEmitter from "eventemitter3";
import { v4 as uuidv4 } from "uuid";
export interface Server {
onConnect: (callback: (con: Connect) => void) => void;
export interface IServer {
onConnect: (callback: (eventName: string, con: IConnect) => void) => void;
}
export interface Connect {
export interface IConnect {
postMessage: (message: unknown) => void;
onMessage: (callback: (message: unknown) => void) => void;
onDisconnect: (callback: () => void) => void;
@ -13,51 +13,83 @@ export interface Connect {
}
// 消息通道, 通过连接封装消息通道
export class MessageChannel {
export class Server {
private EE: EventEmitter;
// 实例id
private id: string;
constructor(private connect: Connect) {
constructor(private connect: IServer) {
this.EE = new EventEmitter();
this.id = uuidv4();
connect.onMessage((message) => {
// 判断消息是否为连接消息
if (message && typeof message === 'object') {
const data = message as { type: string; id: string };
// 判断实例
if (data.id == this.id) {
switch (data.type) {
case 'connect':
// 建立连接
break;
case 'message':
// 发送消息
break;
}
}
}
this.connect.onConnect((eventName, con) => {
this.EE.emit(eventName, con);
});
}
// 发送一次消息并接收结果
async send(message: unknown) {
return new Promise((resolve) => {
const callback = (message: unknown) => {
this.connect.onMessage(callback);
resolve(message);
on(eventName: string, callback: (con: IConnect) => void) {
this.EE.on(eventName, callback);
}
}
export class Connect {
private EE: EventEmitter;
constructor(
private id: string | IConnect,
private con: IConnect
) {
this.EE = new EventEmitter();
if (arguments.length === 1) {
this.con = id as IConnect;
this.con.onMessage((message) => {
const data = message as { eventName: string; data: unknown[]; messageId: string };
data.data.push(this.callbackFunc(data.messageId));
this.EE.emit(data.eventName, ...data.data);
});
} else {
// 子连接
this.con.onMessage((message) => {
const data = message as { eventName: string; data: unknown; id: string };
if (data.eventName === "subcon") {
if (data.id !== this.id) return;
this.messageHandler(data.data);
}
});
}
this.con.onDisconnect(() => {
this.EE.emit("disconnect");
this.EE.removeAllListeners();
});
}
private callbackFunc(msgId: string): (...data: unknown[]) => void {
return function (...data: unknown[]) {
this.con.postMessage({ eventName: "callback", data, messageId: msgId });
};
this.connect.onMessage(callback);
this.connect.postMessage(message);
});
}
// 发送消息
postMessage(message: unknown) {
this.connect.postMessage({
type: 'message',
id: this.id,
data: message,
});
private messageHandler(data: unknown) {
const subData = data.data as { eventName: string; data: unknown[]; messageId: string };
subData.data.push(this.callbackFunc(subData.messageId));
this.EE.emit(subData.eventName, ...subData.data);
}
on(eventName: string, callback: (message: unknown) => void) {
this.EE.on(eventName, callback);
}
send(eventName: string, ...data: unknown[]) {
this.con.postMessage({ eventName, data });
}
emit(eventName: string, ...data: unknown[]) {
// 判断最后一个参数是否为函数
const callback = data.pop();
if (typeof callback !== "function") {
data.push(callback);
}
this.con.postMessage({ eventName, data, messageId: uuidv4() });
}
// 子连接
connect() {
return new Connect(uuidv4(), this.con);
}
}

3
pnpm-lock.yaml generated
View File

@ -26,6 +26,9 @@ importers:
react-i18next:
specifier: ^15.1.0
version: 15.1.0(i18next@23.16.4)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
uuid:
specifier: ^11.0.3
version: 11.0.3
devDependencies:
'@eslint/compat':
specifier: ^1.2.0

View File

@ -1,4 +1,5 @@
import { ExtServer } from "@Packages/message/extension";
import { Server } from "@Packages/message";
async function setupOffscreenDocument() {
// 创建运行后台脚本的沙盒环境
@ -20,7 +21,11 @@ async function main() {
// 初始化沙盒环境
await setupOffscreenDocument();
// 监听消息
new ExtServer();
const extServer = new ExtServer();
const server = new Server(extServer);
server.on("", (con) => {
});
}
main();