Feat: Master data induk NIP/NISN

This commit is contained in:
zhadaarsita 2026-02-05 16:45:25 +07:00
parent 779ef38952
commit c1e631c3ce
16 changed files with 531 additions and 336 deletions

View File

@ -0,0 +1,31 @@
<?php
namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use App\Models\MasterInduk;
use Illuminate\Http\Request;
class MasterIndukController extends Controller
{
// Menyimpan Data Induk Baru (Pre-Register)
public function store(Request $request)
{
$request->validate([
'nomor_induk' => 'required|unique:master_induks,nomor_induk',
'role' => 'required|in:siswa,guru',
'nama_pemilik' => 'required|string',
]);
MasterInduk::create($request->all());
return back()->with('success', 'Data Induk berhasil ditambahkan. User dengan NIP/NISN ini sekarang bisa mendaftar.');
}
// Menghapus Data
public function destroy($id)
{
MasterInduk::findOrFail($id)->delete();
return back()->with('success', 'Data Induk dihapus.');
}
}

View File

@ -3,40 +3,42 @@
namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use App\Services\DummyDataService;
use App\Models\User; // Model User Asli
use App\Models\MasterInduk; // Model Whitelist
use Illuminate\Http\Request;
class UserController extends Controller
{
public function index()
{
$semuaSiswa = DummyDataService::getAllSiswa();
$users = User::orderBy('created_at', 'desc')->paginate(10);
$whitelists = MasterInduk::orderBy('created_at', 'desc')->get();
return view('admin.pengguna.index', [
'pageTitle' => 'Manajemen Pengguna',
'semuaSiswa' => $semuaSiswa
'pageTitle' => 'Daftar Pengguna',
'users' => $users,
'whitelists' => $whitelists
]);
}
/**
* Menampilkan form untuk membuat pengguna baru.
*/
public function create()
{
return view('admin.pengguna.create', [
'pageTitle' => 'Tambah Pengguna Baru',
]);
return view('admin.pengguna.create', ['pageTitle' => 'Tambah Pengguna Baru']);
}
/**
* Menampilkan form untuk mengedit pengguna yang ada.
*/
public function edit($id)
{
$pengguna = collect(DummyDataService::getAllSiswa())->firstWhere('id', (int)$id);
abort_if(!$pengguna, 404);
$pengguna = User::findOrFail($id);
return view('admin.pengguna.edit', [
'pageTitle' => 'Edit Pengguna',
'pengguna' => $pengguna,
]);
}
public function destroy($id)
{
User::findOrFail($id)->delete();
return back()->with('success', 'Pengguna berhasil dihapus.');
}
}

View File

@ -19,7 +19,7 @@ public function create(): View
// Memproses login admin
public function store(LoginRequest $request): RedirectResponse
{
$request->authenticate(); // Menjalankan logika ketat di LoginRequest
$request->authenticate();
$request->session()->regenerate();
return redirect()->route('admin.dashboard');
}

View File

@ -4,83 +4,72 @@
use App\Http\Controllers\Controller;
use App\Models\User;
use App\Models\MasterInduk;
use Illuminate\Auth\Events\Registered;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Hash;
use Illuminate\Validation\Rules;
use Illuminate\Validation\ValidationException;
use Illuminate\View\View;
class RegisteredUserController extends Controller
{
/**
* Menampilkan halaman registrasi.
*/
public function create(Request $request): View
{
// Bagian Pengambilan Role dari URL
$role = $request->query('role', 'siswa'); // Ambil 'role' dari URL, default ke 'siswa'
// Kirim $role ke view
$role = $request->query('role', 'siswa');
return view('auth.register', ['role' => $role]);
}
/**
* Menangani permintaan registrasi yang masuk.
*/
public function store(Request $request): RedirectResponse
{
// Bagian Validasi Dinamis
$role = $request->input('role');
$rules = [
'name' => ['required', 'string', 'max:255'],
'email' => ['required', 'string', 'email', 'max:255', 'unique:users'],
'password' => ['required', 'confirmed', Rules\Password::defaults()],
'role' => ['required', 'in:siswa,guru'], // Sesuaikan dengan role yang diizinkan
'role' => ['required', 'in:siswa,guru'],
];
// Tambahkan validasi NISN atau NIP berdasarkan role
if ($role === 'siswa') {
$rules['nisn'] = ['required', 'string', 'max:255']; // Tambahkan 'unique:users' jika perlu
} else { // Asumsi 'guru'
$rules['nip'] = ['required', 'string', 'max:255']; // Tambahkan 'unique:users' jika perlu
$rules['nisn'] = ['required', 'string', 'max:255', 'unique:users,nisn'];
} else {
$rules['nip'] = ['required', 'string', 'max:255', 'unique:users,nip'];
}
$request->validate($rules);
// Bagian Pembuatan User
$userArray = [
'id' => rand(100, 999), // ID unik sementara
'nama_lengkap' => $request->name,
'name' => $request->name,
'password' => Hash::make($request->password), // Gunakan Hash jika login Anda sudah pakai Hash
// 'password' => $request->password, // Gunakan ini jika login (LoginRequest) masih cek teks biasa
'role' => $request->role,
];
// Tambahkan field dinamis (NISN/NIP) dan buat email unik palsu
if ($role === 'siswa') {
$userArray['nisn'] = $request->nisn;
$userArray['email'] = $request->nisn . '@smkn1perpus.sch.id'; // Email unik sementara
} else {
$userArray['nip'] = $request->nip;
$userArray['email'] = $request->nip . '@smkn1perpus.sch.id'; // Email unik sementara
}
$user = new User();
$user->forceFill($userArray);
// $user->save(); // Aktifkan ini jika menggunakan database
$nomorInduk = ($role === 'siswa') ? $request->nisn : $request->nip;
$isWhitelisted = MasterInduk::where('nomor_induk', $nomorInduk)
->where('role', $role)
->exists();
if (!$isWhitelisted) {
throw ValidationException::withMessages([
($role === 'siswa' ? 'nisn' : 'nip') => ['Nomor induk tidak terdaftar di Data Sekolah. Hubungi petugas.'],
]);
}
$user = User::create([
'name' => $request->name,
'email' => $request->email,
'password' => Hash::make($request->password),
'role' => $role,
'nisn' => ($role === 'siswa') ? $nomorInduk : null,
'nip' => ($role === 'guru') ? $nomorInduk : null,
]);
event(new Registered($user));
Auth::login($user);
// Bagian Redirect
if ($user->role === 'penjaga perpus') {
return redirect()->route('admin.dashboard');
} else {
return redirect()->route('dashboard');
}
return redirect()->route('dashboard');
}
}

View File

@ -2,42 +2,36 @@
namespace App\Http\Controllers;
use App\Http\Requests\ProfileUpdateRequest;
use App\Http\Requests\ProfileUpdateRequest;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Redirect;
use Illuminate\View\View;
use Illuminate\Support\Facades\Hash;
use App\Services\DummyDataService;
class ProfileController extends Controller
{
/**
* Menampilkan halaman utama profil pengguna secara dinamis berdasarkan role.
* Tampilkan halaman profil user.
*/
public function index(): RedirectResponse|View
public function index(Request $request): View
{
$user = Auth::user();
if (!$user) {
return redirect()->route(route: 'login');
}
$user = $request->user();
$viewData = ['user' => $user];
// Menyiapkan data berdasarkan role pengguna
if ($user->role === 'penjaga perpus') {
// Data untuk Penjaga Perpus: Statistik global & aktivitas terkini
$viewData['statistik'] = DummyDataService::getAdminDashboardStats();
$viewData['aktivitasTerakhir'] = DummyDataService::getAktivitasTerakhir();
} elseif ($user->role === 'guru') {
// Data untuk Guru: Data personal + ringkasan laporan minat baca
$viewData['bukuOffline'] = DummyDataService::getBukuPinjamOffline($user);
$viewData['bukuOnline'] = DummyDataService::getBacaBukuOnline($user);
$viewData['laporan'] = DummyDataService::getLaporanMinatBaca();
} else {
// Data default untuk Siswa
$viewData['bukuOffline'] = DummyDataService::getBukuPinjamOffline($user);
$viewData['bukuOnline'] = DummyDataService::getBacaBukuOnline($user);
$viewData['statistik'] = DummyDataService::getDashboardStats();
@ -47,7 +41,7 @@ public function index(): RedirectResponse|View
}
/**
* Menampilkan form untuk mengedit profil.
* Tampilkan form edit profil.
*/
public function edit(Request $request): View
{
@ -57,23 +51,31 @@ public function edit(Request $request): View
}
/**
* Memperbarui informasi profil pengguna.
* Update data profil ke Database MySQL.
*/
public function update(ProfileUpdateRequest $request): RedirectResponse
public function update(Request $request): RedirectResponse
{
$request->user()->fill($request->validated());
// Validasi input
$request->validate([
'name' => ['required', 'string', 'max:255'],
'email' => ['required', 'string', 'email', 'max:255', 'unique:users,email,'.Auth::id()],
'nomor_hp' => ['nullable', 'string', 'max:15'],
]);
if ($request->user()->isDirty('email')) {
$request->user()->email_verified_at = null;
$user = $request->user();
$user->fill($request->all());
if ($user->isDirty('email')) {
$user->email_verified_at = null;
}
$request->user()->save();
$user->save();
return Redirect::route('profile.edit')->with('status', 'profile-updated');
}
/**
* Menghapus akun pengguna.
* Hapus akun user.
*/
public function destroy(Request $request): RedirectResponse
{
@ -85,11 +87,11 @@ public function destroy(Request $request): RedirectResponse
Auth::logout();
$user->delete();
$user->delete();
$request->session()->invalidate();
$request->session()->regenerateToken();
return Redirect::to('/');
}
}
}

View File

@ -2,129 +2,74 @@
namespace App\Http\Requests\Auth;
use App\Models\User;
use App\Services\DummyDataService;
use Illuminate\Auth\Events\Lockout;
use Illuminate\Foundation\Http\FormRequest;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\RateLimiter;
use Illuminate\Support\Str;
use Illuminate\Validation\ValidationException;
use Illuminate\Auth\Events\Lockout;
class LoginRequest extends FormRequest
{
/**
* Menentukan apakah pengguna diizinkan untuk membuat request ini.
* Selalu true karena semua orang boleh mencoba login.
*/
public function authorize(): bool
{
return true;
}
/**
* Mendapatkan aturan validasi yang berlaku untuk request ini.
* Aturan ini dinamis, berubah tergantung pada 'role' yang dikirim dari form.
*/
public function rules(): array
{
// Jika form mengirim 'role' dengan nilai 'siswa'...
if ($this->input('role') === 'siswa') {
// ...maka validasi input 'nisn' dan 'password'.
return [
'nisn' => ['required', 'string'],
'password' => ['required', 'string'],
];
}
// Jika tidak (untuk 'guru' dan 'penjaga perpus')
return [
'nip' => ['required', 'string'],
'password' => ['required', 'string'],
];
}
/**
* Mencoba untuk mengautentikasi kredensial dari request.
* Ini adalah "otak" dari proses login yang berisi logika paling penting.
*
* @throws \Illuminate\Validation\ValidationException
*/
public function authenticate(): void
{
// Pastikan pengguna tidak mencoba login terlalu sering (mencegah brute-force).
$this->ensureIsNotRateLimited();
// Ambil data yang dikirim dari form login.
$roleDariForm = $this->input('role');
$allUsers = DummyDataService::getAllSiswa();
$inputPassword = $this->input('password');
$userArray = null;
$loginString = $this->input('email') ?? $this->input('nisn') ?? $this->input('nip');
// Tentukan field mana yang akan menerima pesan error jika gagal (nisn atau email).
$errorField = $this->filled('nisn') ? 'nisn' : 'nip';
$password = $this->input('password');
// Cari data pengguna berdasarkan input yang diberikan.
if ($this->filled('nisn')) {
// Jika form diisi dengan 'nisn', cari pengguna berdasarkan 'nisn'.
$userArray = collect($allUsers)->firstWhere('nisn', $this->input('nisn'));
} else {
// Jika tidak, cari pengguna berdasarkan 'nip'.
$userArray = collect($allUsers)->firstWhere('nip', $this->input('nip'));
if (!$loginString) {
throw ValidationException::withMessages([
'email' => 'Mohon masukkan NISN, NIP, atau Email.',
]);
}
// Lakukan Pengecekan Kredensial dan Role.
if ($userArray && $userArray['password'] === $inputPassword) {
// Cek #2: Jika kredensial benar, apakah role pengguna sesuai dengan form yang digunakan?
if (isset($userArray['role']) && $userArray['role'] === $roleDariForm) {
// Buat objek User dari data dummy.
$userModel = new User();
$userArray['name'] = $userArray['nama_lengkap'];
$userModel->forceFill($userArray);
// Loginkan pengguna secara resmi ke dalam sistem.
Auth::login($userModel);
// Reset hitungan percobaan login yang gagal.
RateLimiter::clear($this->throttleKey());
return; // Proses autentikasi berhasil.
} else {
// Tambah hitungan percobaan login yang gagal.
RateLimiter::hit($this->throttleKey());
// Ambil nama role asli pengguna untuk ditampilkan di pesan error.
$actualRole = Str::title($userArray['role'] ?? 'Tidak Dikenal');
// Lemparkan error validasi khusus 'forbidden' dengan pesan yang jelas.
throw ValidationException::withMessages([
'forbidden' => "Akses ditolak. Akun ini terdaftar sebagai {$actualRole}.",
]);
}
if (Auth::attempt(['nisn' => $loginString, 'password' => $password], $this->boolean('remember'))) {
RateLimiter::clear($this->throttleKey());
return;
}
if (Auth::attempt(['nip' => $loginString, 'password' => $password], $this->boolean('remember'))) {
RateLimiter::clear($this->throttleKey());
return;
}
if (Auth::attempt(['email' => $loginString, 'password' => $password], $this->boolean('remember'))) {
RateLimiter::clear($this->throttleKey());
return;
}
// Jika pengguna tidak ditemukan atau password salah.
RateLimiter::hit($this->throttleKey());
$fieldError = $this->input('nisn') ? 'nisn' : ($this->input('nip') ? 'nip' : 'email');
throw ValidationException::withMessages([
$errorField => trans('auth.failed'), // Pesan error umum "These credentials do not match...".
$fieldError => trans('auth.failed'),
]);
}
/**
* Memastikan request login tidak dibatasi karena terlalu banyak percobaan.
*/
public function ensureIsNotRateLimited(): void
{
// Jika percobaan belum melebihi 5 kali, lanjutkan.
if (!RateLimiter::tooManyAttempts($this->throttleKey(), 5)) {
if (! RateLimiter::tooManyAttempts($this->throttleKey(), 5)) {
return;
}
// Jika sudah lebih dari 5 kali, lemparkan error 'throttle'.
event(new Lockout($this));
$seconds = RateLimiter::availableIn($this->throttleKey());
throw ValidationException::withMessages([
'email' => trans('auth.throttle', [
'seconds' => $seconds,
@ -133,14 +78,9 @@ public function ensureIsNotRateLimited(): void
]);
}
/**
* Mendapatkan kunci throttle untuk request ini.
* Kunci ini unik untuk setiap pengguna (berdasarkan nisn/email) dan alamat IP.
*/
public function throttleKey(): string
{
// Gunakan 'nisn' jika ada, jika tidak, gunakan 'email' sebagai identitas.
$loginIdentifier = $this->input('nisn') ?: $this->input('nip');
return Str::transliterate(Str::lower($loginIdentifier) . '|' . $this->ip());
$field = $this->input('email') ?? $this->input('nisn') ?? $this->input('nip') ?? 'unknown';
return Str::transliterate(Str::lower($field).'|'.$this->ip());
}
}

View File

@ -0,0 +1,12 @@
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;
class MasterInduk extends Model
{
use HasFactory;
protected $guarded = ['id'];
}

View File

@ -21,6 +21,12 @@ class User extends Authenticatable
'name',
'email',
'password',
'role',
'nisn',
'nip',
'nomor_hp',
'kelas',
'nomor_induk'
];
/**
@ -45,4 +51,9 @@ protected function casts(): array
'password' => 'hashed',
];
}
public function getNamaLengkapAttribute()
{
return $this->name;
}
}

View File

@ -2,16 +2,14 @@
namespace App\Providers;
use App\Auth\DummyUserProvider;
use Illuminate\Support\Facades\Auth;
// use App\Auth\DummyUserProvider; // Hapus atau Komen baris ini
use Illuminate\Support\Facades\Gate;
use Illuminate\Foundation\Support\Providers\AuthServiceProvider as ServiceProvider;
class AuthServiceProvider extends ServiceProvider
{
/**
* The model to policy mappings for the application.
*
* @var array<class-string, class-string>
*/
protected $policies = [
//
@ -22,9 +20,13 @@ class AuthServiceProvider extends ServiceProvider
*/
public function boot(): void
{
// Daftarkan provider kustom kita di sini
// $this->registerPolicies(); // Laravel 10/11 biasanya auto-discovery
// HAPUS atau KOMENTARI bagian ini:
/*
Auth::provider('dummy', function ($app, array $config) {
return new DummyUserProvider();
});
*/
}
}

View File

@ -38,7 +38,7 @@
'guards' => [
'web' => [
'driver' => 'session',
'provider' => 'dummy',
'provider' => 'users',
],
],
@ -64,10 +64,6 @@
'driver' => 'eloquent',
'model' => env('AUTH_MODEL', App\Models\User::class),
],
'dummy' => [
'driver' => 'dummy',
],
],
/*

View File

@ -17,6 +17,11 @@ public function up(): void
$table->string('email')->unique();
$table->timestamp('email_verified_at')->nullable();
$table->string('password');
$table->string('role')->default('siswa');
$table->string('nisn')->nullable();
$table->string('nip')->nullable();
$table->string('nomor_hp')->nullable();
$table->string('kelas')->nullable();
$table->rememberToken();
$table->timestamps();
});

View File

@ -0,0 +1,29 @@
<?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()
{
Schema::create('master_induks', function (Blueprint $table) {
$table->id();
$table->string('nomor_induk')->unique();
$table->enum('role', ['siswa', 'guru']);
$table->string('nama_pemilik')->nullable();
$table->timestamps();
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('master_induks');
}
};

View File

@ -2,22 +2,90 @@
namespace Database\Seeders;
use App\Models\User;
// use Illuminate\Database\Console\Seeds\WithoutModelEvents;
use Illuminate\Database\Seeder;
use App\Models\User;
use App\Models\MasterInduk;
use Illuminate\Support\Facades\Hash;
use Illuminate\Support\Facades\DB;
class DatabaseSeeder extends Seeder
{
/**
* Seed the application's database.
*/
public function run(): void
public function run()
{
// User::factory(10)->create();
// Bersihkan data lama
DB::statement('SET FOREIGN_KEY_CHECKS=0;');
User::truncate();
MasterInduk::truncate();
DB::statement('SET FOREIGN_KEY_CHECKS=1;');
User::factory()->create([
'name' => 'Test User',
'email' => 'test@example.com',
// 1. ISI DATA INDUK (WHITELIST)
$whitelist = [
['nomor_induk' => '1234567890', 'role' => 'siswa', 'nama_pemilik' => 'Silvi Rahmawati'],
['nomor_induk' => '9988776655', 'role' => 'siswa', 'nama_pemilik' => 'Siti Nurhaliza'],
['nomor_induk' => '5566778899', 'role' => 'siswa', 'nama_pemilik' => 'Andi Pratama'],
['nomor_induk' => '198506152010012', 'role' => 'guru', 'nama_pemilik' => 'Rina Marlina'],
];
foreach ($whitelist as $w) {
MasterInduk::create($w);
}
// 2. ISI USER ASLI (ID USER DISAMAKAN DENGAN DUMMY)
// ID 1: Silvi (Siswa)
User::create([
'id' => 1,
'name' => 'Silvi Rahmawati', // Pakai 'name' saja, 'nama_lengkap' dihapus
'email' => 'silvi.rahmawati@smkn1perpus.sch.id',
'password' => Hash::make('password'),
'role' => 'siswa',
'nisn' => '1234567890',
'nomor_hp' => '08123456789',
'kelas' => 'XII RPL'
]);
// ID 2: Budi (Admin/Penjaga)
User::create([
'id' => 2,
'name' => 'Budi Santoso',
'email' => 'budi.santoso@smkn1perpus.sch.id',
'password' => Hash::make('password'),
'role' => 'penjaga perpus',
'nip' => '197812312005011',
]);
// ID 3: Siti (Siswa)
User::create([
'id' => 3,
'name' => 'Siti Nurhaliza',
'email' => 'siti.nurhaliza@smkn1perpus.sch.id',
'password' => Hash::make('password'),
'role' => 'siswa',
'nisn' => '9988776655',
'nomor_hp' => '081998877665',
'kelas' => 'XII RPL A'
]);
// ID 4: Andi (Siswa)
User::create([
'id' => 4,
'name' => 'Andi Pratama',
'email' => 'andi.pratama@smkn1perpus.sch.id',
'password' => Hash::make('password'),
'role' => 'siswa',
'nisn' => '5566778899',
'nomor_hp' => '081556677889',
'kelas' => 'XII RPL A'
]);
// ID 5: Rina (Guru)
User::create([
'id' => 5,
'name' => 'Rina Marlina',
'email' => 'rina.marlina@smkn1perpus.sch.id',
'password' => Hash::make('password'),
'role' => 'guru',
'nip' => '198506152010012',
]);
}
}
}

View File

@ -1,116 +1,191 @@
<x-app-layout>
@section('page-title', $pageTitle)
<div class="card shadow-sm border-0">
<div class="card-header bg-white d-flex justify-content-between align-items-center">
<h5 class="my-0 fw-bold">Daftar Semua Pengguna</h5>
<a href="{{ route('admin.pengguna.create') }}" class="btn btn-primary">
<i class="bi bi-plus-circle-fill me-2"></i>Tambah Pengguna
</a>
@section('page-title', content: 'Manajemen Pengguna')
<div class="container-fluid p-0">
<div class="d-flex justify-content-between align-items-center mb-3">
<h1 class="h3 text-gray-800">{{ $pageTitle }}</h1>
</div>
<div class="card-body">
<div class="table-responsive">
<table class="table table-hover">
<thead>
<tr>
<th>No</th>
<th>Nama Lengkap</th>
<th>Email</th>
<th>Role</th>
<th>Aksi</th>
</tr>
</thead>
<tbody>
@forelse($semuaSiswa as $siswa)
<tr>
<td>{{ $loop->iteration }}</td>
<td>{{ $siswa['nama_lengkap'] }}</td>
<td>{{ $siswa['email'] }}</td>
<td>
@if($siswa['role'] == 'penjaga perpus')
<span class="badge bg-success-subtle text-success-emphasis">{{ Str::title($siswa['role']) }}</span>
@else
<span class="badge bg-primary-subtle text-primary-emphasis">{{ Str::title($siswa['role']) }}</span>
@endif
</td>
<td>
<button class="btn btn-sm btn-outline-secondary" data-bs-toggle="modal" data-bs-target="#detailPenggunaModal"
data-id="{{ $siswa['id'] }}"
data-nama="{{ $siswa['nama_lengkap'] }}"
data-email="{{ $siswa['email'] }}"
data-role="{{ Str::title($siswa['role']) }}"
data-nisn="{{ $siswa['nisn'] ?? 'N/A' }}">
<i class="bi bi-eye-fill"></i> Detail
</button>
</td>
</tr>
@empty
<tr>
<td colspan="5" class="text-center">Tidak ada data pengguna.</td>
</tr>
@endforelse
</tbody>
</table>
<div class="card shadow mb-5">
<div class="card-header py-3">
<h6 class="m-0 font-weight-bold text-primary">Daftar Pengguna Aktif</h6>
</div>
<div class="card-body">
<div class="table-responsive">
<table class="table table-bordered" width="100%" cellspacing="0">
<thead>
<tr>
<th>No</th>
<th>Nama Lengkap</th>
<th>Email</th>
<th>Role</th>
<th>Nomor Induk</th>
<th>Aksi</th>
</tr>
</thead>
<tbody>
@forelse($users as $index => $user)
<tr>
<td>{{ $index + 1 }}</td>
<td>{{ $user->name }}</td>
<td>{{ $user->email }}</td>
<td>
<span class="badge {{ $user->role == 'guru' ? 'bg-info' : 'bg-primary' }}">
{{ ucfirst($user->role) }}
</span>
</td>
<td>
{{ $user->nisn ?? $user->nip ?? '-' }}
</td>
<td>
<a href="{{ route('admin.pengguna.edit', $user->id) }}" class="btn btn-sm btn-warning">
<i class="bi bi-pencil"></i> Edit
</a>
<form action="{{ route('admin.pengguna.destroy', $user->id) }}" method="POST" class="d-inline" onsubmit="return confirm('Yakin hapus user ini?')">
@csrf
@method('DELETE')
<button class="btn btn-sm btn-danger"><i class="bi bi-trash"></i></button>
</form>
</td>
</tr>
@empty
<tr>
<td colspan="6" class="text-center">Belum ada pengguna terdaftar.</td>
</tr>
@endforelse
</tbody>
</table>
{{-- Pagination --}}
<div class="mt-3">
{{ $users->links() }}
</div>
</div>
</div>
</div>
<hr class="my-5 border-4">
<div class="d-flex justify-content-between align-items-center mb-3">
<div>
<h4 class="fw-bold text-success mb-1">
<i class="bi bi-database-lock me-2"></i>Data Induk (Whitelist)
</h4>
<p class="text-muted mb-0">Daftar NIP/NISN yang <b>diizinkan</b> untuk mendaftar.</p>
</div>
<button type="button" class="btn btn-primary" data-bs-toggle="modal" data-bs-target="#modalMasterInduk">
<i class="bi bi-plus-lg me-1"></i> Tambah Data Induk
</button>
</div>
<div class="card shadow mb-4 border-left-success">
<div class="card-body">
<div class="table-responsive">
<table class="table table-hover align-middle mb-0">
<thead class="table-light">
<tr>
<th>No</th>
<th>NIP / NISN</th>
<th>Nama Pemilik</th>
<th>Role</th>
<th class="text-center">Status Akun</th>
<th class="text-end">Aksi</th>
</tr>
</thead>
<tbody>
@forelse($whitelists as $index => $item)
<tr>
<td>{{ $index + 1 }}</td>
<td class="fw-bold font-monospace">{{ $item->nomor_induk }}</td>
<td>{{ $item->nama_pemilik }}</td>
<td>
<span class="badge {{ $item->role == 'guru' ? 'bg-info' : 'bg-secondary' }}">
{{ ucfirst($item->role) }}
</span>
</td>
<td class="text-center">
{{-- Cek apakah user sudah daftar pakai NIP ini --}}
@php
$isRegistered = \App\Models\User::where('nisn', $item->nomor_induk)
->orWhere('nip', $item->nomor_induk)->exists();
@endphp
@if($isRegistered)
<span class="badge bg-success text-white">
<i class="bi bi-check-circle-fill me-1"></i>Terdaftar
</span>
@else
<span class="badge bg-warning text-dark">
<i class="bi bi-hourglass-split me-1"></i>Belum Daftar
</span>
@endif
</td>
<td class="text-end">
<form action="{{ route('admin.master-induk.destroy', $item->id) }}" method="POST" onsubmit="return confirm('Hapus data ini? User dengan NIP/NISN ini tidak akan bisa daftar lagi.');">
@csrf
@method('DELETE')
<button type="submit" class="btn btn-sm btn-outline-danger">
<i class="bi bi-trash"></i> Hapus
</button>
</form>
</td>
</tr>
@empty
<tr>
<td colspan="6" class="text-center py-4 text-muted">
Belum ada data whitelist. Silakan tambah data.
</td>
</tr>
@endforelse
</tbody>
</table>
</div>
</div>
</div>
</div>
<div class="modal fade" id="detailPenggunaModal" tabindex="-1">
{{-- MODAL TAMBAH DATA INDUK --}}
<div class="modal fade" id="modalMasterInduk" tabindex="-1">
<div class="modal-dialog">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title fw-bold" id="modalNama"></h5>
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<div class="modal-body">
<table class="table table-borderless table-sm">
<tr><th width="80px">Email</th><td id="modalEmail"></td></tr>
<tr><th>Role</th><td><span id="modalRole" class="badge"></span></td></tr>
<tr><th>NISN</th><td id="modalNisn"></td></tr>
</table>
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Tutup</button>
<a href="#" id="modalEditPengguna" class="btn btn-primary"><i class="bi bi-pencil-fill me-2"></i>Edit</a>
<h5 class="modal-title fw-bold">Tambah Whitelist (NIP/NISN)</h5>
<button type="button" class="btn-close" data-bs-dismiss="modal"></button>
</div>
<form action="{{ route('admin.master-induk.store') }}" method="POST">
@csrf
<div class="modal-body">
<div class="alert alert-info small mb-3">
<i class="bi bi-info-circle-fill me-1"></i>
Masukkan data siswa/guru yang valid agar mereka bisa mendaftar.
</div>
<div class="mb-3">
<label class="form-label">Role</label>
<select name="role" class="form-select" required>
<option value="siswa">Siswa</option>
<option value="guru">Guru</option>
</select>
</div>
<div class="mb-3">
<label class="form-label">NIP / NISN</label>
<input type="number" name="nomor_induk" class="form-control" placeholder="Contoh: 1234567890" required>
</div>
<div class="mb-3">
<label class="form-label">Nama Pemilik</label>
<input type="text" name="nama_pemilik" class="form-control" placeholder="Nama Siswa/Guru..." required>
</div>
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Batal</button>
<button type="submit" class="btn btn-success">Simpan Data</button>
</div>
</form>
</div>
</div>
</div>
@push('scripts')
<script>
document.addEventListener('DOMContentLoaded', function () {
const detailPenggunaModal = document.getElementById('detailPenggunaModal');
if(detailPenggunaModal) {
detailPenggunaModal.addEventListener('show.bs.modal', event => {
const button = event.relatedTarget;
const id = button.getAttribute('data-id');
const nama = button.getAttribute('data-nama');
const email = button.getAttribute('data-email');
const role = button.getAttribute('data-role');
const nisn = button.getAttribute('data-nisn');
const modalNama = detailPenggunaModal.querySelector('#modalNama');
const modalEmail = detailPenggunaModal.querySelector('#modalEmail');
const modalRole = detailPenggunaModal.querySelector('#modalRole');
const modalNisn = detailPenggunaModal.querySelector('#modalNisn');
const modalEditPengguna = detailPenggunaModal.querySelector('#modalEditPengguna');
modalNama.textContent = nama;
modalEmail.textContent = `: ${email}`;
modalRole.textContent = role;
modalNisn.textContent = `: ${nisn}`;
modalEditPengguna.href = `{{ url('admin/pengguna') }}/${id}/edit`;
if (role.toLowerCase() === 'penjaga perpus') {
modalRole.className = 'badge bg-success-subtle text-success-emphasis';
} else {
modalRole.className = 'badge bg-primary-subtle text-primary-emphasis';
}
});
}
});
</script>
@endpush
</x-app-layout>

View File

@ -1,57 +1,81 @@
<x-guest-layout>
<div class="mb-4 text-center">
<h4 class="fw-bold text-primary">Daftar Akun {{ ucfirst($role) }}</h4>
<p class="text-muted small">
Silakan isi data diri lengkap untuk mendaftar.
</p>
</div>
<form method="POST" action="{{ route('register') }}">
@csrf
{{-- Input tersembunyi untuk mengirimkan peran (role) ke backend --}}
<input type="hidden" name="role" value="{{ $role }}">
<div class="text-center mb-4">
{{-- Judul dinamis --}}
<h3 class="fw-bold text-primary">Buat Akun {{ Str::title($role) }}</h3>
<p class="text-muted">Daftarkan diri Anda untuk mulai menjelajahi koleksi kami.</p>
</div>
{{-- Form Nama Lengkap (Umum) --}}
<div class="mb-3">
<label for="name" class="form-label">Nama Lengkap</label>
<input id="name" class="form-control bg-body-tertiary @error('name') is-invalid @enderror" type="text" name="name" value="{{ old('name') }}" required autofocus autocomplete="name" />
<label class="form-label">Nama Lengkap</label>
<input type="text" name="name" class="form-control @error('name') is-invalid @enderror"
value="{{ old('name') }}" required autofocus placeholder="Nama sesuai absen">
@error('name')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
{{-- Form dinamis (NISN untuk Siswa, NIP untuk Guru) --}}
@if ($role == 'siswa')
<div class="mb-3">
<label for="nisn" class="form-label">Nomor Induk Siswa Nasional (NISN)</label>
<input id="nisn" class="form-control bg-body-tertiary @error('nisn') is-invalid @enderror" type="text" name="nisn" value="{{ old('nisn') }}" required autocomplete="username" />
@error('nisn')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
@else
<div class="mb-3">
<label for="nip" class="form-label">Nomor Induk Pegawai (NIP)</label>
<input id="nip" class="form-control bg-body-tertiary @error('nip') is-invalid @enderror" type="text" name="nip" value="{{ old('nip') }}" required autocomplete="username" />
@error('nip')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
@endif
<div class="mb-3">
<label class="form-label">
@if ($role == 'siswa')
NISN (Nomor Induk Siswa)
@else
NIP / NIK Sekolah
@endif
</label>
<input type="number" name="{{ $role == 'siswa' ? 'nisn' : 'nip' }}"
class="form-control @error($role == 'siswa' ? 'nisn' : 'nip') is-invalid @enderror"
value="{{ old($role == 'siswa' ? 'nisn' : 'nip') }}" required placeholder="Cth: 1234567890">
@error($role == 'siswa' ? 'nisn' : 'nip')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
<div class="mb-3">
<label class="form-label">Email Aktif</label>
<input type="email" name="email" class="form-control @error('email') is-invalid @enderror"
value="{{ old('email') }}" required placeholder="contoh@gmail.com">
<div class="form-text text-muted small">
<i class="bi bi-bell me-1"></i>
Notifikasi denda & peminjaman akan dikirim ke email ini.
</div>
@error('email')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
{{-- Form Password (Umum) --}}
<div class="mb-3">
<label for="password" class="form-label">Password</label>
<input id="password" class="form-control bg-body-tertiary @error('password') is-invalid @enderror" type="password" name="password" required autocomplete="new-password" />
<div class="input-group">
<input id="password" class="form-control bg-body-tertiary @error('password') is-invalid @enderror"
type="password" name="password" required autocomplete="new-password" />
<button class="btn btn-outline-secondary" type="button" id="togglePassword">
<i class="bi bi-eye-slash-fill"></i>
</button>
</div>
@error('password')
<div class="invalid-feedback">{{ $message }}</div>
@enderror
</div>
{{-- Form Konfirmasi Password (Umum) --}}
<div class="mb-3">
<label for="password_confirmation" class="form-label">Konfirmasi Password</label>
<input id="password_confirmation" class="form-control bg-body-tertiary" type="password" name="password_confirmation" required autocomplete="new-password" />
<div class="input-group">
<input id="password_confirmation" class="form-control bg-body-tertiary" type="password"
name="password_confirmation" required autocomplete="new-password" />
<button class="btn btn-outline-secondary" type="button" id="togglePassword">
<i class="bi bi-eye-slash-fill"></i>
</button>
</div>
</div>
<div class="d-grid mt-4">
@ -63,7 +87,8 @@
{{-- Link Login dinamis --}}
<p class="mt-4 text-center text-muted small">
Sudah punya akun?
<a href="{{ route('login', ['role' => $role]) }}" class="fw-semibold text-decoration-none">Masuk di sini</a>
<a href="{{ route('login', ['role' => $role]) }}" class="fw-semibold text-decoration-none">Masuk di
sini</a>
</p>
</form>
</x-guest-layout>
</x-guest-layout>

View File

@ -15,6 +15,7 @@
// Admin Controllers
use App\Http\Controllers\Admin\DashboardController as AdminDashboardController;
use App\Http\Controllers\Admin\BookController as AdminBookController;
use App\Http\Controllers\Admin\MasterIndukController;
use App\Http\Controllers\Admin\UserController as AdminUserController;
use App\Http\Controllers\Admin\PengumumanController as AdminPengumumanController;
use App\Http\Controllers\RekomendasiController;
@ -37,7 +38,7 @@
// --- RUTE UNTUK PENGGUNA TERAUTENTIKASI (SISWA, GURU, & PENJAGA PERPUS) ---
Route::middleware(['auth'])->group(function () {
// Rute Umum untuk Siswa & Guru
Route::get('/dashboard', [DashboardController::class, 'index'])->name('dashboard');
Route::get('/katalog', [KatalogController::class, 'index'])->name('katalog.index');
@ -80,33 +81,40 @@
// --- GRUP RUTE KHUSUS UNTUK ADMIN / PENJAGA PERPUSTAKAAN ---
Route::middleware(['auth', 'role:penjaga perpus'])->prefix('admin')->name('admin.')->group(function () {
Route::get('/dashboard', [AdminDashboardController::class, 'index'])->name('dashboard');
Route::get('/buku', [AdminBookController::class, 'index'])->name('buku.index');
Route::get('/buku/tambah', [AdminBookController::class, 'create'])->name('buku.create');
Route::get('/buku/{id}/edit', [AdminBookController::class, 'edit'])->name('buku.edit');
Route::get('/pengguna', [AdminUserController::class, 'index'])->name('pengguna.index');
Route::get('/pengguna/tambah', [AdminUserController::class, 'create'])->name('pengguna.create');
Route::get('/pengguna/{id}/edit', [AdminUserController::class, 'edit'])->name('pengguna.edit');
Route::delete('/pengguna/{id}', [AdminUserController::class, 'destroy'])->name('pengguna.destroy');
Route::get('/pengumuman', [AdminPengumumanController::class, 'index'])->name('pengumuman.index');
Route::get('/pengumuman/tambah', [AdminPengumumanController::class, 'create'])->name('pengumuman.create');
Route::get('/pengumuman/{id}/edit', [AdminPengumumanController::class, 'edit'])->name('pengumuman.edit');
Route::get('/rekomendasi', [AdminRekomendasiController
::class, 'index'])->name('rekomendasi.index');
Route::get('/rekomendasi', [
AdminRekomendasiController
::class,
'index'
])->name('rekomendasi.index');
Route::get('/rekomendasi/tambah', [AdminRekomendasiController::class, 'create'])->name('rekomendasi.create');
Route::get('/rekomendasi/{id}/edit', [AdminRekomendasiController::class, 'edit'])->name('rekomendasi.edit');
Route::get('/peminjaman', [AdminPeminjamanController::class, 'index'])->name('peminjaman.index');
Route::get('/peminjaman/tambah', [AdminPeminjamanController::class, 'create'])->name('peminjaman.create');
Route::get('/peminjaman', [AdminPeminjamanController::class, 'index'])->name('peminjaman.index');
Route::get('/peminjaman/tambah', [AdminPeminjamanController::class, 'create'])->name('peminjaman.create');
Route::get('/denda', [AdminPeminjamanController::class, 'dendaIndex'])->name('denda.index');
Route::post('/denda/sanksi', [AdminPeminjamanController::class, 'berikanSanksi'])->name('denda.sanksi');
Route::post('/master-induk', [MasterIndukController::class, 'store'])->name('master-induk.store');
Route::delete('/master-induk/{id}', [MasterIndukController::class, 'destroy'])->name('master-induk.destroy');
});
// --- RUTE LOGIN KHUSUS ADMIN ---
Route::middleware('guest')->group(function() {
Route::middleware('guest')->group(function () {
Route::get('/admin/login', [AdminLoginController::class, 'create'])->name('admin.login');
Route::post('/admin/login', [AdminLoginController::class, 'store'])->name('admin.login.store');
});