const EditSession = { active: false, hosting: false, BBKey: '1h3sq3hoj6vfkh', ip: 'blckbn.ch', clients: {}, placeholder_names: ['R2D2', 'Tin Man', 'C3PO', 'WALL-E', 'EVE', 'BB-8', 'B1 Battle Droid', 'ASIMO', 'Atlas'], start() { if (EditSession.active) return; var peer = EditSession.peer = new Peer({ key: 'edit_session', host: EditSession.ip, port: 9000, path: '/sessions', secure: true }); EditSession.username = $('#edit_session_username').val() || EditSession.placeholder_names.random(); settings.username.value = EditSession.username; peer.on('open', (token) => { EditSession.hosting = true; Prop.session = true; EditSession.setState(true); EditSession.self = new EditSession.Client({ id: EditSession.peer.id, name: EditSession.username, hosting: true }) $('#edit_session_token').val(token) EditSession.token = token; Clipbench.setText(token) Blockbench.dispatchEvent('create_session', {peer, token}) }) peer.on('connection', (conn) => { conn.on('open', function() { var client = new EditSession.Client({ id: conn.peer, conn: conn, name: conn.metadata.username, hosting: false }) Chat.processMessage({text: tl('edit_session.joined', [client.name]), color: 'green'}) Blockbench.showQuickMessage(tl('edit_session.joined', [client.name])) //New Login client.send({ type: 'chat_message', data: {text: tl('edit_session.welcome', [EditSession.username]), color: 'yellow'} }) var model = Codecs.project.compile({uuids: true, bitmaps: true, backup: true, history: true}) client.send({ type: 'init_model', fromHost: EditSession.hosting, sender: EditSession.peer.id, data: model }) }) }) peer.on('error', error => { console.error('Error in edit session:', error) }) }, join() { if (EditSession.active) return; var token = $('#edit_session_token').val() EditSession.hosting = false; EditSession.peer = new Peer({ key: 'edit_session', host: EditSession.ip, port: 9000, path: '/sessions', secure: true }); EditSession.peer.on('open', function() { EditSession.username = $('#edit_session_username').val() || EditSession.placeholder_names.random(); settings.username.value = EditSession.username; if (!token || !EditSession._matchToken(token)) { Blockbench.showMessageBox({ translateKey: 'invalid_session', icon: 'cloud_off', }, result => { showDialog('edit_sessions'); }) return; } EditSession.token = token; var conn = EditSession.peer.connect(token, {metadata: {username: EditSession.username}}); conn.on('error', (err) => { console.error('peer join error', err) Blockbench.showMessageBox({ translateKey: 'invalid_session', icon: 'cloud_off', }) EditSession.quit() }) conn.on('open', () => { hideDialog() EditSession.host = conn; EditSession.setState(true); EditSession.initConnection(conn) Blockbench.dispatchEvent('join_session', {conn}) }) }) EditSession.peer.on('error', error => { console.error('Error in edit session:', error) }) }, quit() { if (!EditSession.active) return; Blockbench.dispatchEvent('quit_session', {}) if (EditSession.hosting) { EditSession.sendAll('command', 'quit_session') } else { EditSession.host.close() } setTimeout(function() { EditSession.setState(false) Chat.history.purge() EditSession.peer.destroy() Prop.session = false; Prop.connections = 0; Blockbench.showQuickMessage('edit_session.quit_session', 1500) }, 400) }, setState(active) { EditSession.active = active; $('#edit_session_username, #edit_session_token').attr('readonly', active) if (active) { $('.edit_session_inactive').hide() $('.edit_session_active').show() $('#edit_session_status').text(EditSession.hosting ? tl('edit_session.hosting') : tl('edit_session.connected')) $('#edit_session_copy_button .tooltip').text(tl('action.copy')) } else { EditSession.hosting = false; $('.edit_session_active').hide() $('.edit_session_inactive').show() $('#edit_session_copy_button .tooltip').text(tl('action.paste')) $('#edit_session_token').val('') } updateInterface() }, dialog() { showDialog('edit_sessions'); if (!EditSession.active) { var username = settings.username.value; if (isApp) { var token = clipboard.readText() if (EditSession._matchToken(token) && !$('#edit_session_token').val()) { $('#edit_session_token').val(token) } if (!username) { username = process.env.USERNAME } } if (!username) username = EditSession.placeholder_names.random() if (username) { $('#edit_session_username').val(username) } } }, copyToken() { var input = $('#edit_session_token') if (EditSession.active) { input.focus() document.execCommand('selectAll') document.execCommand('copy') } else { if (isApp) { var token = clipboard.readText() if (EditSession._matchToken(token)) { $('#edit_session_token').val(token) } } else { navigator.clipboard.readText().then((token) => { if (EditSession._matchToken(token)) { $('#edit_session_token').val(token) } }) } } }, initNewModel(force) { if (EditSession.active && EditSession.hosting) { var model = Codecs.project.compile({uuids: true, bitmaps: true, backup: true, flag: force ? 'force' : null}) if (force) { model.flag = 'force' } EditSession.sendAll('init_model', model) } }, initConnection(conn) { conn.on('data', EditSession.receiveData) }, sendAll(type, data) { var tag = {type, data} Blockbench.dispatchEvent('send_session_data', tag) for (var key in EditSession.peer.connections) { var conns = EditSession.peer.connections[key]; conns.forEach(conn => { conn.send({ type: tag.type, fromHost: EditSession.hosting, sender: EditSession.peer.id, data: tag.data }); }) } if (Blockbench.hasFlag('log_session')) { console.log('Sent Data:', type, data) } }, sendEdit(entry) { var new_entry = { before: omitKeys(entry.before, ['aspects']), post: omitKeys(entry.post, ['aspects']), save_history: entry.save_history, action: entry.action } EditSession.sendAll('edit', JSON.stringify(new_entry)) }, receiveData(tag) { if (Blockbench.hasFlag('log_session')) { console.log('Received Data:', tag) } if (EditSession.hosting && !tag.hostOnly && Object.keys(EditSession.peer.connections).length > 1) { //Redistribute for (var id in EditSession.peer.connections) { if (id !== tag.sender) { EditSession.peer.connections[id][0].send(tag); } } } var data = tag.data; if (typeof data === 'string' && (data.includes('"') || data.includes('['))) { try { data = tag.data = JSON.parse(data) } catch (err) { console.log(err) return; } } Blockbench.dispatchEvent('receive_session_data', tag) if (tag.type === 'edit') { Undo.remoteEdit(data) } else if (tag.type === 'init_model') { newProject(data.meta.type||'free', data.flag === 'force'); Codecs.project.parse(data); } else if (tag.type === 'command') { switch (data) { case 'undo': Undo.undo(true); break; case 'redo': Undo.redo(true); break; case 'quit_session': EditSession.quit(); break; } } else if (tag.type === 'change_project_meta') { for (var key in data) { Project[key] = data[key]; } } else if (tag.type === 'chat_input' && EditSession.hosting) { Chat.processMessage(tag.data) } else if (tag.type === 'chat_message') { Chat.addMessage(tag.data) } }, updateClientCount() { Prop.connections = Object.keys(EditSession.clients).length-1 }, _matchToken(token) { return !!(token.length === 16 && token.match(/[a-z0-9]{16}/)) } } EditSession.Client = class { constructor(data) { var scope = this; this.id = data.id; this.hosting = data.hosting; this.conn = data.conn; this.name = data.name; EditSession.clients[this.id] = this; EditSession.updateClientCount() if (this.conn) { EditSession.initConnection(this.conn) this.conn.on('close', () => { scope.disconnect() }) this.conn.on('error', (e) => { scope.disconnect() this.conn.close() console.error(e) }) Blockbench.dispatchEvent('user_joins_session', this) } } send(tag) { this.conn.send(tag) } disconnect(e) { if (!EditSession.clients[this.id]) return; Blockbench.dispatchEvent('user_leaves_session', this) delete EditSession.peer.connections[this.conn.peer]; delete EditSession.clients[this.id]; EditSession.updateClientCount(); Chat.processMessage({text: tl('edit_session.left', [this.name]), color: 'red'}) Blockbench.showQuickMessage(tl('edit_session.left', [this.name])) } } const Chat = { history: [], maxlength: 512, send(text) { if (typeof text !== 'string') { text = $('input#chat_input').val() $('input#chat_input').val('') } if (!text) return; Chat.processMessage({ author: EditSession.username, text: text, sender: EditSession.peer.id }) }, addMessage(message) { if (!(message instanceof Chat.Message)) { message = new Chat.Message(message) } if (!message.text) return; Chat.history.push(message) Vue.nextTick(() => { $('#chat_history').scrollTop(10000) }) if (!document.hasFocus() && !message.self) { Blockbench.notification(message.author ? message.author+':' : 'Chat', message.text) } return message; }, processMessage(data) { if (!EditSession.hosting) { EditSession.host.send({ type: 'chat_input', data, sender: EditSession.peer.id }) return; } //Host Only Blockbench.dispatchEvent('process_chat_message', data) EditSession.sendAll('chat_message', data) Chat.addMessage(data) } }; Chat.Message = class { constructor(data) { this.author = data.author||''; this.author = this.author.substr(0, 64) this.sender = data.sender this.self = data.sender == EditSession.peer.id; this.text = data.text.substr(0, Chat.maxlength)||''; this.html = this.text.replace(//g, '>'); this.html = this.html.replace(/https?:\/\/(www\.)?[-a-zA-Z0-9@:%._\+~#=]{1,256}\.[a-zA-Z0-9()]{1,6}\b([-a-zA-Z0-9()@:%_\+.~#?&//=]*)/g, (text, i) => { return `${text}`; }) var date = new Date(); this.timestamp = date.getTimestamp() this.toString = () => (this.author + ': ' + this.content); //Color this.color = data.color switch (this.color) { case 'red': this.hex = '#ff4158'; break; case 'orange': this.hex = '#ff6f10'; break; case 'yellow': this.hex = '#ffd100'; break; case 'green': this.hex = '#00eb3b'; break; case 'blue': this.hex = '#08a3ff'; break; } } toObject() { return { author: this.author, text: this.text, color: this.color, } } showAuthor() { if (!this.author) return false; var this_i = Chat.history.indexOf(this); var prev = Chat.history[this_i-1]; return (!prev) || (prev.author !== this.author); } } BARS.defineActions(function() { new Action('edit_session', { icon: 'people', category: 'blockbench', click: EditSession.dialog }) }) Interface.definePanels(function() { Interface.Panels.chat = new Panel({ id: 'chat', icon: 'chat', condition: {method() {return EditSession.active}}, toolbars: {}, onResize: t => { }, component: { data() {return Chat}, template: `
send
` } }) })