CRUD организаций - Этап 2

 OrganizationController (index, create, store, show, edit, update, destroy)
 OrganizationPolicy (viewAny, view, create, update, delete)
 Маршруты: /admin/organizations (resource)
 Blade-шаблоны:
  - admin/organizations/index.blade.php (список с пагинацией)
  - admin/organizations/create.blade.php (форма создания)
  - admin/organizations/show.blade.php (просмотр + статистика)
  - admin/organizations/edit.blade.php (форма редактирования)
 Обновлённое меню в dashboard/admin.blade.php

Co-authored-by: Qwen-Coder <qwen-coder@alibabacloud.com>
This commit is contained in:
mirivlad 2026-03-26 08:41:45 +08:00
parent 244c56df39
commit 2271abf732
47 changed files with 804 additions and 5 deletions

0
app/Console/Kernel.php Normal file → Executable file
View File

0
app/Exceptions/Handler.php Normal file → Executable file
View File

View File

@ -0,0 +1,108 @@
<?php
namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use App\Models\Organization;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Gate;
class OrganizationController extends Controller
{
public function __construct()
{
$this->middleware('auth');
}
public function index()
{
Gate::authorize('viewAny', Organization::class);
$organizations = Organization::withCount('users', 'groups', 'courses')
->orderBy('created_at', 'desc')
->paginate(20);
return view('admin.organizations.index', compact('organizations'));
}
public function create()
{
Gate::authorize('create', Organization::class);
return view('admin.organizations.create');
}
public function store(Request $request)
{
Gate::authorize('create', Organization::class);
$validated = $request->validate([
'name' => 'required|string|max:255|unique:organizations',
'inn' => 'nullable|string|max:20',
'kpp' => 'nullable|string|max:20',
'address' => 'nullable|string|max:500',
'phone' => 'nullable|string|max:20',
'email' => 'nullable|email|max:255',
'description' => 'nullable|string',
]);
$validated['is_active'] = $request->boolean('is_active');
Organization::create($validated);
return redirect()->route('admin.organizations.index')
->with('success', 'Организация успешно создана.');
}
public function show(Organization $organization)
{
Gate::authorize('view', $organization);
$organization->load(['users', 'groups', 'courseRequests']);
return view('admin.organizations.show', compact('organization'));
}
public function edit(Organization $organization)
{
Gate::authorize('update', $organization);
return view('admin.organizations.edit', compact('organization'));
}
public function update(Request $request, Organization $organization)
{
Gate::authorize('update', $organization);
$validated = $request->validate([
'name' => 'required|string|max:255|unique:organizations,name,' . $organization->id,
'inn' => 'nullable|string|max:20',
'kpp' => 'nullable|string|max:20',
'address' => 'nullable|string|max:500',
'phone' => 'nullable|string|max:20',
'email' => 'nullable|email|max:255',
'description' => 'nullable|string',
]);
$validated['is_active'] = $request->boolean('is_active');
$organization->update($validated);
return redirect()->route('admin.organizations.show', $organization)
->with('success', 'Организация успешно обновлена.');
}
public function destroy(Organization $organization)
{
Gate::authorize('delete', $organization);
if ($organization->users()->count() > 0) {
return back()->with('error', 'Невозможно удалить организацию с пользователями.');
}
$organization->delete();
return redirect()->route('admin.organizations.index')
->with('success', 'Организация успешно удалена.');
}
}

0
app/Http/Controllers/Auth/LoginController.php Normal file → Executable file
View File

0
app/Http/Controllers/Auth/RegisterController.php Normal file → Executable file
View File

0
app/Http/Controllers/Controller.php Normal file → Executable file
View File

0
app/Http/Controllers/DashboardController.php Normal file → Executable file
View File

0
app/Http/Kernel.php Normal file → Executable file
View File

0
app/Http/Middleware/Authenticate.php Normal file → Executable file
View File

0
app/Http/Middleware/EncryptCookies.php Normal file → Executable file
View File

View File

0
app/Http/Middleware/RedirectIfAuthenticated.php Normal file → Executable file
View File

0
app/Http/Middleware/TrimStrings.php Normal file → Executable file
View File

0
app/Http/Middleware/TrustHosts.php Normal file → Executable file
View File

0
app/Http/Middleware/TrustProxies.php Normal file → Executable file
View File

0
app/Http/Middleware/ValidateSignature.php Normal file → Executable file
View File

0
app/Http/Middleware/VerifyCsrfToken.php Normal file → Executable file
View File

0
app/Models/Answer.php Normal file → Executable file
View File

0
app/Models/Course.php Normal file → Executable file
View File

0
app/Models/CourseAssignment.php Normal file → Executable file
View File

0
app/Models/CourseCategory.php Normal file → Executable file
View File

0
app/Models/CourseModule.php Normal file → Executable file
View File

0
app/Models/CourseRequest.php Normal file → Executable file
View File

0
app/Models/CourseRequestItem.php Normal file → Executable file
View File

0
app/Models/Group.php Normal file → Executable file
View File

0
app/Models/Log.php Normal file → Executable file
View File

0
app/Models/Organization.php Normal file → Executable file
View File

0
app/Models/Question.php Normal file → Executable file
View File

0
app/Models/QuestionMatchingPair.php Normal file → Executable file
View File

0
app/Models/ScormData.php Normal file → Executable file
View File

0
app/Models/Test.php Normal file → Executable file
View File

0
app/Models/TestAttempt.php Normal file → Executable file
View File

0
app/Models/TestResponse.php Normal file → Executable file
View File

0
app/Models/User.php Normal file → Executable file
View File

0
app/Models/UserCourseProgress.php Normal file → Executable file
View File

View File

@ -0,0 +1,65 @@
<?php
namespace App\Policies;
use App\Models\Organization;
use App\Models\User;
class OrganizationPolicy
{
/**
* Determine whether the user can view any models.
*/
public function viewAny(User $user): bool
{
return $user->hasRole(['Administrator', 'Manager']);
}
/**
* Determine whether the user can view the model.
*/
public function view(User $user, Organization $organization): bool
{
return $user->hasRole(['Administrator', 'Manager', 'Curator']);
}
/**
* Determine whether the user can create models.
*/
public function create(User $user): bool
{
return $user->hasRole(['Administrator', 'Manager']);
}
/**
* Determine whether the user can update the model.
*/
public function update(User $user, Organization $organization): bool
{
return $user->hasRole(['Administrator', 'Manager']);
}
/**
* Determine whether the user can delete the model.
*/
public function delete(User $user, Organization $organization): bool
{
return $user->hasRole(['Administrator']);
}
/**
* Determine whether the user can restore the model.
*/
public function restore(User $user, Organization $organization): bool
{
return $user->hasRole(['Administrator']);
}
/**
* Determine whether the user can permanently delete the model.
*/
public function forceDelete(User $user, Organization $organization): bool
{
return $user->hasRole(['Administrator']);
}
}

0
app/Providers/AppServiceProvider.php Normal file → Executable file
View File

5
app/Providers/AuthServiceProvider.php Normal file → Executable file
View File

@ -2,7 +2,8 @@
namespace App\Providers; namespace App\Providers;
// use Illuminate\Support\Facades\Gate; use App\Models\Organization;
use App\Policies\OrganizationPolicy;
use Illuminate\Foundation\Support\Providers\AuthServiceProvider as ServiceProvider; use Illuminate\Foundation\Support\Providers\AuthServiceProvider as ServiceProvider;
class AuthServiceProvider extends ServiceProvider class AuthServiceProvider extends ServiceProvider
@ -13,7 +14,7 @@ class AuthServiceProvider extends ServiceProvider
* @var array<class-string, class-string> * @var array<class-string, class-string>
*/ */
protected $policies = [ protected $policies = [
// Organization::class => OrganizationPolicy::class,
]; ];
/** /**

0
app/Providers/BroadcastServiceProvider.php Normal file → Executable file
View File

0
app/Providers/EventServiceProvider.php Normal file → Executable file
View File

0
app/Providers/RouteServiceProvider.php Normal file → Executable file
View File

View File

@ -0,0 +1,143 @@
@extends('layouts.app')
@section('title', 'Добавить организацию')
@section('content')
<div class="container-fluid">
<div class="row">
<nav class="col-md-3 col-lg-2 d-md-block sidebar collapse">
<div class="position-sticky pt-3">
<ul class="nav flex-column">
<li class="nav-item">
<a class="nav-link" href="{{ route('dashboard') }}">
<i class="bi bi-speedometer2"></i> Панель управления
</a>
</li>
<li class="nav-item">
<a class="nav-link" href="{{ route('admin.organizations.index') }}">
<i class="bi bi-building"></i> Организации
</a>
</li>
</ul>
</div>
</nav>
<main class="col-md-9 ms-sm-auto col-lg-10 px-md-4 main-content">
<div class="d-flex justify-content-between flex-wrap flex-md-nowrap align-items-center pt-3 pb-2 mb-3 border-bottom">
<h1 class="h2">Добавить организацию</h1>
<a href="{{ route('admin.organizations.index') }}" class="btn btn-secondary btn-sm">
<i class="bi bi-arrow-left"></i> Назад
</a>
</div>
@if ($errors->any())
<div class="alert alert-danger">
<ul class="mb-0">
@foreach ($errors->all() as $error)
<li>{{ $error }}</li>
@endforeach
</ul>
</div>
@endif
<form action="{{ route('admin.organizations.store') }}" method="POST">
@csrf
<div class="row">
<div class="col-md-8">
<div class="card shadow-sm mb-4">
<div class="card-body">
<h5 class="card-title mb-3">Основная информация</h5>
<div class="mb-3">
<label for="name" class="form-label">Название <span class="text-danger">*</span></label>
<input type="text" class="form-control @error('name') is-invalid @enderror"
id="name" name="name" value="{{ old('name') }}" required autofocus>
@error('name')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3">
<label for="description" class="form-label">Описание</label>
<textarea class="form-control @error('description') is-invalid @enderror"
id="description" name="description" rows="3">{{ old('description') }}</textarea>
@error('description')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3 form-check">
<input type="checkbox" class="form-check-input" id="is_active" name="is_active" value="1" {{ old('is_active', true) ? 'checked' : '' }}>
<label class="form-check-label" for="is_active">Активна</label>
</div>
</div>
</div>
</div>
<div class="col-md-4">
<div class="card shadow-sm mb-4">
<div class="card-body">
<h5 class="card-title mb-3">Контактная информация</h5>
<div class="mb-3">
<label for="inn" class="form-label">ИНН</label>
<input type="text" class="form-control @error('inn') is-invalid @enderror"
id="inn" name="inn" value="{{ old('inn') }}" maxlength="20">
@error('inn')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3">
<label for="kpp" class="form-label">КПП</label>
<input type="text" class="form-control @error('kpp') is-invalid @enderror"
id="kpp" name="kpp" value="{{ old('kpp') }}" maxlength="20">
@error('kpp')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3">
<label for="email" class="form-label">Email</label>
<input type="email" class="form-control @error('email') is-invalid @enderror"
id="email" name="email" value="{{ old('email') }}">
@error('email')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3">
<label for="phone" class="form-label">Телефон</label>
<input type="tel" class="form-control @error('phone') is-invalid @enderror"
id="phone" name="phone" value="{{ old('phone') }}" maxlength="20">
@error('phone')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3">
<label for="address" class="form-label">Адрес</label>
<textarea class="form-control @error('address') is-invalid @enderror"
id="address" name="address" rows="2">{{ old('address') }}</textarea>
@error('address')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
</div>
</div>
</div>
</div>
<div class="mb-3">
<button type="submit" class="btn btn-primary">
<i class="bi bi-check-lg"></i> Создать организацию
</button>
<a href="{{ route('admin.organizations.index') }}" class="btn btn-secondary">
<i class="bi bi-x-lg"></i> Отмена
</a>
</div>
</form>
</main>
</div>
</div>
@endsection

View File

@ -0,0 +1,144 @@
@extends('layouts.app')
@section('title', 'Редактировать организацию')
@section('content')
<div class="container-fluid">
<div class="row">
<nav class="col-md-3 col-lg-2 d-md-block sidebar collapse">
<div class="position-sticky pt-3">
<ul class="nav flex-column">
<li class="nav-item">
<a class="nav-link" href="{{ route('dashboard') }}">
<i class="bi bi-speedometer2"></i> Панель управления
</a>
</li>
<li class="nav-item">
<a class="nav-link" href="{{ route('admin.organizations.index') }}">
<i class="bi bi-building"></i> Организации
</a>
</li>
</ul>
</div>
</nav>
<main class="col-md-9 ms-sm-auto col-lg-10 px-md-4 main-content">
<div class="d-flex justify-content-between flex-wrap flex-md-nowrap align-items-center pt-3 pb-2 mb-3 border-bottom">
<h1 class="h2">Редактировать организацию</h1>
<a href="{{ route('admin.organizations.show', $organization) }}" class="btn btn-secondary btn-sm">
<i class="bi bi-arrow-left"></i> Назад
</a>
</div>
@if ($errors->any())
<div class="alert alert-danger">
<ul class="mb-0">
@foreach ($errors->all() as $error)
<li>{{ $error }}</li>
@endforeach
</ul>
</div>
@endif
<form action="{{ route('admin.organizations.update', $organization) }}" method="POST">
@csrf
@method('PUT')
<div class="row">
<div class="col-md-8">
<div class="card shadow-sm mb-4">
<div class="card-body">
<h5 class="card-title mb-3">Основная информация</h5>
<div class="mb-3">
<label for="name" class="form-label">Название <span class="text-danger">*</span></label>
<input type="text" class="form-control @error('name') is-invalid @enderror"
id="name" name="name" value="{{ old('name', $organization->name) }}" required autofocus>
@error('name')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3">
<label for="description" class="form-label">Описание</label>
<textarea class="form-control @error('description') is-invalid @enderror"
id="description" name="description" rows="3">{{ old('description', $organization->description) }}</textarea>
@error('description')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3 form-check">
<input type="checkbox" class="form-check-input" id="is_active" name="is_active" value="1" {{ old('is_active', $organization->is_active) ? 'checked' : '' }}>
<label class="form-check-label" for="is_active">Активна</label>
</div>
</div>
</div>
</div>
<div class="col-md-4">
<div class="card shadow-sm mb-4">
<div class="card-body">
<h5 class="card-title mb-3">Контактная информация</h5>
<div class="mb-3">
<label for="inn" class="form-label">ИНН</label>
<input type="text" class="form-control @error('inn') is-invalid @enderror"
id="inn" name="inn" value="{{ old('inn', $organization->inn) }}" maxlength="20">
@error('inn')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3">
<label for="kpp" class="form-label">КПП</label>
<input type="text" class="form-control @error('kpp') is-invalid @enderror"
id="kpp" name="kpp" value="{{ old('kpp', $organization->kpp) }}" maxlength="20">
@error('kpp')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3">
<label for="email" class="form-label">Email</label>
<input type="email" class="form-control @error('email') is-invalid @enderror"
id="email" name="email" value="{{ old('email', $organization->email) }}">
@error('email')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3">
<label for="phone" class="form-label">Телефон</label>
<input type="tel" class="form-control @error('phone') is-invalid @enderror"
id="phone" name="phone" value="{{ old('phone', $organization->phone) }}" maxlength="20">
@error('phone')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3">
<label for="address" class="form-label">Адрес</label>
<textarea class="form-control @error('address') is-invalid @enderror"
id="address" name="address" rows="2">{{ old('address', $organization->address) }}</textarea>
@error('address')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
</div>
</div>
</div>
</div>
<div class="mb-3">
<button type="submit" class="btn btn-primary">
<i class="bi bi-check-lg"></i> Сохранить изменения
</button>
<a href="{{ route('admin.organizations.show', $organization) }}" class="btn btn-secondary">
<i class="bi bi-x-lg"></i> Отмена
</a>
</div>
</form>
</main>
</div>
</div>
@endsection

View File

@ -0,0 +1,154 @@
@extends('layouts.app')
@section('title', 'Организации')
@section('content')
<div class="container-fluid">
<div class="row">
<!-- Sidebar -->
<nav class="col-md-3 col-lg-2 d-md-block sidebar collapse">
<div class="position-sticky pt-3">
<ul class="nav flex-column">
<li class="nav-item">
<a class="nav-link" href="{{ route('dashboard') }}">
<i class="bi bi-speedometer2"></i> Панель управления
</a>
</li>
@can('viewAny', App\Models\Organization::class)
<li class="nav-item">
<a class="nav-link active" href="{{ route('admin.organizations.index') }}">
<i class="bi bi-building"></i> Организации
</a>
</li>
@endcan
<li class="nav-item">
<a class="nav-link" href="#">
<i class="bi bi-people"></i> Пользователи
</a>
</li>
<li class="nav-item">
<a class="nav-link" href="#">
<i class="bi bi-book"></i> Курсы
</a>
</li>
</ul>
</div>
</nav>
<!-- Main content -->
<main class="col-md-9 ms-sm-auto col-lg-10 px-md-4 main-content">
<div class="d-flex justify-content-between flex-wrap flex-md-nowrap align-items-center pt-3 pb-2 mb-3 border-bottom">
<h1 class="h2">Организации</h1>
@can('create', App\Models\Organization::class)
<div class="btn-toolbar mb-2 mb-md-0">
<a href="{{ route('admin.organizations.create') }}" class="btn btn-primary btn-sm">
<i class="bi bi-plus-lg"></i> Добавить организацию
</a>
</div>
@endcan
</div>
@if(session('success'))
<div class="alert alert-success alert-dismissible fade show" role="alert">
<i class="bi bi-check-circle"></i> {{ session('success') }}
<button type="button" class="btn-close" data-bs-dismiss="alert"></button>
</div>
@endif
@if(session('error'))
<div class="alert alert-danger alert-dismissible fade show" role="alert">
<i class="bi bi-exclamation-triangle"></i> {{ session('error') }}
<button type="button" class="btn-close" data-bs-dismiss="alert"></button>
</div>
@endif
<div class="card shadow-sm">
<div class="card-body">
<div class="table-responsive">
<table class="table table-hover">
<thead>
<tr>
<th>ID</th>
<th>Название</th>
<th>INN/KPP</th>
<th>Email</th>
<th>Пользователей</th>
<th>Групп</th>
<th>Курсов</th>
<th>Статус</th>
<th>Действия</th>
</tr>
</thead>
<tbody>
@forelse($organizations as $org)
<tr>
<td>{{ $org->id }}</td>
<td>
<a href="{{ route('admin.organizations.show', $org) }}" class="text-decoration-none">
<strong>{{ $org->name }}</strong>
</a>
</td>
<td>
@if($org->inn)
<small>{{ $org->inn }}</small>
@if($org->kpp)<br><small class="text-muted">{{ $org->kpp }}</small>@endif
@else
<span class="text-muted"></span>
@endif
</td>
<td>{{ $org->email ?? '—' }}</td>
<td>{{ $org->users_count }}</td>
<td>{{ $org->groups_count }}</td>
<td>{{ $org->courses_count ?? 0 }}</td>
<td>
@if($org->is_active)
<span class="badge bg-success">Активна</span>
@else
<span class="badge bg-secondary">Не активна</span>
@endif
</td>
<td>
<div class="btn-group btn-group-sm">
<a href="{{ route('admin.organizations.show', $org) }}" class="btn btn-outline-primary" title="Просмотр">
<i class="bi bi-eye"></i>
</a>
@can('update', $org)
<a href="{{ route('admin.organizations.edit', $org) }}" class="btn btn-outline-warning" title="Редактировать">
<i class="bi bi-pencil"></i>
</a>
@endcan
@can('delete', $org)
<form action="{{ route('admin.organizations.destroy', $org) }}" method="POST" class="d-inline" onsubmit="return confirm('Вы уверены?')">
@csrf
@method('DELETE')
<button type="submit" class="btn btn-outline-danger" title="Удалить">
<i class="bi bi-trash"></i>
</button>
</form>
@endcan
</div>
</td>
</tr>
@empty
<tr>
<td colspan="9" class="text-center text-muted py-4">
<i class="bi bi-inbox" style="font-size: 2rem;"></i>
<p class="mt-2">Организаций пока нет</p>
</td>
</tr>
@endforelse
</tbody>
</table>
</div>
</div>
</div>
@if($organizations->hasPages())
<div class="mt-3">
{{ $organizations->links() }}
</div>
@endif
</main>
</div>
</div>
@endsection

View File

@ -0,0 +1,178 @@
@extends('layouts.app')
@section('title', $organization->name)
@section('content')
<div class="container-fluid">
<div class="row">
<nav class="col-md-3 col-lg-2 d-md-block sidebar collapse">
<div class="position-sticky pt-3">
<ul class="nav flex-column">
<li class="nav-item">
<a class="nav-link" href="{{ route('dashboard') }}">
<i class="bi bi-speedometer2"></i> Панель управления
</a>
</li>
<li class="nav-item">
<a class="nav-link" href="{{ route('admin.organizations.index') }}">
<i class="bi bi-building"></i> Организации
</a>
</li>
</ul>
</div>
</nav>
<main class="col-md-9 ms-sm-auto col-lg-10 px-md-4 main-content">
<div class="d-flex justify-content-between flex-wrap flex-md-nowrap align-items-center pt-3 pb-2 mb-3 border-bottom">
<h1 class="h2">{{ $organization->name }}</h1>
<div class="btn-toolbar">
@can('update', $organization)
<a href="{{ route('admin.organizations.edit', $organization) }}" class="btn btn-warning btn-sm me-2">
<i class="bi bi-pencil"></i> Редактировать
</a>
@endcan
<a href="{{ route('admin.organizations.index') }}" class="btn btn-secondary btn-sm">
<i class="bi bi-arrow-left"></i> Назад
</a>
</div>
</div>
@if(session('success'))
<div class="alert alert-success">
<i class="bi bi-check-circle"></i> {{ session('success') }}
</div>
@endif
<div class="row">
<div class="col-md-6 mb-4">
<div class="card shadow-sm h-100">
<div class="card-header bg-primary text-white">
<h5 class="mb-0"><i class="bi bi-info-circle"></i> Информация</h5>
</div>
<div class="card-body">
<table class="table table-sm">
<tr>
<th width="40%">Название:</th>
<td>{{ $organization->name }}</td>
</tr>
<tr>
<th>ИНН/КПП:</th>
<td>{{ $organization->inn ?? '—' }} / {{ $organization->kpp ?? '—' }}</td>
</tr>
<tr>
<th>Адрес:</th>
<td>{{ $organization->address ?? '—' }}</td>
</tr>
<tr>
<th>Email:</th>
<td>{{ $organization->email ?? '—' }}</td>
</tr>
<tr>
<th>Телефон:</th>
<td>{{ $organization->phone ?? '—' }}</td>
</tr>
<tr>
<th>Статус:</th>
<td>
@if($organization->is_active)
<span class="badge bg-success">Активна</span>
@else
<span class="badge bg-secondary">Не активна</span>
@endif
</td>
</tr>
<tr>
<th>Описание:</th>
<td>{{ $organization->description ?? '—' }}</td>
</tr>
</table>
</div>
</div>
</div>
<div class="col-md-6 mb-4">
<div class="card shadow-sm h-100">
<div class="card-header bg-success text-white">
<h5 class="mb-0"><i class="bi bi-graph-up"></i> Статистика</h5>
</div>
<div class="card-body">
<div class="row text-center">
<div class="col-4 mb-3">
<div class="display-4 text-primary">{{ $organization->users->count() }}</div>
<div class="text-muted">Пользователей</div>
</div>
<div class="col-4 mb-3">
<div class="display-4 text-success">{{ $organization->groups->count() }}</div>
<div class="text-muted">Групп</div>
</div>
<div class="col-4 mb-3">
<div class="display-4 text-info">{{ $organization->courseRequests->count() }}</div>
<div class="text-muted">Заявок</div>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="row">
<div class="col-md-6 mb-4">
<div class="card shadow-sm">
<div class="card-header">
<h5 class="mb-0"><i class="bi bi-people"></i> Пользователи</h5>
</div>
<div class="card-body">
@if($organization->users->count() > 0)
<ul class="list-group list-group-flush">
@foreach($organization->users->take(5) as $user)
<li class="list-group-item d-flex justify-content-between align-items-center">
{{ $user->name }}
<small class="text-muted">{{ $user->email }}</small>
</li>
@endforeach
</ul>
@if($organization->users->count() > 5)
<div class="mt-2 text-muted small">
+ ещё {{ $organization->users->count() - 5 }} пользователей
</div>
@endif
@else
<p class="text-muted mb-0">Нет пользователей</p>
@endif
</div>
</div>
</div>
<div class="col-md-6 mb-4">
<div class="card shadow-sm">
<div class="card-header">
<h5 class="mb-0"><i class="bi bi-people-fill"></i> Группы</h5>
</div>
<div class="card-body">
@if($organization->groups->count() > 0)
<ul class="list-group list-group-flush">
@foreach($organization->groups->take(5) as $group)
<li class="list-group-item">
<strong>{{ $group->name }}</strong>
@if($group->description)
<br><small class="text-muted">{{ Str::limit($group->description, 50) }}</small>
@endif
</li>
@endforeach
</ul>
@if($organization->groups->count() > 5)
<div class="mt-2 text-muted small">
+ ещё {{ $organization->groups->count() - 5 }} групп
</div>
@endif
@else
<p class="text-muted mb-0">Нет групп</p>
@endif
</div>
</div>
</div>
</div>
</main>
</div>
</div>
@endsection

View File

@ -15,13 +15,13 @@
</a> </a>
</li> </li>
<li class="nav-item"> <li class="nav-item">
<a class="nav-link" href="#"> <a class="nav-link" href="{{ route('admin.organizations.index') }}">
<i class="bi bi-people"></i> Пользователи <i class="bi bi-building"></i> Организации
</a> </a>
</li> </li>
<li class="nav-item"> <li class="nav-item">
<a class="nav-link" href="#"> <a class="nav-link" href="#">
<i class="bi bi-building"></i> Организации <i class="bi bi-people"></i> Пользователи
</a> </a>
</li> </li>
<li class="nav-item"> <li class="nav-item">

View File

@ -2,6 +2,7 @@
use App\Http\Controllers\Auth\LoginController; use App\Http\Controllers\Auth\LoginController;
use App\Http\Controllers\Auth\RegisterController; use App\Http\Controllers\Auth\RegisterController;
use App\Http\Controllers\Admin\OrganizationController;
use App\Http\Controllers\DashboardController; use App\Http\Controllers\DashboardController;
use Illuminate\Support\Facades\Route; use Illuminate\Support\Facades\Route;
@ -29,4 +30,9 @@ Route::middleware('auth')->group(function () {
Route::post('logout', [LoginController::class, 'logout'])->name('logout'); Route::post('logout', [LoginController::class, 'logout'])->name('logout');
Route::get('/dashboard', [DashboardController::class, 'index'])->name('dashboard'); Route::get('/dashboard', [DashboardController::class, 'index'])->name('dashboard');
// Администрирование организаций
Route::prefix('admin')->name('admin.')->group(function () {
Route::resource('organizations', OrganizationController::class);
});
}); });