Movatterモバイル変換


[0]ホーム

URL:


Skip to content
DEV Community
Log in Create account

DEV Community

Chimezie Enyinnaya
Chimezie Enyinnaya

Posted on

     

How to build a collaborative note app using Laravel

In this tutorial, we'll build an online collaborative note app using Laravel and Pusher. We'll be using Vue.js as our JavaScript framework. The app is going to be basic but will demonstrate the necessary features of a collaborative application since that's the focus of this tutorial.

What We'll Be Building

Before we get our hands busy, let's go over what we'll be building. The app will be a simple note taking app that is accessible only to authenticated users. With the app, a user can create new note, edit the note and/or share the link to the note to other users for editing. In the case of editing a note, the app will be able to keep track of the users editing a particular note, show other users realtime edits that are being made on the note and lastly notify the other users when a user saves the note.

Let's get started!

Setting Up Laravel

Create a new Laravel project by opening your terminal and run the code below:

laravel new laravel-notes
Enter fullscreen modeExit fullscreen mode

Next, we need to setup our new Laravel project. First, we need to register theApp\Providers\BroadcastServiceProvider. Openconfig/app.php and uncommentApp\Providers\BroadcastServiceProvider in the providers array.

We then need to tell Laravel that we are using the Pusher driver in the.env file:

// .envBROADCAST_DRIVER=pusher
Enter fullscreen modeExit fullscreen mode

Since we specified we want to use Pusher as our broadcasting driver, we need to install the Pusher PHP SDK:

 language-bashcomposer require pusher/pusher-php-server Setting Up Pusher
Enter fullscreen modeExit fullscreen mode

Setting Up Pusher

If you don’t have one already, create a free Pusher accounthere then log in to your dashboard and create an app. Take note of your app credentials as we’ll be using them shortly. For the purpose of this tutorial, we'll be triggering some client events in our online collaborative note app.

By default, when you create a Pusher app, client events are not enabled. We have to enable this for our app. To enable client events in your Pusher app, select the app then click on theApp Settings tab, then check the box next toEnable client events.

Now, let’s fill in our Pusher app credentials. Update the.env file to contain our Pusher app credentials:

// .envPUSHER_APP_ID=xxxxxxPUSHER_APP_KEY=xxxxxxxxxxxxxxxxxxxxPUSHER_APP_SECRET=xxxxxxxxxxxxxxxxxxxx
Enter fullscreen modeExit fullscreen mode

Remember to replace thexs with your Pusher app credentials. You can find your app credentials under the Keys section on the overview tab in the Pusher Dashboard.

Also, remember to fill in the cluster of your Pusher app and other additional options.

Installing Frontend Dependencies

For this tutorial, we’ll be using Bootstrap, Vue and Axios, which have been setup for us by Laravel, though we still need to install each of the dependencies. To compile our CSS and JavaScript, we need to install Laravel Mix, which is a wrapper around Webpack. We can install these dependencies through NPM:

 language-bashnpm install
Enter fullscreen modeExit fullscreen mode

We also need to install Laravel Echo, which is a JavaScript library that makes it painless to subscribe to channels and listen for events broadcast by Laravel and of course the Pusher JavaScript library:

 language-bashnpm install --save laravel-echo pusher-js
Enter fullscreen modeExit fullscreen mode

Once installed, we need to tell Laravel Echo to use Pusher. At the bottom of theresources/assets/js/bootstrap.js file, uncomment the Laravel Echo section and update the details with:

 language-javascript// resources/assets/js/bootstrap.jsimport Echo from "laravel-echo"window.Echo = new Echo({    broadcaster: 'pusher',    key: xxxxxxxxxxxxxxxxxxxx,});
Enter fullscreen modeExit fullscreen mode

Remember to replace thexs with your Pusher app key.

Authenticating Users

As mentioned earlier, our collaborative note app will be only accessible to authenticated users. So, we need an authentication system:

 language-bashphp artisan make:auth
Enter fullscreen modeExit fullscreen mode

This will create the necessary routes, views and controllers needed for an authentication system.

Before we go on to create users, we need to run theusers migration that comes with a fresh installation of Laravel. But to do this, we first need to set up our database. Open the.env file and enter your database details:

// .envDB_CONNECTION=mysqlDB_HOST=127.0.0.1DB_PORT=3306DB_DATABASE=laravel-notesDB_USERNAME=rootDB_PASSWORD=
Enter fullscreen modeExit fullscreen mode

Update with your own database details. Now, we can run our migration:

 language-bashphp artisan migrate
Enter fullscreen modeExit fullscreen mode

NOTE: There’s a bug in Laravel 5.4 if you’re running a version of MySQL older than 5.7.7 or MariaDB older than 10.2.2. This can be fixed by replacing theboot() ofapp/Providers/AppServiceProvider.php with:

 language-php// app/Providers/AppServiceProvider.php// remember to useIlluminate\Support\Facades\Schema;/** * Bootstrap any application services. * * @return void */public function boot(){  Schema::defaultStringLength(191);}
Enter fullscreen modeExit fullscreen mode

Note Model and Migration

Create aNote model along with the migration file by running the command:

 language-bashphp artisan make:model Note -m
Enter fullscreen modeExit fullscreen mode

Open theNote model and add the code below to it:

 language-php/** * Fields that can not be mass assigned *  * @var array */protected $guarded = ['id'];/** * Get the route key for the model. * * @return string */public function getRouteKeyName(){  return 'slug';}
Enter fullscreen modeExit fullscreen mode

Instead of manually specifying each field that can be mass assigned in the$fillable array, we simply use$guarded and add theid column as the field that can not be mass assigned, meaning every other field can be mass assigned. Laravel route model bind will by default use theid column on the model, but in this tutorial, we want to use theslug column instead, hence thegetRouteKeyName method which will simply return the column we want to use for route model binding.

Within thedatabases/migrations directory, open thenotes table migration that was created when we ran the command above and update theup method with:

 language-phpSchema::create('notes', function (Blueprint $table) {  $table->increments('id');  $table->unsignedInteger('user_id');  $table->string('title');  $table->string('slug')->unique();  $table->text('body');  $table->timestamps();});
Enter fullscreen modeExit fullscreen mode

Run the migration:

 language-bashphp artisan migrate
Enter fullscreen modeExit fullscreen mode

Defining App Routes

Openroutes/web.php and replace the routes with the code below:

 language-phpAuth::routes();Route::get('/', 'NotesController@index');Route::get('create', 'NotesController@create');Route::post('create', 'NotesController@store');Route::get('edit/{note}', 'NotesController@edit');Route::patch('edit/{note}', 'NotesController@update');
Enter fullscreen modeExit fullscreen mode

The routes are straightforward: routes that will handle authentication, a homepage route to list all notes created a user, routes for creating a new note and lastly routes to update a specified note.

NOTE: Since we have removed the/home route, you might want to update theredirectTo property of bothapp/Http/Controllers/Auth/LoginController.php andapp/Http/Controllers/Auth/RegisterController.php to:

 language-phpprotected $redirectTo = '/';
Enter fullscreen modeExit fullscreen mode

NotesController

Let’s create the controller which will handle the logic of our chat app. Create aNotesController with the command below:

 language-bashphp artisan make:controller NotesController
Enter fullscreen modeExit fullscreen mode

Open the new createapp/Http/Controllers/NotesController.php file and add the following code to it:

 language-php// app/Http/Controllers/NotesController.phpuse App\Note;public function __construct(){  $this->middleware('auth');}/** * Display a listing of all notes. * * @return \Illuminate\Http\Response */public function index(){  $notes = Note::where('user_id', auth()->user()->id)                  ->orderBy('updated_at', 'DESC')                  ->get();  return view('notes.index', compact('notes'));}/** * Show the form for creating a new note. * * @return \Illuminate\Http\Response */public function create(){  return view('notes.create');}/** * Store a newly created note in database. * * @param  \Illuminate\Http\Request  $request * @return \Illuminate\Http\Response */public function store(Request $request){  $this->validate($request, [    'title' => 'required',    'body'  => 'required'  ]);  $note = Note::create([    'user_id' => $request->user()->id,    'title'   => $request->title,    'slug'    => str_slug($request->title) . str_random(10),    'body'    => $request->body  ]);  return redirect('/');}/** * Show the form for editing the specified note. * * @param  \App\Note  $note * @return \Illuminate\Http\Response */public function edit(Note $note){  return view('notes.edit', compact('note'));}/** * Update the specified note. * * @param  \Illuminate\Http\Request  $request * @param  \App\Note  $note * @return \Illuminate\Http\Response */public function update(Request $request, Note $note){  $note->title = $request->title;  $note->body = $request->body;  $note->save();  return 'Saved!';}
Enter fullscreen modeExit fullscreen mode

Using the auth middleware inNotesController‘s__contruct() indicates that all the methods with the controller will only be accessible to authenticated users. Theindex method will fetch the notes created by the currently authenticated user and render a view with notes. Thecreate method will display a form to create new note. Thestore method will do the actual persisting of the note to the database. Notice we're appending a random string to the slug so as to make it unique for each note. Theedit method shows the form for editing a specified note. Lastly, theupdate method handles the actual update and persist to database.

Creating Our Note App Views

When we ranmake:auth, Laravel created a master layout calledapp.blade.php which we are going to leverage with some slight additions. So openresources/view/layouts/app.blade.php and update the left side of the navbar with:

 language-html<!-- resources/view/layouts/app.blade.php --><!-- Left Side Of Navbar --><ul>  <li><a href="{{ url('create') }}">Create Note</a></li></ul>
Enter fullscreen modeExit fullscreen mode

All we did is add a link to create new note on the navbar.

Create New Note View

Now, let's create the view for creating a new note. Create a new directory namednotes within theviews directory. Within the newly creatednotes directory, create a new file namedcreate.blade.php and paste the code below to it:

 language-html<!-- resources/views/notes/create.blade.php -->@extends('layouts.app')@section('content')    <div>        <div>            <div>                <div>                    <div>Create new note</div>                    <div>                        <form action="{{ url('create') }}" method="POST" role="form">                            {{ csrf_field() }}                            <div>                                <input type="text" name="title" value="{{ old('title') }}" placeholder="Give your note a title" required autofocus>                                @if ($errors->has('title'))                                    <span>                                        <strong>{{ $errors->first('title') }}</strong>                                    </span>                                @endif                            </div>                            <div>                                <textarea name="body" rows="15" placeholder="...and here goes your note body" required>{{ old('body') }}</textarea>                                @if ($errors->has('body'))                                    <span>                                        <strong>{{ $errors->first('body') }}</strong>                                    </span>                                @endif                            </div>                            <button>Save</button>                        </form>                    </div>                </div>            </div>        </div>    </div>@endsection
Enter fullscreen modeExit fullscreen mode

This creates a form with two input fields (for title and body of the note respectively) and a save button.

List All Notes View

Let's give our users a way to see all the notes they have created. Within thenotes directory, create a new file namedindex.blade.php and paste the code below into it:

 language-html<!-- resources/views/notes/index.blade.php -->@extends('layouts.app')@section('content')    <div>        <div>            <div>                <div>                    <div>My notes</div>                    <div>                        @if($notes->isEmpty())                            <p>                                You have not created any notes! <a href="{{ url('create') }}">Create one</a> now.                            </p>                        @else                        <ul>                            @foreach($notes as $note)                                <li>                                    <a href="{{ url('edit', [$note->slug]) }}">                                        {{ $note->title }}                                    </a>                                    <span>{{ $note->updated_at->diffForHumans() }}</span>                                </li>                            @endforeach                        </ul>                        @endif                    </div>                </div>            </div>        </div>    </div>@endsection
Enter fullscreen modeExit fullscreen mode

The simply displays a message if the user has not created any notes and a link to create a new note. Otherwise it will display all the notes created by the user in a list.

Edit Note View

Let's create the edit view which will allow users to edit a note. Within thenotes directory, create a new file namededit.blade.php and paste the code below into it:

 language-html<!-- resources/views/notes/edit.blade.php -->@extends('layouts.app')@section('content')    <div>        <div>            <div>                <edit-note :note="{{ $note }}"></edit-note>            </div>        </div>    </div>@endsection
Enter fullscreen modeExit fullscreen mode

You will notice we're using a custom tag `` with the edit view, this is our view component which we'll create shortly.

Now let's create a Vue component. Create a new file namedEditNote.vue withinresources/assets/js/components directory and paste the code below to it:

 language-javascript// resources/assets/js/components/EditNote.vue<template>    <div>        <div>Edit note</div>        <div>            <div>                <input type="text" v-model="title" @keydown="editingNote">            </div>            <div>                <textarea rows="15" v-model="body" @keydown="editingNote"></textarea>            </div>            <button @click="updateNote">Save</button>            <p>                Users editing this note:  <span>{{ usersEditing.length }}</span>                <span v-text="status"></span>            </p>        </div>    </div></template><script>    export default {        props: [            'note',        ],        data() {            return {                title: this.note.title,                body: this.note.body,                usersEditing: [],                status: ''            }        },        mounted() {            Echo.join(`note.${this.note.slug}`)                .here(users => {                    this.usersEditing = users;                })                .joining(user => {                    this.usersEditing.push(user);                })                .leaving(user => {                    this.usersEditing = this.usersEditing.filter(u => u != user);                })                .listenForWhisper('editing', (e) => {                    this.title = e.title;                    this.body = e.body;                })                .listenForWhisper('saved', (e) => {                    this.status = e.status;                    // clear is status after 1s                    setTimeout(() => {                        this.status = '';                    }, 1000);                });        },        methods: {            editingNote() {                let channel = Echo.join(`note.${this.note.slug}`);                // show changes after 1s                setTimeout(() => {                    channel.whisper('editing', {                        title: this.title,                        body: this.body                    });                }, 1000);            },            updateNote() {                let note = {                    title: this.title,                     body:  this.body                };                // persist to database                axios.patch(`/edit/${this.note.slug}`, note)                    .then(response => {                        // show saved status                        this.status = response.data;                        // clear is status after 1s                        setTimeout(() => {                            this.status = '';                        }, 1000);                        // show saved status to others                        Echo.join(`note.${this.note.slug}`)                            .whisper('saved', {                                status: response.data                            });                    });            }        }    }</script>
Enter fullscreen modeExit fullscreen mode

Let's explain each piece of the code. Just like we have in the 'create new note' form, the template section has two input fields: title and body. Each field is bound to data (title and body respectively). Once a user starts typing (that is, a keydown event) in any of the input fields, theeditingNote method will be triggered. Also, when the save button is clicked, theupdateNote method will be triggered. (We'll take a close look at these methods soon) Lastly on the template section, we display the number of users who are currently editing the specified note and also display a status message once the save button is clicked.

Moving to the script section of the component, first we define a property for the component callednote. Thisnote property will be the note that is currently being edited. Recall from the edit view where we used theEditNote component, you will notice we passed the whole note object as the component'snote property. Next we define some data, thetitle and thebody data are bound to respective input fields, theusersEditing will be an array of users editing the note andstatus will serve as an indicator for when a note's edits have been persisted to the database. Themount method will be triggered immediately the component is mounted, so it's a nice place to subscribe and listen to a channel. In our case, because we to be able to keep track of users editing a note, we'll make use of Pusher's presence channel.

Using Laravel Echo, we can subscribe to a presence channel usingEcho.join('channel-name'). As you can see our channel name isnote.note-slug. Once we subscribe to a presence channel, we can get all the users that are subscribed to the channel with thehere method where we simply assign the subscribed users to theusersEditing array. When a user joins the channel, we simply add that user to theusersEditing array. Similarly, when a user leaves the channel, we remove that user from theusersEditing array. To display edits in realtime to other users, we listen for client events that are triggered as a user types usinglistenForWhisper and update the form data accordingly. In the same vein, we listen for when edits are saved and display the "Saved!" status to other users, then after a second we clear the status message.

Next, we define the methods we talked about earlier. TheeditingNote method simply triggers a client event to all users currently subscribed to the channel after a specified time (1 second). TheupdateNote method on the other hand sends aPATCH request with the edits made to persist the edits to the database. Once the request is successful, we display the message saved status to the user that made the save and clear the status message after 1 second. Lastly, we trigger a client event so other users can also see the message saved status.

Since we created a presence channel, only authenticated users will be able to subscribe and listen on the note channel. So, we need a way to authorize that the currently authenticated user can actually subscribe and listen on the channel. This can be done in theroutes/channels.php file:

 language-php// routes/channels.phpBroadcast::channel('note.{slug}', function ($user, $slug) {    return [        'id'   => $user->id,        'name' => $user->name    ];});
Enter fullscreen modeExit fullscreen mode

We pass to thechannel(), the name of our channel and a callback function that will return the details of the user if the current user is authenticated.

Now let's register our new created component with our Vue instance, openresources/assets/js/app.js and add the line below just before Vue instantiation:

 language-javascript// resources/assets/js/app.jsVue.component('edit-note', require('./components/EditNote.vue'));
Enter fullscreen modeExit fullscreen mode

Before testing out our online collaborative note app, we need to compile the JavaScript files using Laravel Mix using:

 language-bashnpm run dev
Enter fullscreen modeExit fullscreen mode

Now we can start our note app by running:

 language-bashphp artisan serve
Enter fullscreen modeExit fullscreen mode

The code of the complete demo is available onGitHub.

Conclusion

We have seen how to build a simple online collaborative note app using Laravel and Pusher. Sure there are other ways of accomplishing what we did in this tutorial, but we have seen how to build a collaborative app with Pusher's real-time features. Also, you will notice our note app doesn’t account for cases like concurrent editing of notes; to achieve that you'd want to look intoOperational Transformation.

This post was originally posted by the author on thePusher blog.

Top comments(0)

Subscribe
pic
Create template

Templates let you quickly answer FAQs or store snippets for re-use.

Dismiss

Are you sure you want to hide this comment? It will become hidden in your post, but will still be visible via the comment'spermalink.

For further actions, you may consider blocking this person and/orreporting abuse

Founder mezielabs.com
  • Location
    Lagos, Nigeria
  • Work
    Software Developer
  • Joined

More fromChimezie Enyinnaya

DEV Community

We're a place where coders share, stay up-to-date and grow their careers.

Log in Create account

[8]ページ先頭

©2009-2025 Movatter.jp