timmcleod / laravel-core-lib
A library for Laravel
Installs: 1 575
Dependents: 0
Suggesters: 0
Security: 0
Stars: 0
Watchers: 2
Forks: 1
Open Issues: 0
Requires
- php: >=7.3
- illuminate/filesystem: ^7.0
- illuminate/support: ^7.0
- ramsey/uuid: ~3.0
Requires (Dev)
- orchestra/testbench: ^5.0
- phpunit/phpunit: ^8.5
README
Installation
This package requires PHP >= 7.3. Require this package with composer:
composer require timmcleod/laravel-core-lib
Then, register the following service providers in your app config (project/config/app.php
) by adding them to the providers
array:
'providers' => [ TimMcLeod\LaravelCoreLib\Providers\LaravelCoreLibServiceProvider::class, TimMcLeod\LaravelCoreLib\Providers\LaravelCoreLibArtisanServiceProvider::class, ]
Change Trackable Models
Track changes that happen in your Eloquent Models
The purpose of this trait is to allow us to keep track of changes that are made to attribute values within our eloquent models, so we can access those changes AFTER the model is saved.
This can be helpful if we want to log the changes somewhere for later review.
To keep track of changes that are made to the properties/fields of your Eloquent models, just add the ChangeTrackable
trait to the models you want to track. If you would like to limit which fields are tracked, just add a property called $trackable
as shown below.
use Illuminate\Database\Eloquent\Model; use TimMcLeod\LaravelCoreLib\Database\Eloquent\ChangeTrackable; class User extends Model { use ChangeTrackable; protected $trackable = ['first_name', 'last_name']; }
When your models are saved, the ChangeTrackable
trait will automatically keep track of the fields that have changed. The following methods can be used to learn more about those changes:
/** * Returns true if any of the $trackable attributes have changed. If the * $trackable property evaluates as empty, this will return true if * any of the attributes' values have changed on the model. * * @return bool */ $model->hasTrackedChanges() /** * Returns a string representation of the attributes that have changed. If a * property called $trackable exists on the model, then that $trackable * array is used to filter the attributes that should be serialized. * * If the $trackable property doesn't exist on the model or evaluates as empty, * then all of the tracked changes will be serialized without being filtered. * * @param string $format * @param string $delimiter * @param string $emptyOld * @param string $emptyNew * @return string */ $model->getTrackedChanges($format = '{attribute}: {old} > {new}', $delimiter = ' | ', $emptyOld = '', $emptyNew = '') /** * Returns an array of the attributes that have changed. If a property called * $trackable exists on the model, then the $trackable array will be used * to limit or filter the attributes that are returned in the result. * * If the $trackable property doesn't exist on the model or evaluates as empty, * then all of the tracked changes will be returned without being filtered. * * @return array */ $model->getTrackedChangesArray() /** * Returns true if any of the given attributes have changed on the model, * regardless of whether or not the $trackable property is defined on * the model. * * @param array $attributes * @return bool */ $model->hasAnyTrackedChangesFor($attributes = []) /** * Returns a string representation of the attributes that have changed. If the * $attributes parameter evaluates as empty, then all tracked changes will * be serialized into a string without being filtered. * * @param array $attributes * @param string $format * @param string $delimiter * @param string $emptyOld * @param string $emptyNew * @return string */ $model->getTrackedChangesFor($attributes = [], $format = '{attribute}: {old} > {new}', $delimiter = ' | ', $emptyOld = '', $emptyNew = '') /** * Returns an array of the attributes that have changed. If the $attributes * parameter evaluates as empty, then all tracked changes are returned. * * @param array $attributes * @return array */ $model->getTrackedChangesArrayFor($attributes = []) /** * Returns true if any of the model's attributes have changed, regardless * of whether or not the $trackable property is defined on the model. * * @return bool */ $model->hasAnyTrackedChanges() /** * Returns a string representation of ALL of the attributes that have changed * on the model, even if the $trackable property does exist on the model. * * @param string $format * @param string $delimiter * @param string $emptyOld * @param string $emptyNew * @return string */ $model->getTrackedChangesForAll($format = '{attribute}: {old} > {new}', $delimiter = ' | ', $emptyOld = '', $emptyNew = '') /** * Returns an array of the attributes that have changed. If the $trackable * property exists on the model, it's ignored, and all tracked changes * are always returned, regardless of whether $trackable exists. * * @return array */ $model->getTrackedChangesArrayForAll()
View Models
Clean up messy views using View Models
This package can be used to help you encapsulate, validate, and manipulate the data that is required by each of the views in your Laravel 5.x applications.
Purpose and Overview
As the views in our applications grow more complex, they tend to require more data that needs to be presented in varying ways. As our view dependencies increase and the view files become more complex, it can be increasingly difficult to keep track of the data that is required by the view.
View Models help by:
- Validating the data that is injected into the view to verify that the data is exactly what the view expects.
- Facilitating the bundling of data with the methods required to manipulate the data in the context of the view.
- Reducing the chance that the "global" variables in your views will be inadvertently overwritten within your views.
- Providing a way to see what data is required by the view at-a-glance.
Getting Started
After you have registered the LaravelCoreLibArtisanServiceProvider
, a new Artisan command will be available to generate your view models. To create a new view model, use the make:view-model
Artisan command:
php artisan make:view-model EditProfileViewModel
This command will place a new EditProfileViewModel
class within your app/Http/ViewModels
directory. If the directory doesn't exist, it will be created for you.
<?php namespace App\Http\ViewModels; use TimMcLeod\LaravelCoreLib\View\BaseViewModel; class EditProfileViewModel extends BaseViewModel { /** @var array */ protected $rules = []; }
If you prefer to subgroup your view models, you can generate them this way:
php artisan make:view-model User/EditProfileViewModel
This command will place a new EditProfileViewModel
class within your app/Http/ViewModels/User
directory.
Basic Usage
To use view models, just create a new instance of a view model and pass the instance into the view.
$vm = new EditProfileViewModel([ 'timezones' => Timezone::all(), 'states' => State::all(), 'cities' => City::all(), 'user' => Auth::user() ]); return view('user.profile')->with('vm', $vm);
Within your views, you can access the data in your view model like this: $vm->timezones
.
Now you can add your own methods to your view model to keep your view-specific logic bundled nicely with the data to keep your views clean.
// Inside of EditProfileViewModel class: /** * Returns true if the user's timezone is the same as the given timezone. * * @param Timezone $timezone * @return bool */ public function isSelectedTimezone(Timezone $timezone) { return $this->user->timezone === $timezone->code; }
<!-- Inside of user.profile view --> <select id="timezone" name="timezone"> @foreach($vm->timezones as $timezone) <option value="{{$timezone->code}}" {{$vm->isSelectedTimezone($timezone) ? 'selected' : ''}}>{{$timezone->label}}</option> @endforeach </select>
Or maybe you want to create a list of checkboxes for cities in 2 separate columns in this particular view. You could do something like this:
// Inside of EditProfileViewModel class: /** * Returns the first half of cities from the cities array. * * @return array */ public function citiesCol1() { $len = count($this->cities); return $this->cities->slice(0, round($len / 2)); } /** * Returns the second half of cities from the cities array. * * @return array */ public function citiesCol2() { $len = count($this->cities); return $this->cities->slice(round($len / 2)); }
<!-- Inside of user.profile view --> <!-- Inside markup for column 1 --> @foreach($vm->citiesCol1() as $city) <label><input name="cities[]" type="checkbox" value="{{$city->id}}">{{"$city->name, $city->state_code"}}</label> @endforeach <!-- Inside markup for column 2 --> @foreach($vm->citiesCol2() as $city) <label><input name="cities[]" type="checkbox" value="{{$city->id}}"> {{"$city->name, $city->state_code"}}</label> @endforeach
As you can see, View Models allow us to bundle the view data with the methods required to manipulate the data in the context of the view.
Optionally, when the new view model is instantiated, the data can be validated using the rules defined in your view model class. You can use any of the available validation rules defined in Laravel documentation.
protected $rules = [ 'total' => 'required|integer', 'name' => 'required|string', ];
If any of the data coming from the controller fails validation, an exception is thrown.
< Side Note >
Optionally, if you require the package, timmcleod/laravel-instance-validator
, it will add a few additional validation rules that can be helpful in your View Models if you would like to validate the data that the View Model receives.
These rules are available within timmcleod/laravel-instance-validator
: instance_of
, collection_of
, and paginator_of
.
protected $rules = [ 'timezones' => 'required|collection_of:App\\Timezone', 'states' => 'required|collection_of:App\\State', 'cities' => 'present|collection_of:App\\City', 'user' => 'required|instance_of:App\\User', ];
< /Side Note >
File Savable Models
TODO: write helpful things about saving a file along with a model
ICS Calendar
TODO: write helpful things about creating calendar events with ICS files
License
LaravelCoreLib for Laravel is open-sourced software licensed under the MIT license.
Laravel is a trademark of Taylor Otwell.