-
Notifications
You must be signed in to change notification settings - Fork 12
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Andre Marques
authored and
Andre Marques
committed
Sep 18, 2020
1 parent
42010e9
commit 0aa8d20
Showing
1 changed file
with
116 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1 +1,116 @@ | ||
# tinymce-collaborative-editing-plugin | ||
## Codepen | ||
|
||
[Example available on CodePen](https://codepen.io/thealmarques/full/YzqOrWV) | ||
|
||
# How to build a real-time collaborative plugin in TinyMCE | ||
|
||
Introduction | ||
|
||
Just like any other developer out there, I had a project idea that I wanted to build, and when I finally had the time to research technologies that I needed I came across an issue. I needed a stylish text editor that allowed me to have multiple users edit at the same time. | ||
|
||
After some time (actually, it was the first that appeared, but it’s comprehensible, it’s super cool) I found TinyMCE. This WYSIWYG editor had a lot of free plugins, but none of them was a collaborative plugin. I found one but it was a paid plugin (more than 10$ a month). In summary, I had a new project idea, create a free collaborative plugin for TinyMCE. | ||
|
||
## Create a plugin | ||
|
||
Following the great, but somehow confuse TinyMCE documentation, it’s very quick to get started with a plugin example. Using the Yeoman generator you create a sample plugin project where you can start testing and deploying your plugin. | ||
|
||
`npm install --global yo generator-tinymce | ||
yo tinymce` | ||
|
||
…and to start the project | ||
|
||
npm start | ||
|
||
I suggest using VSCode with Live Reload server to open the demo HTML file where you can access your plugin and its changes. | ||
|
||
## The Big Picture | ||
|
||
Now it comes the fun part, what it takes to create a collaborative plugin? So if you think about it, we obviously need a way of connecting multiple clients in your TinyMCE editor, and in each one of these clients we have to communicate with a server when where we’ll bind several events, the client opens the TinyMCE editor, the client clicks in the editor and changes cursor position, client selects text, and, the clients adds new input. Each one of these events will have to be spread across all connected clients in the server. | ||
|
||
Some of you, by reading this will think about Socket.io, and you are right, this was the approach that I’ve taken, although you could use other alternatives (**SignalR, SocketCluster**, **Firebase, etc.**). | ||
|
||
## **TinyMCE Plugin — Client** | ||
|
||
Having the sample plugin created, we can start to build our own logic in the code. Let’s start from the beginning, in the demo.js we have to add a new property object to the TinyMCE Editor. In this example let’s call our plugin **budwriter**. | ||
|
||
tinymce.init({ | ||
selector: 'textarea.tinymce_1', | ||
plugins: 'code budwriter', | ||
budwriter: { | ||
name: 'Andre', | ||
photoUrl: 'https://www.biggalyoga.com/wp- content/uploads/2018/07/profilecircle-768x814.png', | ||
key: 'free4all', | ||
socketUrl: 'http://localhost:3000' | ||
}, | ||
toolbar: 'budwriter', | ||
height : "600", | ||
branding: false, | ||
}); | ||
|
||
As you can see, I’ve added four properties into our budwriter plugin, special interest in socket_url that will represent the URL of our socket.io server. The other properties are my custom properties to represent a User. | ||
|
||
Having all these properties created, we can start to develop our logic, let’s create a file inside the core and start creating our collaborative class. The plugin should connect to a socket.io server to listen for event messages. These events will be split into the following: | ||
|
||
* Update clients - Event received when other clients connect to the server. | ||
|
||
* Update cursor — Changes cursor position of a client. | ||
|
||
* Update selection — Changes selection of a client. | ||
|
||
* Delete client — Deletes user container’s when a client disconnects from the server. | ||
|
||
/** | ||
* Socket IO | ||
*/ | ||
io: SocketIOClientStatic; | ||
ioClient: SocketIOClient.Socket; | ||
|
||
this.ioClient = this.io.connect(user.socketUrl, { | ||
query: `name=${user.name}&photoUrl=${user.photoUrl}` | ||
}); | ||
|
||
this.ioClient.on('update_clients', (array: []) => {// logic}); | ||
this.ioClient.on('delete_client', (user: User) => {// logic}); | ||
this.ioClient.on('update_cursor', (obj: string) => {// logic}); | ||
this.ioClient.on('update_selection', (obj: string) => {// logic}); | ||
this.ioClient.on('update_content', (content: string) => {// logic}); | ||
|
||
![Events sent from the clients to the server](https://cdn-images-1.medium.com/max/2000/1*EOcgHALT1FIMiU_Zp4h0OA.png)*Events sent from the clients to the server* | ||
|
||
As depicted in the above image, all the connected clients will send a set of events updating its TinyMCE editor status. Following my other article regarding ranges and selections we can easily implement the logic of cursor’s and selection’s, you can access the article from [here](https://medium.com/@almarques/cursors-and-selections-how-do-they-work-in-modern-browsers-5c132029b044). | ||
> **Note**: Since our socket.ioclient uses node.js modules, such as **require**, I added a step to our Grunt file ([Browserify](http://browserify.org/)) to compile these modules and use it as a bundle in the browser. | ||
## TinyMCE Plugin — Server | ||
|
||
The server side of this project, as we already discussed will be implemented using socket.io. For that, I’ve created a Node.js project where I added the socket.io package into the project. | ||
|
||
Once the socket.io package is ready to go, we have to create the event listeners for the events sent from the clients. | ||
|
||
io.on("connection", (socket: Socket) => { | ||
io.sockets.emit('update_clients', list_of_users); | ||
socket.on('disconnect', () => { | ||
io.sockets.emit('delete_client', deleted_client); | ||
}); | ||
|
||
/** | ||
* TinyMCE Clients event types | ||
*/ | ||
socket.on("set_cursor", (message: string) => { | ||
io.sockets.emit('update_cursor', message); | ||
}); | ||
socket.on("set_selection", (message: string) => { | ||
io.sockets.emit('update_selection', message); | ||
}); | ||
socket.on("set_content", (message: string) => { | ||
io.sockets.emit('update_content', message); | ||
}); | ||
}); | ||
|
||
In each one of these event listeners we have to propagate the change to the other clients, so they can update their plugin status (as depicted in the following image). | ||
|
||
![Events sent from the server to the clients](https://cdn-images-1.medium.com/max/2000/1*-OUE-sXj7PG_0K2X_QSJ2g.png)*Events sent from the server to the clients* | ||
|
||
## Repositories | ||
|
||
This implementation is available on GitHub. | ||
> [TinyMCE Plugin](https://github.com/thealmarques/tinymce-collaborative-editing-plugin) |