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; namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller; 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 class UserController extends Controller
{ {
public function index() 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', [ return view('admin.pengguna.index', [
'pageTitle' => 'Manajemen Pengguna', 'pageTitle' => 'Daftar Pengguna',
'semuaSiswa' => $semuaSiswa 'users' => $users,
'whitelists' => $whitelists
]); ]);
} }
/**
* Menampilkan form untuk membuat pengguna baru.
*/
public function create() public function create()
{ {
return view('admin.pengguna.create', [ return view('admin.pengguna.create', ['pageTitle' => 'Tambah Pengguna Baru']);
'pageTitle' => 'Tambah Pengguna Baru',
]);
} }
/**
* Menampilkan form untuk mengedit pengguna yang ada.
*/
public function edit($id) public function edit($id)
{ {
$pengguna = collect(DummyDataService::getAllSiswa())->firstWhere('id', (int)$id); $pengguna = User::findOrFail($id);
abort_if(!$pengguna, 404);
return view('admin.pengguna.edit', [ return view('admin.pengguna.edit', [
'pageTitle' => 'Edit Pengguna', 'pageTitle' => 'Edit Pengguna',
'pengguna' => $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 // Memproses login admin
public function store(LoginRequest $request): RedirectResponse public function store(LoginRequest $request): RedirectResponse
{ {
$request->authenticate(); // Menjalankan logika ketat di LoginRequest $request->authenticate();
$request->session()->regenerate(); $request->session()->regenerate();
return redirect()->route('admin.dashboard'); return redirect()->route('admin.dashboard');
} }

View File

@ -4,83 +4,72 @@
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
use App\Models\User; use App\Models\User;
use App\Models\MasterInduk;
use Illuminate\Auth\Events\Registered; use Illuminate\Auth\Events\Registered;
use Illuminate\Http\RedirectResponse; use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use Illuminate\Support\Facades\Auth; use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Hash; use Illuminate\Support\Facades\Hash;
use Illuminate\Validation\Rules; use Illuminate\Validation\Rules;
use Illuminate\Validation\ValidationException;
use Illuminate\View\View; use Illuminate\View\View;
class RegisteredUserController extends Controller class RegisteredUserController extends Controller
{ {
/**
* Menampilkan halaman registrasi.
*/
public function create(Request $request): View public function create(Request $request): View
{ {
// Bagian Pengambilan Role dari URL $role = $request->query('role', 'siswa');
$role = $request->query('role', 'siswa'); // Ambil 'role' dari URL, default ke 'siswa'
// Kirim $role ke view
return view('auth.register', ['role' => $role]); return view('auth.register', ['role' => $role]);
} }
/**
* Menangani permintaan registrasi yang masuk.
*/
public function store(Request $request): RedirectResponse public function store(Request $request): RedirectResponse
{ {
// Bagian Validasi Dinamis
$role = $request->input('role'); $role = $request->input('role');
$rules = [ $rules = [
'name' => ['required', 'string', 'max:255'], 'name' => ['required', 'string', 'max:255'],
'email' => ['required', 'string', 'email', 'max:255', 'unique:users'],
'password' => ['required', 'confirmed', Rules\Password::defaults()], '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') { if ($role === 'siswa') {
$rules['nisn'] = ['required', 'string', 'max:255']; // Tambahkan 'unique:users' jika perlu $rules['nisn'] = ['required', 'string', 'max:255', 'unique:users,nisn'];
} else { // Asumsi 'guru' } else {
$rules['nip'] = ['required', 'string', 'max:255']; // Tambahkan 'unique:users' jika perlu $rules['nip'] = ['required', 'string', 'max:255', 'unique:users,nip'];
} }
$request->validate($rules); $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 $nomorInduk = ($role === 'siswa') ? $request->nisn : $request->nip;
if ($role === 'siswa') {
$userArray['nisn'] = $request->nisn; $isWhitelisted = MasterInduk::where('nomor_induk', $nomorInduk)
$userArray['email'] = $request->nisn . '@smkn1perpus.sch.id'; // Email unik sementara ->where('role', $role)
} else { ->exists();
$userArray['nip'] = $request->nip;
$userArray['email'] = $request->nip . '@smkn1perpus.sch.id'; // Email unik sementara if (!$isWhitelisted) {
throw ValidationException::withMessages([
($role === 'siswa' ? 'nisn' : 'nip') => ['Nomor induk tidak terdaftar di Data Sekolah. Hubungi petugas.'],
]);
} }
$user = new User();
$user->forceFill($userArray); $user = User::create([
// $user->save(); // Aktifkan ini jika menggunakan database '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)); event(new Registered($user));
Auth::login($user); Auth::login($user);
// Bagian Redirect return redirect()->route('dashboard');
if ($user->role === 'penjaga perpus') {
return redirect()->route('admin.dashboard');
} else {
return redirect()->route('dashboard');
}
} }
} }

View File

@ -8,36 +8,30 @@
use Illuminate\Support\Facades\Auth; use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Redirect; use Illuminate\Support\Facades\Redirect;
use Illuminate\View\View; use Illuminate\View\View;
use Illuminate\Support\Facades\Hash;
use App\Services\DummyDataService; use App\Services\DummyDataService;
class ProfileController extends Controller 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(); $user = $request->user();
if (!$user) {
return redirect()->route(route: 'login');
}
$viewData = ['user' => $user]; $viewData = ['user' => $user];
// Menyiapkan data berdasarkan role pengguna
if ($user->role === 'penjaga perpus') { if ($user->role === 'penjaga perpus') {
// Data untuk Penjaga Perpus: Statistik global & aktivitas terkini
$viewData['statistik'] = DummyDataService::getAdminDashboardStats(); $viewData['statistik'] = DummyDataService::getAdminDashboardStats();
$viewData['aktivitasTerakhir'] = DummyDataService::getAktivitasTerakhir(); $viewData['aktivitasTerakhir'] = DummyDataService::getAktivitasTerakhir();
} elseif ($user->role === 'guru') { } elseif ($user->role === 'guru') {
// Data untuk Guru: Data personal + ringkasan laporan minat baca
$viewData['bukuOffline'] = DummyDataService::getBukuPinjamOffline($user); $viewData['bukuOffline'] = DummyDataService::getBukuPinjamOffline($user);
$viewData['bukuOnline'] = DummyDataService::getBacaBukuOnline($user); $viewData['bukuOnline'] = DummyDataService::getBacaBukuOnline($user);
$viewData['laporan'] = DummyDataService::getLaporanMinatBaca(); $viewData['laporan'] = DummyDataService::getLaporanMinatBaca();
} else { } else {
// Data default untuk Siswa
$viewData['bukuOffline'] = DummyDataService::getBukuPinjamOffline($user); $viewData['bukuOffline'] = DummyDataService::getBukuPinjamOffline($user);
$viewData['bukuOnline'] = DummyDataService::getBacaBukuOnline($user); $viewData['bukuOnline'] = DummyDataService::getBacaBukuOnline($user);
$viewData['statistik'] = DummyDataService::getDashboardStats(); $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 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')) { $user = $request->user();
$request->user()->email_verified_at = null; $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'); return Redirect::route('profile.edit')->with('status', 'profile-updated');
} }
/** /**
* Menghapus akun pengguna. * Hapus akun user.
*/ */
public function destroy(Request $request): RedirectResponse public function destroy(Request $request): RedirectResponse
{ {

View File

@ -2,129 +2,74 @@
namespace App\Http\Requests\Auth; namespace App\Http\Requests\Auth;
use App\Models\User; use Illuminate\Auth\Events\Lockout;
use App\Services\DummyDataService;
use Illuminate\Foundation\Http\FormRequest; use Illuminate\Foundation\Http\FormRequest;
use Illuminate\Support\Facades\Auth; use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\RateLimiter; use Illuminate\Support\Facades\RateLimiter;
use Illuminate\Support\Str; use Illuminate\Support\Str;
use Illuminate\Validation\ValidationException; use Illuminate\Validation\ValidationException;
use Illuminate\Auth\Events\Lockout;
class LoginRequest extends FormRequest class LoginRequest extends FormRequest
{ {
/**
* Menentukan apakah pengguna diizinkan untuk membuat request ini.
* Selalu true karena semua orang boleh mencoba login.
*/
public function authorize(): bool public function authorize(): bool
{ {
return true; 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 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 [ return [
'nip' => ['required', 'string'],
'password' => ['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 public function authenticate(): void
{ {
// Pastikan pengguna tidak mencoba login terlalu sering (mencegah brute-force).
$this->ensureIsNotRateLimited(); $this->ensureIsNotRateLimited();
// Ambil data yang dikirim dari form login. $loginString = $this->input('email') ?? $this->input('nisn') ?? $this->input('nip');
$roleDariForm = $this->input('role');
$allUsers = DummyDataService::getAllSiswa();
$inputPassword = $this->input('password');
$userArray = null;
// Tentukan field mana yang akan menerima pesan error jika gagal (nisn atau email). $password = $this->input('password');
$errorField = $this->filled('nisn') ? 'nisn' : 'nip';
// Cari data pengguna berdasarkan input yang diberikan. if (!$loginString) {
if ($this->filled('nisn')) { throw ValidationException::withMessages([
// Jika form diisi dengan 'nisn', cari pengguna berdasarkan 'nisn'. 'email' => 'Mohon masukkan NISN, NIP, atau Email.',
$userArray = collect($allUsers)->firstWhere('nisn', $this->input('nisn')); ]);
} else {
// Jika tidak, cari pengguna berdasarkan 'nip'.
$userArray = collect($allUsers)->firstWhere('nip', $this->input('nip'));
} }
// Lakukan Pengecekan Kredensial dan Role. if (Auth::attempt(['nisn' => $loginString, 'password' => $password], $this->boolean('remember'))) {
if ($userArray && $userArray['password'] === $inputPassword) { RateLimiter::clear($this->throttleKey());
return;
// Cek #2: Jika kredensial benar, apakah role pengguna sesuai dengan form yang digunakan? }
if (isset($userArray['role']) && $userArray['role'] === $roleDariForm) {
if (Auth::attempt(['nip' => $loginString, 'password' => $password], $this->boolean('remember'))) {
// Buat objek User dari data dummy. RateLimiter::clear($this->throttleKey());
$userModel = new User(); return;
$userArray['name'] = $userArray['nama_lengkap']; }
$userModel->forceFill($userArray);
if (Auth::attempt(['email' => $loginString, 'password' => $password], $this->boolean('remember'))) {
// Loginkan pengguna secara resmi ke dalam sistem. RateLimiter::clear($this->throttleKey());
Auth::login($userModel); return;
}
// Reset hitungan percobaan login yang gagal.
RateLimiter::clear($this->throttleKey()); RateLimiter::hit($this->throttleKey());
return; // Proses autentikasi berhasil.
$fieldError = $this->input('nisn') ? 'nisn' : ($this->input('nip') ? 'nip' : 'email');
} else {
// Tambah hitungan percobaan login yang gagal. throw ValidationException::withMessages([
RateLimiter::hit($this->throttleKey()); $fieldError => trans('auth.failed'),
]);
// Ambil nama role asli pengguna untuk ditampilkan di pesan error. }
$actualRole = Str::title($userArray['role'] ?? 'Tidak Dikenal');
public function ensureIsNotRateLimited(): void
// Lemparkan error validasi khusus 'forbidden' dengan pesan yang jelas. {
throw ValidationException::withMessages([ if (! RateLimiter::tooManyAttempts($this->throttleKey(), 5)) {
'forbidden' => "Akses ditolak. Akun ini terdaftar sebagai {$actualRole}.",
]);
}
}
// Jika pengguna tidak ditemukan atau password salah.
RateLimiter::hit($this->throttleKey());
throw ValidationException::withMessages([
$errorField => trans('auth.failed'), // Pesan error umum "These credentials do not match...".
]);
}
/**
* 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)) {
return; return;
} }
// Jika sudah lebih dari 5 kali, lemparkan error 'throttle'.
event(new Lockout($this)); event(new Lockout($this));
$seconds = RateLimiter::availableIn($this->throttleKey()); $seconds = RateLimiter::availableIn($this->throttleKey());
throw ValidationException::withMessages([ throw ValidationException::withMessages([
'email' => trans('auth.throttle', [ 'email' => trans('auth.throttle', [
'seconds' => $seconds, '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 public function throttleKey(): string
{ {
// Gunakan 'nisn' jika ada, jika tidak, gunakan 'email' sebagai identitas. $field = $this->input('email') ?? $this->input('nisn') ?? $this->input('nip') ?? 'unknown';
$loginIdentifier = $this->input('nisn') ?: $this->input('nip'); return Str::transliterate(Str::lower($field).'|'.$this->ip());
return Str::transliterate(Str::lower($loginIdentifier) . '|' . $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', 'name',
'email', 'email',
'password', 'password',
'role',
'nisn',
'nip',
'nomor_hp',
'kelas',
'nomor_induk'
]; ];
/** /**
@ -45,4 +51,9 @@ protected function casts(): array
'password' => 'hashed', 'password' => 'hashed',
]; ];
} }
public function getNamaLengkapAttribute()
{
return $this->name;
}
} }

View File

@ -2,16 +2,14 @@
namespace App\Providers; namespace App\Providers;
use App\Auth\DummyUserProvider; // use App\Auth\DummyUserProvider; // Hapus atau Komen baris ini
use Illuminate\Support\Facades\Auth; use Illuminate\Support\Facades\Gate;
use Illuminate\Foundation\Support\Providers\AuthServiceProvider as ServiceProvider; use Illuminate\Foundation\Support\Providers\AuthServiceProvider as ServiceProvider;
class AuthServiceProvider extends ServiceProvider class AuthServiceProvider extends ServiceProvider
{ {
/** /**
* The model to policy mappings for the application. * The model to policy mappings for the application.
*
* @var array<class-string, class-string>
*/ */
protected $policies = [ protected $policies = [
// //
@ -22,9 +20,13 @@ class AuthServiceProvider extends ServiceProvider
*/ */
public function boot(): void 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) { Auth::provider('dummy', function ($app, array $config) {
return new DummyUserProvider(); return new DummyUserProvider();
}); });
*/
} }
} }

View File

@ -38,7 +38,7 @@
'guards' => [ 'guards' => [
'web' => [ 'web' => [
'driver' => 'session', 'driver' => 'session',
'provider' => 'dummy', 'provider' => 'users',
], ],
], ],
@ -64,10 +64,6 @@
'driver' => 'eloquent', 'driver' => 'eloquent',
'model' => env('AUTH_MODEL', App\Models\User::class), '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->string('email')->unique();
$table->timestamp('email_verified_at')->nullable(); $table->timestamp('email_verified_at')->nullable();
$table->string('password'); $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->rememberToken();
$table->timestamps(); $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; namespace Database\Seeders;
use App\Models\User;
// use Illuminate\Database\Console\Seeds\WithoutModelEvents;
use Illuminate\Database\Seeder; 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 class DatabaseSeeder extends Seeder
{ {
/** public function run()
* Seed the application's database.
*/
public function run(): void
{ {
// 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([ // 1. ISI DATA INDUK (WHITELIST)
'name' => 'Test User', $whitelist = [
'email' => 'test@example.com', ['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> <x-app-layout>
@section('page-title', $pageTitle) @section('page-title', content: 'Manajemen Pengguna')
<div class="container-fluid p-0">
<div class="card shadow-sm border-0"> <div class="d-flex justify-content-between align-items-center mb-3">
<div class="card-header bg-white d-flex justify-content-between align-items-center"> <h1 class="h3 text-gray-800">{{ $pageTitle }}</h1>
<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>
</div> </div>
<div class="card-body">
<div class="table-responsive"> <div class="card shadow mb-5">
<table class="table table-hover"> <div class="card-header py-3">
<thead> <h6 class="m-0 font-weight-bold text-primary">Daftar Pengguna Aktif</h6>
<tr> </div>
<th>No</th> <div class="card-body">
<th>Nama Lengkap</th> <div class="table-responsive">
<th>Email</th> <table class="table table-bordered" width="100%" cellspacing="0">
<th>Role</th> <thead>
<th>Aksi</th> <tr>
</tr> <th>No</th>
</thead> <th>Nama Lengkap</th>
<tbody> <th>Email</th>
@forelse($semuaSiswa as $siswa) <th>Role</th>
<tr> <th>Nomor Induk</th>
<td>{{ $loop->iteration }}</td> <th>Aksi</th>
<td>{{ $siswa['nama_lengkap'] }}</td> </tr>
<td>{{ $siswa['email'] }}</td> </thead>
<td> <tbody>
@if($siswa['role'] == 'penjaga perpus') @forelse($users as $index => $user)
<span class="badge bg-success-subtle text-success-emphasis">{{ Str::title($siswa['role']) }}</span> <tr>
@else <td>{{ $index + 1 }}</td>
<span class="badge bg-primary-subtle text-primary-emphasis">{{ Str::title($siswa['role']) }}</span> <td>{{ $user->name }}</td>
@endif <td>{{ $user->email }}</td>
</td> <td>
<td> <span class="badge {{ $user->role == 'guru' ? 'bg-info' : 'bg-primary' }}">
<button class="btn btn-sm btn-outline-secondary" data-bs-toggle="modal" data-bs-target="#detailPenggunaModal" {{ ucfirst($user->role) }}
data-id="{{ $siswa['id'] }}" </span>
data-nama="{{ $siswa['nama_lengkap'] }}" </td>
data-email="{{ $siswa['email'] }}" <td>
data-role="{{ Str::title($siswa['role']) }}" {{ $user->nisn ?? $user->nip ?? '-' }}
data-nisn="{{ $siswa['nisn'] ?? 'N/A' }}"> </td>
<i class="bi bi-eye-fill"></i> Detail <td>
</button> <a href="{{ route('admin.pengguna.edit', $user->id) }}" class="btn btn-sm btn-warning">
</td> <i class="bi bi-pencil"></i> Edit
</tr> </a>
@empty <form action="{{ route('admin.pengguna.destroy', $user->id) }}" method="POST" class="d-inline" onsubmit="return confirm('Yakin hapus user ini?')">
<tr> @csrf
<td colspan="5" class="text-center">Tidak ada data pengguna.</td> @method('DELETE')
</tr> <button class="btn btn-sm btn-danger"><i class="bi bi-trash"></i></button>
@endforelse </form>
</tbody> </td>
</table> </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>
</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>
<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-dialog">
<div class="modal-content"> <div class="modal-content">
<div class="modal-header"> <div class="modal-header">
<h5 class="modal-title fw-bold" id="modalNama"></h5> <h5 class="modal-title fw-bold">Tambah Whitelist (NIP/NISN)</h5>
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button> <button type="button" class="btn-close" data-bs-dismiss="modal"></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>
</div> </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> </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> </x-app-layout>

View File

@ -1,57 +1,81 @@
<x-guest-layout> <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') }}"> <form method="POST" action="{{ route('register') }}">
@csrf @csrf
{{-- Input tersembunyi untuk mengirimkan peran (role) ke backend --}}
<input type="hidden" name="role" value="{{ $role }}"> <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"> <div class="mb-3">
<label for="name" class="form-label">Nama Lengkap</label> <label 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" /> <input type="text" name="name" class="form-control @error('name') is-invalid @enderror"
value="{{ old('name') }}" required autofocus placeholder="Nama sesuai absen">
@error('name') @error('name')
<div class="invalid-feedback">{{ $message }}</div> <div class="invalid-feedback">{{ $message }}</div>
@enderror @enderror
</div> </div>
{{-- Form dinamis (NISN untuk Siswa, NIP untuk Guru) --}} <div class="mb-3">
@if ($role == 'siswa') <label class="form-label">
<div class="mb-3"> @if ($role == 'siswa')
<label for="nisn" class="form-label">Nomor Induk Siswa Nasional (NISN)</label> NISN (Nomor Induk Siswa)
<input id="nisn" class="form-control bg-body-tertiary @error('nisn') is-invalid @enderror" type="text" name="nisn" value="{{ old('nisn') }}" required autocomplete="username" /> @else
@error('nisn') NIP / NIK Sekolah
<div class="invalid-feedback">{{ $message }}</div> @endif
@enderror </label>
</div>
@else <input type="number" name="{{ $role == 'siswa' ? 'nisn' : 'nip' }}"
<div class="mb-3"> class="form-control @error($role == 'siswa' ? 'nisn' : 'nip') is-invalid @enderror"
<label for="nip" class="form-label">Nomor Induk Pegawai (NIP)</label> value="{{ old($role == 'siswa' ? 'nisn' : 'nip') }}" required placeholder="Cth: 1234567890">
<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') @error($role == 'siswa' ? 'nisn' : 'nip')
<div class="invalid-feedback">{{ $message }}</div> <div class="invalid-feedback">{{ $message }}</div>
@enderror @enderror
</div> </div>
@endif
<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"> <div class="mb-3">
<label for="password" class="form-label">Password</label> <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') @error('password')
<div class="invalid-feedback">{{ $message }}</div> <div class="invalid-feedback">{{ $message }}</div>
@enderror @enderror
</div> </div>
{{-- Form Konfirmasi Password (Umum) --}}
<div class="mb-3"> <div class="mb-3">
<label for="password_confirmation" class="form-label">Konfirmasi Password</label> <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>
<div class="d-grid mt-4"> <div class="d-grid mt-4">
@ -63,7 +87,8 @@
{{-- Link Login dinamis --}} {{-- Link Login dinamis --}}
<p class="mt-4 text-center text-muted small"> <p class="mt-4 text-center text-muted small">
Sudah punya akun? 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> </p>
</form> </form>
</x-guest-layout> </x-guest-layout>

View File

@ -15,6 +15,7 @@
// Admin Controllers // Admin Controllers
use App\Http\Controllers\Admin\DashboardController as AdminDashboardController; use App\Http\Controllers\Admin\DashboardController as AdminDashboardController;
use App\Http\Controllers\Admin\BookController as AdminBookController; 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\UserController as AdminUserController;
use App\Http\Controllers\Admin\PengumumanController as AdminPengumumanController; use App\Http\Controllers\Admin\PengumumanController as AdminPengumumanController;
use App\Http\Controllers\RekomendasiController; use App\Http\Controllers\RekomendasiController;
@ -88,13 +89,17 @@
Route::get('/pengguna', [AdminUserController::class, 'index'])->name('pengguna.index'); Route::get('/pengguna', [AdminUserController::class, 'index'])->name('pengguna.index');
Route::get('/pengguna/tambah', [AdminUserController::class, 'create'])->name('pengguna.create'); Route::get('/pengguna/tambah', [AdminUserController::class, 'create'])->name('pengguna.create');
Route::get('/pengguna/{id}/edit', [AdminUserController::class, 'edit'])->name('pengguna.edit'); 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', [AdminPengumumanController::class, 'index'])->name('pengumuman.index');
Route::get('/pengumuman/tambah', [AdminPengumumanController::class, 'create'])->name('pengumuman.create'); Route::get('/pengumuman/tambah', [AdminPengumumanController::class, 'create'])->name('pengumuman.create');
Route::get('/pengumuman/{id}/edit', [AdminPengumumanController::class, 'edit'])->name('pengumuman.edit'); Route::get('/pengumuman/{id}/edit', [AdminPengumumanController::class, 'edit'])->name('pengumuman.edit');
Route::get('/rekomendasi', [AdminRekomendasiController Route::get('/rekomendasi', [
::class, 'index'])->name('rekomendasi.index'); AdminRekomendasiController
::class,
'index'
])->name('rekomendasi.index');
Route::get('/rekomendasi/tambah', [AdminRekomendasiController::class, 'create'])->name('rekomendasi.create'); Route::get('/rekomendasi/tambah', [AdminRekomendasiController::class, 'create'])->name('rekomendasi.create');
Route::get('/rekomendasi/{id}/edit', [AdminRekomendasiController::class, 'edit'])->name('rekomendasi.edit'); Route::get('/rekomendasi/{id}/edit', [AdminRekomendasiController::class, 'edit'])->name('rekomendasi.edit');
@ -103,10 +108,13 @@
Route::get('/denda', [AdminPeminjamanController::class, 'dendaIndex'])->name('denda.index'); Route::get('/denda', [AdminPeminjamanController::class, 'dendaIndex'])->name('denda.index');
Route::post('/denda/sanksi', [AdminPeminjamanController::class, 'berikanSanksi'])->name('denda.sanksi'); 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 --- // --- 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::get('/admin/login', [AdminLoginController::class, 'create'])->name('admin.login');
Route::post('/admin/login', [AdminLoginController::class, 'store'])->name('admin.login.store'); Route::post('/admin/login', [AdminLoginController::class, 'store'])->name('admin.login.store');
}); });