mirror of
https://github.com/hexedtech/codemp-vscode.git
synced 2024-11-22 07:24:49 +01:00
feat: added core codemp commands for text sync
note that there's an awful OP_CACHE set to store our edits because we get them as events too... will have a ton of edge cases! apart from this text synchronization works pretty well, also autocompletions
This commit is contained in:
parent
7d497c224d
commit
340eafb432
1 changed files with 97 additions and 6 deletions
|
@ -2,17 +2,108 @@ const vscode = require("vscode");
|
||||||
const codemp = require("./codemp.node");
|
const codemp = require("./codemp.node");
|
||||||
|
|
||||||
var CLIENT = null
|
var CLIENT = null
|
||||||
|
var CONTROLLER
|
||||||
|
var OP_CACHE = new Set()
|
||||||
|
|
||||||
async function activate(context) {
|
async function activate(context) {
|
||||||
// This must match the command property in the package.json
|
context.subscriptions.push(
|
||||||
const commandID = "codemp.connect";
|
vscode.commands.registerCommand("codemp.connect", connect),
|
||||||
let disposable = vscode.commands.registerCommand(commandID, connect);
|
vscode.commands.registerCommand("codemp.share", share),
|
||||||
context.subscriptions.push(disposable);
|
vscode.commands.registerCommand("codemp.join", join),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
async function connect() {
|
async function connect() {
|
||||||
CLIENT = await codemp.connect("http://fantabos.co:50051")
|
let host = await vscode.window.showInputBox({prompt: "server host (default to http://fantabos.co:50051)"})
|
||||||
vscode.window.showInformationMessage("Connected to CodeMP!");
|
if (host === undefined) return // user cancelled with ESC
|
||||||
|
if (host.length == 0) host = "http://fantabos.co:50051"
|
||||||
|
CLIENT = await codemp.connect(host)
|
||||||
|
vscode.window.showInformationMessage(`Connected to codemp @[${host}]`);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function share() {
|
||||||
|
if (CLIENT === null) {
|
||||||
|
vscode.window.showErrorMessage("No connected client");
|
||||||
|
}
|
||||||
|
|
||||||
|
let path = await vscode.window.showInputBox({prompt: "buffer uri (default to file path)"})
|
||||||
|
if (path === undefined) return // user cancelled with ESC
|
||||||
|
if (path.length == 0) path = doc.uri.toString()
|
||||||
|
|
||||||
|
let doc = vscode.window.activeTextEditor.document;
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (!await CLIENT.create(path, doc.getText())) {
|
||||||
|
vscode.window.showErrorMessage("Could not share buffer");
|
||||||
|
}
|
||||||
|
|
||||||
|
await _attach(path)
|
||||||
|
|
||||||
|
vscode.window.showInformationMessage(`Shared document on buffer "${path}"`);
|
||||||
|
} catch (err) {
|
||||||
|
vscode.window.showErrorMessage("Error sharing: " + err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function join() {
|
||||||
|
if (CLIENT === null) {
|
||||||
|
vscode.window.showErrorMessage("No connected client");
|
||||||
|
}
|
||||||
|
|
||||||
|
let path = await vscode.window.showInputBox({prompt: "buffer uri"})
|
||||||
|
|
||||||
|
try {
|
||||||
|
let controller = await _attach(path)
|
||||||
|
|
||||||
|
vscode.window.showInformationMessage(`Joined buffer "${path}"`);
|
||||||
|
|
||||||
|
let editor = vscode.window.activeTextEditor
|
||||||
|
|
||||||
|
let range = new vscode.Range(
|
||||||
|
editor.document.positionAt(0),
|
||||||
|
editor.document.positionAt(editor.document.getText().length)
|
||||||
|
)
|
||||||
|
let content = controller.content()
|
||||||
|
OP_CACHE.add((range, content))
|
||||||
|
editor.edit(editBuilder => editBuilder.replace(range, content))
|
||||||
|
} catch (err) {
|
||||||
|
vscode.window.showErrorMessage("error joining " + err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function _attach(path) {
|
||||||
|
let doc = vscode.window.activeTextEditor.document;
|
||||||
|
CONTROLLER = await CLIENT.attach(path)
|
||||||
|
vscode.workspace.onDidChangeTextDocument(async (e) => {
|
||||||
|
if (e.document != doc) return
|
||||||
|
for (let change of e.contentChanges) {
|
||||||
|
if (OP_CACHE.has((change.range, change.text))) {
|
||||||
|
OP_CACHE.delete((change.range, change.text))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
await CONTROLLER.apply(change.rangeOffset, change.text, change.rangeOffset + change.rangeLength)
|
||||||
|
} catch (err) {
|
||||||
|
vscode.window.showErrorMessage("failed sending change: " + err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
let editor = vscode.window.activeTextEditor
|
||||||
|
CONTROLLER.set_callback((start, end) => {
|
||||||
|
// TODO only change affected document range
|
||||||
|
let content = CONTROLLER.content()
|
||||||
|
let range = new vscode.Range(
|
||||||
|
editor.document.positionAt(0),
|
||||||
|
editor.document.positionAt(editor.document.getText().length)
|
||||||
|
)
|
||||||
|
try {
|
||||||
|
OP_CACHE.add((range, content))
|
||||||
|
editor.edit(editBuilder => editBuilder.replace(range, content))
|
||||||
|
} catch (err) {
|
||||||
|
vscode.window.showErrorMessage("could not set buffer: " + err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return CONTROLLER
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
|
|
Loading…
Reference in a new issue