Create "profile" controller and routes (WIP)

Views are incomplete and components need to be broken out from user edit.
This commit is contained in:
Christopher C. Wells 2021-04-21 16:37:10 -07:00
parent 104bbcd614
commit a378936b72
9 changed files with 199 additions and 32 deletions

View File

@ -0,0 +1,42 @@
<?php
namespace App\Http\Controllers;
use App\Http\Controllers\Traits\UpdatesUser;
use App\Http\Requests\UpdateUserRequest;
use App\Models\User;
use Illuminate\Contracts\View\View;
use Illuminate\Http\RedirectResponse;
class ProfileController extends Controller
{
use UpdatesUser;
/**
* Display a user profile page.
*/
public function show(User $user): View
{
return view('profiles.show')->with('user', $user);
}
/**
* Show the form for editing a user's profile data.
*/
public function edit(User $user): View
{
return view('profiles.edit')->with('user', $user);
}
/**
* Update the user profile data.
*/
public function update(UpdateUserRequest $request, User $user): RedirectResponse
{
$this->updateUser($request, $user);
$user->refresh();
session()->flash('message', "Profile updated!");
return redirect()->route('profiles.show', $user);
}
}

View File

@ -0,0 +1,45 @@
<?php
namespace App\Http\Controllers\Traits;
use App\Http\Requests\UpdateUserRequest;
use App\Models\User;
use Illuminate\Support\Facades\Hash;
use Illuminate\Support\Str;
trait UpdatesUser
{
/**
* Updates a user from a user update request.
*/
public function updateUser(UpdateUserRequest $request, User $user): void {
$input = $request->validated();
$input['remember_token'] = Str::random(10);
if (!empty($input['password'])) {
$input['password'] = Hash::make($input['password']);
}
else {
unset($input['password']);
}
$input['admin'] = $input['admin'] ?? false;
$user->fill($input)->save();
// Handle image.
if (!empty($input['image'])) {
/** @var \Illuminate\Http\UploadedFile $file */
$file = $input['image'];
$user->clearMediaCollection();
$user
->addMediaFromRequest('image')
->usingName($user->username)
->usingFileName("{$user->slug}.{$file->extension()}")
->toMediaCollection();
}
elseif (isset($input['remove_image']) && $input['remove_image']) {
$user->clearMediaCollection();
}
}
}

View File

@ -2,15 +2,16 @@
namespace App\Http\Controllers;
use App\Http\Controllers\Traits\UpdatesUser;
use App\Http\Requests\UpdateUserRequest;
use App\Models\User;
use Illuminate\Contracts\View\View;
use Illuminate\Http\RedirectResponse;
use Illuminate\Support\Facades\Hash;
use Illuminate\Support\Str;
class UserController extends Controller
{
use UpdatesUser;
/**
* Display a listing of the resource.
*/
@ -50,33 +51,7 @@ class UserController extends Controller
*/
public function update(UpdateUserRequest $request, User $user): RedirectResponse
{
$input = $request->validated();
$input['remember_token'] = Str::random(10);
if (!empty($input['password'])) {
$input['password'] = Hash::make($input['password']);
}
else {
unset($input['password']);
}
$input['admin'] = $input['admin'] ?? false;
$user->fill($input)->save();
// Handle image.
if (!empty($input['image'])) {
/** @var \Illuminate\Http\UploadedFile $file */
$file = $input['image'];
$user->clearMediaCollection();
$user
->addMediaFromRequest('image')
->usingName($user->username)
->usingFileName("{$user->slug}.{$file->extension()}")
->toMediaCollection();
}
elseif (isset($input['remove_image']) && $input['remove_image']) {
$user->clearMediaCollection();
}
$this->updateUser($request, $user);
session()->flash('message', "User {$user->name} updated!");
return redirect()->route('users.index');
}

View File

@ -83,6 +83,14 @@ final class User extends Authenticatable implements HasMedia
'admin' => 'bool',
];
/**
* @inheritdoc
*/
public function sluggable(): array
{
return ['slug' => ['source' => 'username']];
}
/**
* Get the User's goals.
*/

View File

@ -16,6 +16,13 @@ class UserPolicy
return $user->admin;
}
/**
* Determine whether the user can edit a user via the "profile".
*/
public function editProfile(User $user, User $model): bool {
return $user->id === $model->id;
}
/**
* Determine whether the user can delete the model.
*/

View File

@ -8,10 +8,9 @@ use Illuminate\Foundation\Support\Providers\AuthServiceProvider as ServiceProvid
class AuthServiceProvider extends ServiceProvider
{
/**
* The policy mappings for the application.
*
* @var array
* @inheritdoc
*/
protected $policies = [
User::class => UserPolicy::class,

View File

@ -0,0 +1,70 @@
<x-app-layout>
<x-slot name="title">Edit Profile</x-slot>
<x-slot name="header">
<h1 class="font-semibold text-xl text-gray-800 leading-tight">Edit Profile</h1>
</x-slot>
<form method="POST" enctype="multipart/form-data" action="{{ route('profiles.update', $user) }}">
@method('put')
@csrf
<div class="flex flex-col space-y-4">
<div class="flex flex-col space-y-4 md:flex-row md:space-x-4 md:space-y-0">
<!-- Username -->
<div class="flex-auto">
<x-inputs.label for="username" value="Username"/>
<x-inputs.input name="username"
type="text"
class="block mt-1 w-full"
autocapitalize="none"
:value="old('username', $user->username)"
:hasError="$errors->has('username')"
required />
</div>
<!-- Name -->
<div class="flex-auto">
<x-inputs.label for="name" value="Display name"/>
<x-inputs.input name="name"
type="text"
class="block mt-1 w-full"
:value="old('name', $user->name)"/>
</div>
</div>
<div class="flex flex-col space-y-4 md:flex-row md:space-x-4 md:space-y-0">
<!-- Password -->
<div class="flex-auto">
<x-inputs.label for="password" value="Password"/>
<x-inputs.input name="password"
type="password"
class="block mt-1 w-full"
:hasError="$errors->has('password')"
:required="!$user->exists"/>
</div>
<!-- Password confirm -->
<div class="flex-auto">
<x-inputs.label for="password_confirmation" value="Confirm Password"/>
<x-inputs.input name="password_confirmation"
type="password"
class="block mt-1 w-full"
:hasError="$errors->has('password')"
:required="!$user->exists"/>
</div>
</div>
<!-- Image -->
<div class="flex flex-col space-y-4 md:flex-row md:space-x-4 md:space-y-0">
<x-inputs.image :model="$user" preview_name="icon" />
</div>
</div>
<div class="flex items-center justify-end mt-4">
<x-inputs.button>Save</x-inputs.button>
</div>
</form>
</x-app-layout>

View File

@ -0,0 +1,9 @@
<x-app-layout>
<x-slot name="title">Profile</x-slot>
<x-slot name="header">
<div class="flex justify-between items-center">
<h1 class="font-semibold text-2xl text-gray-800 leading-tight">Profile</h1>
</div>
</x-slot>
{{ $user->name }}
</x-app-layout>

View File

@ -4,6 +4,7 @@ use App\Http\Controllers\FoodController;
use App\Http\Controllers\GoalController;
use App\Http\Controllers\IngredientPickerController;
use App\Http\Controllers\JournalEntryController;
use App\Http\Controllers\ProfileController;
use App\Http\Controllers\RecipeController;
use App\Http\Controllers\Auth\AuthenticatedSessionController;
use Illuminate\Support\Facades\Route;
@ -15,7 +16,9 @@ use Illuminate\Support\Facades\Route;
*/
Route::middleware(['auth'])->group(function () {
// Auth.
Route::post('/logout', [AuthenticatedSessionController::class, 'destroy'])->name('logout');
// Foods.
Route::resource('foods', FoodController::class);
Route::get('/foods/{food}/delete', [FoodController::class, 'delete'])->name('foods.delete');
@ -36,4 +39,13 @@ Route::middleware(['auth'])->group(function () {
// Recipes.
Route::resource('recipes', RecipeController::class);
Route::get('/recipes/{recipe}/delete', [RecipeController::class, 'delete'])->name('recipes.delete');
// Users.
Route::get('/profile/{user}', [ProfileController::class, 'show'])->name('profiles.show');
});
Route::middleware(['auth', 'can:editProfile,user'])->group(function () {
// Profiles (non-admin Users variant).
Route::get('/profile/{user}/edit', [ProfileController::class, 'edit'])->name('profiles.edit');
Route::put('/profile/{user}', [ProfileController::class, 'update'])->name('profiles.update');
});