Fix: Улучшена форма заявки на курсы

 Организация - необязательна
 Новая структура элемента: Курс → Получатели → Даты
 Поддержка множественных user_ids/group_ids
 Обновлены create, edit, store, update, createAssignments

Co-authored-by: Qwen-Coder <qwen-coder@alibabacloud.com>
This commit is contained in:
mirivlad 2026-04-01 16:19:04 +08:00
parent 7ad06e85fa
commit c6f303fac7
3 changed files with 206 additions and 93 deletions

View File

@ -58,13 +58,16 @@ class CourseRequestController extends Controller
Gate::authorize('create', CourseRequest::class);
$validated = $request->validate([
'organization_id' => 'required|exists:organizations,id',
'organization_id' => 'nullable|exists:organizations,id',
'status' => 'nullable|in:pending,approved,rejected',
'note' => 'nullable|string',
'items' => 'required|array',
'items.*.course_id' => 'required|exists:courses,id',
'items.*.organization_id' => 'nullable|exists:organizations,id',
'items.*.user_id' => 'nullable|exists:users,id',
'items.*.user_ids' => 'nullable|string',
'items.*.group_id' => 'nullable|exists:groups,id',
'items.*.group_ids' => 'nullable|string',
'items.*.start_date' => 'required|date',
'items.*.end_date' => 'nullable|date|after:items.*.start_date',
]);
@ -80,9 +83,12 @@ class CourseRequestController extends Controller
DB::beginTransaction();
try {
// Определяем организацию из первого элемента или общую
$organizationId = $validated['organization_id'] ?? null;
// Создаём заявку
$courseRequest = CourseRequest::create([
'organization_id' => $validated['organization_id'],
'organization_id' => $organizationId,
'requested_by_user_id' => $user->id,
'status' => $status,
'note' => $validated['note'] ?? null,
@ -92,14 +98,64 @@ class CourseRequestController extends Controller
// Создаём элементы заявки
foreach ($validated['items'] as $itemData) {
CourseRequestItem::create([
'course_request_id' => $courseRequest->id,
'course_id' => $itemData['course_id'],
'user_id' => $itemData['user_id'] ?? null,
'group_id' => $itemData['group_id'] ?? null,
'start_date' => $itemData['start_date'],
'end_date' => $itemData['end_date'] ?? null,
]);
// Определяем тип назначения
$hasUsers = !empty($itemData['user_ids']);
$hasGroups = !empty($itemData['group_ids']);
$hasOrganization = !empty($itemData['organization_id']);
// Создаём отдельные записи для каждого пользователя
if ($hasUsers) {
$userIds = array_map('intval', array_filter(explode(',', $itemData['user_ids'])));
foreach ($userIds as $userId) {
CourseRequestItem::create([
'course_request_id' => $courseRequest->id,
'course_id' => $itemData['course_id'],
'user_id' => $userId,
'group_id' => null,
'start_date' => $itemData['start_date'],
'end_date' => $itemData['end_date'] ?? null,
]);
}
}
// Создаём отдельные записи для каждой группы
if ($hasGroups) {
$groupIds = array_map('intval', array_filter(explode(',', $itemData['group_ids'])));
foreach ($groupIds as $groupId) {
CourseRequestItem::create([
'course_request_id' => $courseRequest->id,
'course_id' => $itemData['course_id'],
'user_id' => null,
'group_id' => $groupId,
'start_date' => $itemData['start_date'],
'end_date' => $itemData['end_date'] ?? null,
]);
}
}
// Если только организация (без пользователей и групп)
if (!$hasUsers && !$hasGroups && $hasOrganization) {
CourseRequestItem::create([
'course_request_id' => $courseRequest->id,
'course_id' => $itemData['course_id'],
'user_id' => null,
'group_id' => null,
'start_date' => $itemData['start_date'],
'end_date' => $itemData['end_date'] ?? null,
]);
}
// Если ничего не указано - создаём запись без привязки (для организации заявки)
if (!$hasUsers && !$hasGroups && !$hasOrganization) {
CourseRequestItem::create([
'course_request_id' => $courseRequest->id,
'course_id' => $itemData['course_id'],
'user_id' => null,
'group_id' => null,
'start_date' => $itemData['start_date'],
'end_date' => $itemData['end_date'] ?? null,
]);
}
}
// Если заявка одобрена - сразу создаём назначения
@ -155,6 +211,7 @@ class CourseRequestController extends Controller
'items' => 'required|array',
'items.*.id' => 'nullable|exists:course_request_items,id',
'items.*.course_id' => 'required|exists:courses,id',
'items.*.organization_id' => 'nullable|exists:organizations,id',
'items.*.user_id' => 'nullable|exists:users,id',
'items.*.group_id' => 'nullable|exists:groups,id',
'items.*.start_date' => 'required|date',
@ -173,6 +230,7 @@ class CourseRequestController extends Controller
$item = CourseRequestItem::find($itemData['id']);
$item->update([
'course_id' => $itemData['course_id'],
'organization_id' => $itemData['organization_id'] ?? null,
'user_id' => $itemData['user_id'] ?? null,
'group_id' => $itemData['group_id'] ?? null,
'start_date' => $itemData['start_date'],
@ -183,6 +241,7 @@ class CourseRequestController extends Controller
CourseRequestItem::create([
'course_request_id' => $courseRequest->id,
'course_id' => $itemData['course_id'],
'organization_id' => $itemData['organization_id'] ?? null,
'user_id' => $itemData['user_id'] ?? null,
'group_id' => $itemData['group_id'] ?? null,
'start_date' => $itemData['start_date'],
@ -263,11 +322,20 @@ class CourseRequestController extends Controller
private function createAssignments(CourseRequest $courseRequest): void
{
foreach ($courseRequest->items as $item) {
$type = $item->type;
// Определяем тип назначения
if ($item->user_id) {
$type = 'individual';
} elseif ($item->group_id) {
$type = 'group';
} elseif ($item->organization_id) {
$type = 'organization';
} else {
$type = 'organization'; // По умолчанию на организацию заявки
}
CourseAssignment::create([
'course_id' => $item->course_id,
'organization_id' => $type === 'organization' ? $courseRequest->organization_id : null,
'organization_id' => $item->organization_id ?? ($type === 'organization' ? $courseRequest->organization_id : null),
'group_id' => $item->group_id,
'user_id' => $item->user_id,
'type' => $type,

View File

@ -15,27 +15,14 @@
<div class="row">
<div class="col-md-8 mb-4">
<div class="card shadow-sm">
<div class="card-header bg-primary text-white"><h5 class="mb-0">Курсы и получатели</h5></div>
<div class="card-header bg-primary text-white"><h5 class="mb-0">Элементы заявки</h5></div>
<div class="card-body">
<div class="mb-3">
<label class="form-label">Организация *</label>
<x-searchable-select
name="organization_id"
url="{{ route('api.organizations.search') }}"
placeholder="Начните вводить название организации..."
:required="true"
/>
@error('organization_id')<div class="invalid-feedback d-block">{{ $message }}</div>@enderror
</div>
<hr class="my-4">
<h6 class="mb-3">Элементы заявки</h6>
<div id="items-container">
<div class="item-row border rounded p-3 mb-3 bg-light">
<div class="row g-2">
<div class="col-md-4">
<label class="form-label small">Курс *</label>
<!-- Строка 1: Курс -->
<div class="col-md-12">
<label class="form-label">Курс *</label>
<x-searchable-select
name="items[0][course_id]"
url="{{ route('api.courses.search') }}"
@ -43,32 +30,45 @@
:required="true"
/>
</div>
<div class="col-md-3">
<label class="form-label small">Пользователь</label>
<x-tags-input
name="items[0][user_ids]"
url="{{ route('api.users.search') }}"
placeholder="Или выберите пользователей..."
badge_color="success"
<!-- Строка 2: Получатели -->
<div class="col-md-4">
<label class="form-label">Организация</label>
<x-searchable-select
name="items[0][organization_id]"
url="{{ route('api.organizations.search') }}"
placeholder="Или оставьте пустым..."
/>
<small class="text-muted">Или</small>
<small class="text-muted">И</small>
</div>
<div class="col-md-3">
<label class="form-label small">Группа</label>
<div class="col-md-4">
<label class="form-label">Группа</label>
<x-tags-input
name="items[0][group_ids]"
url="{{ route('api.groups.search') }}"
placeholder="Или выберите группы..."
badge_color="info"
/>
<small class="text-muted">И</small>
</div>
<div class="col-md-2">
<label class="form-label small">Дата начала *</label>
<input type="date" name="items[0][start_date]" class="form-control form-control-sm" required>
<div class="col-md-4">
<label class="form-label">Пользователь</label>
<x-tags-input
name="items[0][user_ids]"
url="{{ route('api.users.search') }}"
placeholder="Или выберите пользователей..."
badge_color="success"
/>
</div>
<div class="col-md-2">
<label class="form-label small">Дата окончания</label>
<input type="date" name="items[0][end_date]" class="form-control form-control-sm">
<!-- Строка 3: Даты -->
<div class="col-md-6">
<label class="form-label">Дата начала *</label>
<input type="date" name="items[0][start_date]" class="form-control" required>
</div>
<div class="col-md-6">
<label class="form-label">Дата окончания</label>
<input type="date" name="items[0][end_date]" class="form-control">
</div>
</div>
<button type="button" class="btn btn-sm btn-outline-danger mt-2 remove-item" style="display:none;"><i class="bi bi-trash"></i> Удалить</button>
@ -123,31 +123,46 @@ document.getElementById('add-item-btn').addEventListener('click', function() {
newItem.className = 'item-row border rounded p-3 mb-3 bg-light';
newItem.innerHTML = `
<div class="row g-2">
<div class="col-md-4">
<label class="form-label small">Курс *</label>
<select name="items[${itemCount}][course_id]" class="form-select form-select-sm" required>
<!-- Строка 1: Курс -->
<div class="col-md-12">
<label class="form-label">Курс *</label>
<select name="items[${itemCount}][course_id]" class="form-select" required>
<option value="">Выберите курс</option>
@foreach($courses as $id => $title)
<option value="{{ $id }}">{{ $title }}</option>
@endforeach
</select>
</div>
<div class="col-md-3">
<label class="form-label small">Пользователь</label>
<input type="text" class="form-control form-control-sm" placeholder="ID пользователей (через запятую)" name="items[${itemCount}][user_ids]">
<small class="text-muted">Или</small>
<!-- Строка 2: Получатели -->
<div class="col-md-4">
<label class="form-label">Организация</label>
<select name="items[${itemCount}][organization_id]" class="form-select">
<option value="">Не выбрано</option>
@foreach($organizations as $id => $name)
<option value="{{ $id }}">{{ $name }}</option>
@endforeach
</select>
<small class="text-muted">И</small>
</div>
<div class="col-md-3">
<label class="form-label small">Группа</label>
<input type="text" class="form-control form-control-sm" placeholder="ID групп (через запятую)" name="items[${itemCount}][group_ids]">
<div class="col-md-4">
<label class="form-label">Группа</label>
<input type="text" class="form-control" placeholder="ID групп (через запятую)" name="items[${itemCount}][group_ids]">
<small class="text-muted">И</small>
</div>
<div class="col-md-2">
<label class="form-label small">Дата начала *</label>
<input type="date" name="items[${itemCount}][start_date]" class="form-control form-control-sm" required>
<div class="col-md-4">
<label class="form-label">Пользователь</label>
<input type="text" class="form-control" placeholder="ID пользователей (через запятую)" name="items[${itemCount}][user_ids]">
</div>
<div class="col-md-2">
<label class="form-label small">Дата окончания</label>
<input type="date" name="items[${itemCount}][end_date]" class="form-control form-control-sm">
<!-- Строка 3: Даты -->
<div class="col-md-6">
<label class="form-label">Дата начала *</label>
<input type="date" name="items[${itemCount}][start_date]" class="form-control" required>
</div>
<div class="col-md-6">
<label class="form-label">Дата окончания</label>
<input type="date" name="items[${itemCount}][end_date]" class="form-control">
</div>
</div>
<button type="button" class="btn btn-sm btn-outline-danger mt-2 remove-item"><i class="bi bi-trash"></i> Удалить</button>

View File

@ -23,30 +23,45 @@
<div class="item-row border rounded p-3 mb-3 bg-light">
<input type="hidden" name="items[{{ $index }}][id]" value="{{ $item->id }}">
<div class="row g-2">
<div class="col-md-4">
<label class="form-label small">Курс *</label>
<select name="items[{{ $index }}][course_id]" class="form-select form-select-sm" required>
<!-- Строка 1: Курс -->
<div class="col-md-12">
<label class="form-label">Курс *</label>
<select name="items[{{ $index }}][course_id]" class="form-select" required>
@foreach($courses as $id => $title)
<option value="{{ $id }}" {{ $item->course_id == $id ? 'selected' : '' }}>{{ $title }}</option>
@endforeach
</select>
</div>
<div class="col-md-3">
<label class="form-label small">Пользователь</label>
<input type="text" class="form-control form-control-sm" value="{{ $item->user_id }}" placeholder="ID пользователей" name="items[{{ $index }}][user_id]">
<small class="text-muted">Или</small>
<!-- Строка 2: Получатели -->
<div class="col-md-4">
<label class="form-label">Организация</label>
<select name="items[{{ $index }}][organization_id]" class="form-select">
<option value="">Не выбрано</option>
@foreach($organizations as $id => $name)
<option value="{{ $id }}" {{ $item->organization_id == $id ? 'selected' : '' }}>{{ $name }}</option>
@endforeach
</select>
<small class="text-muted">И</small>
</div>
<div class="col-md-3">
<label class="form-label small">Группа</label>
<input type="text" class="form-control form-control-sm" value="{{ $item->group_id }}" placeholder="ID групп" name="items[{{ $index }}][group_id]">
<div class="col-md-4">
<label class="form-label">Группа</label>
<input type="text" class="form-control" value="{{ $item->group_id }}" placeholder="ID групп" name="items[{{ $index }}][group_id]">
<small class="text-muted">И</small>
</div>
<div class="col-md-2">
<label class="form-label small">Дата начала *</label>
<input type="date" name="items[{{ $index }}][start_date]" class="form-control form-control-sm" value="{{ $item->start_date->format('Y-m-d') }}" required>
<div class="col-md-4">
<label class="form-label">Пользователь</label>
<input type="text" class="form-control" value="{{ $item->user_id }}" placeholder="ID пользователей" name="items[{{ $index }}][user_id]">
</div>
<div class="col-md-2">
<label class="form-label small">Дата окончания</label>
<input type="date" name="items[{{ $index }}][end_date]" class="form-control form-control-sm" value="{{ $item->end_date?->format('Y-m-d') }}">
<!-- Строка 3: Даты -->
<div class="col-md-6">
<label class="form-label">Дата начала *</label>
<input type="date" name="items[{{ $index }}][start_date]" class="form-control" value="{{ $item->start_date->format('Y-m-d') }}" required>
</div>
<div class="col-md-6">
<label class="form-label">Дата окончания</label>
<input type="date" name="items[{{ $index }}][end_date]" class="form-control" value="{{ $item->end_date?->format('Y-m-d') }}">
</div>
</div>
<button type="button" class="btn btn-sm btn-outline-danger mt-2 remove-item"><i class="bi bi-trash"></i> Удалить</button>
@ -91,31 +106,46 @@ document.getElementById('add-item-btn').addEventListener('click', function() {
newItem.className = 'item-row border rounded p-3 mb-3 bg-light';
newItem.innerHTML = `
<div class="row g-2">
<div class="col-md-4">
<label class="form-label small">Курс *</label>
<select name="items[${itemCount}][course_id]" class="form-select form-select-sm" required>
<!-- Строка 1: Курс -->
<div class="col-md-12">
<label class="form-label">Курс *</label>
<select name="items[${itemCount}][course_id]" class="form-select" required>
<option value="">Выберите курс</option>
@foreach($courses as $id => $title)
<option value="{{ $id }}">{{ $title }}</option>
@endforeach
</select>
</div>
<div class="col-md-3">
<label class="form-label small">Пользователь</label>
<input type="text" class="form-control form-control-sm" placeholder="ID пользователей" name="items[${itemCount}][user_id]">
<small class="text-muted">Или</small>
<!-- Строка 2: Получатели -->
<div class="col-md-4">
<label class="form-label">Организация</label>
<select name="items[${itemCount}][organization_id]" class="form-select">
<option value="">Не выбрано</option>
@foreach($organizations as $id => $name)
<option value="{{ $id }}">{{ $name }}</option>
@endforeach
</select>
<small class="text-muted">И</small>
</div>
<div class="col-md-3">
<label class="form-label small">Группа</label>
<input type="text" class="form-control form-control-sm" placeholder="ID групп" name="items[${itemCount}][group_id]">
<div class="col-md-4">
<label class="form-label">Группа</label>
<input type="text" class="form-control" placeholder="ID групп" name="items[${itemCount}][group_id]">
<small class="text-muted">И</small>
</div>
<div class="col-md-2">
<label class="form-label small">Дата начала *</label>
<input type="date" name="items[${itemCount}][start_date]" class="form-control form-control-sm" required>
<div class="col-md-4">
<label class="form-label">Пользователь</label>
<input type="text" class="form-control" placeholder="ID пользователей" name="items[${itemCount}][user_id]">
</div>
<div class="col-md-2">
<label class="form-label small">Дата окончания</label>
<input type="date" name="items[${itemCount}][end_date]" class="form-control form-control-sm">
<!-- Строка 3: Даты -->
<div class="col-md-6">
<label class="form-label">Дата начала *</label>
<input type="date" name="items[${itemCount}][start_date]" class="form-control" required>
</div>
<div class="col-md-6">
<label class="form-label">Дата окончания</label>
<input type="date" name="items[${itemCount}][end_date]" class="form-control">
</div>
</div>
<button type="button" class="btn btn-sm btn-outline-danger mt-2 remove-item"><i class="bi bi-trash"></i> Удалить</button>