Resto de las vistas y plantillas del sistema
This commit is contained in:
@@ -0,0 +1,441 @@
|
||||
<!doctype html>
|
||||
<html lang="es">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<title>Nuevo Profesional</title>
|
||||
<script>
|
||||
(function () {
|
||||
const root = document.documentElement;
|
||||
root.classList.add('sidebar-nav-pending');
|
||||
|
||||
const releasePending = function () {
|
||||
root.classList.remove('sidebar-nav-pending');
|
||||
};
|
||||
|
||||
const timeoutId = window.setTimeout(releasePending, 1200);
|
||||
|
||||
if (document.querySelector('.admin-sidebar')) {
|
||||
window.clearTimeout(timeoutId);
|
||||
releasePending();
|
||||
return;
|
||||
}
|
||||
|
||||
const observer = new MutationObserver(function () {
|
||||
if (!document.querySelector('.admin-sidebar')) {
|
||||
return;
|
||||
}
|
||||
|
||||
observer.disconnect();
|
||||
window.clearTimeout(timeoutId);
|
||||
releasePending();
|
||||
});
|
||||
|
||||
observer.observe(document.documentElement, {
|
||||
childList: true,
|
||||
subtree: true,
|
||||
});
|
||||
})();
|
||||
</script>
|
||||
<style>
|
||||
@media (min-width: 992px) {
|
||||
html.sidebar-nav-pending main {
|
||||
visibility: hidden;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@vite(['resources/css/app.css', 'resources/js/app.js'])
|
||||
</head>
|
||||
<body id="top" class="d-flex flex-column min-vh-100 bg-light">
|
||||
<header class="app-navbar">
|
||||
<nav class="navbar navbar-expand-lg">
|
||||
<div class="container">
|
||||
<a class="navbar-brand d-flex align-items-center" href="/administrador/dashboard">
|
||||
<div class="d-flex align-items-center justify-content-center" style="width: 130px; height: 52px;">
|
||||
<img src="{{ asset('images/logo.png') }}" alt="Logo" class="img-fluid" style="max-height: 70px; width: auto; object-fit: contain;">
|
||||
</div>
|
||||
<span class="app-navbar-greeting ms-2">¡Hola, Administrador!</span>
|
||||
</a>
|
||||
|
||||
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#menuPrincipal" aria-controls="menuPrincipal" aria-expanded="false" aria-label="Toggle navigation">
|
||||
<span class="navbar-toggler-icon"></span>
|
||||
</button>
|
||||
|
||||
<div class="collapse navbar-collapse" id="menuPrincipal">
|
||||
<ul class="navbar-nav mx-auto mb-2 mb-lg-0">
|
||||
<li class="nav-item"><a class="btn app-navbar-link" href="/administrador/profesionales">Profesionales</a></li>
|
||||
<li class="nav-item"><a class="btn app-navbar-link" href="/administrador/dashboard">Mis Datos</a></li>
|
||||
<li class="nav-item"><a class="btn app-navbar-link" href="/administrador/contenido-web">Contenido</a></li>
|
||||
<li class="nav-item"><a class="btn app-navbar-link" href="/administrador/emails">Emails</a></li>
|
||||
<li class="nav-item"><a class="btn app-navbar-link" href="/administrador/logs">Logs</a></li>
|
||||
<li class="nav-item"><a class="btn app-navbar-link" href="/administrador/fallas">Fallas</a></li>
|
||||
<li class="nav-item">
|
||||
<a class="btn app-navbar-link position-relative" href="/administrador/bugs">
|
||||
Bugs
|
||||
@if(($bugsPendientesCount ?? 0) > 0)
|
||||
<span class="position-absolute top-0 start-100 translate-middle p-1 bg-danger rounded-circle">
|
||||
<span class="visually-hidden">Hay bugs pendientes</span>
|
||||
</span>
|
||||
@endif
|
||||
</a>
|
||||
</li> <li class="nav-item"><a class="btn app-navbar-link" href="/administrador/backups">Backup</a></li> </ul>
|
||||
|
||||
<a class="btn app-navbar-link" href="/logout">Cerrar Sesión</a>
|
||||
</div>
|
||||
</div>
|
||||
</nav>
|
||||
</header>
|
||||
|
||||
<main class="container py-4 flex-grow-1">
|
||||
<div class="row justify-content-center">
|
||||
<div class="col-12 col-lg-9">
|
||||
<h1 class="h4 mb-3">Agregar nuevo profesional</h1>
|
||||
|
||||
@if($errors->any())
|
||||
<div class="alert alert-danger" role="alert">
|
||||
<ul class="mb-0 ps-3">
|
||||
@foreach($errors->all() as $error)
|
||||
<li>{{ $error }}</li>
|
||||
@endforeach
|
||||
</ul>
|
||||
</div>
|
||||
@endif
|
||||
|
||||
<div class="card border shadow-sm">
|
||||
<div class="card-body">
|
||||
<form method="POST" action="/administrador/profesionales" enctype="multipart/form-data" class="row g-3">
|
||||
@csrf
|
||||
|
||||
{{-- DNI con botón de verificación --}}
|
||||
<div class="col-12 col-md-6">
|
||||
<label for="dni" class="form-label">DNI</label>
|
||||
<div class="input-group">
|
||||
<input id="dni" name="dni" type="text" class="form-control" value="{{ old('dni') }}" pattern="[0-9A-Za-z]{7,20}" maxlength="20" title="N�meros y letras: 7 a 20 caracteres" required>
|
||||
<button type="button" class="btn btn-outline-secondary" id="btn-verificar-dni">Verificar</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{{-- Espacio para el mensaje de estado de la búsqueda --}}
|
||||
<div class="col-12" id="dni-estado" @if(!old('dni')) style="display:none" @endif>
|
||||
@if(old('persona_existe') === '1')
|
||||
<div class="alert alert-info mb-0">
|
||||
<strong>Persona ya registrada en el sistema.</strong> Complete solo los datos faltantes para registrar a este profesional (foto, correo, matrícula, profesión, servicios y contraseña).
|
||||
</div>
|
||||
@elseif(old('dni'))
|
||||
<div class="alert alert-warning mb-0">
|
||||
<strong>Persona no encontrada.</strong> Complete todos los datos para registrar a esta persona.
|
||||
</div>
|
||||
@endif
|
||||
</div>
|
||||
|
||||
{{-- Input oculto que indica si la persona ya existía --}}
|
||||
<input type="hidden" id="persona_existe" name="persona_existe" value="{{ old('persona_existe', '0') }}">
|
||||
<input type="hidden" id="dni_verificado" name="dni_verificado" value="{{ old('dni_verificado', '') }}">
|
||||
|
||||
{{-- Campos básicos de la persona --}}
|
||||
<div class="col-12 col-md-6" data-campo-persona>
|
||||
<label for="nombre" class="form-label">Nombre</label>
|
||||
<input id="nombre" name="nombre" type="text" class="form-control {{ old('persona_existe') === '1' ? 'bg-light' : '' }}" value="{{ old('nombre') }}" pattern="[A-Za-z������������\s]+" oninput="this.value=this.value.replace(/[^A-Za-z������������\s]/g,'')" {{ old('persona_existe') !== '1' ? 'required' : 'readonly' }}>
|
||||
</div>
|
||||
|
||||
<div class="col-12 col-md-6" data-campo-persona>
|
||||
<label for="apellido" class="form-label">Apellido</label>
|
||||
<input id="apellido" name="apellido" type="text" class="form-control {{ old('persona_existe') === '1' ? 'bg-light' : '' }}" value="{{ old('apellido') }}" pattern="[A-Za-z������������\s]+" oninput="this.value=this.value.replace(/[^A-Za-z������������\s]/g,'')" {{ old('persona_existe') !== '1' ? 'required' : 'readonly' }}>
|
||||
</div>
|
||||
|
||||
<div class="col-12 col-md-6" data-campo-persona>
|
||||
<label for="cuil" class="form-label">CUIL</label>
|
||||
<input id="cuil" name="cuil" type="text" class="form-control {{ old('persona_existe') === '1' ? 'bg-light' : '' }}" value="{{ old('cuil') }}" inputmode="numeric" pattern="[0-9]+" oninput="this.value=this.value.replace(/\D/g,'')" {{ old('persona_existe') !== '1' ? 'required' : 'readonly' }}>
|
||||
</div>
|
||||
|
||||
<div class="col-12 col-md-6" data-campo-persona>
|
||||
<label for="celular" class="form-label">Celular</label>
|
||||
<input id="celular" name="celular" type="text" class="form-control" value="{{ old('celular') }}" inputmode="numeric" pattern="[0-9]+" oninput="this.value=this.value.replace(/\D/g,'')" required>
|
||||
</div>
|
||||
|
||||
<div class="col-12 col-md-6" data-campo-persona>
|
||||
<label for="fechanac" class="form-label">Fecha de nacimiento</label>
|
||||
<input id="fechanac" name="fechanac" type="date" class="form-control {{ old('persona_existe') === '1' ? 'bg-light' : '' }}" value="{{ old('fechanac') }}" {{ old('persona_existe') !== '1' ? 'required' : 'readonly' }}>
|
||||
</div>
|
||||
|
||||
<div class="col-12 col-md-6" id="foto-section">
|
||||
<label for="foto" class="form-label">Foto</label>
|
||||
<input id="foto" name="foto" type="file" class="form-control" accept=".jpg,.jpeg,.png,.webp">
|
||||
<div class="form-text">Opcional. Formatos permitidos: JPG, PNG, WEBP.</div>
|
||||
</div>
|
||||
|
||||
{{-- Correo y matrícula siempre requeridos --}}
|
||||
<div class="col-12 col-md-6">
|
||||
<label for="correo" class="form-label">Correo</label>
|
||||
<input id="correo" name="correo" type="email" class="form-control" value="{{ old('correo') }}" required>
|
||||
</div>
|
||||
|
||||
<div class="col-12 col-md-6">
|
||||
<label for="matricula" class="form-label">Matrícula</label>
|
||||
<input id="matricula" name="matricula" type="text" class="form-control" value="{{ old('matricula') }}" required>
|
||||
</div>
|
||||
|
||||
<div class="col-12 col-md-6">
|
||||
<label for="profesion_id" class="form-label">Profesión</label>
|
||||
<select id="profesion_id" name="profesion_id" class="form-select" required>
|
||||
<option value="">Seleccionar profesión</option>
|
||||
@foreach($profesiones as $profesion)
|
||||
<option value="{{ $profesion->id }}" @selected(old('profesion_id') == $profesion->id)>
|
||||
{{ $profesion->titulo }}
|
||||
</option>
|
||||
@endforeach
|
||||
</select>
|
||||
</div>
|
||||
|
||||
{{-- Usuario autogenerado (solo lectura) --}}
|
||||
<div class="col-12 col-md-6">
|
||||
<label class="form-label">Usuario (autogenerado)</label>
|
||||
<input id="usuario-preview" type="text" class="form-control bg-light" readonly
|
||||
value="{{ old('dni') && old('profesion_id') ? old('dni').'-'.old('profesion_id') : '' }}"
|
||||
placeholder="Se calcula al ingresar DNI y Profesión">
|
||||
<div class="form-text">Formato: <strong>DNI-CódigoProfesión</strong>. Ej.: 43293244-1</div>
|
||||
</div>
|
||||
|
||||
<div class="col-12">
|
||||
<label class="form-label d-block">Servicios que ofrece</label>
|
||||
<div id="servicios-container" class="d-flex flex-wrap gap-3 border rounded p-3 bg-light">
|
||||
@foreach($servicios as $servicio)
|
||||
<div class="form-check" data-profesion-id="{{ $servicio->profesion_id }}">
|
||||
<input
|
||||
class="form-check-input"
|
||||
type="checkbox"
|
||||
id="servicio_{{ $servicio->id }}"
|
||||
name="servicio_ids[]"
|
||||
value="{{ $servicio->id }}"
|
||||
@checked(in_array((string) $servicio->id, array_map('strval', old('servicio_ids', [])), true))
|
||||
>
|
||||
<label class="form-check-label" for="servicio_{{ $servicio->id }}">{{ $servicio->titulo }}</label>
|
||||
</div>
|
||||
@endforeach
|
||||
</div>
|
||||
<div class="form-text">Primero selecciona una profesión para ver sus servicios.</div>
|
||||
</div>
|
||||
|
||||
<div class="col-12 col-md-6">
|
||||
<label for="contra" class="form-label">Contraseña</label>
|
||||
<input id="contra" name="contra" type="password" class="form-control" minlength="6" required>
|
||||
</div>
|
||||
|
||||
<div class="col-12 d-flex gap-2 justify-content-end mt-2">
|
||||
<a href="/administrador/profesionales" class="btn btn-outline-secondary">Cancelar</a>
|
||||
<button type="submit" class="btn btn-primary">Guardar profesional</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</main>
|
||||
|
||||
<script>
|
||||
(function () {
|
||||
const dniInput = document.getElementById('dni');
|
||||
const celularInput = document.getElementById('celular');
|
||||
const btnVerificar = document.getElementById('btn-verificar-dni');
|
||||
const dniEstado = document.getElementById('dni-estado');
|
||||
const personaExisteInput = document.getElementById('persona_existe');
|
||||
const dniVerificadoInput = document.getElementById('dni_verificado');
|
||||
const usuarioPreview = document.getElementById('usuario-preview');
|
||||
const profesionSelect = document.getElementById('profesion_id');
|
||||
const serviciosContainer = document.getElementById('servicios-container');
|
||||
const formulario = document.querySelector('form[action="/administrador/profesionales"]');
|
||||
const controlesBloqueables = formulario
|
||||
? Array.from(formulario.querySelectorAll('input, select, textarea, button'))
|
||||
: [];
|
||||
|
||||
/* --- Preview del usuario autogenerado --- */
|
||||
const actualizarUsuario = () => {
|
||||
const dni = dniInput.value.trim();
|
||||
const profesionId = profesionSelect.value;
|
||||
usuarioPreview.value = (dni && profesionId) ? `${dni}-${profesionId}` : '';
|
||||
};
|
||||
|
||||
const dniEstaVerificado = () => {
|
||||
const dniActual = dniInput.value.trim();
|
||||
return dniActual !== '' && dniVerificadoInput.value === dniActual;
|
||||
};
|
||||
|
||||
const aplicarBloqueoVerificacion = () => {
|
||||
const verificado = dniEstaVerificado();
|
||||
|
||||
controlesBloqueables.forEach((control) => {
|
||||
const esPermitido = control.id === 'dni'
|
||||
|| control.id === 'btn-verificar-dni'
|
||||
|| control.id === 'persona_existe'
|
||||
|| control.id === 'dni_verificado'
|
||||
|| control.type === 'hidden'
|
||||
|| control.name === '_token'
|
||||
|| control.name === '_method';
|
||||
|
||||
if (esPermitido) {
|
||||
return;
|
||||
}
|
||||
|
||||
control.disabled = !verificado;
|
||||
});
|
||||
};
|
||||
|
||||
dniInput.addEventListener('input', () => {
|
||||
dniInput.value = dniInput.value.replace(/[^0-9A-Za-z]/g, '');
|
||||
actualizarUsuario();
|
||||
personaExisteInput.value = '0';
|
||||
dniVerificadoInput.value = '';
|
||||
aplicarBloqueoVerificacion();
|
||||
filtrarServicios();
|
||||
});
|
||||
if (celularInput) {
|
||||
celularInput.addEventListener('input', () => {
|
||||
celularInput.value = celularInput.value.replace(/\D/g, '');
|
||||
});
|
||||
}
|
||||
profesionSelect.addEventListener('change', actualizarUsuario);
|
||||
|
||||
/* --- Filtro de servicios por profesi�n --- */
|
||||
const itemsServicio = Array.from(serviciosContainer.querySelectorAll('[data-profesion-id]'));
|
||||
|
||||
const filtrarServicios = () => {
|
||||
const profesionId = profesionSelect.value;
|
||||
const verificado = dniEstaVerificado();
|
||||
|
||||
itemsServicio.forEach((item) => {
|
||||
const checkbox = item.querySelector('input[type="checkbox"]');
|
||||
const visible = verificado && Boolean(profesionId) && item.dataset.profesionId === profesionId;
|
||||
item.hidden = !visible;
|
||||
if (checkbox) {
|
||||
checkbox.disabled = !visible;
|
||||
if (!visible) checkbox.checked = false;
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
filtrarServicios();
|
||||
profesionSelect.addEventListener('change', filtrarServicios);
|
||||
|
||||
/* --- Helpers para aplicar el estado encontrada/no-encontrada --- */
|
||||
const aplicarPersonaEncontrada = (persona) => {
|
||||
personaExisteInput.value = '1';
|
||||
dniVerificadoInput.value = dniInput.value.trim();
|
||||
aplicarBloqueoVerificacion();
|
||||
|
||||
dniEstado.innerHTML = `<div class="alert alert-info mb-0">
|
||||
<strong>Persona ya registrada en el sistema.</strong>
|
||||
Complete solo los datos faltantes para registrar a este profesional (foto, correo, matrícula, profesión, servicios y contraseña).
|
||||
</div>`;
|
||||
dniEstado.style.display = '';
|
||||
|
||||
document.getElementById('nombre').value = persona.nombre;
|
||||
document.getElementById('apellido').value = persona.apellido;
|
||||
document.getElementById('cuil').value = persona.cuil;
|
||||
document.getElementById('celular').value = persona.celular ?? '';
|
||||
document.getElementById('fechanac').value = persona.fechanac ?? '';
|
||||
|
||||
['nombre', 'apellido', 'cuil', 'fechanac'].forEach((id) => {
|
||||
const el = document.getElementById(id);
|
||||
el.setAttribute('readonly', true);
|
||||
el.removeAttribute('required');
|
||||
el.classList.add('bg-light');
|
||||
});
|
||||
|
||||
actualizarUsuario();
|
||||
filtrarServicios();
|
||||
};
|
||||
|
||||
const aplicarPersonaNoEncontrada = (limpiar = true) => {
|
||||
personaExisteInput.value = '0';
|
||||
dniVerificadoInput.value = dniInput.value.trim();
|
||||
aplicarBloqueoVerificacion();
|
||||
|
||||
dniEstado.innerHTML = `<div class="alert alert-warning mb-0">
|
||||
<strong>Persona no encontrada.</strong>
|
||||
Complete todos los datos para registrar a esta persona.
|
||||
</div>`;
|
||||
dniEstado.style.display = '';
|
||||
|
||||
['nombre', 'apellido', 'cuil', 'fechanac'].forEach((id) => {
|
||||
const el = document.getElementById(id);
|
||||
el.removeAttribute('disabled');
|
||||
el.removeAttribute('readonly');
|
||||
el.setAttribute('required', true);
|
||||
el.classList.remove('bg-light');
|
||||
if (limpiar) el.value = '';
|
||||
});
|
||||
|
||||
actualizarUsuario();
|
||||
filtrarServicios();
|
||||
};
|
||||
|
||||
/* --- Verificaci�n contra el backend --- */
|
||||
const verificarDni = async () => {
|
||||
const dni = dniInput.value.trim();
|
||||
if (dni.length < 3) return;
|
||||
|
||||
btnVerificar.disabled = true;
|
||||
btnVerificar.textContent = '...';
|
||||
|
||||
try {
|
||||
const res = await fetch(`/administrador/profesionales/buscar-persona?dni=${encodeURIComponent(dni)}`);
|
||||
const data = await res.json();
|
||||
|
||||
if (data.encontrada) {
|
||||
aplicarPersonaEncontrada(data.persona);
|
||||
} else {
|
||||
aplicarPersonaNoEncontrada();
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('Error al verificar DNI:', err);
|
||||
} finally {
|
||||
btnVerificar.disabled = false;
|
||||
btnVerificar.textContent = 'Verificar';
|
||||
}
|
||||
};
|
||||
|
||||
btnVerificar.addEventListener('click', verificarDni);
|
||||
dniInput.addEventListener('keydown', (e) => {
|
||||
if (e.key === 'Enter') { e.preventDefault(); verificarDni(); }
|
||||
});
|
||||
|
||||
/* Si al cargar la p�gina ya hay un valor en DNI (re-submit con errores),
|
||||
y persona_existe ya era 1, restauramos el estado visual */
|
||||
const existeAlCargar = personaExisteInput.value === '1';
|
||||
aplicarBloqueoVerificacion();
|
||||
if (dniEstaVerificado() && dniInput.value.trim()) {
|
||||
actualizarUsuario();
|
||||
filtrarServicios();
|
||||
}
|
||||
|
||||
if (existeAlCargar && dniEstaVerificado()) {
|
||||
['nombre', 'apellido', 'cuil', 'fechanac'].forEach((id) => {
|
||||
const el = document.getElementById(id);
|
||||
el.removeAttribute('disabled');
|
||||
el.setAttribute('readonly', true);
|
||||
el.removeAttribute('required');
|
||||
el.classList.add('bg-light');
|
||||
});
|
||||
}
|
||||
|
||||
formulario?.addEventListener('submit', (e) => {
|
||||
const dniActual = dniInput.value.trim();
|
||||
|
||||
if (!dniActual || dniVerificadoInput.value !== dniActual) {
|
||||
e.preventDefault();
|
||||
dniEstado.innerHTML = `<div class="alert alert-warning mb-0">
|
||||
<strong>Debe verificar el DNI.</strong>
|
||||
Presione el bot�n Verificar antes de guardar el profesional.
|
||||
</div>`;
|
||||
dniEstado.style.display = '';
|
||||
btnVerificar.focus();
|
||||
}
|
||||
});
|
||||
})();
|
||||
</script>
|
||||
@include('partials.reportar-falla-boton')
|
||||
</body>
|
||||
</html>
|
||||
|
||||
|
||||
|
||||
Reference in New Issue
Block a user