/livewire-laravel-chirper

My first time playing with Livewire will be to follow the example project from the Laravel Bootcamp #buildInPublic

Primary LanguageBladeMIT LicenseMIT

livewire-laravel-chirper

My first time playing with Livewire will be to follow the example project from the Laravel Bootcamp. This is a #buildInPublic project.

Build this project yourself


Getting Started

  • Created a github repo
  • Cloned to my laptop and moved to that directory
  • I use docker for playing with PHP projects in general: curl -s "https://laravel.build/chirper" | bash
  • Change to the new directory and bring up the base project ./vendor/bin/sail up
  • Use Breeze (a minimal authentication implementation) composer require laravel/breeze --dev
  • And bring in livewire php artisan breeze:install livewire

With the basic project in place, we can move on to Models, Controllers and Components.

Using artisan we can create Model/Migration/Controller at the same time:

  • Let's dive in php artisan make:model -mc Chirp

    INFO Model [app/Models/Chirp.php] created successfully. Eloquent Model

    INFO Migration [database/migrations/2024_04_17_095543_create_chirps_table.php] created successfully. DB

    INFO Controller [app/Http/Controllers/ChirpController.php] created successfully. Handle requests, return response

Making Routes, Controller, View = standard Laravel stuff.

Render with Livewire

When we have something to render via Livewire we use the make:volt command: php artisan make:volt chirps/create --class

This command will create a new Livewire component at resources/views/livewire/chirps/create.blade.php

Navigation

I'm using Breeze for this project;

  • update navigation.blade.php to add items

This file has duplicated links, one for desktop, one for responsive (mobile) devices. So update twice in this example anyway.

I am impressed by how easy it is to get fully functional navigation and user profile sections with breeze.

Mass Assignment Protection

Important to control what data a user can pass in order to prevent/avoid mass assignment vulnerability. Laravel protects against this by default, but this default protection can make some deliberate mass assignments inconvenient. Therefore it is important to understand your database and usecases to decide which attributes are safe for mass assignment, we mark these as $fillable.

Database migrations

When application was created, laravel applied any default migrations to create starting database for us. Can review with:

php artisan db:show php artisan db:show php artisan db:table users php artisan db:table users

We need to add any new columns to our migration file:

databases/migrations/<timestamp>_create_chirps_table.php

After any changes, we need to push those changes to the database with php artisan migrate .

NB For dev only, can use php artisan migrate:refresh which will dump your current data and recreate from scratch, useful if also using a factory.

Tinker

We haven't built a 'show' view yet. We can still review our chirps so far by using Tinker.

  • Start Tinker in a new console window php artisan tinker
  • At the prompt: App\Models\Chirp:all();

You can then see any chirps you sent during testing.

Showing Current Chirps

  • Add a list option for resources/views/chirps.blade.php
  • Using volt to create a new livewire component: php artisan make:volt chirps/list --class
  • Update that new file here resources/views/livewire/chirps/list.blade.php

New Chirps

When a new chirp is created, we want to make use of livewire events.

  • add a dispatch to resources/views/livewire/chirps/create.blade.php to announce something new
  • add a listener to resources/views/livewire/chirps/list.blade.php to listen for new chirps and update list

We also need to show a relationship between the user and the chirp.

  • update the app/Models/Chirp.php
  • adding a belongs to relationship

Once the relationship between chirps and users is established, we can refresh the page and see the following:

Chirps timeline view

Editing Chirps

That's right folks, you don't need to subscribe to the premium membership in order to edit your previously posted chirps 😜

  • Update the list view to have an edit button resources/views/livewire/chirps/list.blade.php

First, we'll use the x-dropdown component that is included with Breeze. In addition, we will make this dropdown only visible to the Chirp's original author. In this dropdown, we'll add a link that will trigger the edit action on the component. This method will set the editing property to the Chirp that we want to edit. We'll use this property to conditionally display the edit form.

  • using volt we need to create the edit component php artisan make:volt chirps/edit --class
  • build out the edit functionality
  • we then need to update the list view to show the edited items

If the chirp-updated event is dispatched, we'll need to update the list of Chirps. If the chirp-edit-canceled event is dispatched, we'll need to set the editing property to null so that the edit form is no longer displayed:

  • by default, Authorise method weill prevent anyone from updating a chirp, we need to allow the original author. create a Model Policy: php artisan make:policy ChirpPolicy --model=Chirp
  • update the new file: app/Policies/ChirpPolicy.php to include return $chirp->user()->is($user);

List of edited chirps (still shows original post time)

Delete that Chirp

Too many late-night or Drink fueld chirps, worry not with the ability to delete a chirp.

  • Update the list view to add a button and delet action resources/views/livewire/chirps/list.blade.php
  • Update the Chirp Policy to ensure this is limited to chirp owners app/Policies/ChirpPolicy.php

Rather than repeating the logic from the update method, we can define the same logic by calling the update method from our destroy method. Anyone that is authorized to update a Chirp will now be authorized to delete it as well.

Create / Update / Delete

Notifications & Events

How about email notifications when a new Chirp is created, Laravel has your back.

  • Artisan php artisan make:notification NewChirp
  • Edit the new file app/Notifications/NewChirp.php
  • Let's dispatch an event that we can listen for and process in a background queue to keep Chirper snappy.

Create an event

  • Create the new event php artisan make:event ChirpCreated
  • Edit the new here app/Events/ChirpCreated.php

Dispatch the event

You may dispatch events anywhere in your application lifecycle, but as our event relates to the creation of an Eloquent model, we can configure our Chirp model to dispatch the event for us.

  • Update app/Models/Chirp.php
  • When a chirp is created, ChirpCreated will be dispatched

Event Listener

We need something that can listen for those dispatched events.

  • php artisan make:listener SendChirpCreatedNotifications --event=ChirpCreated
  • new file app/Listeners/SendChirpCreatedNotifications.php

We've marked our listener with the ShouldQueue interface, which tells Laravel that the listener should be run in a queue. By default, the "sync" queue will be used to process jobs synchronously; however, you may configure a queue worker to process jobs in the background.

We've used a database cursor to avoid loading every user into memory at once.


Useful for me to remember (accessing composer and artisan commands via cli if not added to shell).

When developing applications using Sail, you may execute Artisan, NPM, and Composer commands via the Sail CLI instead of invoking them directly

    ./vendor/bin/sail php --version
    ./vendor/bin/sail artisan --version
    ./vendor/bin/sail composer --version
    ./vendor/bin/sail npm --version