Browse Source
aqui hice el nuevo catalogo de departamaneto para que el usuario lo agregue manual y de ahi se refleje en la base de datos y la tabla de usuarios la jale y muestre la informacion ingresada mas aparte los letreros de mis catalogos y todo lo que hice le meti validaciones y tambien en españolmain
18 changed files with 619 additions and 31 deletions
@ -0,0 +1,23 @@ |
|||||
|
<?php |
||||
|
|
||||
|
namespace App\Exports; |
||||
|
|
||||
|
use App\Models\Despartamento; // Asegúrate de que el modelo esté correctamente importado |
||||
|
use Maatwebsite\Excel\Concerns\FromCollection; |
||||
|
use Maatwebsite\Excel\Concerns\WithHeadings; |
||||
|
|
||||
|
class DespartamentosExport implements FromCollection, WithHeadings |
||||
|
{ |
||||
|
public function collection() |
||||
|
{ |
||||
|
return Despartamento::where('eliminado', 0)->get(); // Obtiene todos los departamentos activos |
||||
|
} |
||||
|
|
||||
|
public function headings(): array |
||||
|
{ |
||||
|
return [ |
||||
|
'ID', |
||||
|
'Departamento' |
||||
|
]; |
||||
|
} |
||||
|
} |
@ -0,0 +1,127 @@ |
|||||
|
<?php |
||||
|
|
||||
|
namespace App\Http\Controllers; |
||||
|
|
||||
|
use App\Models\despartamento; |
||||
|
use Illuminate\Http\Request; |
||||
|
use App\Exports\DespartamentosExport; |
||||
|
use Maatwebsite\Excel\Facades\Excel; |
||||
|
use PDF; |
||||
|
|
||||
|
class DespartamentoController extends Controller |
||||
|
{ |
||||
|
/** |
||||
|
* Display a listing of the resource. |
||||
|
*/ |
||||
|
public function index(Request $request) |
||||
|
{ |
||||
|
$busqueda = $request->busqueda; |
||||
|
|
||||
|
if ($busqueda) { |
||||
|
// Busca en la columna 'nombre' de la tabla 'despartamentos' |
||||
|
$despartamentos = despartamento::where('nombre', 'LIKE', "%{$busqueda}%")->where('eliminado', 0)->get(); |
||||
|
|
||||
|
if ($despartamentos->isEmpty()) { |
||||
|
return redirect()->route('despartamento.index') |
||||
|
->with('error', 'No existe ningún departamento con el nombre "' . $busqueda . '". Por favor, inténtalo de nuevo.'); |
||||
|
} |
||||
|
} else { |
||||
|
// Si no hay búsqueda, mostrar todos los departamentos |
||||
|
$despartamentos = despartamento::where('eliminado', 0)->get(); |
||||
|
} |
||||
|
|
||||
|
return view('despartamentos', ['despartamentos' => $despartamentos]); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Show the form for creating a new resource. |
||||
|
*/ |
||||
|
public function create() |
||||
|
{ |
||||
|
return view('despartamentosCrearEditar', ['despartamento' => null]); // No se necesita pasar departamentos |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Store a newly created resource in storage. |
||||
|
*/ |
||||
|
public function store(Request $request) |
||||
|
{ |
||||
|
// Valida la entrada |
||||
|
$request->validate([ |
||||
|
'departamento' => 'required|string|max:255|unique:despartamentos,departamento', // Asegúrate de que 'departamento' sea único |
||||
|
], [ |
||||
|
'departamento.required' => 'El campo departamento es obligatorio.', |
||||
|
'departamento.string' => 'El campo departamento debe ser una cadena de texto.', |
||||
|
'departamento.max' => 'El campo departamento no puede tener más de 255 caracteres.', |
||||
|
'departamento.unique' => 'Ya existe un departamento con ese nombre.', |
||||
|
]); |
||||
|
|
||||
|
// Crea un nuevo departamento |
||||
|
$despartamento = new despartamento(); |
||||
|
$despartamento->departamento = $request->departamento; // Asigna el nombre ingresado por el usuario |
||||
|
$despartamento->eliminado = 0; // Departamento como activo por defecto |
||||
|
$despartamento->save(); |
||||
|
|
||||
|
return redirect()->route('despartamentos.index')->with('success', 'Departamento creado exitosamente.'); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Show the form for editing the specified resource. |
||||
|
*/ |
||||
|
public function edit($id) |
||||
|
{ |
||||
|
$despartamento = despartamento::findOrFail($id); // Busca el departamento por ID |
||||
|
return view('despartamentosCrearEditar', ['despartamento' => $despartamento]); // Pasa el departamento a la vista |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Update the specified resource in storage. |
||||
|
*/ |
||||
|
public function update(Request $request, $id) |
||||
|
{ |
||||
|
$request->validate([ |
||||
|
'departamento' => 'required|string|max:255|unique:despartamentos,departamento', // Asegúrate de que 'departamento' sea único |
||||
|
], [ |
||||
|
'departamento.required' => 'El campo departamento es obligatorio.', |
||||
|
'departamento.string' => 'El campo departamento debe ser una cadena de texto.', |
||||
|
'departamento.max' => 'El campo departamento no puede tener más de 255 caracteres.', |
||||
|
'departamento.unique' => 'Ya existe un departamento con ese nombre, por favor elige otro.', |
||||
|
]); |
||||
|
|
||||
|
$despartamento = despartamento::findOrFail($id); // Encuentra el departamento por ID |
||||
|
|
||||
|
// Verifica si el nombre del departamento ha cambiado |
||||
|
if ($despartamento->departamento !== $request->departamento) { |
||||
|
$despartamento->departamento = $request->departamento; // Actualiza el nombre del departamento |
||||
|
} |
||||
|
|
||||
|
$despartamento->eliminado = 0; // Cambia el estado a activo si se está editando |
||||
|
$despartamento->save(); // Guarda los cambios |
||||
|
|
||||
|
return redirect()->route('despartamentos.index')->with('success', 'Departamento actualizado correctamente.'); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Remove the specified resource from storage. |
||||
|
*/ |
||||
|
public function destroy($id) |
||||
|
{ |
||||
|
$despartamento = despartamento::findOrFail($id); // Encuentra el departamento por ID |
||||
|
$despartamento->eliminado = 1; // Cambia el estado a inactivo |
||||
|
$despartamento->save(); // Guarda los cambios |
||||
|
|
||||
|
return redirect()->route('despartamentos.index')->with('success', 'Departamento inactivado correctamente.'); |
||||
|
} |
||||
|
|
||||
|
public function exportExcel() |
||||
|
{ |
||||
|
return Excel::download(new DespartamentosExport, 'despartamentos.xlsx'); |
||||
|
} |
||||
|
|
||||
|
public function exportPDF() |
||||
|
{ |
||||
|
$departamentos = despartamento::where('eliminado', 0)->get(); |
||||
|
$pdf = PDF::loadView('exports.departamentos', ['departamentos' => $departamentos]); |
||||
|
return $pdf->download('departamentos.pdf'); |
||||
|
} |
||||
|
} |
@ -0,0 +1,13 @@ |
|||||
|
<?php |
||||
|
|
||||
|
namespace App\Models; |
||||
|
|
||||
|
use Illuminate\Database\Eloquent\Factories\HasFactory; |
||||
|
use Illuminate\Database\Eloquent\Model; |
||||
|
|
||||
|
class despartamento extends Model |
||||
|
{ |
||||
|
use HasFactory; |
||||
|
protected $table = 'despartamentos'; |
||||
|
protected $fillable = ['departamento']; |
||||
|
} |
@ -0,0 +1,23 @@ |
|||||
|
<?php |
||||
|
|
||||
|
namespace Database\Factories; |
||||
|
|
||||
|
use Illuminate\Database\Eloquent\Factories\Factory; |
||||
|
|
||||
|
/** |
||||
|
* @extends \Illuminate\Database\Eloquent\Factories\Factory<\App\Models\despartamento> |
||||
|
*/ |
||||
|
class DespartamentoFactory extends Factory |
||||
|
{ |
||||
|
/** |
||||
|
* Define the model's default state. |
||||
|
* |
||||
|
* @return array<string, mixed> |
||||
|
*/ |
||||
|
public function definition(): array |
||||
|
{ |
||||
|
return [ |
||||
|
// |
||||
|
]; |
||||
|
} |
||||
|
} |
@ -0,0 +1,32 @@ |
|||||
|
<?php |
||||
|
|
||||
|
use Illuminate\Database\Migrations\Migration; |
||||
|
use Illuminate\Database\Schema\Blueprint; |
||||
|
use Illuminate\Support\Facades\Schema; |
||||
|
|
||||
|
return new class extends Migration |
||||
|
{ |
||||
|
/** |
||||
|
* Run the migrations. |
||||
|
*/ |
||||
|
public function up(): void |
||||
|
{ |
||||
|
Schema::create('despartamentos', function (Blueprint $table) { |
||||
|
$table->id(); |
||||
|
$table->string('departamento'); |
||||
|
$table->timestamps(); |
||||
|
}); |
||||
|
DB::table('despartamentos')->insert(['departamento'=> 'fw3f']); |
||||
|
DB::table('despartamentos')->insert(['departamento'=> 'wfaf']); |
||||
|
DB::table('despartamentos')->insert(['departamento'=> 'asef']); |
||||
|
DB::table('despartamentos')->insert(['departamento'=> 'FE fe']); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Reverse the migrations. |
||||
|
*/ |
||||
|
public function down(): void |
||||
|
{ |
||||
|
Schema::dropIfExists('despartamentos'); |
||||
|
} |
||||
|
}; |
@ -0,0 +1,28 @@ |
|||||
|
<?php |
||||
|
|
||||
|
use Illuminate\Database\Migrations\Migration; |
||||
|
use Illuminate\Database\Schema\Blueprint; |
||||
|
use Illuminate\Support\Facades\Schema; |
||||
|
|
||||
|
return new class extends Migration |
||||
|
{ |
||||
|
/** |
||||
|
* Run the migrations. |
||||
|
*/ |
||||
|
public function up(): void |
||||
|
{ |
||||
|
Schema::table('despartamentos', function (Blueprint $table) { |
||||
|
$table->boolean('eliminado')->default(false); |
||||
|
}); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Reverse the migrations. |
||||
|
*/ |
||||
|
public function down(): void |
||||
|
{ |
||||
|
Schema::table('despartamentos', function (Blueprint $table) { |
||||
|
$table->dropColumn('eliminado'); |
||||
|
}); |
||||
|
} |
||||
|
}; |
@ -0,0 +1,142 @@ |
|||||
|
@extends('layouts.dashboard') |
||||
|
|
||||
|
@section('content') |
||||
|
<div class="container mx-auto px-4 py-6"> |
||||
|
<!-- Mensajes de éxito y error --> |
||||
|
@if(session('success')) |
||||
|
<div id="success-message" class="bg-green-100 border border-green-400 text-green-700 px-4 py-3 rounded relative mb-4" role="alert"> |
||||
|
<span class="block sm:inline">{{ session('success') }}</span> |
||||
|
</div> |
||||
|
@endif |
||||
|
|
||||
|
@if(session('error')) |
||||
|
<div class="bg-red-100 border border-red-400 text-red-700 px-4 py-3 rounded relative mb-4" role="alert"> |
||||
|
<span class="block sm:inline">{{ session('error') }}</span> |
||||
|
</div> |
||||
|
@endif |
||||
|
|
||||
|
<div class="bg-white rounded-lg shadow-lg"> |
||||
|
<!-- Encabezado con título y botones de acción --> |
||||
|
<div class="p-4 border-b border-gray-200 flex justify-between items-center"> |
||||
|
<h2 class="text-2xl font-bold">Gestión de Departamentos</h2> |
||||
|
<div class="flex items-center space-x-6"> |
||||
|
<!-- Íconos de agregar --> |
||||
|
<div class="flex space-x-4"> |
||||
|
<a href="{{ route('despartamentos.excel') }}" |
||||
|
class="text-green-600 hover:text-green-700 transition-colors duration-200" |
||||
|
title="Exportar a Excel"> |
||||
|
<i class="fas fa-file-excel text-xl"></i> |
||||
|
</a> |
||||
|
|
||||
|
<a href="{{ route('despartamentos.pdf') }}" |
||||
|
class="text-red-600 hover:text-red-700 transition-colors duration-200" |
||||
|
title="Exportar a PDF"> |
||||
|
<i class="fas fa-file-pdf text-xl"></i> |
||||
|
</a> |
||||
|
<!-- Agregar nuevo departamento --> |
||||
|
<a href="{{ route('despartamentos.create') }}" |
||||
|
class="text-blue-500 hover:text-blue-600 transition-colors duration-200" |
||||
|
title="Agregar nuevo departamento"> |
||||
|
<i class="fas fa-plus text-xl"></i> |
||||
|
</a> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
|
||||
|
<!-- Barra de búsqueda --> |
||||
|
<div class="p-4 border-b border-gray-200 bg-gray-50"> |
||||
|
<form action="{{ route('despartamentos.index') }}" method="GET" class="flex gap-2"> |
||||
|
<div class="relative w-full sm:w-64"> |
||||
|
<input type="text" |
||||
|
name="busqueda" |
||||
|
placeholder="Buscar departamento..." |
||||
|
value="{{ request('busqueda') }}" |
||||
|
class="w-full pl-10 pr-4 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500"> |
||||
|
<div class="absolute left-3 top-2.5 text-gray-400"> |
||||
|
<i class="fas fa-search"></i> |
||||
|
</div> |
||||
|
</div> |
||||
|
<button type="submit" class="px-4 py-2 bg-blue-500 text-white rounded-lg hover:bg-blue-600"> |
||||
|
Buscar |
||||
|
</button> |
||||
|
@if(request('busqueda')) |
||||
|
<a href="{{ route('despartamentos.index') }}" class="px-4 py-2 bg-gray-500 text-white rounded-lg hover:bg-gray-600"> |
||||
|
Limpiar |
||||
|
</a> |
||||
|
@endif |
||||
|
</form> |
||||
|
</div> |
||||
|
|
||||
|
<!-- Tabla de departamentos --> |
||||
|
<div class="overflow-x-auto"> |
||||
|
<table class="min-w-full divide-y divide-gray-200"> |
||||
|
<thead class="bg-gray-50"> |
||||
|
<tr> |
||||
|
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Número</th> |
||||
|
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Departamento</th> |
||||
|
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Acciones</th> |
||||
|
</tr> |
||||
|
</thead> |
||||
|
<tbody class="bg-white divide-y divide-gray-200"> |
||||
|
@foreach($despartamentos as $index => $despartamento) |
||||
|
<tr class="hover:bg-gray-50"> |
||||
|
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">{{ $index + 1 }}</td> |
||||
|
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-900"> |
||||
|
<i class="fas fa-building text-blue-500 mr-2"></i> |
||||
|
{{ $despartamento->departamento }} |
||||
|
</td> |
||||
|
<td class="flex space-x-2 px-6 py-4 whitespace-nowrap text-sm"> |
||||
|
<a href="{{ route('despartamentos.edit', $despartamento->id) }}" |
||||
|
class="text-yellow-600 hover:text-yellow-700 transition-colors duration-200" |
||||
|
title="Editar departamento"> |
||||
|
<i class="fas fa-edit"></i> |
||||
|
</a> |
||||
|
<form action="{{ route('despartamentos.destroy', $despartamento->id) }}" method="POST" class="d-inline"> |
||||
|
@csrf |
||||
|
@method('DELETE') |
||||
|
<a href="#" onclick="event.preventDefault(); confirmarEliminacion(this);" |
||||
|
class="text-red-600 hover:text-red-700 transition-colors duration-200" |
||||
|
title="Eliminar departamento"> |
||||
|
<i class="fas fa-trash"></i> |
||||
|
</a> |
||||
|
</form> |
||||
|
</td> |
||||
|
</tr> |
||||
|
@endforeach |
||||
|
</tbody> |
||||
|
</table> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
|
||||
|
<script> |
||||
|
// Desaparecer el mensaje después de 3 segundos |
||||
|
setTimeout(function() { |
||||
|
var message = document.getElementById('success-message'); |
||||
|
if (message) { |
||||
|
message.style.transition = 'opacity 0.5s ease'; |
||||
|
message.style.opacity = '0'; |
||||
|
setTimeout(function() { |
||||
|
message.remove(); |
||||
|
}, 500); |
||||
|
} |
||||
|
}, 3000); |
||||
|
|
||||
|
function confirmarEliminacion(button) { |
||||
|
Swal.fire({ |
||||
|
title: '¿Estás seguro?', |
||||
|
text: "Esta acción no se puede deshacer", |
||||
|
icon: 'warning', |
||||
|
showCancelButton: true, |
||||
|
confirmButtonColor: '#3085d6', |
||||
|
cancelButtonColor: '#d33', |
||||
|
confirmButtonText: 'Sí, eliminar', |
||||
|
cancelButtonText: 'Cancelar' |
||||
|
}).then((result) => { |
||||
|
if (result.isConfirmed) { |
||||
|
button.closest('form').submit(); |
||||
|
} |
||||
|
}); |
||||
|
} |
||||
|
</script> |
||||
|
@endsection |
@ -0,0 +1,82 @@ |
|||||
|
@extends('layouts.dashboard') |
||||
|
|
||||
|
@section('content') |
||||
|
<div class="container mx-auto px-4 py-6"> |
||||
|
<div class="max-w-lg mx-auto"> |
||||
|
<div class="bg-white rounded-lg shadow-lg overflow-hidden"> |
||||
|
<div class="p-6"> |
||||
|
<!-- Encabezado del formulario --> |
||||
|
<div class="flex items-center justify-between mb-6"> |
||||
|
<h2 class="text-2xl font-bold text-gray-800"> |
||||
|
{{ isset($despartamento) ? 'Editar Departamento' : 'Nuevo Departamento' }} |
||||
|
</h2> |
||||
|
<div class="h-10 w-10 bg-blue-100 rounded-full flex items-center justify-center"> |
||||
|
<i class="fas fa-building text-blue-600"></i> |
||||
|
</div> |
||||
|
</div> |
||||
|
|
||||
|
<!-- Mensajes de error --> |
||||
|
@if($errors->any()) |
||||
|
<div class="mb-6 bg-red-50 border-l-4 border-red-500 p-4 rounded-r-lg"> |
||||
|
<div class="flex items-center"> |
||||
|
<i class="fas fa-exclamation-circle text-red-500 mr-3"></i> |
||||
|
<div class="text-red-700"> |
||||
|
<ul> |
||||
|
@foreach($errors->all() as $error) |
||||
|
<li>{{ $error }}</li> |
||||
|
@endforeach |
||||
|
</ul> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
@endif |
||||
|
<!-- Formulario --> |
||||
|
<form id="despartamentoForm" |
||||
|
action="{{ isset($despartamento) ? route('despartamentos.update', $despartamento->id) : route('despartamentos.store') }}" |
||||
|
method="POST"> |
||||
|
@csrf |
||||
|
@if(isset($despartamento)) |
||||
|
@method('PUT') |
||||
|
@endif |
||||
|
|
||||
|
<div class="space-y-6"> |
||||
|
<!-- Campo Nombre --> |
||||
|
<div> |
||||
|
<label for="departamento" class="block text-sm font-medium text-gray-700 mb-2"> |
||||
|
Nombre del Departamento |
||||
|
</label> |
||||
|
<div class="relative rounded-md shadow-sm"> |
||||
|
<div class="absolute inset-y-0 left-0 pl-3 flex items-center pointer-events-none"> |
||||
|
<i class="fas fa-tag text-gray-400"></i> |
||||
|
</div> |
||||
|
<input type="text" |
||||
|
name="departamento" |
||||
|
id="departamento" |
||||
|
class="block w-full pl-10 pr-3 py-2 border border-gray-300 rounded-md focus:ring-2 focus:ring-blue-500 focus:border-blue-500" |
||||
|
required |
||||
|
placeholder="Ingresa el nombre del departamento" |
||||
|
value="{{ isset($despartamento) ? $despartamento->departamento : old('departamento') }}"> |
||||
|
</div> |
||||
|
@error('departamento') |
||||
|
<p class="mt-1 text-sm text-red-600">{{ $message }}</p> |
||||
|
@enderror |
||||
|
</div> |
||||
|
|
||||
|
<!-- Botones de acción --> |
||||
|
<div class="flex justify-end space-x-2 pt-4 border-t border-gray-200"> |
||||
|
<a href="{{ route('despartamentos.index') }}" |
||||
|
class="px-4 py-2 border border-gray-300 rounded-md text-sm font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500"> |
||||
|
Cancelar |
||||
|
</a> |
||||
|
<button type="submit" |
||||
|
class="px-4 py-2 border border-transparent rounded-md shadow-sm text-sm font-medium text-white bg-blue-600 hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500"> |
||||
|
{{ isset($despartamento) ? 'Actualizar' : 'Guardar' }} |
||||
|
</button> |
||||
|
</div> |
||||
|
</div> |
||||
|
</form> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
@endsection |
@ -0,0 +1,36 @@ |
|||||
|
<!DOCTYPE html> |
||||
|
<html> |
||||
|
<head> |
||||
|
<title>Departamentos</title> |
||||
|
<style> |
||||
|
table { |
||||
|
width: 100%; |
||||
|
border-collapse: collapse; |
||||
|
} |
||||
|
th, td { |
||||
|
border: 1px solid black; |
||||
|
padding: 8px; |
||||
|
text-align: left; |
||||
|
} |
||||
|
</style> |
||||
|
</head> |
||||
|
<body> |
||||
|
<h1>Lista de Departamentos</h1> |
||||
|
<table> |
||||
|
<thead> |
||||
|
<tr> |
||||
|
<th>ID</th> |
||||
|
<th>Departamento</th> |
||||
|
</tr> |
||||
|
</thead> |
||||
|
<tbody> |
||||
|
@foreach($departamentos as $departamento) |
||||
|
<tr> |
||||
|
<td>{{ $departamento->id }}</td> |
||||
|
<td>{{ $departamento->departamento }}</td> |
||||
|
</tr> |
||||
|
@endforeach |
||||
|
</tbody> |
||||
|
</table> |
||||
|
</body> |
||||
|
</html> |
Loading…
Reference in new issue