Initial commit
This commit is contained in:
347
app/Http/Controllers/AdminMemberController.php
Normal file
347
app/Http/Controllers/AdminMemberController.php
Normal file
@@ -0,0 +1,347 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Controllers;
|
||||
|
||||
use App\Models\Member;
|
||||
use App\Support\AuditLogger;
|
||||
use Spatie\Permission\Models\Role;
|
||||
use Symfony\Component\HttpFoundation\StreamedResponse;
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\Support\Facades\Artisan;
|
||||
|
||||
class AdminMemberController extends Controller
|
||||
{
|
||||
public function index(Request $request)
|
||||
{
|
||||
$query = Member::query()->with('user');
|
||||
|
||||
// Text search (name, email, phone, national ID)
|
||||
if ($search = $request->string('search')->toString()) {
|
||||
$query->where(function ($q) use ($search) {
|
||||
$q->where('full_name', 'like', "%{$search}%")
|
||||
->orWhere('email', 'like', "%{$search}%")
|
||||
->orWhere('phone', 'like', "%{$search}%");
|
||||
|
||||
// Search by national ID hash if provided
|
||||
if (!empty($search)) {
|
||||
$q->orWhere('national_id_hash', hash('sha256', $search));
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// Membership status filter
|
||||
if ($status = $request->string('status')->toString()) {
|
||||
if ($status === 'active') {
|
||||
$query->whereDate('membership_expires_at', '>=', now()->toDateString());
|
||||
} elseif ($status === 'expired') {
|
||||
$query->where(function ($q) {
|
||||
$q->whereNull('membership_expires_at')
|
||||
->orWhereDate('membership_expires_at', '<', now()->toDateString());
|
||||
});
|
||||
} elseif ($status === 'expiring_soon') {
|
||||
$query->whereBetween('membership_expires_at', [
|
||||
now()->toDateString(),
|
||||
now()->addDays(30)->toDateString()
|
||||
]);
|
||||
}
|
||||
}
|
||||
|
||||
// Date range filters
|
||||
if ($startedFrom = $request->string('started_from')->toString()) {
|
||||
$query->whereDate('membership_started_at', '>=', $startedFrom);
|
||||
}
|
||||
if ($startedTo = $request->string('started_to')->toString()) {
|
||||
$query->whereDate('membership_started_at', '<=', $startedTo);
|
||||
}
|
||||
|
||||
// Payment status filter
|
||||
if ($paymentStatus = $request->string('payment_status')->toString()) {
|
||||
if ($paymentStatus === 'has_payments') {
|
||||
$query->whereHas('payments');
|
||||
} elseif ($paymentStatus === 'no_payments') {
|
||||
$query->whereDoesntHave('payments');
|
||||
}
|
||||
}
|
||||
|
||||
$members = $query->orderBy('full_name')->paginate(15)->withQueryString();
|
||||
|
||||
return view('admin.members.index', [
|
||||
'members' => $members,
|
||||
'filters' => $request->only(['search', 'status', 'started_from', 'started_to', 'payment_status']),
|
||||
]);
|
||||
}
|
||||
|
||||
public function show(Member $member)
|
||||
{
|
||||
$member->load('user.roles', 'payments');
|
||||
|
||||
$roles = Role::orderBy('name')->get();
|
||||
|
||||
return view('admin.members.show', [
|
||||
'member' => $member,
|
||||
'roles' => $roles,
|
||||
]);
|
||||
}
|
||||
|
||||
public function create()
|
||||
{
|
||||
return view('admin.members.create');
|
||||
}
|
||||
|
||||
public function store(Request $request)
|
||||
{
|
||||
$validated = $request->validate([
|
||||
'full_name' => ['required', 'string', 'max:255'],
|
||||
'email' => ['required', 'email', 'max:255', 'unique:users,email'],
|
||||
'national_id' => ['nullable', 'string', 'max:50'],
|
||||
'phone' => ['nullable', 'string', 'max:50'],
|
||||
'address_line_1' => ['nullable', 'string', 'max:255'],
|
||||
'address_line_2' => ['nullable', 'string', 'max:255'],
|
||||
'city' => ['nullable', 'string', 'max:120'],
|
||||
'postal_code' => ['nullable', 'string', 'max:20'],
|
||||
'emergency_contact_name' => ['nullable', 'string', 'max:255'],
|
||||
'emergency_contact_phone' => ['nullable', 'string', 'max:50'],
|
||||
'membership_started_at' => ['nullable', 'date'],
|
||||
'membership_expires_at' => ['nullable', 'date', 'after_or_equal:membership_started_at'],
|
||||
]);
|
||||
|
||||
// Create user account
|
||||
$user = \App\Models\User::create([
|
||||
'name' => $validated['full_name'],
|
||||
'email' => $validated['email'],
|
||||
'password' => \Illuminate\Support\Str::random(32),
|
||||
]);
|
||||
|
||||
// Create member record
|
||||
$member = Member::create(array_merge($validated, [
|
||||
'user_id' => $user->id,
|
||||
]));
|
||||
|
||||
// Send activation email
|
||||
$token = \Illuminate\Support\Facades\Password::createToken($user);
|
||||
\Illuminate\Support\Facades\Mail::to($user)->queue(new \App\Mail\MemberActivationMail($user, $token));
|
||||
|
||||
// Log the action
|
||||
AuditLogger::log('member.created', $member, $validated);
|
||||
AuditLogger::log('user.activation_link_sent', $user, ['email' => $user->email]);
|
||||
|
||||
return redirect()
|
||||
->route('admin.members.show', $member)
|
||||
->with('status', __('Member created successfully. Activation email has been sent.'));
|
||||
}
|
||||
|
||||
public function edit(Member $member)
|
||||
{
|
||||
return view('admin.members.edit', [
|
||||
'member' => $member,
|
||||
]);
|
||||
}
|
||||
|
||||
public function update(Request $request, Member $member)
|
||||
{
|
||||
$validated = $request->validate([
|
||||
'full_name' => ['required', 'string', 'max:255'],
|
||||
'email' => ['required', 'email', 'max:255'],
|
||||
'national_id' => ['nullable', 'string', 'max:50'],
|
||||
'phone' => ['nullable', 'string', 'max:50'],
|
||||
'address_line_1' => ['nullable', 'string', 'max:255'],
|
||||
'address_line_2' => ['nullable', 'string', 'max:255'],
|
||||
'city' => ['nullable', 'string', 'max:120'],
|
||||
'postal_code' => ['nullable', 'string', 'max:20'],
|
||||
'emergency_contact_name' => ['nullable', 'string', 'max:255'],
|
||||
'emergency_contact_phone' => ['nullable', 'string', 'max:50'],
|
||||
'membership_started_at' => ['nullable', 'date'],
|
||||
'membership_expires_at' => ['nullable', 'date', 'after_or_equal:membership_started_at'],
|
||||
]);
|
||||
|
||||
$member->update($validated);
|
||||
AuditLogger::log('member.updated', $member, $validated);
|
||||
|
||||
return redirect()
|
||||
->route('admin.members.show', $member)
|
||||
->with('status', __('Member updated successfully.'));
|
||||
}
|
||||
|
||||
public function importForm()
|
||||
{
|
||||
return view('admin.members.import');
|
||||
}
|
||||
|
||||
public function import(Request $request)
|
||||
{
|
||||
$validated = $request->validate([
|
||||
'file' => ['required', 'file', 'mimes:csv,txt'],
|
||||
]);
|
||||
|
||||
$path = $validated['file']->store('imports');
|
||||
$fullPath = storage_path('app/'.$path);
|
||||
|
||||
Artisan::call('members:import', ['path' => $fullPath]);
|
||||
|
||||
$output = Artisan::output();
|
||||
|
||||
AuditLogger::log('members.imported', null, [
|
||||
'path' => $fullPath,
|
||||
'output' => $output,
|
||||
]);
|
||||
|
||||
return redirect()
|
||||
->route('admin.members.index')
|
||||
->with('status', __('Import completed.')."\n".$output);
|
||||
}
|
||||
|
||||
public function updateRoles(Request $request, Member $member)
|
||||
{
|
||||
$user = $member->user;
|
||||
|
||||
if (! $user) {
|
||||
abort(400, 'Member is not linked to a user.');
|
||||
}
|
||||
|
||||
$validated = $request->validate([
|
||||
'roles' => ['nullable', 'array'],
|
||||
'roles.*' => ['exists:roles,name'],
|
||||
]);
|
||||
|
||||
$roleNames = $validated['roles'] ?? [];
|
||||
$user->syncRoles($roleNames);
|
||||
|
||||
AuditLogger::log('member.roles_updated', $member, ['roles' => $roleNames]);
|
||||
|
||||
return redirect()->route('admin.members.show', $member)->with('status', __('Roles updated.'));
|
||||
}
|
||||
|
||||
/**
|
||||
* Show membership activation form
|
||||
*/
|
||||
public function showActivate(Member $member)
|
||||
{
|
||||
// Check if user has permission
|
||||
if (!auth()->user()->can('activate_memberships') && !auth()->user()->is_admin) {
|
||||
abort(403, 'You do not have permission to activate memberships.');
|
||||
}
|
||||
|
||||
// Check if member has fully approved payment
|
||||
$approvedPayment = $member->payments()
|
||||
->where('status', \App\Models\MembershipPayment::STATUS_APPROVED_CHAIR)
|
||||
->latest()
|
||||
->first();
|
||||
|
||||
if (!$approvedPayment && !auth()->user()->is_admin) {
|
||||
return redirect()->route('admin.members.show', $member)
|
||||
->with('error', __('Member must have an approved payment before activation.'));
|
||||
}
|
||||
|
||||
return view('admin.members.activate', compact('member', 'approvedPayment'));
|
||||
}
|
||||
|
||||
/**
|
||||
* Activate membership
|
||||
*/
|
||||
public function activate(Request $request, Member $member)
|
||||
{
|
||||
// Check if user has permission
|
||||
if (!auth()->user()->can('activate_memberships') && !auth()->user()->is_admin) {
|
||||
abort(403, 'You do not have permission to activate memberships.');
|
||||
}
|
||||
|
||||
$validated = $request->validate([
|
||||
'membership_started_at' => ['required', 'date'],
|
||||
'membership_expires_at' => ['required', 'date', 'after:membership_started_at'],
|
||||
'membership_type' => ['required', 'in:regular,honorary,lifetime,student'],
|
||||
]);
|
||||
|
||||
// Update member
|
||||
$member->update([
|
||||
'membership_started_at' => $validated['membership_started_at'],
|
||||
'membership_expires_at' => $validated['membership_expires_at'],
|
||||
'membership_type' => $validated['membership_type'],
|
||||
'membership_status' => Member::STATUS_ACTIVE,
|
||||
]);
|
||||
|
||||
AuditLogger::log('member.activated', $member, [
|
||||
'started_at' => $validated['membership_started_at'],
|
||||
'expires_at' => $validated['membership_expires_at'],
|
||||
'type' => $validated['membership_type'],
|
||||
'activated_by' => auth()->id(),
|
||||
]);
|
||||
|
||||
// Send activation confirmation email
|
||||
\Illuminate\Support\Facades\Mail::to($member->email)
|
||||
->queue(new \App\Mail\MembershipActivatedMail($member));
|
||||
|
||||
return redirect()->route('admin.members.show', $member)
|
||||
->with('status', __('Membership activated successfully! Member has been notified.'));
|
||||
}
|
||||
|
||||
public function export(Request $request): StreamedResponse
|
||||
{
|
||||
$query = Member::query()->with('user');
|
||||
|
||||
if ($search = $request->string('search')->toString()) {
|
||||
$query->where(function ($q) use ($search) {
|
||||
$q->where('full_name', 'like', "%{$search}%")
|
||||
->orWhere('email', 'like', "%{$search}%");
|
||||
});
|
||||
}
|
||||
|
||||
if ($status = $request->string('status')->toString()) {
|
||||
if ($status === 'active') {
|
||||
$query->whereDate('membership_expires_at', '>=', now()->toDateString());
|
||||
} elseif ($status === 'expired') {
|
||||
$query->where(function ($q) {
|
||||
$q->whereNull('membership_expires_at')
|
||||
->orWhereDate('membership_expires_at', '<', now()->toDateString());
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
$headers = [
|
||||
'ID',
|
||||
'Full Name',
|
||||
'Email',
|
||||
'Phone',
|
||||
'Address Line 1',
|
||||
'Address Line 2',
|
||||
'City',
|
||||
'Postal Code',
|
||||
'Emergency Contact Name',
|
||||
'Emergency Contact Phone',
|
||||
'Membership Start',
|
||||
'Membership Expiry',
|
||||
];
|
||||
|
||||
$response = new StreamedResponse(function () use ($query, $headers) {
|
||||
$handle = fopen('php://output', 'w');
|
||||
fputcsv($handle, $headers);
|
||||
|
||||
$query->chunk(500, function ($members) use ($handle) {
|
||||
foreach ($members as $member) {
|
||||
fputcsv($handle, [
|
||||
$member->id,
|
||||
$member->full_name,
|
||||
$member->email,
|
||||
$member->phone,
|
||||
$member->address_line_1,
|
||||
$member->address_line_2,
|
||||
$member->city,
|
||||
$member->postal_code,
|
||||
$member->emergency_contact_name,
|
||||
$member->emergency_contact_phone,
|
||||
optional($member->membership_started_at)->toDateString(),
|
||||
optional($member->membership_expires_at)->toDateString(),
|
||||
]);
|
||||
}
|
||||
});
|
||||
|
||||
fclose($handle);
|
||||
});
|
||||
|
||||
$filename = 'members-export-'.now()->format('Ymd_His').'.csv';
|
||||
|
||||
$response->headers->set('Content-Type', 'text/csv');
|
||||
$response->headers->set('Content-Disposition', "attachment; filename=\"{$filename}\"");
|
||||
|
||||
return $response;
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user