2023-06-13 18:47:05 +00:00
|
|
|
import { nanoid } from "https://esm.sh/nanoid@4.0.0";
|
|
|
|
import type { Editor } from "./editor.tsx";
|
|
|
|
|
|
|
|
const collabPingInterval = 2500;
|
|
|
|
|
|
|
|
export class CollabManager {
|
|
|
|
clientId = nanoid();
|
|
|
|
localCollabServer: string;
|
|
|
|
|
|
|
|
constructor(private editor: Editor) {
|
|
|
|
this.localCollabServer = location.protocol === "http:"
|
|
|
|
? `ws://${location.host}/.ws-collab`
|
|
|
|
: `wss://${location.host}/.ws-collab`;
|
|
|
|
editor.eventHook.addLocalListener(
|
|
|
|
"editor:pageLoaded",
|
|
|
|
(pageName, previousPage) => {
|
|
|
|
console.log("Page loaded", pageName, previousPage);
|
2023-07-02 09:25:32 +00:00
|
|
|
this.updatePresence(pageName).catch(console.error);
|
2023-06-13 18:47:05 +00:00
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
start() {
|
|
|
|
setInterval(() => {
|
|
|
|
this.updatePresence(this.editor.currentPage!).catch(console.error);
|
|
|
|
}, collabPingInterval);
|
|
|
|
}
|
|
|
|
|
2023-07-02 09:25:32 +00:00
|
|
|
async updatePresence(currentPage: string) {
|
2023-06-13 18:47:05 +00:00
|
|
|
try {
|
2023-07-02 09:25:32 +00:00
|
|
|
// This is signaled through an OPTIONS call on the file we have open
|
2023-06-13 18:47:05 +00:00
|
|
|
const resp = await this.editor.remoteSpacePrimitives.authenticatedFetch(
|
2023-07-02 09:25:32 +00:00
|
|
|
`${this.editor.remoteSpacePrimitives.url}/${currentPage}.md`,
|
2023-06-13 18:47:05 +00:00
|
|
|
{
|
2023-07-02 09:25:32 +00:00
|
|
|
method: "OPTIONS",
|
2023-06-13 18:47:05 +00:00
|
|
|
headers: {
|
2023-07-02 09:25:32 +00:00
|
|
|
"X-Client-Id": this.clientId,
|
2023-06-13 18:47:05 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
);
|
2023-07-02 09:25:32 +00:00
|
|
|
const collabId = resp.headers.get("X-Collab-Id");
|
|
|
|
// Not reading body at all, is that a problem?
|
2023-06-13 18:47:05 +00:00
|
|
|
|
|
|
|
if (this.editor.collabState && !this.editor.collabState.isLocalCollab) {
|
|
|
|
// We're in a remote collab mode, don't do anything
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// console.log("Collab ID", collabId);
|
|
|
|
const previousCollabId = this.editor.collabState?.token.split("/")[0];
|
|
|
|
if (!collabId && this.editor.collabState) {
|
|
|
|
// Stop collab
|
|
|
|
console.log("Stopping collab");
|
|
|
|
if (this.editor.collabState.path === `${currentPage}.md`) {
|
|
|
|
this.editor.flashNotification(
|
|
|
|
"Other users have left this page, switched back to single-user mode.",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
this.editor.stopCollab();
|
|
|
|
} else if (collabId && collabId !== previousCollabId) {
|
|
|
|
// Start collab
|
|
|
|
console.log("Starting collab");
|
|
|
|
this.editor.flashNotification(
|
|
|
|
"Opening page in multi-user mode.",
|
|
|
|
);
|
|
|
|
this.editor.startCollab(
|
|
|
|
this.localCollabServer,
|
|
|
|
`${collabId}/${currentPage}.md`,
|
|
|
|
this.editor.getUsername(),
|
|
|
|
true,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} catch (e: any) {
|
|
|
|
// console.error("Ping error", e);
|
|
|
|
if (
|
|
|
|
e.message.toLowerCase().includes("failed") && this.editor.collabState
|
|
|
|
) {
|
|
|
|
console.log("Offline, stopping collab");
|
|
|
|
this.editor.stopCollab();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|