File "NotificationController.php"

Full Path: /home/trinadezambia/public_html/admin_panel/app/Http/Controllers/NotificationController.php
File size: 12.98 KB
MIME-type: text/x-php
Charset: utf-8

<?php

namespace App\Http\Controllers;

use App\Models\Notification;
use App\Models\Role;
use App\Models\UserNotification;
use App\Repositories\Fees\FeesInterface;
use App\Repositories\Notification\NotificationInterface;
use App\Repositories\User\UserInterface;
use App\Services\BootstrapTableService;
use App\Services\CachingService;
use App\Services\ResponseService;
use Illuminate\Support\Facades\Validator;
use Illuminate\Support\Facades\Auth;
use Carbon\Carbon;
use DB;
use Illuminate\Http\Request;
use Illuminate\Support\Str;
use Throwable;

class NotificationController extends Controller
{

    private NotificationInterface $notification;
    private CachingService $cache;
    private UserInterface $user;
    private FeesInterface $fees;

    public function __construct(NotificationInterface $notification, CachingService $cache, UserInterface $user, FeesInterface $fees)
    {
        $this->notification = $notification;
        $this->cache = $cache;
        $this->user = $user;
        $this->fees = $fees;
    }

    /**
     * Display a listing of the resource.
     */
    public function index()
    {
        //
        ResponseService::noFeatureThenRedirect('Announcement Management');
        ResponseService::noAnyPermissionThenRedirect(['notification-create', 'notification-list']);

        $sessionYearId = $this->cache->getSessionYear()->id;
        $roles = Role::whereNot('name', 'School Admin')->pluck('name', 'id');
        $over_due_fees_roles = Role::whereIn('name', ['Student', 'Guardian'])->pluck('name', 'id');
        $users = $this->user->guardian()->with('roles')->whereHas('child', function ($q) use ($sessionYearId) {
            $q->where('session_year_id', $sessionYearId);
        })->orWhere(function ($q) use ($roles) {
            $q->where('school_id', Auth::user()->school_id)
                ->whereHas('roles', function ($q) use ($roles) {
                    $q->whereIn('name', $roles);
                });
        })->get();

        $all_users = $users->pluck('id')->toArray();
        $all_users = implode(",", $all_users);

        return view('notification.index', compact('users', 'roles', 'all_users', 'over_due_fees_roles'));
    }

    /**
     * Show the form for creating a new resource.
     */
    public function create()
    {
        //
        ResponseService::noFeatureThenRedirect('Announcement Management');
    }

    /**
     * Store a newly created resource in storage.
     */
    public function store(Request $request)
    {
        //dd($request->all());
        ResponseService::noFeatureThenSendJson('Announcement Management');
        ResponseService::noPermissionThenSendJson('notification-create');
        // Validate request data
        $validator = Validator::make($request->all(), [
            'title' => 'required',
            'message' => 'required',
            'roles' => 'required|array',
            'user_id' => 'required|not_in:0,null',
        ], [
            'title.required' => 'Title is required.',
            'message.required' => 'Message is required.',
            'roles.required_if' => 'Please select roles',
            'user_id.required' => 'Please select users',
        ]);

        if ($validator->fails()) {
            ResponseService::errorResponse($validator->errors()->first());
        }

        try {
            DB::beginTransaction();
            $sessionYear = $this->cache->getSessionYear();
            $roles = Role::whereNot('name', 'School Admin')->whereIn('id', $request->roles)->pluck('name');
            $rolesArray = $roles->toarray();
            $roles = implode(', ', $rolesArray);
            $data = [
                'title' => $request->title,
                'message' => $request->message,
                'send_to' => $roles ?? '',
                'is_custom' => 1,
                'image' => $request->hasFile('image') ? $request->file('image')->store('notification', 'public') : null,
                'session_year_id' => $sessionYear->id
            ];
            $notification = $this->notification->create($data);
            $notifyUser = [];

            if ($request->has('user_id')) {
                $notifyUser = explode(',', $request->user_id);
                // Store user notifications for user-wise storage
                $userNotifications = [];
                foreach ($notifyUser as $userId) {
                    $userNotifications[] = [
                        'notification_id' => $notification->id,
                        'user_id' => $userId,
                        'created_at' => now(),
                        'updated_at' => now(),
                    ];
                }
                if (!empty($userNotifications)) {
                    UserNotification::insert($userNotifications);
                }
            }

            $customData = [];
            if ($notification->image) {
                $customData = [
                    'image' => $notification->image
                ];
            }

            $title = $request->title; // Title for Notification
            $body = $request->message;
            $type = 'custom';
            send_notification($notifyUser, $title, $body, $type, $customData); // Send Notification
            DB::commit();

            ResponseService::successResponse('Data Stored Successfully');
        } catch (Throwable $e) {
            if (
                Str::contains($e->getMessage(), [
                    'does not exist',
                    'file_get_contents'
                ])
            ) {
                DB::commit();
                ResponseService::warningResponse("Data Stored successfully. But App push notification not send.");
            } else {
                DB::rollBack();
                ResponseService::logErrorResponse($e, "Notification Controller -> Store Method");
                ResponseService::errorResponse();
            }
        }
    }

    /**
     * Display the specified resource.
     */
    public function show($id)
    {
        //
        ResponseService::noFeatureThenRedirect('Announcement Management');
        ResponseService::noPermissionThenRedirect('notification-list');
        $offset = request('offset', 0);
        $limit = request('limit', 10);
        $sort = request('sort', 'id');
        $order = request('order', 'DESC');
        $search = request('search');
        $sessionYearId = $this->cache->getSessionYear()->id;

        $sql = $this->notification->builder()
            ->where('school_id', Auth::user()->school_id)
            ->where('session_year_id', $sessionYearId)
            ->where(function ($query) use ($search) {
                $query->when($search, function ($q) use ($search) {
                    $q->where('title', 'LIKE', "%$search%")->orwhere('message', 'LIKE', "%$search%");
                });
            });

        // dd($sql->get()->toArray());
        $total = $sql->count();
        if ($offset >= $total && $total > 0) {
            $lastPage = floor(($total - 1) / $limit) * $limit; // calculate last page offset
            $offset = $lastPage;
        }
        $sql->orderBy($sort, $order)->skip($offset)->take($limit);
        $res = $sql->get();

        $bulkData = array();
        $bulkData['total'] = $total;
        $rows = array();
        $no = 1;
        foreach ($res as $row) {
            $operate = BootstrapTableService::deleteButton(route('notifications.destroy', $row->id));
            $tempRow = $row->toArray();
            $tempRow['send_to'] = trans($row->send_to);
            $tempRow['no'] = $no++;
            $tempRow['operate'] = $operate;
            $rows[] = $tempRow;
        }

        $bulkData['rows'] = $rows;
        return response()->json($bulkData);
    }

    /**
     * Show the form for editing the specified resource.
     */
    public function edit($id)
    {
        //
        ResponseService::noFeatureThenRedirect('Announcement Management');
    }

    /**
     * Update the specified resource in storage.
     */
    public function update(Request $request, $id)
    {
        //
        ResponseService::noFeatureThenRedirect('Announcement Management');
    }

    /**
     * Remove the specified resource from storage.
     */
    public function destroy($id)
    {
        //
        ResponseService::noFeatureThenSendJson('Announcement Management');
        ResponseService::noPermissionThenSendJson('notification-delete');
        try {
            $this->notification->deleteById($id);
            ResponseService::successResponse('Data Deleted Successfully');
        } catch (Throwable $e) {
            ResponseService::logErrorResponse($e, "Notification Controller -> Delete Method");
            ResponseService::errorResponse();
        }
    }

    public function userShow(Request $request)
    {

        ResponseService::noFeatureThenSendJson('Announcement Management');
        ResponseService::noPermissionThenSendJson('notification-list');
        $offset = request('offset', 0);
        $limit = request('limit', 10);
        $sort = request('sort', 'id');
        $order = request('order', 'DESC');
        $search = request('search');
        $type = request('type');

        $sessionYearId = $this->cache->getSessionYear()->id;
        $sql = $this->user->builder()->whereHas('roles', function ($q) {
            $q->whereNot('name', 'School Admin');
        })
            ->where(function ($q) use ($sessionYearId) {
                $q->whereHas('student', function ($q) use ($sessionYearId) {
                    $q->where('session_year_id', $sessionYearId);
                })->orWhereHas('child', function ($q) use ($sessionYearId) {
                    $q->where('session_year_id', $sessionYearId);
                })->orWhereHas('roles', function ($q) {
                    $q->whereNotIn('name', ['Student', 'Guardian']);
                });
            })
            ->where(function ($query) use ($search) {
                $query->when($search, function ($q) use ($search) {
                    $q->where('first_name', 'LIKE', "%$search%")
                        ->orwhere('last_name', 'LIKE', "%$search%")
                        ->orWhereRaw("concat(first_name,' ',last_name) LIKE '%" . $search . "%'");
                });
            })
            ->Owner();

        if ($type == 'Roles' && $request->roles) {
            $sql->whereHas('roles', function ($q) use ($request) {
                $q->whereIn('id', $request->roles);
            });
        }

        if ($type == 'OverDueFees') {
            $today = Carbon::now()->format('Y-m-d');
            $users_ids = [];

            $fees = $this->fees->builder()->where('session_year_id', $sessionYearId)->whereDate('due_date', '<', $today)->get();

            if ($fees->isNotEmpty()) {
                foreach ($fees as $fee) {
                    $overdueStudents = $this->user->builder()
                        ->role('Student')
                        ->select('id', 'first_name', 'last_name')
                        ->with([
                            'fees_paid' => function ($q) use ($fee) {
                                $q->where('fees_id', $fee->id);
                            },
                            'student:id,guardian_id,user_id',
                            'student.guardian:id'
                        ])
                        ->whereHas('student.class_section', function ($q) use ($fee, $sessionYearId) {
                            $q->where('class_id', $fee->class_id)->where('session_year_id', $sessionYearId);
                        })
                        ->whereDoesntHave('fees_paid', function ($q) use ($fee) {
                            $q->where('fees_id', $fee->id);
                        })
                        ->orWhereHas('fees_paid', function ($q) use ($fee) {
                            $q->where(['fees_id' => $fee->id, 'is_fully_paid' => 0]);
                        })
                        ->get();

                    $users_ids = array_unique(array_merge($overdueStudents->pluck('id')->toArray() ?? [], $overdueStudents->pluck('student.guardian.id')->toArray() ?? []));
                    $sql = $this->user->builder()->whereIn("id", $users_ids);

                    if (is_array($request->over_due_fees_roles)) {
                        $sql->whereHas('roles', function ($q) use ($request) {
                            $q->whereIn('id', $request->over_due_fees_roles);
                        });
                    }
                }
            } else {
                $sql = $this->user->builder()->whereIn("id", $users_ids);
            }
        }
        $total = $sql->count();
        if ($offset >= $total && $total > 0) {
            $lastPage = floor(($total - 1) / $limit) * $limit; // calculate last page offset
            $offset = $lastPage;
        }
        $sql->orderBy($sort, $order)->skip($offset)->take($limit);
        $res = $sql->get();

        $bulkData = array();
        $bulkData['total'] = $total;
        $rows = array();
        $no = 1;
        foreach ($res as $row) {

            $tempRow = $row->toArray();
            $tempRow['no'] = $no++;
            $rows[] = $tempRow;
        }

        $bulkData['rows'] = $rows;
        return response()->json($bulkData);

    }
}