164 lines
4.4 KiB
TypeScript
164 lines
4.4 KiB
TypeScript
import { PageMeta } from "./types";
|
|
import { Socket } from "socket.io-client";
|
|
import { Update } from "@codemirror/collab";
|
|
import { ChangeSet, Text, Transaction } from "@codemirror/state";
|
|
|
|
import { CollabDocument, CollabEvents } from "./collab";
|
|
import { cursorEffect } from "./cursorEffect";
|
|
import { EventEmitter } from "../common/event";
|
|
import { Manifest } from "../common/manifest";
|
|
import { SystemJSON } from "../plugos/system";
|
|
|
|
export type SpaceEvents = {
|
|
connect: () => void;
|
|
pageCreated: (meta: PageMeta) => void;
|
|
pageChanged: (meta: PageMeta) => void;
|
|
pageDeleted: (name: string) => void;
|
|
pageListUpdated: (pages: Set<PageMeta>) => void;
|
|
loadSystem: (systemJSON: SystemJSON<any>) => void;
|
|
plugLoaded: (plugName: string, plug: Manifest) => void;
|
|
plugUnloaded: (plugName: string) => void;
|
|
} & CollabEvents;
|
|
|
|
export type KV = {
|
|
key: string;
|
|
value: any;
|
|
};
|
|
|
|
export class Space extends EventEmitter<SpaceEvents> {
|
|
socket: Socket;
|
|
reqId = 0;
|
|
allPages = new Set<PageMeta>();
|
|
|
|
constructor(socket: Socket) {
|
|
super();
|
|
this.socket = socket;
|
|
|
|
[
|
|
"connect",
|
|
"cursorSnapshot",
|
|
"pageCreated",
|
|
"pageChanged",
|
|
"pageDeleted",
|
|
"loadSystem",
|
|
"plugLoaded",
|
|
"plugUnloaded",
|
|
].forEach((eventName) => {
|
|
socket.on(eventName, (...args) => {
|
|
this.emit(eventName as keyof SpaceEvents, ...args);
|
|
});
|
|
});
|
|
this.wsCall("page.listPages").then((pages) => {
|
|
this.allPages = new Set(pages);
|
|
this.emit("pageListUpdated", this.allPages);
|
|
});
|
|
this.on({
|
|
pageCreated: (meta) => {
|
|
// Cannot reply on equivalence in set, need to iterate over all pages
|
|
let found = false;
|
|
for (const page of this.allPages) {
|
|
if (page.name === meta.name) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!found) {
|
|
this.allPages.add(meta);
|
|
console.log("New page created", meta);
|
|
this.emit("pageListUpdated", this.allPages);
|
|
}
|
|
},
|
|
pageDeleted: (name) => {
|
|
console.log("Page delete", name);
|
|
this.allPages.forEach((meta) => {
|
|
if (name === meta.name) {
|
|
this.allPages.delete(meta);
|
|
}
|
|
});
|
|
this.emit("pageListUpdated", this.allPages);
|
|
},
|
|
});
|
|
}
|
|
|
|
public wsCall(eventName: string, ...args: any[]): Promise<any> {
|
|
return new Promise((resolve, reject) => {
|
|
this.reqId++;
|
|
this.socket!.once(`${eventName}Resp${this.reqId}`, (err, result) => {
|
|
if (err) {
|
|
reject(new Error(err));
|
|
} else {
|
|
resolve(result);
|
|
}
|
|
});
|
|
this.socket!.emit(eventName, this.reqId, ...args);
|
|
});
|
|
}
|
|
|
|
async pushUpdates(
|
|
pageName: string,
|
|
version: number,
|
|
fullUpdates: readonly (Update & { origin: Transaction })[]
|
|
): Promise<boolean> {
|
|
if (this.socket) {
|
|
let updates = fullUpdates.map((u) => ({
|
|
clientID: u.clientID,
|
|
changes: u.changes.toJSON(),
|
|
cursors: u.effects?.map((e) => e.value),
|
|
}));
|
|
return this.wsCall("page.pushUpdates", pageName, version, updates);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
async pullUpdates(
|
|
pageName: string,
|
|
version: number
|
|
): Promise<readonly Update[]> {
|
|
let updates: Update[] = await this.wsCall(
|
|
"page.pullUpdates",
|
|
pageName,
|
|
version
|
|
);
|
|
return updates.map((u) => ({
|
|
changes: ChangeSet.fromJSON(u.changes),
|
|
effects: u.effects?.map((e) => cursorEffect.of(e.value)),
|
|
clientID: u.clientID,
|
|
}));
|
|
}
|
|
|
|
async listPages(): Promise<PageMeta[]> {
|
|
return Array.from(this.allPages);
|
|
}
|
|
|
|
async openPage(name: string): Promise<CollabDocument> {
|
|
this.reqId++;
|
|
let pageJSON = await this.wsCall("page.openPage", name);
|
|
|
|
return new CollabDocument(
|
|
Text.of(pageJSON.text),
|
|
pageJSON.version,
|
|
new Map(Object.entries(pageJSON.cursors))
|
|
);
|
|
}
|
|
|
|
async closePage(name: string): Promise<void> {
|
|
this.socket.emit("page.closePage", name);
|
|
}
|
|
|
|
async readPage(name: string): Promise<{ text: string; meta: PageMeta }> {
|
|
return this.wsCall("page.readPage", name);
|
|
}
|
|
|
|
async writePage(name: string, text: string): Promise<PageMeta> {
|
|
return this.wsCall("page.writePage", name, text);
|
|
}
|
|
|
|
async deletePage(name: string): Promise<void> {
|
|
return this.wsCall("page.deletePage", name);
|
|
}
|
|
|
|
async getPageMeta(name: string): Promise<PageMeta> {
|
|
return this.wsCall("page.getPageMeta", name);
|
|
}
|
|
}
|