Compare commits
11 Commits
3b4de82f92
...
0518c06ebe
| Author | SHA1 | Date |
|---|---|---|
|
|
0518c06ebe | |
|
|
0aae934dd4 | |
|
|
58453d316f | |
|
|
da91153143 | |
|
|
f489c6f751 | |
|
|
36c2ccae2f | |
|
|
5275620b8d | |
|
|
b5d14206c8 | |
|
|
7d867f1140 | |
|
|
05c2d193e2 | |
|
|
dbc8de4ede |
|
|
@ -3,53 +3,43 @@
|
|||
namespace App\Livewire\Admin;
|
||||
|
||||
use App\Models\Client;
|
||||
use App\Services\ClientService;
|
||||
use App\Livewire\Forms\ClientForm; // 1. Importa seu Form Object
|
||||
use Livewire\Component;
|
||||
use Livewire\WithFileUploads;
|
||||
use Livewire\Attributes\On;
|
||||
use Illuminate\Support\Facades\Crypt;
|
||||
|
||||
|
||||
class AddClient extends Component
|
||||
{
|
||||
use WithFileUploads;
|
||||
|
||||
// 1. Declara as propriedades principais
|
||||
public ClientForm $form;
|
||||
/**
|
||||
* Método principal chamado pelo wire:submit="save".
|
||||
*/
|
||||
public function save()
|
||||
public function save(ClientService $clientService)
|
||||
{
|
||||
|
||||
|
||||
// 2. Valida os dados usando as 'rules' do ClientForm.php
|
||||
$this->form->validate();
|
||||
|
||||
try {
|
||||
$data = $this->form->all();
|
||||
$data['name'] = $data['client_name'];
|
||||
// 4. Lida com o upload do arquivo de imagem
|
||||
|
||||
|
||||
if ($this->form->profile_image_path) {
|
||||
$path = $this->form->profile_image_path->store('client_logos', 'public');
|
||||
$data['profile_image_path'] = $path;
|
||||
}
|
||||
|
||||
// 5. Cria o cliente no banco de dados
|
||||
Client::create($data);
|
||||
// 6. Despacha um evento para atualizar outros componentes (ex: o grid de clientes)
|
||||
$data['root_password'] = Crypt::encryptString($data['root_password']);
|
||||
|
||||
$clientService->addClient($data);
|
||||
|
||||
$this->dispatch('client-added');
|
||||
// (Opcional) Envia uma notificação de sucesso
|
||||
$this->dispatch('notify', message: 'Cliente adicionado com sucesso!');
|
||||
} catch (\Exception $e) {
|
||||
dd($e);
|
||||
$this->dispatch('notify', message: 'Ocorreu um erro inesperado ao salvar.', type: 'error');
|
||||
}
|
||||
// 3. Pega todos os dados validados
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Renderiza a view do modal.
|
||||
*/
|
||||
public function render()
|
||||
{
|
||||
return view('livewire.admin.add-client');
|
||||
|
|
|
|||
|
|
@ -2,27 +2,24 @@
|
|||
|
||||
namespace App\Livewire\Admin;
|
||||
|
||||
use App\Services\UserService; // <-- Importe seu Service
|
||||
use App\Services\UserService;
|
||||
use Illuminate\Support\Facades\Auth;
|
||||
use Livewire\Component;
|
||||
|
||||
class CreateUser extends Component
|
||||
{
|
||||
// 1. Propriedades públicas (os campos do formulário)
|
||||
// Elas substituem o 'Request $request'
|
||||
public string $name = '';
|
||||
public string $email = '';
|
||||
public string $password = '';
|
||||
public string $password_confirm = ''; // <-- Você também precisa deste
|
||||
public bool $permission_level = false;
|
||||
public string $password_confirm = '';
|
||||
public bool $permissions = false;
|
||||
|
||||
// 2. As regras de validação (copiadas do seu controller)
|
||||
// Nota: Adicionei 'same:password' para garantir que as senhas batem.
|
||||
protected $rules = [
|
||||
'name' => 'required|string|max:255',
|
||||
'email' => 'required|email|unique:users,email',
|
||||
'password' => 'required|string|min:8',
|
||||
'password_confirm' => 'required|string|same:password', // <-- Regra importante!
|
||||
'permission_level' => 'required|boolean' // ou 'required|in:0,1'
|
||||
'password_confirm' => 'required|string|same:password',
|
||||
'permissions' => 'required|boolean'
|
||||
];
|
||||
|
||||
protected $messages = [
|
||||
|
|
@ -31,48 +28,32 @@ class CreateUser extends Component
|
|||
'email.unique' => 'O email informado já foi cadastrado anteriormente.',
|
||||
'password' => 'A senha precisa ter 8 ou mais caracteres.',
|
||||
'password_confirm' => 'As senhas não coincidem.',
|
||||
'permission_level' => 'Defina o nível de autorização do usuário.'
|
||||
'permissions' => 'Defina o nível de autorização do usuário.',
|
||||
];
|
||||
|
||||
/**
|
||||
* O método de "salvar", que substitui o seu 'createUsers'.
|
||||
*
|
||||
* Note como injetamos o UserService direto no método!
|
||||
* O Livewire cuida disso para você, assim como o Laravel faz nos controllers.
|
||||
*/
|
||||
public function createUser(UserService $userService)
|
||||
{
|
||||
|
||||
// 3. Valida as propriedades públicas ($this->name, $this->email, etc.)
|
||||
$validated = $this->validate($this->rules, $this->messages);
|
||||
// 4. Seu 'try...catch' - praticamente idêntico
|
||||
|
||||
if ($validated['permissions'] === true) {
|
||||
$validated['permissions'] = array('admin');
|
||||
} else {
|
||||
$validated['permissions'] = array('user');
|
||||
}
|
||||
|
||||
try {
|
||||
|
||||
// 5. CHAMA O MESMO SERVICE! Nenhuma lógica de negócio é duplicada.
|
||||
$this->authorize('createUser', Auth::user());
|
||||
$user = $userService->createUser($validated);
|
||||
|
||||
// 6. O "Sucesso" (Tradução do Redirect)
|
||||
|
||||
// Limpa o formulário
|
||||
$this->reset();
|
||||
|
||||
// Dispara um evento para o Alpine.js fechar o modal
|
||||
$this->dispatch('user-created');
|
||||
|
||||
// Envia a mesma mensagem de sucesso do seu controller
|
||||
$this->dispatch('notify', message: 'Usuário cadastrado com sucesso!');
|
||||
|
||||
|
||||
// (Opcional) Se sua tabela de usuários for outro componente Livewire,
|
||||
// você pode mandar ela atualizar assim:
|
||||
// $this->dispatch('refreshUserList');
|
||||
|
||||
} catch (\Exception $e) {
|
||||
|
||||
// 7. O "Erro" (Tradução do Redirect de Erro)
|
||||
// Em vez de redirecionar, adicionamos o erro ao formulário
|
||||
// para que o usuário veja na tela, sem refresh.
|
||||
|
||||
$this->addError('general', $e->getMessage());
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -0,0 +1,13 @@
|
|||
<?php
|
||||
|
||||
namespace App\Livewire\Admin;
|
||||
|
||||
use Livewire\Component;
|
||||
|
||||
class ShowUsers extends Component
|
||||
{
|
||||
public function render()
|
||||
{
|
||||
return view('livewire.admin.show-users');
|
||||
}
|
||||
}
|
||||
|
|
@ -21,6 +21,7 @@ class User extends Authenticatable
|
|||
protected $fillable = [
|
||||
'name',
|
||||
'email',
|
||||
'permissions',
|
||||
'password',
|
||||
];
|
||||
|
||||
|
|
|
|||
|
|
@ -23,7 +23,7 @@ public function register(): void
|
|||
public function boot(): void
|
||||
{
|
||||
Gate::define('createUser', function (User $user) {
|
||||
return $user->permissions;
|
||||
return isset($user->permissions) ? in_array('admin', $user->permissions) : false;
|
||||
});
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -3,6 +3,7 @@
|
|||
namespace Database\Seeders;
|
||||
|
||||
use App\Models\User;
|
||||
use App\Models\Client;
|
||||
use Illuminate\Support\Facades\Hash;
|
||||
// use Illuminate\Database\Console\Seeds\WithoutModelEvents;
|
||||
use Illuminate\Database\Seeder;
|
||||
|
|
@ -20,5 +21,7 @@ public function run(): void
|
|||
'email' => 'inglinesystemsadmin@inglinesystems.com.br',
|
||||
'password' => Hash::make('*Ingline.Sys#9420%SECURITY#')
|
||||
]);
|
||||
|
||||
Client::factory()->create([]);
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -60,6 +60,7 @@ @layer components {
|
|||
@apply absolute right-5;
|
||||
@apply rounded-xl;
|
||||
@apply max-w-7 max-h-7;
|
||||
@apply shadow-md shadow-blue-400;
|
||||
}
|
||||
|
||||
.profile-list-items {
|
||||
|
|
|
|||
Binary file not shown.
|
After Width: | Height: | Size: 32 KiB |
|
|
@ -3,6 +3,7 @@
|
|||
@section('content')
|
||||
<livewire:admin.create-user />
|
||||
<livewire:admin.add-client />
|
||||
<livewire:admin.show-users />
|
||||
|
||||
<div class="container grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-4 gap-6">
|
||||
|
||||
|
|
|
|||
|
|
@ -22,7 +22,7 @@
|
|||
<a href="{{ route('dashboard') }}">Início</a>
|
||||
<div class="nav-bar-logo">
|
||||
<a href="{{ route('dashboard') }}">
|
||||
<img src="{{ Vite::asset('resources/images/logo.png') }}" alt="Logo">
|
||||
<img src="{{ Vite::asset('resources/images/newlogo.png') }}" alt="Logo">
|
||||
</a>
|
||||
</div>
|
||||
|
||||
|
|
@ -42,8 +42,25 @@
|
|||
src="{{ Vite::asset('resources/images/avatar-nexus.svg') }}" alt="Avatar">
|
||||
</a>
|
||||
|
||||
<ul class="profile-list-items" x-show="open">
|
||||
<h2 class="font-weight-bold text-center">Nexus</h2>
|
||||
<ul class="profile-list-items" x-show="open" x-transition:enter.duration.300ms
|
||||
x-transition:leave.duration.300ms>
|
||||
<h2 class="font-bold text-center" x-show="open" x-transition:enter="transition ease-out duration-300"
|
||||
x-transition:enter-start="opacity-0 scale-90" x-transition:enter-end="opacity-100 scale-100"
|
||||
x-transition:leave="transition ease-in duration-300"
|
||||
x-transition:leave-start="opacity-100 scale-100" x-transition:leave-end="opacity-0 scale-90">Nexus
|
||||
</h2>
|
||||
|
||||
<li class="profile-items">
|
||||
<a href="" class="profile-link">
|
||||
Início
|
||||
</a>
|
||||
</li>
|
||||
|
||||
<li class="profile-items">
|
||||
<a @click="$dispatch('show-users')" class="profile-link">
|
||||
Administração
|
||||
</a>
|
||||
</li>
|
||||
|
||||
<li class="profile-items">
|
||||
<a @click="$dispatch('open-create-user')" class="profile-link">
|
||||
|
|
@ -51,18 +68,6 @@
|
|||
</a>
|
||||
</li>
|
||||
|
||||
<li class="profile-items">
|
||||
<a href="" class="profile-link">
|
||||
Visualizar Clientes
|
||||
</a>
|
||||
</li>
|
||||
|
||||
<li class="profile-items">
|
||||
<a href="" class="profile-link">
|
||||
Administração
|
||||
</a>
|
||||
</li>
|
||||
|
||||
<li class="profile-items">
|
||||
<a @click="$dispatch('open-add-client')" class="profile-link">
|
||||
Adicionar clientes
|
||||
|
|
|
|||
|
|
@ -2,7 +2,7 @@
|
|||
<div x-data="{ showModal: false }" x-on:user-created.window="showModal = false"
|
||||
x-on:open-create-user.window="showModal = true">
|
||||
|
||||
<div x-show="showModal" class="modal-overlay">
|
||||
<div x-show="showModal" class="modal-overlay" x-transition:enter.duration.300ms x-transition:leave.duration.300ms>
|
||||
<div x-on:click.outside="showModal = false" x-show="showModal" x-transition:enter="transition-enter"
|
||||
x-transition:enter-start="transition-enter-start" x-transition:enter-end="transition-enter-end"
|
||||
x-transition:leave="transition-leave" x-transition:leave-start="transition-leave-start"
|
||||
|
|
@ -46,7 +46,7 @@
|
|||
|
||||
<div x-data="{
|
||||
open: false,
|
||||
selected: @entangle('permission_level'),
|
||||
selected: @entangle('permissions'),
|
||||
options: { '0': 'Usuário', '1': 'Admin' }
|
||||
}" x-on:click.outside="open = false" class="select-wrapper">
|
||||
|
||||
|
|
@ -93,7 +93,7 @@ class="absolute inset-y-0 left-0 flex items-center pl-3 text-blue-600">
|
|||
</div>
|
||||
</div>
|
||||
|
||||
@error('permission_level') <span class="error-text">{{ $message }}</span> @enderror
|
||||
@error('permissions') <span class="error-text">{{ $message }}</span> @enderror
|
||||
</div>
|
||||
|
||||
<div class="form-footer">
|
||||
|
|
|
|||
|
|
@ -0,0 +1,4 @@
|
|||
<div x-data="{showUsers: false}" x-on:show-users.window="showUsers = true">
|
||||
|
||||
|
||||
</div>
|
||||
|
|
@ -1,19 +0,0 @@
|
|||
@extends('layouts.app')
|
||||
|
||||
|
||||
@section('content')
|
||||
|
||||
<div class="container">
|
||||
<h1>Bem vindo a página de criação dos usuários, <span style="background-color:rgb(89, 255, 255); border-radius: 0.3em; padding: 0.2em">{{Auth::user()->name}}!</span></h1>
|
||||
|
||||
<form action="{{ route('users.create') }}" method="POST" class="form-class">
|
||||
@csrf
|
||||
<input type="text" name="name" placeholder="Nome do usuário" class="form-input-class">
|
||||
<input type="email" name="email" placeholder="Email do usuário" class="form-input-class">
|
||||
<input type="password" name="password" placeholder="8 a 20 caracteres" class="form-input-class">
|
||||
<button type="submit" class="form-button-class">Criar usuário</button>
|
||||
</form>
|
||||
|
||||
</div>
|
||||
|
||||
@endsection
|
||||
Loading…
Reference in New Issue