Fix: Улучшена форма заявки на курсы
✅ Организация - необязательна ✅ Новая структура элемента: Курс → Получатели → Даты ✅ Поддержка множественных user_ids/group_ids ✅ Обновлены create, edit, store, update, createAssignments Co-authored-by: Qwen-Coder <qwen-coder@alibabacloud.com>
This commit is contained in:
parent
7ad06e85fa
commit
c6f303fac7
|
|
@ -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,
|
||||
|
|
|
|||
|
|
@ -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>
|
||||
|
|
|
|||
|
|
@ -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>
|
||||
|
|
|
|||
Loading…
Reference in New Issue