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;
|
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.');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -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');
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -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');
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -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
|
||||||
{
|
{
|
||||||
|
|
|
||||||
|
|
@ -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());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -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',
|
'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;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -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();
|
||||||
});
|
});
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -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',
|
|
||||||
],
|
|
||||||
],
|
],
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
||||||
|
|
@ -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();
|
||||||
});
|
});
|
||||||
|
|
|
||||||
|
|
@ -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;
|
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',
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -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>
|
||||||
|
|
@ -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>
|
||||||
|
|
@ -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');
|
||||||
});
|
});
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue