Feat: Master data induk NIP/NISN
This commit is contained in:
parent
779ef38952
commit
c1e631c3ce
|
|
@ -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.');
|
||||
}
|
||||
}
|
||||
|
|
@ -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.');
|
||||
}
|
||||
}
|
||||
|
|
@ -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');
|
||||
}
|
||||
|
|
|
|||
|
|
@ -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');
|
||||
}
|
||||
}
|
||||
|
|
@ -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('/');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -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());
|
||||
}
|
||||
}
|
||||
|
|
@ -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'];
|
||||
}
|
||||
|
|
@ -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;
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -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();
|
||||
});
|
||||
*/
|
||||
}
|
||||
}
|
||||
|
|
@ -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',
|
||||
],
|
||||
],
|
||||
|
||||
/*
|
||||
|
|
|
|||
|
|
@ -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();
|
||||
});
|
||||
|
|
|
|||
|
|
@ -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');
|
||||
}
|
||||
};
|
||||
|
|
@ -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',
|
||||
]);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -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>
|
||||
|
|
@ -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>
|
||||
|
|
|
|||
|
|
@ -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');
|
||||
});
|
||||
|
|
|
|||
Loading…
Reference in New Issue