first commit

This commit is contained in:
2026-05-14 09:08:09 +08:00
commit 919b86c8ec
111 changed files with 14085 additions and 0 deletions

View File

@@ -0,0 +1,27 @@
<?php
namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use App\Models\Permohonan;
class ApprovalController extends Controller
{
public function index()
{
$permohonan = Permohonan::with(['user', 'jabatan', 'vot'])->latest()->paginate(10);
return view('reports.permohonan', compact('permohonan'));
}
public function approve(Permohonan $permohonan)
{
$permohonan->update(['status' => 'Approved', 'approved_by' => auth()->id(), 'approved_at' => now()]);
return back()->with('success', 'Permohonan diluluskan.');
}
public function reopen(Permohonan $permohonan)
{
$permohonan->update(['status' => 'Under Review', 'approved_by' => null, 'approved_at' => null]);
return back()->with('success', 'Permohonan dibuka semula.');
}
}

View File

@@ -0,0 +1,53 @@
<?php
namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use App\Models\Jabatan;
use Illuminate\Http\Request;
class JabatanController extends Controller
{
public function index(Request $request)
{
$jabatan = Jabatan::query()
->when($request->search, fn ($q, $v) => $q->where('nama', 'like', "%$v%")->orWhere('kod', 'like', "%$v%"))
->when($request->peringkat, fn ($q, $v) => $q->where('peringkat', $v))
->latest()->paginate(10)->withQueryString();
return view('admin.jabatan.index', compact('jabatan'));
}
public function create() { return view('admin.jabatan.form', ['jabatan' => new Jabatan()]); }
public function store(Request $request)
{
Jabatan::create($this->validated($request));
return redirect()->route('jabatan.index')->with('success', 'Jabatan berjaya ditambah.');
}
public function edit(Jabatan $jabatan) { return view('admin.jabatan.form', compact('jabatan')); }
public function update(Request $request, Jabatan $jabatan)
{
$jabatan->update($this->validated($request, $jabatan->id));
return redirect()->route('jabatan.index')->with('success', 'Jabatan berjaya dikemaskini.');
}
public function destroy(Jabatan $jabatan)
{
$jabatan->delete();
return back()->with('success', 'Jabatan dipadam secara soft delete.');
}
private function validated(Request $request, ?int $id = null): array
{
return $request->validate([
'nama' => ['required', 'max:255'],
'peringkat' => ['required', 'in:Jabatan,Bahagian,Seksyen,Unit,Sub Unit'],
'kod' => ['nullable', 'max:50', 'unique:jabatan,kod,'.$id],
'penerangan' => ['nullable'],
'status' => ['nullable', 'boolean'],
]) + ['status' => $request->boolean('status')];
}
}

View File

@@ -0,0 +1,29 @@
<?php
namespace App\Http\Controllers\Admin;
use App\Exports\PermohonanExport;
use App\Http\Controllers\Controller;
use App\Models\Permohonan;
use Barryvdh\DomPDF\Facade\Pdf;
use Maatwebsite\Excel\Facades\Excel;
class ReportController extends Controller
{
public function index()
{
$permohonan = Permohonan::with(['user', 'jabatan', 'vot'])->latest()->paginate(20);
return view('reports.index', compact('permohonan'));
}
public function excel()
{
return Excel::download(new PermohonanExport(), 'laporan-permohonan.xlsx');
}
public function pdf()
{
$permohonan = Permohonan::with(['user', 'jabatan', 'vot'])->latest()->get();
return Pdf::loadView('reports.pdf', compact('permohonan'))->download('laporan-permohonan.pdf');
}
}

View File

@@ -0,0 +1,50 @@
<?php
namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use Illuminate\Http\Request;
use Spatie\Permission\Models\Permission;
use Spatie\Permission\Models\Role;
class RoleController extends Controller
{
public function index()
{
return view('admin.roles.index', ['roles' => Role::with('permissions')->paginate(10)]);
}
public function create()
{
return view('admin.roles.form', ['role' => new Role(), 'permissions' => Permission::orderBy('name')->get()]);
}
public function store(Request $request)
{
$validated = $request->validate(['name' => ['required', 'unique:roles,name'], 'permissions' => ['array']]);
$role = Role::create(['name' => $validated['name'], 'guard_name' => 'web']);
$role->syncPermissions($validated['permissions'] ?? []);
return redirect()->route('roles.index')->with('success', 'Role berjaya ditambah.');
}
public function edit(Role $role)
{
return view('admin.roles.form', ['role' => $role, 'permissions' => Permission::orderBy('name')->get()]);
}
public function update(Request $request, Role $role)
{
$validated = $request->validate(['name' => ['required', 'unique:roles,name,'.$role->id], 'permissions' => ['array']]);
$role->update(['name' => $validated['name']]);
$role->syncPermissions($validated['permissions'] ?? []);
return redirect()->route('roles.index')->with('success', 'Role berjaya dikemaskini.');
}
public function destroy(Role $role)
{
$role->delete();
return back()->with('success', 'Role berjaya dipadam.');
}
}

View File

@@ -0,0 +1,95 @@
<?php
namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use App\Models\Jabatan;
use App\Models\User;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Hash;
use Spatie\Permission\Models\Role;
class UserController extends Controller
{
public function index(Request $request)
{
$users = User::with(['jabatan', 'roles'])
->when($request->search, fn ($q, $v) => $q->where('name', 'like', "%$v%")->orWhere('email', 'like', "%$v%"))
->latest()->paginate(10)->withQueryString();
return view('admin.users.index', compact('users'));
}
public function create()
{
return view('admin.users.form', $this->formData(new User()));
}
public function store(Request $request)
{
$validated = $this->validated($request);
$role = $validated['role'] ?? null;
unset($validated['role']);
$validated['password'] = Hash::make($validated['password']);
$user = User::create($validated);
$user->syncRoles($role ? [$role] : []);
return redirect()->route('users.index')->with('success', 'User berjaya ditambah.');
}
public function edit(User $user)
{
return view('admin.users.form', $this->formData($user));
}
public function update(Request $request, User $user)
{
$validated = $this->validated($request, $user->id);
$role = $validated['role'] ?? null;
unset($validated['role']);
if (empty($validated['password'])) {
unset($validated['password']);
} else {
$validated['password'] = Hash::make($validated['password']);
}
$user->update($validated);
$user->syncRoles($role ? [$role] : []);
return redirect()->route('users.index')->with('success', 'User berjaya dikemaskini.');
}
public function destroy(User $user)
{
$user->update(['status' => ! $user->status]);
return back()->with('success', 'Status user berjaya dikemaskini.');
}
public function resetPassword(User $user)
{
$user->update(['password' => Hash::make('password')]);
return back()->with('success', 'Kata laluan diset semula kepada: password');
}
private function formData(User $user): array
{
return [
'user' => $user,
'jabatan' => Jabatan::orderBy('nama')->get(),
'roles' => Role::orderBy('name')->get(),
];
}
private function validated(Request $request, ?int $id = null): array
{
return $request->validate([
'name' => ['required', 'max:255'],
'email' => ['required', 'email', 'unique:users,email,'.$id],
'no_telefon' => ['nullable', 'max:30'],
'jawatan' => ['nullable', 'max:255'],
'jabatan_id' => ['nullable', 'exists:jabatan,id'],
'role' => ['nullable', 'exists:roles,name'],
'password' => [$id ? 'nullable' : 'required', 'confirmed', 'min:8'],
'status' => ['nullable', 'boolean'],
]) + ['status' => $request->boolean('status')];
}
}

View File

@@ -0,0 +1,55 @@
<?php
namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use App\Models\Jabatan;
use App\Models\Vot;
use Illuminate\Http\Request;
class VotController extends Controller
{
public function index(Request $request)
{
$vot = Vot::with('jabatan')
->when($request->search, fn ($q, $v) => $q->where('nama', 'like', "%$v%")->orWhere('kod', 'like', "%$v%"))
->when($request->jabatan_id, fn ($q, $v) => $q->where('jabatan_id', $v))
->latest()->paginate(10)->withQueryString();
$jabatan = Jabatan::orderBy('nama')->get();
return view('admin.vot.index', compact('vot', 'jabatan'));
}
public function create() { return view('admin.vot.form', ['vot' => new Vot(), 'jabatan' => Jabatan::orderBy('nama')->get()]); }
public function store(Request $request)
{
Vot::create($this->validated($request));
return redirect()->route('vot.index')->with('success', 'VOT berjaya ditambah.');
}
public function edit(Vot $vot) { return view('admin.vot.form', ['vot' => $vot, 'jabatan' => Jabatan::orderBy('nama')->get()]); }
public function update(Request $request, Vot $vot)
{
$vot->update($this->validated($request, $vot->id));
return redirect()->route('vot.index')->with('success', 'VOT berjaya dikemaskini.');
}
public function destroy(Vot $vot)
{
$vot->delete();
return back()->with('success', 'VOT berjaya dipadam.');
}
private function validated(Request $request, ?int $id = null): array
{
return $request->validate([
'kod' => ['required', 'max:50', 'unique:vot,kod,'.$id],
'nama' => ['required', 'max:255'],
'jabatan_id' => ['nullable', 'exists:jabatan,id'],
'penerangan' => ['nullable'],
'status' => ['nullable', 'boolean'],
]) + ['status' => $request->boolean('status')];
}
}