Compare commits
	
		
			35 Commits
		
	
	
		
			36ec099cd9
			...
			main
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 8b833e7295 | |||
| 0b79bd665c | |||
| 81efe25ce2 | |||
| fbbcf86509 | |||
| 18897186e0 | |||
| 136db1a158 | |||
| f4202ce146 | |||
| a2c821d4bd | |||
| 1331e4a46e | |||
| 111c3e35a1 | |||
| 4b5cb88e4a | |||
| 0284d25af4 | |||
| a92d0c632f | |||
| 77e5b345b7 | |||
| a3f42315e4 | |||
| 65f7cc1ebf | |||
| e1153c375d | |||
| 0fe1cac8bf | |||
| 0621b14b77 | |||
| 08776e5a1b | |||
| b3a2467629 | |||
| dae0954891 | |||
| d013eb6dd1 | |||
| 8445a18416 | |||
| d1f90af6f6 | |||
| f6d11ce5e9 | |||
| 993b529331 | |||
| 5c687fc24f | |||
| bc92ed5234 | |||
| 5d585a6e26 | |||
| 64235d08f5 | |||
| 85d29d4142 | |||
| 620486de25 | |||
| 48fe5e3ef6 | |||
| e49309f709 | 
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -13,3 +13,4 @@ npm-debug.log | |||||||
| yarn-error.log | yarn-error.log | ||||||
| .vscode | .vscode | ||||||
| .DS_Store | .DS_Store | ||||||
|  | .sql | ||||||
| @ -9,6 +9,7 @@ class Helper | |||||||
|     const EARTH_RADIUS_M = 6371000; |     const EARTH_RADIUS_M = 6371000; | ||||||
|     const EARTH_RADIUS_KM = 6371; |     const EARTH_RADIUS_KM = 6371; | ||||||
|     const EARTH_RADIUS_MILES = 3959; // 3958.756 || 3959 || 3963 |     const EARTH_RADIUS_MILES = 3959; // 3958.756 || 3959 || 3963 | ||||||
|  | 	const TIMEFIX = 25200; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Calculates the great-circle distance between two points, with |      * Calculates the great-circle distance between two points, with | ||||||
|  | |||||||
| @ -13,6 +13,7 @@ use App\Models\Users; | |||||||
| use Illuminate\Support\Facades\Auth; | use Illuminate\Support\Facades\Auth; | ||||||
| use Illuminate\Http\JsonResponse; | use Illuminate\Http\JsonResponse; | ||||||
| use Illuminate\Validation\ValidationException; | use Illuminate\Validation\ValidationException; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  |  | ||||||
| class LoginController extends Controller | class LoginController extends Controller | ||||||
| { | { | ||||||
| @ -143,6 +144,12 @@ class LoginController extends Controller | |||||||
|         // return $req->wantsJson() |         // return $req->wantsJson() | ||||||
|         //     ? new JsonResponse([], 204) |         //     ? new JsonResponse([], 204) | ||||||
|         //     : redirect(route('view_dashboard')); |         //     : redirect(route('view_dashboard')); | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Auth", | ||||||
|  | 			"action" => "Login", | ||||||
|  | 			"desc" => "User login", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
|  |  | ||||||
|         if ($req->wantsJson()) { |         if ($req->wantsJson()) { | ||||||
|             return new JsonResponse([], 204); |             return new JsonResponse([], 204); | ||||||
| @ -173,7 +180,14 @@ class LoginController extends Controller | |||||||
|     public function logout(Request $req) |     public function logout(Request $req) | ||||||
|     { |     { | ||||||
|         $user = Auth::user(); |         $user = Auth::user(); | ||||||
|         if ($user->role == Users::ROLE_ADMIN) { | 		$log = [ | ||||||
|  | 			"module" => "Auth", | ||||||
|  | 			"action" => "Logout", | ||||||
|  | 			"desc" => "User logout", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
|  |  | ||||||
|  | 		if ($user->role == Users::ROLE_ADMIN) { | ||||||
|             $this->guard()->logout(); |             $this->guard()->logout(); | ||||||
|             $req->session()->invalidate(); |             $req->session()->invalidate(); | ||||||
|             $req->session()->regenerateToken(); |             $req->session()->regenerateToken(); | ||||||
|  | |||||||
| @ -12,6 +12,8 @@ use App\Responses; | |||||||
| use App\Helper; | use App\Helper; | ||||||
| use App\Models\Clients; | use App\Models\Clients; | ||||||
| use App\Models\Users; | use App\Models\Users; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  | use Auth; | ||||||
|  |  | ||||||
| class ClientController extends Controller | class ClientController extends Controller | ||||||
| { | { | ||||||
| @ -25,6 +27,13 @@ class ClientController extends Controller | |||||||
|         $data = [ |         $data = [ | ||||||
|             "disc_types" => Clients::select2DiscountTypes(), |             "disc_types" => Clients::select2DiscountTypes(), | ||||||
|         ]; |         ]; | ||||||
|  |  | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Company", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Company Menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
|         return view("menu_v1.clients", $data); |         return view("menu_v1.clients", $data); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -250,6 +259,13 @@ class ClientController extends Controller | |||||||
|             $apiResp = Responses::created("success add new client"); |             $apiResp = Responses::created("success add new client"); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Company", | ||||||
|  | 				"action" => "Create", | ||||||
|  | 				"desc" => "Add new company: ".$req->cname, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             Storage::disk("public")->delete($url_clogo); |             Storage::disk("public")->delete($url_clogo); | ||||||
| @ -481,6 +497,13 @@ class ClientController extends Controller | |||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::success("success update client"); |             $apiResp = Responses::success("success update client"); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Company", | ||||||
|  | 				"action" => "Update", | ||||||
|  | 				"desc" => "Update company: ".$req->cname, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             Storage::disk("public")->delete($url_clogo); |             Storage::disk("public")->delete($url_clogo); | ||||||
| @ -541,6 +564,13 @@ class ClientController extends Controller | |||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::success("success delete client"); |             $apiResp = Responses::success("success delete client"); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Company", | ||||||
|  | 				"action" => "Delete", | ||||||
|  | 				"desc" => "Delete company: ".$client[0]->c_name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
|  | |||||||
| @ -15,6 +15,8 @@ use App\Helper; | |||||||
| use App\Models\ConfRates; | use App\Models\ConfRates; | ||||||
| use App\Models\ConfTruckTypes; | use App\Models\ConfTruckTypes; | ||||||
| use App\Models\Vehicles; | use App\Models\Vehicles; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  | use Auth; | ||||||
|  |  | ||||||
| class ConfDistributionController extends Controller | class ConfDistributionController extends Controller | ||||||
| { | { | ||||||
| @ -26,6 +28,12 @@ class ConfDistributionController extends Controller | |||||||
|     { |     { | ||||||
|         $data = []; |         $data = []; | ||||||
|  |  | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Distribution Category", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Distribution Category menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
|         return view("menu_v1.configs.distribution_category", $data); |         return view("menu_v1.configs.distribution_category", $data); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -99,6 +107,13 @@ class ConfDistributionController extends Controller | |||||||
|             $apiResp = Responses::created("success " . ($tipe == "new" ? "add new" : "edit") . " distribution category"); |             $apiResp = Responses::created("success " . ($tipe == "new" ? "add new" : "edit") . " distribution category"); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Distribution Category", | ||||||
|  | 				"action" => "".($tipe == "new" ? "Create" : "Update")."", | ||||||
|  | 				"desc" => "".($tipe == "new" ? "Add new" : "Update")." distribution category: ".$req->dc_name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
| @ -200,6 +215,13 @@ class ConfDistributionController extends Controller | |||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::success("success delete vehicle type"); |             $apiResp = Responses::success("success delete vehicle type"); | ||||||
|  | 			 | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Distribution Category", | ||||||
|  | 				"action" => "Delete", | ||||||
|  | 				"desc" => "Delete distribution category: ".$dtl[0]->dc_name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, "Delete distribution category: ".$dtl[0]->dc_name); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
|  | |||||||
| @ -15,6 +15,8 @@ use App\Helper; | |||||||
| use App\Models\ConfRates; | use App\Models\ConfRates; | ||||||
| use App\Models\ConfTruckTypes; | use App\Models\ConfTruckTypes; | ||||||
| use App\Models\Vehicles; | use App\Models\Vehicles; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  | use Auth; | ||||||
|  |  | ||||||
| class ConfPoolController extends Controller | class ConfPoolController extends Controller | ||||||
| { | { | ||||||
| @ -26,6 +28,13 @@ class ConfPoolController extends Controller | |||||||
|     { |     { | ||||||
|         $data = []; |         $data = []; | ||||||
|  |  | ||||||
|  | 		 | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Pool", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Pool menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
|         return view("menu_v1.configs.pool", $data); |         return view("menu_v1.configs.pool", $data); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -99,6 +108,13 @@ class ConfPoolController extends Controller | |||||||
|             $apiResp = Responses::created("success " . ($tipe == "new" ? "add new" : "edit") . " distribution category"); |             $apiResp = Responses::created("success " . ($tipe == "new" ? "add new" : "edit") . " distribution category"); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Pool", | ||||||
|  | 				"action" => "".($tipe == "new" ? "Create" : "Update")."", | ||||||
|  | 				"desc" => "".($tipe == "new" ? "Add new" : "Update")." pool: ".$req->pool_name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
| @ -200,6 +216,13 @@ class ConfPoolController extends Controller | |||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::success("success delete vehicle type"); |             $apiResp = Responses::success("success delete vehicle type"); | ||||||
|  | 		 | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Pool", | ||||||
|  | 				"action" => "Delete", | ||||||
|  | 				"desc" => "Delete pool: ".$dtl[0]->pool_name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
|  | |||||||
| @ -15,6 +15,8 @@ use App\Helper; | |||||||
| use App\Models\ConfRates; | use App\Models\ConfRates; | ||||||
| use App\Models\ConfTruckTypes; | use App\Models\ConfTruckTypes; | ||||||
| use App\Models\Vehicles; | use App\Models\Vehicles; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  | use Auth; | ||||||
|  |  | ||||||
| class ConfTruckTypeController extends Controller | class ConfTruckTypeController extends Controller | ||||||
| { | { | ||||||
| @ -26,6 +28,12 @@ class ConfTruckTypeController extends Controller | |||||||
|     { |     { | ||||||
|         $data = []; |         $data = []; | ||||||
|  |  | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Vehicle Type", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Vehicle Type menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
|         return view("menu_v1.configs.truck_types", $data); |         return view("menu_v1.configs.truck_types", $data); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -163,6 +171,14 @@ class ConfTruckTypeController extends Controller | |||||||
|             $apiResp = Responses::created("success add new vehicle type"); |             $apiResp = Responses::created("success add new vehicle type"); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Vehicle Type", | ||||||
|  | 				"action" => "Create", | ||||||
|  | 				"desc" => "Add new vehicle type: ".$req->type_name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
| @ -256,6 +272,13 @@ class ConfTruckTypeController extends Controller | |||||||
|             $apiResp = Responses::created("success update vehicle type"); |             $apiResp = Responses::created("success update vehicle type"); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  | 			 | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Vehicle Type", | ||||||
|  | 				"action" => "Update", | ||||||
|  | 				"desc" => "Update vehicle type: ".$req->type_name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
| @ -310,6 +333,13 @@ class ConfTruckTypeController extends Controller | |||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::success("success delete vehicle type"); |             $apiResp = Responses::success("success delete vehicle type"); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Vehicle Type", | ||||||
|  | 				"action" => "Delete", | ||||||
|  | 				"desc" => "Delete vehicle type: ".$truckType[0]->name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
|  | |||||||
| @ -11,6 +11,8 @@ use App\Responses; | |||||||
| use App\Helper; | use App\Helper; | ||||||
| use App\Models\Devices; | use App\Models\Devices; | ||||||
| use App\Models\Vehicles; | use App\Models\Vehicles; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  | use Auth; | ||||||
|  |  | ||||||
| class DevicesController extends Controller | class DevicesController extends Controller | ||||||
| { | { | ||||||
| @ -20,6 +22,13 @@ class DevicesController extends Controller | |||||||
|         $data = [ |         $data = [ | ||||||
|             "vhcs" => $vhcs, |             "vhcs" => $vhcs, | ||||||
|         ]; |         ]; | ||||||
|  |  | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Device", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Device menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
|         return view("menu_v1.configs.devices", $data); |         return view("menu_v1.configs.devices", $data); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -202,6 +211,13 @@ class DevicesController extends Controller | |||||||
|             $apiResp = Responses::created("success add new device"); |             $apiResp = Responses::created("success add new device"); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Device", | ||||||
|  | 				"action" => "Create", | ||||||
|  | 				"desc" => "Add new device: ".$device_id, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
| @ -354,6 +370,13 @@ class DevicesController extends Controller | |||||||
|             $apiResp = Responses::created("success update device"); |             $apiResp = Responses::created("success update device"); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Device", | ||||||
|  | 				"action" => "Update", | ||||||
|  | 				"desc" => "Edit device: ".$device_id, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
| @ -403,6 +426,13 @@ class DevicesController extends Controller | |||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::success("success delete device"); |             $apiResp = Responses::success("success delete device"); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Device", | ||||||
|  | 				"action" => "Delete", | ||||||
|  | 				"desc" => "Delete device : ".$device[0]->device_id, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ use App\Models\Drivers; | |||||||
| use App\Models\DriversDetail; | use App\Models\DriversDetail; | ||||||
| use App\Models\Users; | use App\Models\Users; | ||||||
| use App\Models\DrvPhoneDevices; | use App\Models\DrvPhoneDevices; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  |  | ||||||
| class DriversController extends Controller | class DriversController extends Controller | ||||||
| { | { | ||||||
| @ -252,6 +253,13 @@ class DriversController extends Controller | |||||||
|             $apiResp = Responses::created("success add new driver"); |             $apiResp = Responses::created("success add new driver"); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Driver", | ||||||
|  | 				"action" => "View", | ||||||
|  | 				"desc" => "Add new driver: ".$req->fullname, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             Storage::disk("public")->delete($url_ktp); |             Storage::disk("public")->delete($url_ktp); | ||||||
| @ -451,6 +459,13 @@ class DriversController extends Controller | |||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::success("success update driver"); |             $apiResp = Responses::success("success update driver"); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Driver", | ||||||
|  | 				"action" => "Update", | ||||||
|  | 				"desc" => "Edit driver: ".$req->fullname, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             Storage::disk("public")->delete($url_ktp); |             Storage::disk("public")->delete($url_ktp); | ||||||
| @ -501,6 +516,13 @@ class DriversController extends Controller | |||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::success("success delete driver"); |             $apiResp = Responses::success("success delete driver"); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Driver", | ||||||
|  | 				"action" => "Delete", | ||||||
|  | 				"desc" => "Delete driver : ".$driver[0]->fullname, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
|  | |||||||
| @ -30,12 +30,21 @@ use App\Models\Finance; | |||||||
| use App\Models\OrdersCheckpoints; | use App\Models\OrdersCheckpoints; | ||||||
| use App\Models\OrdersInvoices; | use App\Models\OrdersInvoices; | ||||||
| use App\Models\OrdersDriversUploads; | use App\Models\OrdersDriversUploads; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  | use Auth;	 | ||||||
|  |  | ||||||
| class MenuController extends Controller | class MenuController extends Controller | ||||||
| { | { | ||||||
|     public function view_dashboard(Request $req) |     public function view_dashboard(Request $req) | ||||||
|     { |     { | ||||||
|         $data = [ | 		$log = [ | ||||||
|  | 			"module" => "Dashboard", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Dashboard menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert($req->auth->uid, $log); | ||||||
|  |      | ||||||
|  | 		$data = [ | ||||||
|             "client_group" => Clients::getClientById($req->auth->client_group_id), |             "client_group" => Clients::getClientById($req->auth->client_group_id), | ||||||
|         ]; |         ]; | ||||||
|  |  | ||||||
| @ -45,12 +54,20 @@ class MenuController extends Controller | |||||||
|             $data["client_group"] = null; |             $data["client_group"] = null; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |  | ||||||
|         return view("menu_v1.dashboard", $data); |         return view("menu_v1.dashboard", $data); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function view_drivers(Request $req) |     public function view_drivers(Request $req) | ||||||
|     { |     { | ||||||
|         $data = [ | 		$log = [ | ||||||
|  | 			"module" => "Driver", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Driver menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert($req->auth->uid, $log); | ||||||
|  |  | ||||||
|  | 		$data = [ | ||||||
|             "bloods" => Helper::listBloods(), |             "bloods" => Helper::listBloods(), | ||||||
|             "relationships" => Drivers::listRelationships(), |             "relationships" => Drivers::listRelationships(), | ||||||
|             "vendors" => Users::listUsersByRole(Users::ROLE_VENDOR), |             "vendors" => Users::listUsersByRole(Users::ROLE_VENDOR), | ||||||
| @ -61,6 +78,13 @@ class MenuController extends Controller | |||||||
|  |  | ||||||
|     public function view_vehicles(Request $req) |     public function view_vehicles(Request $req) | ||||||
|     { |     { | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Vehicle", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Vehicle menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert($req->auth->uid, $log); | ||||||
|  |  | ||||||
| 		$listPool = DB::select("SELECT * FROM t_conf_pool WHERE dlt IS NULL ORDER BY pool_code ASC"); | 		$listPool = DB::select("SELECT * FROM t_conf_pool WHERE dlt IS NULL ORDER BY pool_code ASC"); | ||||||
| 		$listDistribution = DB::select("SELECT * FROM t_conf_distribution_category WHERE dlt IS NULL ORDER BY dc_code ASC"); | 		$listDistribution = DB::select("SELECT * FROM t_conf_distribution_category WHERE dlt IS NULL ORDER BY dc_code ASC"); | ||||||
|  |  | ||||||
| @ -89,6 +113,13 @@ class MenuController extends Controller | |||||||
|  |  | ||||||
|     public function view_transactions() |     public function view_transactions() | ||||||
|     { |     { | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Transactions", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Transactions menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
|  |  | ||||||
|         $data = [ |         $data = [ | ||||||
|             "availOrdToMerge" => Finance::availOrdToMerge(), |             "availOrdToMerge" => Finance::availOrdToMerge(), | ||||||
|         ]; |         ]; | ||||||
| @ -174,6 +205,14 @@ class MenuController extends Controller | |||||||
|     } |     } | ||||||
|     public function view_transactions_view(Request $req) |     public function view_transactions_view(Request $req) | ||||||
|     { |     { | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Transactions", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Transactions menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert($req->auth->uid, $log); | ||||||
|  |  | ||||||
|  |  | ||||||
|         $codes = explode(",", $req->code); |         $codes = explode(",", $req->code); | ||||||
|         $limit = count($codes); |         $limit = count($codes); | ||||||
|         if ($limit > 2) { |         if ($limit > 2) { | ||||||
| @ -335,6 +374,13 @@ class MenuController extends Controller | |||||||
|  |  | ||||||
|     public function view_logs_gps() |     public function view_logs_gps() | ||||||
|     { |     { | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Logs GPS", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Logs GPS menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
|  |  | ||||||
|         return view("menu_v1.configs.index_logs_gps"); |         return view("menu_v1.configs.index_logs_gps"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | |||||||
| @ -17,23 +17,31 @@ use Maatwebsite\Excel\Concerns\WithHeadings; | |||||||
| use Maatwebsite\Excel\Concerns\WithStyles; | use Maatwebsite\Excel\Concerns\WithStyles; | ||||||
| use Maatwebsite\Excel\Concerns\WithCustomStartCell; | use Maatwebsite\Excel\Concerns\WithCustomStartCell; | ||||||
| use PhpOffice\PhpSpreadsheet\Worksheet\Worksheet; | use PhpOffice\PhpSpreadsheet\Worksheet\Worksheet; | ||||||
|  | use Carbon\Carbon; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  |  | ||||||
| class ReportsController extends Controller | class ReportsController extends Controller | ||||||
| { | { | ||||||
| 	public function view_report_vehicle_trips(Request $req) | 	public function view_report_vehicle_trips(Request $req) | ||||||
| 	{ | 	{ | ||||||
| 		$q = "select id, nopol1 from t_vehicles WHERE dlt is null order by nopol1"; | 		$q = "SELECT id, nopol1 from t_vehicles WHERE dlt is null order by nopol1"; | ||||||
| 		$listNopol = DB::select($q); | 		$listNopol = DB::select($q); | ||||||
|  |  | ||||||
| 		$data = [ | 		$data = [ | ||||||
| 			'listNopol' => $listNopol, | 			'listNopol' => $listNopol, | ||||||
| 		]; | 		]; | ||||||
|  |  | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Vehicle Trips Report", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Vehicle Trips Report menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
| 		return view('menu_v1.reports.vehicle_trips', $data); | 		return view('menu_v1.reports.vehicle_trips', $data); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	public function api_report_vehicle_trips_list(Request $req) | 	public function api_report_vehicle_trips_list(Request $req) | ||||||
| 	{ | 	{ | ||||||
|  | 		// $TIMEFIX = Helper::TIMEFIX; | ||||||
| 		// Validate input | 		// Validate input | ||||||
| 		// date in unix datetime format | 		// date in unix datetime format | ||||||
| 		// dd($req->type); | 		// dd($req->type); | ||||||
| @ -49,66 +57,106 @@ class ReportsController extends Controller | |||||||
| 			return new Response($apiResp, $apiResp["meta"]["code"]); | 			return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		$from_date = $req->input('from_date'); | 		// $from_date = $req->input('from_date'); | ||||||
| 		$to_date = $req->input('to_date'); | 		// $to_date = $req->input('to_date'); | ||||||
|  | 		$from_date = $req->input('from_date') - Helper::TIMEFIX; | ||||||
|  | 		$to_date = $req->input('to_date') - Helper::TIMEFIX; | ||||||
| 		$vid = $req->input('vid'); | 		$vid = $req->input('vid'); | ||||||
| 		// $from_date = 1756054800; | 		// $from_date = 1756054800; | ||||||
| 		// $to_date = 1756745940; | 		// $to_date = 1756745940; | ||||||
|  |  | ||||||
|  | 		// get month year | ||||||
|  | 		$date = Carbon::createFromTimestamp($from_date); | ||||||
|  | 		$yymm = $date->format('ym'); | ||||||
|  |  | ||||||
| 		try { | 		try { | ||||||
| 			$q = " | 			// $list = DB::select("WITH  | ||||||
| 				WITH trips AS ( | 			// 	gaps AS ( | ||||||
| 					SELECT | 			// 		SELECT | ||||||
| 						t.*, | 			// 			-- previous gap since previous row > 1 hour (3600s) | ||||||
| 						-- mark the start of a trip when ignition=4 and previous ignition <> 4 | 			// 			CASE | ||||||
| 						CASE | 			// 				WHEN (crt_d - LAG(crt_d, 1, NULL) OVER (PARTITION BY vhc_id ORDER BY crt_d)) > 3600 | ||||||
| 							WHEN ignition = 4  | 			// 				THEN 1 ELSE 0 | ||||||
| 								AND LAG(ignition, 1, 0) OVER (PARTITION BY vhc_id ORDER BY crt_d) <> 4 | 			// 			END AS isStop, | ||||||
| 							THEN 1 ELSE 0 | 			// 			t.* | ||||||
| 						END AS trip_start | 			// 		FROM tracks_2509 t | ||||||
| 					FROM t_gps_tracks t | 			// 		WHERE  | ||||||
| 					WHERE  | 			// 		t.latitude IS NOT NULL | ||||||
| 					t.latitude IS NOT NULL | 			// 		AND t.longitude IS NOT NULL | ||||||
| 					AND t.longitude IS NOT NULL | 			// 		AND t.action = 'location' | ||||||
| 					AND t.action = 'location' | 			// 		AND t.crt_d BETWEEN ? AND ? | ||||||
| 					AND t.crt_d BETWEEN ? AND ? | 			// 	) | ||||||
| 				) | 			// 	, trips AS ( | ||||||
| 				, numbered AS ( | 			// 		SELECT | ||||||
| 					SELECT | 			// 			-- mark the start of a trip when ignition=4 and previous ignition <> 4 | ||||||
| 						*, | 			// 			CASE | ||||||
| 						-- assign a trip_id by cumulative sum of trip_start | 			// 				WHEN ignition = 4  | ||||||
| 						SUM(trip_start) OVER (PARTITION BY vhc_id ORDER BY crt_d) AS trip_id | 			// 					AND LAG(ignition, 1, 0) OVER (PARTITION BY vhc_id ORDER BY crt_d) <> 4 | ||||||
|  | 			// 					or LAG(isStop, 1, 0) over (PARTITION BY vhc_id ORDER BY crt_d) = 1 | ||||||
|  | 			// 				THEN 1 ELSE 0 | ||||||
|  | 			// 			END AS trip_start, | ||||||
|  | 			// 			g.* | ||||||
|  | 			// 		FROM gaps g | ||||||
|  | 			// 	) | ||||||
|  | 			// 	, numbered AS ( | ||||||
|  | 			// 		SELECT | ||||||
|  | 			// 			*, | ||||||
|  | 			// 			-- assign a trip_id by cumulative sum of trip_start | ||||||
|  | 			// 			SUM(trip_start) OVER (PARTITION BY vhc_id ORDER BY crt_d) AS trip_id | ||||||
|  | 			// 		FROM trips | ||||||
|  | 			// 		where  | ||||||
|  | 			// 			ignition = 4 | ||||||
|  | 			// 			and isStop = 0 | ||||||
|  | 			// 	), | ||||||
|  | 			// 	agg AS ( | ||||||
|  | 			// 		SELECT | ||||||
|  | 			// 			COUNT(*) AS row_count, | ||||||
|  | 			// 			v.name, | ||||||
|  | 			// 			v.nopol1, | ||||||
|  | 			// 			vhc_id, | ||||||
|  | 			// 			-- trip_id, | ||||||
|  | 			// 			ROW_NUMBER() OVER (PARTITION BY v.id ORDER BY MIN(a.crt_d)) AS trip_id, | ||||||
|  | 			// 			SUM(pre_milleage) AS mileage, | ||||||
|  | 			// 			MIN(a.crt_d) AS start, | ||||||
|  | 			// 			MAX(a.crt_d) AS finish, | ||||||
|  | 			// 			MIN(a.vhc_milleage) AS startMileage, | ||||||
|  | 			// 			MAX(a.vhc_milleage) AS finishMileage, | ||||||
|  | 			// 			(SELECT fulladdress FROM t_gps_tracks_address WHERE master_id = MIN(a.id) LIMIT 1) AS startLoc, | ||||||
|  | 			// 			(SELECT fulladdress FROM t_gps_tracks_address WHERE master_id = MAX(a.id) LIMIT 1) AS finishLoc | ||||||
|  | 			// 		FROM t_vehicles v | ||||||
|  | 			// 		LEFT JOIN numbered a ON a.vhc_id = v.id | ||||||
|  | 			// 		WHERE  | ||||||
|  | 			// 			v.dlt is null and trip_id != 0  | ||||||
|  | 			// 			and if(? , v.id = ? , 1=1) | ||||||
|  | 			// 		GROUP BY v.id, a.trip_id | ||||||
|  | 			// 		HAVING COUNT(*) > 1 | ||||||
|  | 			// 	) | ||||||
|  | 			// 	SELECT | ||||||
|  | 			// 		*, | ||||||
|  | 			// 		SUM(mileage) OVER (PARTITION BY agg.id) AS total_mileage, | ||||||
|  | 			// 		COUNT(trip_id) OVER (PARTITION BY agg.id) AS total_trip, | ||||||
|  | 			// 		tvd.pool_code, tvd.dc_code | ||||||
|  | 			// 	FROM agg agg | ||||||
|  | 			// 		join t_vehicles_detail tvd on tvd.vid = agg.vhc_id | ||||||
|  | 			// 	ORDER BY agg.id, trip_id			 | ||||||
|  | 			// ", [$from_date, $to_date, $vid, $vid]); | ||||||
|  |  | ||||||
|  | 			$list = DB::select("WITH TotalMileage AS ( | ||||||
|  | 					SELECT id, SUM(mileage) AS total_mileage, count(*) total_trip | ||||||
| 					FROM trips | 					FROM trips | ||||||
| 				), | 					WHERE start BETWEEN ? AND ? | ||||||
| 				agg AS ( | 					GROUP BY id | ||||||
| 					SELECT |  | ||||||
| 						v.id, |  | ||||||
| 						v.name, |  | ||||||
| 						v.nopol1, |  | ||||||
| 						vhc_id, |  | ||||||
| 						trip_id, |  | ||||||
| 						SUM(pre_milleage) AS mileage, |  | ||||||
| 						MIN(a.crt_d) AS start, |  | ||||||
| 						MAX(a.crt_d) AS finish, |  | ||||||
| 						(SELECT fulladdress FROM t_gps_tracks_address WHERE master_id = MIN(a.id) LIMIT 1) AS startLoc, |  | ||||||
| 						(SELECT fulladdress FROM t_gps_tracks_address WHERE master_id = MAX(a.id) LIMIT 1) AS finishLoc |  | ||||||
| 					FROM t_vehicles v |  | ||||||
| 					LEFT JOIN numbered a ON a.vhc_id = v.id |  | ||||||
| 					WHERE  |  | ||||||
| 						v.dlt is null and trip_id != 0  |  | ||||||
| 						and if(? , v.id = ? , 1=1) |  | ||||||
| 					GROUP BY v.id, a.trip_id |  | ||||||
| 				) | 				) | ||||||
| 				SELECT  | 				SELECT  | ||||||
| 					*, | 					t.*, | ||||||
| 					SUM(mileage) OVER (PARTITION BY id) AS total_mileage, | 					tm.total_mileage, total_trip, | ||||||
| 					COUNT(trip_id) OVER (PARTITION BY id) AS total_trip | 					ROW_NUMBER() OVER (PARTITION BY t.id ORDER BY t.start) AS trip_id | ||||||
| 				FROM agg | 				FROM trips t | ||||||
| 				ORDER BY id, trip_id; | 					JOIN TotalMileage tm ON t.id = tm.id | ||||||
| 			"; | 				WHERE | ||||||
| 			$d = [$from_date, $to_date, $vid, $vid]; | 					t.start BETWEEN ? AND ? | ||||||
|  | 					and if(? , t.id = ? , 1=1) | ||||||
| 			$list = DB::select($q, $d); | 			", [$from_date, $to_date, $from_date, $to_date, $vid, $vid]); | ||||||
|  |  | ||||||
| 			// // RETURN 1 - LIST | 			// // RETURN 1 - LIST | ||||||
| 			// if($req->type != 'report'){ | 			// if($req->type != 'report'){ | ||||||
| @ -189,6 +237,64 @@ class ReportsController extends Controller | |||||||
| 			// return Responses::json(Responses::SERVER_ERROR, 'An error occurred while generating the report.', (object)[]); | 			// return Responses::json(Responses::SERVER_ERROR, 'An error occurred while generating the report.', (object)[]); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 	public function view_report_trip_detail(Request $req) | ||||||
|  | 	{ | ||||||
|  | 		$vid = $req->vid; | ||||||
|  | 		$tgl0 = $req->tgl0; | ||||||
|  | 		$tgl1 = $req->tgl1; | ||||||
|  | 		$nopol1 = $req->nopol1; | ||||||
|  |  | ||||||
|  | 		$d = [$vid, $tgl0, $tgl1]; | ||||||
|  | 		$list = DB::select("SELECT | ||||||
|  | 				t.crt_d , t.latitude, t.longitude, t.speed, | ||||||
|  | 				tgta.fulladdress, | ||||||
|  | 				t.pre_milleage, t.vhc_milleage  | ||||||
|  | 			FROM  | ||||||
|  | 				t_gps_tracks t | ||||||
|  | 				left join t_gps_tracks_address tgta on tgta.master_id = t.id  | ||||||
|  | 			WHERE  | ||||||
|  | 				t.vhc_id = ? | ||||||
|  | 				and t.latitude IS NOT NULL | ||||||
|  | 				AND t.longitude IS NOT NULL | ||||||
|  | 				AND t.action = 'location' | ||||||
|  | 				AND t.crt_d BETWEEN ? AND ? | ||||||
|  | 			ORDER BY t.crt_d asc | ||||||
|  | 		", $d); | ||||||
|  |  | ||||||
|  | 		$start = [ | ||||||
|  | 			'time' => $list[0]->crt_d, | ||||||
|  | 			'fulladdress' => urldecode($list[0]->fulladdress), | ||||||
|  | 			'mileage' => $list[0]->vhc_milleage, | ||||||
|  | 		]; | ||||||
|  | 		$finish = [ | ||||||
|  | 			'time' => $list[count($list) - 1]->crt_d, | ||||||
|  | 			'fulladdress' => urldecode($list[count($list) - 1]->fulladdress), | ||||||
|  | 			'mileage' => $list[count($list) - 1]->vhc_milleage, | ||||||
|  | 		]; | ||||||
|  |  | ||||||
|  | 		$t0 = Carbon::createFromTimestamp($list[0]->crt_d); | ||||||
|  | 		$t1 = Carbon::createFromTimestamp($list[count($list) - 1]->crt_d); | ||||||
|  | 		$diff = $t1->diff($t0); | ||||||
|  | 		$hours = $diff->h + ($diff->days * 24); // include days converted to hours | ||||||
|  | 		$minutes = $diff->i; | ||||||
|  | 		$duration = "{$hours} hour" . ($hours > 1 ? 's' : '') . " {$minutes} minute" . ($minutes > 1 ? 's' : ''); | ||||||
|  |  | ||||||
|  | 		$distance = $list[count($list) - 1]->vhc_milleage - $list[0]->vhc_milleage; | ||||||
|  |  | ||||||
|  | 		$data = [ | ||||||
|  | 			'nopol1' => $nopol1, | ||||||
|  | 			'vid' => $vid, | ||||||
|  | 			'tgl0' => $tgl0, | ||||||
|  | 			'tgl1' => $tgl1, | ||||||
|  | 			'list' => $list, | ||||||
|  | 			'start' => $start, | ||||||
|  | 			'finish' => $finish, | ||||||
|  | 			'duration' => $duration, | ||||||
|  | 			'distance' => $distance, | ||||||
|  | 		]; | ||||||
|  | 		// dd($list); | ||||||
|  | 		return view('menu_v1.reports._trip_detail', $data); | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	public function view_report_abnormalities(Request $req) | 	public function view_report_abnormalities(Request $req) | ||||||
| 	{ | 	{ | ||||||
| @ -199,6 +305,12 @@ class ReportsController extends Controller | |||||||
| 			'listNopol' => $listNopol, | 			'listNopol' => $listNopol, | ||||||
| 		]; | 		]; | ||||||
|  |  | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Abnormalities Report", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Abnormalities Report menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
| 		return view('menu_v1.reports.abnormalities', $data); | 		return view('menu_v1.reports.abnormalities', $data); | ||||||
| 	} | 	} | ||||||
| 	public function api_report_abnormalities_list(Request $req) | 	public function api_report_abnormalities_list(Request $req) | ||||||
| @ -216,16 +328,15 @@ class ReportsController extends Controller | |||||||
| 			return new Response($apiResp, $apiResp["meta"]["code"]); | 			return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		$from_date = $req->input('from_date'); | 		$from_date = $req->input('from_date') - Helper::TIMEFIX; | ||||||
| 		$to_date = $req->input('to_date'); | 		$to_date = $req->input('to_date') - Helper::TIMEFIX; | ||||||
| 		$vid = $req->input('vid'); | 		$vid = $req->input('vid'); | ||||||
|  |  | ||||||
| 		try { | 		try { | ||||||
| 			$q = " | 			$list = DB::select("SELECT | ||||||
| 				select |  | ||||||
| 					tv.name, tv.nopol1,  | 					tv.name, tv.nopol1,  | ||||||
| 					t.crt_d, t.speed, tgta.fulladdress, | 					t.crt_d, t.speed, tgta.fulladdress, | ||||||
| 					tvd.speed_limit | 					tvd.speed_limit, tvd.pool_code, tvd.dc_code | ||||||
| 				from | 				from | ||||||
| 					t_gps_tracks t | 					t_gps_tracks t | ||||||
| 					left join t_vehicles tv on tv.id = t.vhc_id  | 					left join t_vehicles tv on tv.id = t.vhc_id  | ||||||
| @ -239,10 +350,7 @@ class ReportsController extends Controller | |||||||
| 					-- and t.speed > tvd.speed_limit  | 					-- and t.speed > tvd.speed_limit  | ||||||
| 				having t.speed >= tvd.speed_limit | 				having t.speed >= tvd.speed_limit | ||||||
| 				ORDER BY t.crt_d | 				ORDER BY t.crt_d | ||||||
| 			"; | 			", [$from_date, $to_date, $vid, $vid]); | ||||||
| 			$d = [$from_date, $to_date, $vid, $vid]; |  | ||||||
|  |  | ||||||
| 			$list = DB::select($q, $d); |  | ||||||
|  |  | ||||||
| 			// // RETURN 1 - LIST | 			// // RETURN 1 - LIST | ||||||
| 			// if($req->type != 'report'){ | 			// if($req->type != 'report'){ | ||||||
|  | |||||||
| @ -15,6 +15,8 @@ use App\Models\Vehicles; | |||||||
| use App\Models\Banks; | use App\Models\Banks; | ||||||
| use App\Models\UsersMenuPermissions; | use App\Models\UsersMenuPermissions; | ||||||
| use Spatie\Permission\PermissionRegistrar; | use Spatie\Permission\PermissionRegistrar; | ||||||
|  | use Illuminate\Support\Facades\Auth; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  |  | ||||||
| class RolesController extends Controller | class RolesController extends Controller | ||||||
| { | { | ||||||
| @ -34,6 +36,13 @@ class RolesController extends Controller | |||||||
|             }), |             }), | ||||||
|         ]; |         ]; | ||||||
|  |  | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Role", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Role menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
|  |  | ||||||
|         return view("menu_v1.roles", $data); |         return view("menu_v1.roles", $data); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -118,6 +127,14 @@ class RolesController extends Controller | |||||||
|             app()[PermissionRegistrar::class]->forgetCachedPermissions(); |             app()[PermissionRegistrar::class]->forgetCachedPermissions(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::created("success add new role"); |             $apiResp = Responses::created("success add new role"); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Role", | ||||||
|  | 				"action" => "Create", | ||||||
|  | 				"desc" => "Add new role : ".$req->name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|  |  | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
| @ -156,6 +173,13 @@ class RolesController extends Controller | |||||||
|             app()[PermissionRegistrar::class]->forgetCachedPermissions(); |             app()[PermissionRegistrar::class]->forgetCachedPermissions(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::created("success update role"); |             $apiResp = Responses::created("success update role"); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Role", | ||||||
|  | 				"action" => "Update", | ||||||
|  | 				"desc" => "Update role : ".$req->name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
| @ -198,6 +222,13 @@ class RolesController extends Controller | |||||||
|             app()[PermissionRegistrar::class]->forgetCachedPermissions(); |             app()[PermissionRegistrar::class]->forgetCachedPermissions(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::created("success delete role"); |             $apiResp = Responses::created("success delete role"); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Role", | ||||||
|  | 				"action" => "Delete", | ||||||
|  | 				"desc" => "Delete role : ".$role[0]->name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
|  | |||||||
							
								
								
									
										71
									
								
								app/Http/Controllers/UserLogsController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								app/Http/Controllers/UserLogsController.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,71 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | namespace App\Http\Controllers; | ||||||
|  |  | ||||||
|  | use App\Http\Controllers\Controller; | ||||||
|  | use Illuminate\Http\Request; | ||||||
|  | use Illuminate\Http\Response; | ||||||
|  | use Illuminate\Support\Facades\DB; | ||||||
|  | use Illuminate\Support\Facades\Storage; | ||||||
|  | use Validator; | ||||||
|  | use Hidehalo\Nanoid\Client as Nanoid; | ||||||
|  | use Hidehalo\Nanoid\GeneratorInterface as NanoidInterface; | ||||||
|  | use App\Responses; | ||||||
|  | use App\Helper; | ||||||
|  | use App\Models\ConfRates; | ||||||
|  | use App\Models\ConfTruckTypes; | ||||||
|  | use App\Models\Vehicles; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  | use Auth; | ||||||
|  | use App\Models\Users; | ||||||
|  |  | ||||||
|  | class UserLogsController extends Controller | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * View | ||||||
|  |      */ | ||||||
|  |  | ||||||
|  |     public function view_user_logs(Request $req) | ||||||
|  |     { | ||||||
|  | 		$users = Users::listUsers(); | ||||||
|  | 		$data = [ | ||||||
|  | 			"users" => $users, | ||||||
|  | 		]; | ||||||
|  |  | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "User Logs", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open User Logs menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
|  |         return view("menu_v1.userLogs", $data); | ||||||
|  |     } | ||||||
|  |     public function api_user_logs(Request $req) | ||||||
|  |     { | ||||||
|  |         try { | ||||||
|  | 			$tgl0 = $req->tgl0; | ||||||
|  | 			$tgl1 = $req->tgl1; | ||||||
|  | 			$userId = $req->userId; | ||||||
|  |  | ||||||
|  | 			$d = [$tgl0, $tgl1, $userId, $userId]; | ||||||
|  | 			$list = DB::select("SELECT | ||||||
|  | 					a.*, b.email | ||||||
|  | 				FROM t_user_log a | ||||||
|  | 					join t_users b on a.userId = b.id | ||||||
|  | 				WHERE  | ||||||
|  | 					a.crt BETWEEN ? AND ? | ||||||
|  | 					and if(? , a.userId = ? , 1=1) | ||||||
|  | 				order by a.crt desc | ||||||
|  | 			", $d); | ||||||
|  |  | ||||||
|  | 			$apiResp = Responses::success("success user logs"); | ||||||
|  | 			$apiResp["data"] = $list; | ||||||
|  | 			return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|  | 		} catch (\Exception $e) { | ||||||
|  |             DB::rollBack(); | ||||||
|  |             $apiResp = Responses::error($e->getMessage()); | ||||||
|  |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @ -15,6 +15,8 @@ use App\Models\Clients; | |||||||
| use App\Models\Vehicles; | use App\Models\Vehicles; | ||||||
| use App\Models\Banks; | use App\Models\Banks; | ||||||
| use App\Models\UsersMenuPermissions; | use App\Models\UsersMenuPermissions; | ||||||
|  | use Illuminate\Support\Facades\Auth; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  |  | ||||||
| class UsersController extends Controller | class UsersController extends Controller | ||||||
| { | { | ||||||
| @ -50,6 +52,13 @@ class UsersController extends Controller | |||||||
|             $data["clients"] = Clients::select2Client($req->auth->client_group_id); |             $data["clients"] = Clients::select2Client($req->auth->client_group_id); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "User", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open User menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert(Auth::user()->id, $log); | ||||||
|  |  | ||||||
|         return view("menu_v1.users", $data); |         return view("menu_v1.users", $data); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -328,6 +337,14 @@ class UsersController extends Controller | |||||||
|             $apiResp = Responses::created("success add new user"); |             $apiResp = Responses::created("success add new user"); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "User", | ||||||
|  | 				"action" => "Create", | ||||||
|  | 				"desc" => "Add new user : ".$req->email, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|  |  | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
| @ -529,7 +546,15 @@ class UsersController extends Controller | |||||||
|             $apiResp = Responses::created("success update user"); |             $apiResp = Responses::created("success update user"); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "User", | ||||||
|  | 				"action" => "Update", | ||||||
|  | 				"desc" => "Update user : ".$req->email, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|  |  | ||||||
|  | 			return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
|             $apiResp = Responses::error($e->getMessage()); |             $apiResp = Responses::error($e->getMessage()); | ||||||
| @ -710,6 +735,14 @@ class UsersController extends Controller | |||||||
|             $apiResp = Responses::created("success delete user"); |             $apiResp = Responses::created("success delete user"); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "User", | ||||||
|  | 				"action" => "Delete", | ||||||
|  | 				"desc" => "Delete user : ".$getUser[0]->email, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|  |  | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
| @ -776,4 +809,38 @@ class UsersController extends Controller | |||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | 	public function view_user_logs1(Request $req) | ||||||
|  | 	{ | ||||||
|  | 		$id = $req->id; | ||||||
|  |  | ||||||
|  | 		$user = Users::getUserById($id)[0]; | ||||||
|  | 		$data = [ | ||||||
|  | 			'user' => $user | ||||||
|  | 		]; | ||||||
|  | 		// dd($dtl); | ||||||
|  | 		return view('menu_v1._userLogs', $data); | ||||||
|  | 	} | ||||||
|  |     public function api_user_logs1(Request $req) | ||||||
|  |     { | ||||||
|  |         try { | ||||||
|  | 			$id = $req->id; | ||||||
|  | 			$tgl0 = $req->tgl0; | ||||||
|  | 			$tgl1 = $req->tgl1; | ||||||
|  |  | ||||||
|  | 			$d = [$id, $tgl0, $tgl1]; | ||||||
|  | 			$list = DB::select("SELECT * FROM t_user_log  | ||||||
|  | 				WHERE userId = ? AND crt BETWEEN ? AND ? | ||||||
|  | 				order by crt desc | ||||||
|  | 			", $d); | ||||||
|  |  | ||||||
|  | 			$apiResp = Responses::success("success user logs"); | ||||||
|  | 			$apiResp["data"] = $list; | ||||||
|  | 			return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|  | 		} catch (\Exception $e) { | ||||||
|  |             DB::rollBack(); | ||||||
|  |             $apiResp = Responses::error($e->getMessage()); | ||||||
|  |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -15,6 +15,7 @@ use App\Models\Vehicles; | |||||||
| use App\Models\Devices; | use App\Models\Devices; | ||||||
| use App\Models\VehiclesDetail; | use App\Models\VehiclesDetail; | ||||||
| use App\Models\Users; | use App\Models\Users; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  |  | ||||||
| class VehiclesController extends Controller | class VehiclesController extends Controller | ||||||
| { | { | ||||||
| @ -330,7 +331,15 @@ class VehiclesController extends Controller | |||||||
|             $apiResp = Responses::created("success add new vehicle"); |             $apiResp = Responses::created("success add new vehicle"); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); | 	 | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Vehicle", | ||||||
|  | 				"action" => "Create", | ||||||
|  | 				"desc" => "Add new vehicle: ".$req->vhc_name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|  |      | ||||||
|  | 			return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             Storage::disk("public")->delete($url_fvhc); |             Storage::disk("public")->delete($url_fvhc); | ||||||
|             Storage::disk("public")->delete($url_stnk); |             Storage::disk("public")->delete($url_stnk); | ||||||
| @ -589,6 +598,13 @@ class VehiclesController extends Controller | |||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::success("success update vehicle"); |             $apiResp = Responses::success("success update vehicle"); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Vehicle", | ||||||
|  | 				"action" => "Update", | ||||||
|  | 				"desc" => "Update vehicle: ".$req->vhc_name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             Storage::disk("public")->delete($url_fvhc); |             Storage::disk("public")->delete($url_fvhc); | ||||||
| @ -646,6 +662,13 @@ class VehiclesController extends Controller | |||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::success("success delete vehicle"); |             $apiResp = Responses::success("success delete vehicle"); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Vehicle", | ||||||
|  | 				"action" => "Delete", | ||||||
|  | 				"desc" => "Delete vehicle: ".$vehicle[0]->name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert(Auth::user()->id, $log); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             DB::rollBack(); |             DB::rollBack(); | ||||||
|  | |||||||
| @ -17,6 +17,8 @@ use App\Models\Orders; | |||||||
| use App\Models\OrdersDrops; | use App\Models\OrdersDrops; | ||||||
| // use App\Models\OrdersPckDrop; | // use App\Models\OrdersPckDrop; | ||||||
| use App\Models\OrdersAItems; | use App\Models\OrdersAItems; | ||||||
|  | use App\Models\UserLogs; | ||||||
|  |  | ||||||
|  |  | ||||||
| class ZoneController extends Controller | class ZoneController extends Controller | ||||||
| { | { | ||||||
| @ -25,8 +27,14 @@ class ZoneController extends Controller | |||||||
|      * |      * | ||||||
|      * @return \Illuminate\Http\Response |      * @return \Illuminate\Http\Response | ||||||
|      */ |      */ | ||||||
|     public function view_zone() |     public function view_zone(Request $req) | ||||||
|     { |     { | ||||||
|  | 		$log = [ | ||||||
|  | 			"module" => "Zone", | ||||||
|  | 			"action" => "View", | ||||||
|  | 			"desc" => "Open Zone menu", | ||||||
|  | 		]; | ||||||
|  | 		UserLogs::insert($req->auth->uid, $log); | ||||||
|         return view("menu_v1.zone"); |         return view("menu_v1.zone"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -355,6 +363,13 @@ class ZoneController extends Controller | |||||||
|  |  | ||||||
|             $apiResp = Responses::created("success add new zone"); |             $apiResp = Responses::created("success add new zone"); | ||||||
|  |  | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Zone", | ||||||
|  | 				"action" => "Create", | ||||||
|  | 				"desc" => "Add Zone ".$req->zone_name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert($req->auth->uid, $log); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
| @ -481,7 +496,14 @@ class ZoneController extends Controller | |||||||
|  |  | ||||||
|             Zone::updateZone($zid, $updtZone); |             Zone::updateZone($zid, $updtZone); | ||||||
|  |  | ||||||
|             DB::commit(); | 			$log = [ | ||||||
|  | 				"module" => "Zone", | ||||||
|  | 				"action" => "Update", | ||||||
|  | 				"desc" => "Update Zone ".$req->zone_name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert($req->auth->uid, $log); | ||||||
|  |  | ||||||
|  | 			DB::commit(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::success("success update zone"); |             $apiResp = Responses::success("success update zone"); | ||||||
|             return new Response($apiResp, $apiResp["meta"]["code"]); |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
| @ -524,6 +546,13 @@ class ZoneController extends Controller | |||||||
|                 "dlt_by" => $req->auth->uid, |                 "dlt_by" => $req->auth->uid, | ||||||
|             ]); |             ]); | ||||||
| 			 | 			 | ||||||
|  | 			$log = [ | ||||||
|  | 				"module" => "Zone", | ||||||
|  | 				"action" => "Delete	", | ||||||
|  | 				"desc" => "Delete Zone ".$zone[0]->name, | ||||||
|  | 			]; | ||||||
|  | 			UserLogs::insert($req->auth->uid, $log); | ||||||
|  |  | ||||||
|             DB::commit(); |             DB::commit(); | ||||||
|  |  | ||||||
|             $apiResp = Responses::success("success delete zone"); |             $apiResp = Responses::success("success delete zone"); | ||||||
|  | |||||||
| @ -386,10 +386,10 @@ class Tracks extends Model | |||||||
|             $query .= " AND v.nopol1 = ? AND v.nopol2 = ? AND v.nopol3 = ?"; |             $query .= " AND v.nopol1 = ? AND v.nopol2 = ? AND v.nopol3 = ?"; | ||||||
|             array_push($params, $filter["nopol1"], $filter["nopol2"], $filter["nopol3"]); |             array_push($params, $filter["nopol1"], $filter["nopol2"], $filter["nopol3"]); | ||||||
|         } |         } | ||||||
|         if (isset($filter["vid"])) { |         // if (isset($filter["vid"])) { | ||||||
|             $query .= " AND v.id = ?"; |         //     $query .= " AND v.id = ?"; | ||||||
|             $params[] = $filter["vid"]; |         //     $params[] = $filter["vid"]; | ||||||
|         } |         // } | ||||||
|         if (isset($filter["vids"])) { |         if (isset($filter["vids"])) { | ||||||
|             if ($filter["vids"] && count($filter["vids"]) > 0) { |             if ($filter["vids"] && count($filter["vids"]) > 0) { | ||||||
|                 $placeholders = rtrim(str_repeat("?,", count($filter["vids"])), ","); |                 $placeholders = rtrim(str_repeat("?,", count($filter["vids"])), ","); | ||||||
| @ -433,7 +433,7 @@ class Tracks extends Model | |||||||
|             $params[] = $filter["company"]; |             $params[] = $filter["company"]; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         $query .= " GROUP BY v.id ORDER BY tr.crt_d DESC LIMIT 500"; |         $query .= " GROUP BY v.id ORDER BY tr.crt_s DESC LIMIT 500"; | ||||||
|  |  | ||||||
|         $list = DB::select($query, $params); |         $list = DB::select($query, $params); | ||||||
| 		// dd($list); | 		// dd($list); | ||||||
| @ -543,7 +543,9 @@ class Tracks extends Model | |||||||
|  |  | ||||||
|         if (isset($filter["start_date"]) && isset($filter["end_date"])) { |         if (isset($filter["start_date"]) && isset($filter["end_date"])) { | ||||||
|             $query .= " AND tr.crt_d BETWEEN ? AND ?"; |             $query .= " AND tr.crt_d BETWEEN ? AND ?"; | ||||||
|             array_push($params, $filter["start_date"], $filter["end_date"]); | 			$tgl0 = $filter["start_date"] - Helper::TIMEFIX; | ||||||
|  | 			$tgl1 = $filter["end_date"] - Helper::TIMEFIX; | ||||||
|  |             array_push($params, $tgl0, $tgl1); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (isset($filter["start_at"])) { |         if (isset($filter["start_at"])) { | ||||||
|  | |||||||
							
								
								
									
										20
									
								
								app/Models/UserLogs.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								app/Models/UserLogs.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,20 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | namespace App\Models; | ||||||
|  |  | ||||||
|  | use Illuminate\Database\Eloquent\Model; | ||||||
|  | use Illuminate\Support\Facades\DB; | ||||||
|  |  | ||||||
|  | class UserLogs extends Model | ||||||
|  | { | ||||||
|  |     public static function insert($userId, $log) | ||||||
|  |     { | ||||||
|  | 		$now = time(); | ||||||
|  |         return DB::insert("INSERT into t_user_log | ||||||
|  | 			set | ||||||
|  | 				userId = ?, | ||||||
|  | 				log = ?, | ||||||
|  | 				crt = ? | ||||||
|  | 		", [$userId, json_encode($log), $now]); | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										350
									
								
								public/assets/vendor/printThis.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										350
									
								
								public/assets/vendor/printThis.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,350 @@ | |||||||
|  | /* | ||||||
|  |  * printThis v2.0.0 | ||||||
|  |  * @desc Printing plug-in for jQuery | ||||||
|  |  * @author Jason Day | ||||||
|  |  * @author Samuel Rouse | ||||||
|  |  * | ||||||
|  |  * Resources (based on): | ||||||
|  |  * - jPrintArea: http://plugins.jquery.com/project/jPrintArea | ||||||
|  |  * - jqPrint: https://github.com/permanenttourist/jquery.jqprint | ||||||
|  |  * - Ben Nadal: http://www.bennadel.com/blog/1591-Ask-Ben-Print-Part-Of-A-Web-Page-With-jQuery.htm | ||||||
|  |  * | ||||||
|  |  * Licensed under the MIT licence: | ||||||
|  |  *              http://www.opensource.org/licenses/mit-license.php | ||||||
|  |  * | ||||||
|  |  * (c) Jason Day 2015-2022 | ||||||
|  |  * | ||||||
|  |  * Usage: | ||||||
|  |  * | ||||||
|  |  *  $("#mySelector").printThis({ | ||||||
|  |  *      debug: false,                   // show the iframe for debugging | ||||||
|  |  *      importCSS: true,                // import parent page css | ||||||
|  |  *      importStyle: true,              // import style tags | ||||||
|  |  *      printContainer: true,           // grab outer container as well as the contents of the selector | ||||||
|  |  *      loadCSS: "path/to/my.css",      // path to additional css file - use an array [] for multiple | ||||||
|  |  *      pageTitle: "",                  // add title to print page | ||||||
|  |  *      removeInline: false,            // remove all inline styles from print elements | ||||||
|  |  *      removeInlineSelector: "body *", // custom selectors to filter inline styles. removeInline must be true | ||||||
|  |  *      printDelay: 1000,               // variable print delay | ||||||
|  |  *      header: null,                   // prefix to html | ||||||
|  |  *      footer: null,                   // postfix to html | ||||||
|  |  *      base: false,                    // preserve the BASE tag, or accept a string for the URL | ||||||
|  |  *      formValues: true,               // preserve input/form values | ||||||
|  |  *      canvas: true,                   // copy canvas elements | ||||||
|  |  *      doctypeString: '...',           // enter a different doctype for older markup | ||||||
|  |  *      removeScripts: false,           // remove script tags from print content | ||||||
|  |  *      copyTagClasses: true            // copy classes from the html & body tag | ||||||
|  |  *      copyTagStyles: true,            // copy styles from html & body tag (for CSS Variables) | ||||||
|  |  *      beforePrintEvent: null,         // callback function for printEvent in iframe | ||||||
|  |  *      beforePrint: null,              // function called before iframe is filled | ||||||
|  |  *      afterPrint: null                // function called before iframe is removed | ||||||
|  |  *  }); | ||||||
|  |  * | ||||||
|  |  * Notes: | ||||||
|  |  *  - the loadCSS will load additional CSS (with or without @media print) into the iframe, adjusting layout | ||||||
|  |  */ | ||||||
|  | ; | ||||||
|  | (function($) { | ||||||
|  |  | ||||||
|  |     function appendContent($el, content) { | ||||||
|  |         if (!content) return; | ||||||
|  |  | ||||||
|  |         // Simple test for a jQuery element | ||||||
|  |         $el.append(content.jquery ? content.clone() : content); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function appendBody($body, $element, opt) { | ||||||
|  |         // Clone for safety and convenience | ||||||
|  |         // Calls clone(withDataAndEvents = true) to copy form values. | ||||||
|  |         var $content = $element.clone(opt.formValues); | ||||||
|  |  | ||||||
|  |         if (opt.formValues) { | ||||||
|  |             // Copy original select and textarea values to their cloned counterpart | ||||||
|  |             // Makes up for inability to clone select and textarea values with clone(true) | ||||||
|  |             copyValues($element, $content, 'select, textarea'); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (opt.removeScripts) { | ||||||
|  |             $content.find('script').remove(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (opt.printContainer) { | ||||||
|  |             // grab $.selector as container | ||||||
|  |             $content.appendTo($body); | ||||||
|  |         } else { | ||||||
|  |             // otherwise just print interior elements of container | ||||||
|  |             $content.each(function() { | ||||||
|  |                 $(this).children().appendTo($body) | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Copies values from origin to clone for passed in elementSelector | ||||||
|  |     function copyValues(origin, clone, elementSelector) { | ||||||
|  |         var $originalElements = origin.find(elementSelector); | ||||||
|  |  | ||||||
|  |         clone.find(elementSelector).each(function(index, item) { | ||||||
|  |             $(item).val($originalElements.eq(index).val()); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var opt; | ||||||
|  |     $.fn.printThis = function(options) { | ||||||
|  |         opt = $.extend({}, $.fn.printThis.defaults, options); | ||||||
|  |         var $element = this instanceof jQuery ? this : $(this); | ||||||
|  |  | ||||||
|  |         var strFrameName = "printThis-" + (new Date()).getTime(); | ||||||
|  |  | ||||||
|  |         if (window.location.hostname !== document.domain && navigator.userAgent.match(/msie/i)) { | ||||||
|  |             // Ugly IE hacks due to IE not inheriting document.domain from parent | ||||||
|  |             // checks if document.domain is set by comparing the host name against document.domain | ||||||
|  |             var iframeSrc = "javascript:document.write(\"<head><script>document.domain=\\\"" + document.domain + "\\\";</s" + "cript></head><body></body>\")"; | ||||||
|  |             var printI = document.createElement('iframe'); | ||||||
|  |             printI.name = "printIframe"; | ||||||
|  |             printI.id = strFrameName; | ||||||
|  |             printI.className = "MSIE"; | ||||||
|  |             document.body.appendChild(printI); | ||||||
|  |             printI.src = iframeSrc; | ||||||
|  |  | ||||||
|  |         } else { | ||||||
|  |             // other browsers inherit document.domain, and IE works if document.domain is not explicitly set | ||||||
|  |             var $frame = $("<iframe id='" + strFrameName + "' name='printIframe' />"); | ||||||
|  |             $frame.appendTo("body"); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         var $iframe = $("#" + strFrameName); | ||||||
|  |  | ||||||
|  |         // show frame if in debug mode | ||||||
|  |         if (!opt.debug) $iframe.css({ | ||||||
|  |             position: "absolute", | ||||||
|  |             width: "0px", | ||||||
|  |             height: "0px", | ||||||
|  |             left: "-600px", | ||||||
|  |             top: "-600px" | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         // before print callback | ||||||
|  |         if (typeof opt.beforePrint === "function") { | ||||||
|  |             opt.beforePrint(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // $iframe.ready() and $iframe.load were inconsistent between browsers | ||||||
|  |         setTimeout(function() { | ||||||
|  |  | ||||||
|  |             // Add doctype to fix the style difference between printing and render | ||||||
|  |             function setDocType($iframe, doctype){ | ||||||
|  |                 var win, doc; | ||||||
|  |                 win = $iframe.get(0); | ||||||
|  |                 win = win.contentWindow || win.contentDocument || win; | ||||||
|  |                 doc = win.document || win.contentDocument || win; | ||||||
|  |                 doc.open(); | ||||||
|  |                 doc.write(doctype); | ||||||
|  |                 doc.close(); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (opt.doctypeString){ | ||||||
|  |                 setDocType($iframe, opt.doctypeString); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             var $doc = $iframe.contents(), | ||||||
|  |                 $head = $doc.find("head"), | ||||||
|  |                 $body = $doc.find("body"), | ||||||
|  |                 $base = $('base'), | ||||||
|  |                 baseURL; | ||||||
|  |  | ||||||
|  |             // add base tag to ensure elements use the parent domain | ||||||
|  |             if (opt.base === true && $base.length > 0) { | ||||||
|  |                 // take the base tag from the original page | ||||||
|  |                 baseURL = $base.attr('href'); | ||||||
|  |             } else if (typeof opt.base === 'string') { | ||||||
|  |                 // An exact base string is provided | ||||||
|  |                 baseURL = opt.base; | ||||||
|  |             } else { | ||||||
|  |                 // Use the page URL as the base | ||||||
|  |                 baseURL = document.location.protocol + '//' + document.location.host; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             $head.append('<base href="' + baseURL + '">'); | ||||||
|  |  | ||||||
|  |             // import page stylesheets | ||||||
|  |             if (opt.importCSS) $("link[rel=stylesheet]").each(function() { | ||||||
|  |                 var href = $(this).attr("href"); | ||||||
|  |                 if (href) { | ||||||
|  |                     var media = $(this).attr("media") || "all"; | ||||||
|  |                     $head.append("<link type='text/css' rel='stylesheet' href='" + href + "' media='" + media + "'>"); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             // import style tags | ||||||
|  |             if (opt.importStyle) $("style").each(function() { | ||||||
|  |                 $head.append(this.outerHTML); | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             // add title of the page | ||||||
|  |             if (opt.pageTitle) $head.append("<title>" + opt.pageTitle + "</title>"); | ||||||
|  |  | ||||||
|  |             // import additional stylesheet(s) | ||||||
|  |             if (opt.loadCSS) { | ||||||
|  |                 if ($.isArray(opt.loadCSS)) { | ||||||
|  |                     jQuery.each(opt.loadCSS, function(index, value) { | ||||||
|  |                         $head.append("<link type='text/css' rel='stylesheet' href='" + this + "'>"); | ||||||
|  |                     }); | ||||||
|  |                 } else { | ||||||
|  |                     $head.append("<link type='text/css' rel='stylesheet' href='" + opt.loadCSS + "'>"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             var pageHtml = $('html')[0]; | ||||||
|  |  | ||||||
|  |             // CSS VAR in html tag when dynamic apply e.g.  document.documentElement.style.setProperty("--foo", bar); | ||||||
|  |             $doc.find('html').prop('style', pageHtml.style.cssText); | ||||||
|  |  | ||||||
|  |             // copy 'root' tag classes | ||||||
|  |             var tag = opt.copyTagClasses; | ||||||
|  |             if (tag) { | ||||||
|  |                 tag = tag === true ? 'bh' : tag; | ||||||
|  |                 if (tag.indexOf('b') !== -1) { | ||||||
|  |                     $body.addClass($('body')[0].className); | ||||||
|  |                 } | ||||||
|  |                 if (tag.indexOf('h') !== -1) { | ||||||
|  |                     $doc.find('html').addClass(pageHtml.className); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // copy ':root' tag classes | ||||||
|  |             tag = opt.copyTagStyles; | ||||||
|  |             if (tag) { | ||||||
|  |                 tag = tag === true ? 'bh' : tag; | ||||||
|  |                 if (tag.indexOf('b') !== -1) { | ||||||
|  |                     $body.attr('style', $('body')[0].style.cssText); | ||||||
|  |                 } | ||||||
|  |                 if (tag.indexOf('h') !== -1) { | ||||||
|  |                     $doc.find('html').attr('style', pageHtml.style.cssText); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // print header | ||||||
|  |             appendContent($body, opt.header); | ||||||
|  |  | ||||||
|  |             if (opt.canvas) { | ||||||
|  |                 // add canvas data-ids for easy access after cloning. | ||||||
|  |                 var canvasId = 0; | ||||||
|  |                 // .addBack('canvas') adds the top-level element if it is a canvas. | ||||||
|  |                 $element.find('canvas').addBack('canvas').each(function(){ | ||||||
|  |                     $(this).attr('data-printthis', canvasId++); | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             appendBody($body, $element, opt); | ||||||
|  |  | ||||||
|  |             if (opt.canvas) { | ||||||
|  |                 // Re-draw new canvases by referencing the originals | ||||||
|  |                 $body.find('canvas').each(function(){ | ||||||
|  |                     var cid = $(this).data('printthis'), | ||||||
|  |                         $src = $('[data-printthis="' + cid + '"]'); | ||||||
|  |  | ||||||
|  |                     this.getContext('2d').drawImage($src[0], 0, 0); | ||||||
|  |  | ||||||
|  |                     // Remove the markup from the original | ||||||
|  |                     if ($.isFunction($.fn.removeAttr)) { | ||||||
|  |                         $src.removeAttr('data-printthis'); | ||||||
|  |                     } else { | ||||||
|  |                         $.each($src, function(i, el) { | ||||||
|  |                             el.removeAttribute('data-printthis'); | ||||||
|  |                         }); | ||||||
|  |                     } | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // remove inline styles | ||||||
|  |             if (opt.removeInline) { | ||||||
|  |                 // Ensure there is a selector, even if it's been mistakenly removed | ||||||
|  |                 var selector = opt.removeInlineSelector || '*'; | ||||||
|  |                 // $.removeAttr available jQuery 1.7+ | ||||||
|  |                 if ($.isFunction($.removeAttr)) { | ||||||
|  |                     $body.find(selector).removeAttr("style"); | ||||||
|  |                 } else { | ||||||
|  |                     $body.find(selector).attr("style", ""); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // print "footer" | ||||||
|  |             appendContent($body, opt.footer); | ||||||
|  |  | ||||||
|  |             // attach event handler function to beforePrint event | ||||||
|  |             function attachOnBeforePrintEvent($iframe, beforePrintHandler) { | ||||||
|  |                 var win = $iframe.get(0); | ||||||
|  |                 win = win.contentWindow || win.contentDocument || win; | ||||||
|  |  | ||||||
|  |                 if (typeof beforePrintHandler === "function") { | ||||||
|  |                     if ('matchMedia' in win) { | ||||||
|  |                         win.matchMedia('print').addListener(function(mql) { | ||||||
|  |                             if(mql.matches)  beforePrintHandler(); | ||||||
|  |                         }); | ||||||
|  |                     } else { | ||||||
|  |                         win.onbeforeprint = beforePrintHandler; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             attachOnBeforePrintEvent($iframe, opt.beforePrintEvent); | ||||||
|  |  | ||||||
|  |             setTimeout(function() { | ||||||
|  |                 if ($iframe.hasClass("MSIE")) { | ||||||
|  |                     // check if the iframe was created with the ugly hack | ||||||
|  |                     // and perform another ugly hack out of neccessity | ||||||
|  |                     window.frames["printIframe"].focus(); | ||||||
|  |                     $head.append("<script>  window.print(); </s" + "cript>"); | ||||||
|  |                 } else { | ||||||
|  |                     // proper method | ||||||
|  |                     if (document.queryCommandSupported("print")) { | ||||||
|  |                         $iframe[0].contentWindow.document.execCommand("print", false, null); | ||||||
|  |                     } else { | ||||||
|  |                         $iframe[0].contentWindow.focus(); | ||||||
|  |                         $iframe[0].contentWindow.print(); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 // remove iframe after print | ||||||
|  |                 if (!opt.debug) { | ||||||
|  |                     setTimeout(function() { | ||||||
|  |                         $iframe.remove(); | ||||||
|  |  | ||||||
|  |                     }, 1000); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 // after print callback | ||||||
|  |                 if (typeof opt.afterPrint === "function") { | ||||||
|  |                     opt.afterPrint(); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |             }, opt.printDelay); | ||||||
|  |  | ||||||
|  |         }, 333); | ||||||
|  |  | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     // defaults | ||||||
|  |     $.fn.printThis.defaults = { | ||||||
|  |         debug: false,                       // show the iframe for debugging | ||||||
|  |         importCSS: true,                    // import parent page css | ||||||
|  |         importStyle: true,                  // import style tags | ||||||
|  |         printContainer: true,               // print outer container/$.selector | ||||||
|  |         loadCSS: "",                        // path to additional css file - use an array [] for multiple | ||||||
|  |         pageTitle: "",                      // add title to print page | ||||||
|  |         removeInline: false,                // remove inline styles from print elements | ||||||
|  |         removeInlineSelector: "*",          // custom selectors to filter inline styles. removeInline must be true | ||||||
|  |         printDelay: 1000,                   // variable print delay | ||||||
|  |         header: null,                       // prefix to html | ||||||
|  |         footer: null,                       // postfix to html | ||||||
|  |         base: false,                        // preserve the BASE tag or accept a string for the URL | ||||||
|  |         formValues: true,                   // preserve input/form values | ||||||
|  |         canvas: true,                       // copy canvas content | ||||||
|  |         doctypeString: '<!DOCTYPE html>',   // enter a different doctype for older markup | ||||||
|  |         removeScripts: false,               // remove script tags from print content | ||||||
|  |         copyTagClasses: true,               // copy classes from the html & body tag | ||||||
|  |         copyTagStyles: true,                // copy styles from html & body tag (for CSS Variables) | ||||||
|  |         beforePrintEvent: null,             // callback function for printEvent in iframe | ||||||
|  |         beforePrint: null,                  // function called before iframe is filled | ||||||
|  |         afterPrint: null                    // function called before iframe is removed | ||||||
|  |     }; | ||||||
|  | })(jQuery); | ||||||
| @ -77,6 +77,7 @@ | |||||||
|     <script src="{{ asset('assets/js/bootstrap-datepicker.min.js') }}"></script> |     <script src="{{ asset('assets/js/bootstrap-datepicker.min.js') }}"></script> | ||||||
|     <script src="{{ asset('assets/js/helper.js') }}"></script> |     <script src="{{ asset('assets/js/helper.js') }}"></script> | ||||||
| 	<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery-datetimepicker/2.5.20/jquery.datetimepicker.full.min.js" integrity="sha512-AIOTidJAcHBH2G/oZv9viEGXRqDNmfdPVPYOYKGy3fti0xIplnlgMHUGfuNRzC6FkzIo0iIxgFnr9RikFxK+sw==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> | 	<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery-datetimepicker/2.5.20/jquery.datetimepicker.full.min.js" integrity="sha512-AIOTidJAcHBH2G/oZv9viEGXRqDNmfdPVPYOYKGy3fti0xIplnlgMHUGfuNRzC6FkzIo0iIxgFnr9RikFxK+sw==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> | ||||||
|  | 	<!-- <script src="https://cdnjs.cloudflare.com/ajax/libs/moment-timezone/0.5.34/moment-timezone-with-data.min.js"></script> --> | ||||||
|  |  | ||||||
|     <script> |     <script> | ||||||
|         const AppState = { |         const AppState = { | ||||||
| @ -87,6 +88,8 @@ | |||||||
|             route_segment1: "{{ Request::segment(1) }}", |             route_segment1: "{{ Request::segment(1) }}", | ||||||
|             route_segment2: "{{ Request::segment(2) }}", |             route_segment2: "{{ Request::segment(2) }}", | ||||||
|             current_company: 0, // cptid |             current_company: 0, // cptid | ||||||
|  | 			// TIMEFIX: new Date().getTimezoneOffset() * -60  | ||||||
|  | 			TIMEFIX: 25200 | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         // function startTime() { |         // function startTime() { | ||||||
|  | |||||||
							
								
								
									
										208
									
								
								resources/views/menu_v1/_userLogs.blade.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										208
									
								
								resources/views/menu_v1/_userLogs.blade.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,208 @@ | |||||||
|  | <style> | ||||||
|  | 	#map { | ||||||
|  | 		height: 100%; | ||||||
|  | 		margin: 0; | ||||||
|  | 	} | ||||||
|  | 	.my-leaflet-map-container img { | ||||||
|  | 		max-height: none; | ||||||
|  | 	} | ||||||
|  | 	.dtl-text{ | ||||||
|  | 		font-size: 11px; | ||||||
|  | 	} | ||||||
|  | 	.head-text{ | ||||||
|  | 		font-size: 12px !important; | ||||||
|  | 	} | ||||||
|  | 	/* .leaflet-overlay-pane svg { | ||||||
|  | 		transform: none !important; | ||||||
|  | 	} */ | ||||||
|  | 	.leaflet-routing-container { | ||||||
|  | 		display: none !important; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	#viewPdf { | ||||||
|  | 		display: flex; | ||||||
|  | 		justify-content: center; | ||||||
|  | 		width: 794px; | ||||||
|  | 		/* height: auto; */ | ||||||
|  | 		max-width: 100%; /* Ensures it is responsive */ | ||||||
|  | 	} | ||||||
|  | 	/* .modal-dialog{ | ||||||
|  | 		width: 794px; | ||||||
|  | 	} */ | ||||||
|  |  | ||||||
|  | </style> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | <div class="modal-dialog modal-dialog modal-dialog-centered modal-dialog-scrollable modal-lg"> | ||||||
|  | 	<div class="modal-content"> | ||||||
|  | 		<div class="modal-header"> | ||||||
|  | 			<h5 class="modal-title" id="mdlDetailTripLabel">{{$user->first_name}} Logs</h5> | ||||||
|  | 			<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="modal-body"> | ||||||
|  | 			<div class="row"> | ||||||
|  | 				<div class="col-4"> | ||||||
|  | 					<div class="form-group"> | ||||||
|  | 						<label class="text-muted">From</label> | ||||||
|  | 						<!-- default today --> | ||||||
|  | 						<!-- <input class="form-control" id="tgl0" value="02-09-2025 00:00"> --> | ||||||
|  | 						<input class="form-control" id="tgl0" value="{{ date('d-m-Y 00:00') }}"> | ||||||
|  | 					</div> | ||||||
|  | 				</div> | ||||||
|  | 				<div class="col-4"> | ||||||
|  | 					<div class="form-group"> | ||||||
|  | 						<label class="text-muted">To</label> | ||||||
|  | 						<!-- <input class="form-control" id="tgl1" value="02-09-2025 23:00"> --> | ||||||
|  | 						<input class="form-control" id="tgl1" value="{{ date('d-m-Y 23:59') }}"> | ||||||
|  | 					</div> | ||||||
|  | 				</div> | ||||||
|  | 				<div class="col-4 d-flex align-items-end"> | ||||||
|  | 					<button class="btn btn-primary" id="submitFilter">Submit</button> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="table-responsive"> | ||||||
|  | 				<table id="tLogs" class="table table-hover dataTable w-100"> | ||||||
|  | 					<thead> | ||||||
|  | 						<tr class=""> | ||||||
|  | 							<th class="">Time</th> | ||||||
|  | 							<th class="">Module</th> | ||||||
|  | 							<th class="">Description</th> | ||||||
|  | 							<th class="">Action</th> | ||||||
|  | 						</tr> | ||||||
|  | 					</thead> | ||||||
|  | 				</table> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="modal-footer"> | ||||||
|  | 			<button type="button" class="btn btn-sm btn-danger" data-bs-dismiss="modal">Close</button> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | <script> | ||||||
|  | 	$(document).ready(function(){ | ||||||
|  | 		$('.time').each(function () { | ||||||
|  | 			const unix = parseInt($(this).text().trim()) + AppState.TIMEFIX; | ||||||
|  | 			$(this).text(moment.unix(unix).format('DD MMM YYYY HH:mm:ss')); | ||||||
|  | 		}); | ||||||
|  | 		$('#tgl0, #tgl1').datetimepicker({ | ||||||
|  | 			format:'d-m-Y H:i', | ||||||
|  | 			defaultTime:'00:00', | ||||||
|  | 			closeOnDateSelect: true, | ||||||
|  | 			// mask:true | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		$('#submitFilter').on('click', function() { | ||||||
|  | 			DTable.reload(); | ||||||
|  | 		}); | ||||||
|  | 		// $('#submitFilter').trigger('click'); | ||||||
|  |  | ||||||
|  |         const DTable = { | ||||||
|  | 			table: null, | ||||||
|  |     		lastAjax: null, // keep track of the last ajax request | ||||||
|  |             activate: function() { | ||||||
|  |                 DTable.reload(); | ||||||
|  |             }, | ||||||
|  |             reload: function() { | ||||||
|  | 				// Abort the last request if it's still running | ||||||
|  | 				if (DTable.lastAjax) { | ||||||
|  | 					DTable.lastAjax.abort(); | ||||||
|  | 				} | ||||||
|  | 				if (DTable.table) { | ||||||
|  | 					// If table already exists → reload | ||||||
|  | 					DTable.table.ajax.reload(); | ||||||
|  | 					return; | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				DTable.table = $('#tLogs').DataTable({ | ||||||
|  | 					searching: false,   // 🔹 remove search box | ||||||
|  | 					ordering: false,    // 🔹 disable sorting for all columns | ||||||
|  | 					processing: true, | ||||||
|  | 					serverSide: false, | ||||||
|  | 					bLengthChange: true, | ||||||
|  | 					deferRender: true, | ||||||
|  | 					destroy: true, | ||||||
|  | 					ajax: function(data, callback, settings) { | ||||||
|  | 						// Abort previous | ||||||
|  | 						if (DTable.lastAjax) { | ||||||
|  | 							DTable.lastAjax.abort(); | ||||||
|  | 						} | ||||||
|  |  | ||||||
|  | 						// Fire new request | ||||||
|  | 						DTable.lastAjax = $.ajax({ | ||||||
|  | 							url: `{{ route('api_user_logs1') }}? | ||||||
|  | 								tgl0=${moment($('#tgl0').val(), "DD-MM-YYYY HH:mm").unix()} | ||||||
|  | 								&tgl1=${moment($('#tgl1').val(), "DD-MM-YYYY HH:mm").unix()} | ||||||
|  | 								&id={{$user->id}} | ||||||
|  | 							`, | ||||||
|  | 							type: 'GET', | ||||||
|  | 							success: function(json) { | ||||||
|  | 								callback(json); | ||||||
|  | 							}, | ||||||
|  | 							error: function(xhr, status, error) { | ||||||
|  | 								if (status !== 'abort') { | ||||||
|  | 									console.error("AJAX error:", error); | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							complete: function() { | ||||||
|  | 								DTable.lastAjax = null; | ||||||
|  | 							} | ||||||
|  | 						}); | ||||||
|  | 					}, | ||||||
|  |                     deferRender: true, | ||||||
|  | 					columns: [ | ||||||
|  | 						{ | ||||||
|  | 							data: 'crt', | ||||||
|  | 							render: (data, type, row, meta) => { | ||||||
|  | 								// let addr = row | ||||||
|  | 								return ` | ||||||
|  | 									${moment.unix(data).format('DD MMM YYYY HH:mm:ss')}<br> | ||||||
|  | 								`; | ||||||
|  | 							}  | ||||||
|  |  | ||||||
|  | 						}, | ||||||
|  | 						{  | ||||||
|  | 							data: 'log', | ||||||
|  | 							render: (data, type, row, meta) => { | ||||||
|  | 								let log = JSON.parse(data); | ||||||
|  | 								let module = log.module || '-'; | ||||||
|  | 								return module; | ||||||
|  | 							} | ||||||
|  | 						}, | ||||||
|  | 						{  | ||||||
|  | 							data: 'log', | ||||||
|  | 							render: (data, type, row, meta) => { | ||||||
|  | 								let log = JSON.parse(data); | ||||||
|  | 								let desc = log.desc || '-'; | ||||||
|  | 								return desc; | ||||||
|  | 							} | ||||||
|  | 						}, | ||||||
|  | 						{  | ||||||
|  | 							data: 'log', | ||||||
|  | 							render: (data, type, row, meta) => { | ||||||
|  | 								let log = JSON.parse(data); | ||||||
|  | 								let action = log.action || '-'; | ||||||
|  |  | ||||||
|  | 								// use badge for action | ||||||
|  | 								// action = `<span class="badge bg-secondary">${action}</span>`; | ||||||
|  | 								let badge = ''; | ||||||
|  | 								if (action.toLowerCase() === 'create') { | ||||||
|  | 									badge = `<span class="badge bg-success">${action}</span>`; | ||||||
|  | 								} else if (action.toLowerCase() === 'update') { | ||||||
|  | 									badge = `<span class="badge bg-warning">${action}</span>`; | ||||||
|  | 								} else if (action.toLowerCase() === 'delete') { | ||||||
|  | 									badge = `<span class="badge bg-danger">${action}</span>`; | ||||||
|  | 								} else { | ||||||
|  | 									badge = `<span class="badge bg-secondary">${action}</span>`; | ||||||
|  | 								} | ||||||
|  | 								return badge; | ||||||
|  | 							} | ||||||
|  | 						}, | ||||||
|  | 					], | ||||||
|  | 					paging: false, | ||||||
|  | 				}); | ||||||
|  |             }, | ||||||
|  |         }; | ||||||
|  | 		DTable.activate(); | ||||||
|  | 	}); | ||||||
|  | </script> | ||||||
| @ -46,6 +46,7 @@ | |||||||
|                                     <th class="text-end">Type</th> |                                     <th class="text-end">Type</th> | ||||||
|                                     <th class="text-center">Status</th> |                                     <th class="text-center">Status</th> | ||||||
|                                     <th class="text-center">Installation</th> |                                     <th class="text-center">Installation</th> | ||||||
|  |                                     <th class="text-center">Vehicle</th> | ||||||
|                                     <th class="text-center">Availability</th> |                                     <th class="text-center">Availability</th> | ||||||
|                                 </tr> |                                 </tr> | ||||||
|                             </thead> |                             </thead> | ||||||
| @ -491,6 +492,13 @@ | |||||||
|                                 } |                                 } | ||||||
|                             } |                             } | ||||||
|                         }, |                         }, | ||||||
|  | 						{ | ||||||
|  | 							data: 'vhc_nopol1', | ||||||
|  | 							className: 'text-left text-nowrap', | ||||||
|  | 							visible: true, | ||||||
|  | 							orderable: true, | ||||||
|  | 							searchable: true, | ||||||
|  | 						}, | ||||||
|                         { |                         { | ||||||
|                             data: 'is_available', |                             data: 'is_available', | ||||||
|                             className: 'text-center text-nowrap', |                             className: 'text-center text-nowrap', | ||||||
|  | |||||||
							
								
								
									
										3541
									
								
								resources/views/menu_v1/dashboard.blade copy.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3541
									
								
								resources/views/menu_v1/dashboard.blade copy.php
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -408,10 +408,12 @@ | |||||||
|                         <div class="mb-2"> |                         <div class="mb-2"> | ||||||
|                             <label class="text-muted">From</label> |                             <label class="text-muted">From</label> | ||||||
|                             <input class="form-control" id="historyStartDate"> |                             <input class="form-control" id="historyStartDate"> | ||||||
|  |                             <!-- <input class="form-control" id="historyStartDate" value="11-08-2025 00:00"> --> | ||||||
|                         </div> |                         </div> | ||||||
|                         <div class="mb-2"> |                         <div class="mb-2"> | ||||||
|                             <label class="text-muted">To</label> |                             <label class="text-muted">To</label> | ||||||
|                             <input class="form-control" id="historyEndDate"> |                             <input class="form-control" id="historyEndDate"> | ||||||
|  |                             <!-- <input class="form-control" id="historyEndDate" value="11-08-2025 23:00"> --> | ||||||
|                         </div> |                         </div> | ||||||
| 						<button class="btn btn-primary btn-sm w-100 mb-3" id="historySearch">Search</button> | 						<button class="btn btn-primary btn-sm w-100 mb-3" id="historySearch">Search</button> | ||||||
|                     </div> |                     </div> | ||||||
| @ -533,6 +535,7 @@ | |||||||
|     <script src="https://cdnjs.cloudflare.com/ajax/libs/leaflet.draw/1.0.4/leaflet.draw.js" integrity="sha512-ozq8xQKq6urvuU6jNgkfqAmT7jKN2XumbrX1JiB3TnF7tI48DPI4Gy1GXKD/V3EExgAs1V+pRO7vwtS1LHg0Gw==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> |     <script src="https://cdnjs.cloudflare.com/ajax/libs/leaflet.draw/1.0.4/leaflet.draw.js" integrity="sha512-ozq8xQKq6urvuU6jNgkfqAmT7jKN2XumbrX1JiB3TnF7tI48DPI4Gy1GXKD/V3EExgAs1V+pRO7vwtS1LHg0Gw==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> | ||||||
|     <script src="https://cdn.jsdelivr.net/npm/leaflet-rotatedmarker@0.2.0/leaflet.rotatedMarker.min.js"></script> |     <script src="https://cdn.jsdelivr.net/npm/leaflet-rotatedmarker@0.2.0/leaflet.rotatedMarker.min.js"></script> | ||||||
|     <script src="https://unpkg.com/leaflet-routing-machine@3.2.12/dist/leaflet-routing-machine.js"></script> |     <script src="https://unpkg.com/leaflet-routing-machine@3.2.12/dist/leaflet-routing-machine.js"></script> | ||||||
|  | 	<script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script> | ||||||
| 					 | 					 | ||||||
|     <script> |     <script> | ||||||
|         'use strict'; |         'use strict'; | ||||||
| @ -941,31 +944,120 @@ | |||||||
|                 if (cb) cb(polyline); |                 if (cb) cb(polyline); | ||||||
|                 return polyline; |                 return polyline; | ||||||
|             }, |             }, | ||||||
|             addRoutes: function(locs = [], cb = null) { | 			addRoutes: async function(locs = [], cb = null) { | ||||||
|                 let latLngs = []; | 				let latLngs = []; | ||||||
|                 for (let i = 0; i < locs.length; i++) { | 				for (let i = 0; i < locs.length; i++) { | ||||||
|                     latLngs.push([locs[i].lat, locs[i].lng]); | 					latLngs.push([locs[i].lat, locs[i].lng]); | ||||||
|                 } | 				} | ||||||
| 				let routes = L.Routing.control({ |  | ||||||
| 					waypoints: latLngs.reverse(), // reverse to make first point as start |  | ||||||
| 					// router: L.Routing.osrmv1(), |  | ||||||
| 					router: L.Routing.osrmv1({ |  | ||||||
| 						serviceUrl: "https://brilianapps.britimorleste.tl:5001/route/v1", |  | ||||||
| 					}), |  | ||||||
| 					show: false, |  | ||||||
| 					itinerary: null, // 👈 completely removes the panel |  | ||||||
| 					addWaypoints: false, // ❌ prevent adding points by clicking |  | ||||||
| 					draggableWaypoints: false, // ❌ prevent dragging markers |  | ||||||
| 					routeWhileDragging: false, // optional: don’t reroute while dragging |  | ||||||
| 					createMarker: () => null, |  | ||||||
| 					lineOptions:{ |  | ||||||
| 						styles: [{ color: locs[0]?.options?.color, weight: 3, opacity: 0.7 }], |  | ||||||
| 					}, |  | ||||||
| 				}).addTo(Leaflet.map) |  | ||||||
|  |  | ||||||
|                 if (cb) cb(routes); | 				function chunkArray(arr, size) { | ||||||
|                 return routes; | 					const result = []; | ||||||
|             }, | 					for (let i = 0; i < arr.length; i += size) { | ||||||
|  | 						result.push(arr.slice(i, i + size)); | ||||||
|  | 					} | ||||||
|  | 					return result; | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				function fetchOsrm(points) { | ||||||
|  | 					const coords = points; | ||||||
|  | 					const hints = ";".repeat(points.length - 1); | ||||||
|  |  | ||||||
|  | 					const body = { | ||||||
|  | 						coordinates: coords, | ||||||
|  | 						overview: "false", | ||||||
|  | 						alternatives: "true", | ||||||
|  | 						steps: "true", | ||||||
|  | 						hints: hints | ||||||
|  | 					}; | ||||||
|  |  | ||||||
|  | 					let config = { | ||||||
|  | 						method: 'post', | ||||||
|  | 						maxBodyLength: Infinity, | ||||||
|  | 						url: 'https://brilianapps.britimorleste.tl/osrm-backend/post-route/v1/driving/', | ||||||
|  | 						headers: { | ||||||
|  | 							'Content-Type': 'application/json' | ||||||
|  | 						}, | ||||||
|  | 						data: body | ||||||
|  | 					}; | ||||||
|  |  | ||||||
|  | 					return axios.request(config) | ||||||
|  | 						.then((response) => { | ||||||
|  | 							return response.data; | ||||||
|  | 						}) | ||||||
|  | 						.catch((err) => { | ||||||
|  | 							console.error("Error:", err.message); | ||||||
|  | 							return null; | ||||||
|  | 						}); | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				function decodeOSRMGeometry(encoded) { | ||||||
|  | 					const coordinates = []; | ||||||
|  | 					let index = 0, lat = 0, lng = 0; | ||||||
|  |  | ||||||
|  | 					while (index < encoded.length) { | ||||||
|  | 						let result = 1, shift = 0, b; | ||||||
|  | 						do { | ||||||
|  | 							b = encoded.charCodeAt(index++) - 63 - 1; | ||||||
|  | 							result += b << shift; | ||||||
|  | 							shift += 5; | ||||||
|  | 						} while (b >= 0x1f); | ||||||
|  | 						lat += (result & 1 ? ~(result >> 1) : result >> 1); | ||||||
|  |  | ||||||
|  | 						result = 1; | ||||||
|  | 						shift = 0; | ||||||
|  | 						do { | ||||||
|  | 							b = encoded.charCodeAt(index++) - 63 - 1; | ||||||
|  | 							result += b << shift; | ||||||
|  | 							shift += 5; | ||||||
|  | 						} while (b >= 0x1f); | ||||||
|  | 						lng += (result & 1 ? ~(result >> 1) : result >> 1); | ||||||
|  |  | ||||||
|  | 						coordinates.push([lat / 1e5, lng / 1e5]); | ||||||
|  | 					} | ||||||
|  |  | ||||||
|  | 					return coordinates; | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				async function getCoordinates(points) { | ||||||
|  | 					const chunkSize = 500; | ||||||
|  | 					const chunks = chunkArray(points, chunkSize); // Split the points array into chunks of 500 | ||||||
|  | 					let allCoords = []; | ||||||
|  |  | ||||||
|  | 					for (const chunk of chunks) { | ||||||
|  | 						const osrm = await fetchOsrm(chunk); // Fetch OSRM data for each chunk | ||||||
|  |  | ||||||
|  | 						if (!osrm) { | ||||||
|  | 							console.log("OSRM failed for chunk"); | ||||||
|  | 							return; | ||||||
|  | 						} | ||||||
|  |  | ||||||
|  | 						const coords = osrm.routes[0].legs.flatMap(leg => | ||||||
|  | 							leg.steps.flatMap(step => | ||||||
|  | 								decodeOSRMGeometry(step.geometry) | ||||||
|  | 							) | ||||||
|  | 						); | ||||||
|  |  | ||||||
|  | 						allCoords = allCoords.concat(coords); // Combine the result | ||||||
|  | 					} | ||||||
|  |  | ||||||
|  | 					// Now add the polyline to the map | ||||||
|  | 					return L.polyline(allCoords, { | ||||||
|  | 						color: locs[0]?.options?.color, | ||||||
|  | 						weight: 3, | ||||||
|  | 						opacity: 0.8 | ||||||
|  | 					}).addTo(Leaflet.map); | ||||||
|  | 					// map.fitBounds(allCoords.map(c => L.latLng(c[0], c[1]))); | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				// Usage: Pass the array of coordinates to the function | ||||||
|  | 				const routes = await getCoordinates(latLngs); | ||||||
|  |  | ||||||
|  | 				// optionally fit bounds | ||||||
|  | 				// Leaflet.map.fitBounds(coords.map(c => L.latLng(c[0], c[1]))); | ||||||
|  |  | ||||||
|  | 				if (cb) cb(routes); | ||||||
|  | 				return routes; | ||||||
|  | 			}, | ||||||
|             // start custom polylines |             // start custom polylines | ||||||
|             addCustomPolylines: function(locs = [], cb = null) { |             addCustomPolylines: function(locs = [], cb = null) { | ||||||
|                 const radius = 0.5 /* corner smooth radius, keep value in range 0 - 0.5 to avoid artifacts */ |                 const radius = 0.5 /* corner smooth radius, keep value in range 0 - 0.5 to avoid artifacts */ | ||||||
| @ -1310,8 +1402,9 @@ | |||||||
|                     $('#infoMovement').addClass('d-block'); |                     $('#infoMovement').addClass('d-block'); | ||||||
|                     if (State.inShowVid) { |                     if (State.inShowVid) { | ||||||
|                         let cache = !(State.inShowVid != State.loadedLastMoveVid); |                         let cache = !(State.inShowVid != State.loadedLastMoveVid); | ||||||
|                         Trucks.bundleShowRouteTruck(cache); // jika data yang diload sebelumnya beda, maka tidak mengambil dari cache |                         Trucks.bundleShowRouteTruck(cache, false); // jika data yang diload sebelumnya beda, maka tidak mengambil dari cache | ||||||
|                     } |                     } | ||||||
|  | 					// Menu.clearListMovements(); | ||||||
|                 } |                 } | ||||||
|             }, |             }, | ||||||
|             eventListVehicle: function() { |             eventListVehicle: function() { | ||||||
| @ -1354,48 +1447,14 @@ | |||||||
|                 }); |                 }); | ||||||
|             }, |             }, | ||||||
|             showToListMovement: function(obj) { |             showToListMovement: function(obj) { | ||||||
|                 /** |  | ||||||
|                  * fix DOMException: Failed to execute 'btoa' on 'Window': The string to be encoded contains |  | ||||||
|                  * because of string Cawang–Pluit |  | ||||||
|                  * solution: https://stackoverflow.com/questions/23223718/failed-to-execute-btoa-on-window-the-string-to-be-encoded-contains-characte |  | ||||||
|                  * window.btoa( unescape(encodeURIComponent(str) ) is deprecated => window.btoa(encodeURIComponent(str)) |  | ||||||
|                  */ |  | ||||||
|                 // <p class="text-muted mb-0">${(obj?.city_text || obj?.state_text || 'address')} - ${(obj?.postcode || 'postcode')}</p> |  | ||||||
|                 // <p class="text-muted mb-0">${moment.unix(obj?.lst_loc_crt).format('DD MMM YYYY HH:mm')}</p> // yang ini mix |  | ||||||
|  |  | ||||||
|                 // <p class="text-success mb-0">Device Time: ${moment.unix(obj?.lst_loc_crt_d).format('DD MMM YYYY HH:mm')}</p> |  | ||||||
|                 // <p class="text-danger mb-0">Server Time: ${moment.unix(obj?.lst_loc_crt_s).format('DD MMM YYYY HH:mm')}</p> |  | ||||||
|                 // <p class="${(obj.pre_milleage >= 3) ? 'text-warning' : 'text-muted'} mb-0">Pre Milleage: ${(obj.pre_milleage).toFixed(3)} Km</p> |  | ||||||
|  |  | ||||||
|                 // $('#infoMove-plots').append(` |  | ||||||
|             //     <a href="#" class="plotMove-item" data-obj="${window.btoa( encodeURIComponent( JSON.stringify(obj) ) )}"> |  | ||||||
|             //         <li class="list-group-item p-1 px-2"> |  | ||||||
|             //             <p class="text-bold mb-0">${obj.key_index}</p> |  | ||||||
|             //             <p class="text-bold mb-0">Time: ${moment.unix(obj?.lst_loc_crt_d).format('DD MMM YYYY HH:mm')}</p> |  | ||||||
|             //             <p class="text-muted mb-0">Speed: ${(typeof obj.speed != 'undefined') ? obj.speed : '0'}</p> |  | ||||||
|             //             <p class="text-muted mb-0">${Number(obj.latitude).toFixed(5)} - ${Number(obj.longitude).toFixed(6)}</p> |  | ||||||
|             //             <p class="text-muted mb-0">${Helper.shortenText(decodeURIComponent(obj?.fulladdress || 'address'), 25)}</p> |  | ||||||
|             //         </li> |  | ||||||
|             //     </a> |  | ||||||
|             // `); |  | ||||||
|                 let arrIdx = Helper.getIndexReversedSequence(obj.key_index - 1, Trucks.last_move.length - 1); |                 let arrIdx = Helper.getIndexReversedSequence(obj.key_index - 1, Trucks.last_move.length - 1); | ||||||
|                 // +7 | 				const unix = parseInt(obj?.lst_loc_crt_d) + AppState.TIMEFIX; | ||||||
|                 // $('#infoMove-plots').append(` |  | ||||||
|             //     <a href="#" class="plotMove-item" data-obj="${window.btoa( encodeURIComponent( JSON.stringify(obj) ) )}"> |  | ||||||
|             //         <li class="list-group-item p-1 px-2"> |  | ||||||
|             //             <p class="text-bold mb-0">${arrIdx + 1}</p> |  | ||||||
|             //             <p class="text-bold mb-0">Time: ${moment.unix(obj?.lst_loc_crt_d).format('DD MMM YYYY HH:mm:ss')}</p> |  | ||||||
|             //             <p class="text-muted mb-0">${Number(obj.latitude).toFixed(5)} - ${Number(obj.longitude).toFixed(6)}</p> |  | ||||||
|             //             <p class="text-muted mb-2">${Helper.shortenText(decodeURIComponent(obj?.fulladdress || 'address'), 255)}</p> |  | ||||||
|             //             <p class="mb-0">Current speed: ${Number(obj.speed)}km/h</p> |  | ||||||
|             //         </li> |  | ||||||
|             //     </a> |  | ||||||
|             // `); |  | ||||||
|                 $('#infoMove-plots').append(` |                 $('#infoMove-plots').append(` | ||||||
|                     <a href="#" class="plotMove-item" data-obj="${window.btoa( encodeURIComponent( JSON.stringify(obj) ) )}"> |                     <a href="#" class="plotMove-item" data-obj="${window.btoa( encodeURIComponent( JSON.stringify(obj) ) )}"> | ||||||
|                         <li class="list-group-item p-1 px-2"> |                         <li class="list-group-item p-1 px-2"> | ||||||
|                             <p class="text-bold mb-0">${arrIdx + 1}</p> |                             <p class="text-bold mb-0">${arrIdx + 1}</p> | ||||||
|                             <p class="text-bold mb-0">Time: ${moment.unix(obj?.lst_loc_crt_d).format('DD MMM YYYY HH:mm:ss')}</p> |                             <p class="text-bold mb-0">Time: ${moment.unix(unix).format('DD MMM YYYY HH:mm:ss')}</p> | ||||||
|                             <p class="text-muted mb-0">${Number(obj.latitude).toFixed(5)} - ${Number(obj.longitude).toFixed(6)}</p> |                             <p class="text-muted mb-0">${Number(obj.latitude).toFixed(5)} - ${Number(obj.longitude).toFixed(6)}</p> | ||||||
|                             <p class="text-muted mb-2">${Helper.shortenText(decodeURIComponent(obj?.fulladdress || 'address'), 255)}</p> |                             <p class="text-muted mb-2">${Helper.shortenText(decodeURIComponent(obj?.fulladdress || 'address'), 255)}</p> | ||||||
|                             <p class="mb-0">Current speed: ${Number(obj.speed)}km/h</p> |                             <p class="mb-0">Current speed: ${Number(obj.speed)}km/h</p> | ||||||
| @ -1417,11 +1476,12 @@ | |||||||
|             }, |             }, | ||||||
|             createMarkerDetailPlotMovement: function(tr, opt = {}) { |             createMarkerDetailPlotMovement: function(tr, opt = {}) { | ||||||
|                 Leaflet.clearLayer('eventRemoveDetailPlotMovement'); |                 Leaflet.clearLayer('eventRemoveDetailPlotMovement'); | ||||||
|  | 				const unix = parseInt(tr?.lst_loc_crt_d) + AppState.TIMEFIX; | ||||||
|                 let marker = Leaflet.addMarkers({ |                 let marker = Leaflet.addMarkers({ | ||||||
|                     lat: tr.latitude, |                     lat: tr.latitude, | ||||||
|                     lng: tr.longitude, |                     lng: tr.longitude, | ||||||
|                     // label: `<b>${tr.key_index}</b><br>${tr.nopol1} ${tr.nopol2} ${tr.nopol3}<br>${moment.unix(tr?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br>Speed: ${(typeof tr.lst_speed != 'undefined') ? tr.lst_speed : '0'}<br>${tr.latitude},${tr.longitude}<br>${decodeURIComponent(tr?.fulladdress || 'address')}`, |                     // label: `<b>${tr.key_index}</b><br>${tr.nopol1} ${tr.nopol2} ${tr.nopol3}<br>${moment.unix(tr?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br>Speed: ${(typeof tr.lst_speed != 'undefined') ? tr.lst_speed : '0'}<br>${tr.latitude},${tr.longitude}<br>${decodeURIComponent(tr?.fulladdress || 'address')}`, | ||||||
|                     label: `<b>${tr.nopol1} ${tr.nopol2} ${tr.nopol3}</b><br>${moment.unix(tr?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br>${decodeURIComponent(tr?.fulladdress || 'address')}.<br><br>Current speed: ${tr?.speed}km/h`, |                     label: `<b>${tr.nopol1} ${tr.nopol2} ${tr.nopol3}</b><br>${moment.unix(unix).format('DD MMM YYYY HH:mm')}<br>${decodeURIComponent(tr?.fulladdress || 'address')}.<br><br>Current speed: ${tr?.speed}km/h`, | ||||||
|                     //label: `<b>${tr.nopol1} ${tr.nopol2} ${tr.nopol3}</b><br>${moment.unix(tr?.lst_loc_crt_d).utcOffset(9 * 60).format('DD MMM YYYY HH:mm:ss')}<br>${decodeURIComponent(tr?.fulladdress || 'address')}.<br><br>Current speed: ${tr?.speed}km/h`, |                     //label: `<b>${tr.nopol1} ${tr.nopol2} ${tr.nopol3}</b><br>${moment.unix(tr?.lst_loc_crt_d).utcOffset(9 * 60).format('DD MMM YYYY HH:mm:ss')}<br>${decodeURIComponent(tr?.fulladdress || 'address')}.<br><br>Current speed: ${tr?.speed}km/h`, | ||||||
|                     options: { |                     options: { | ||||||
|                         // icon: Icon.destination() |                         // icon: Icon.destination() | ||||||
| @ -1559,7 +1619,7 @@ | |||||||
| 					$('#historyStartDate').val(moment().startOf('day').format('DD-MM-YYYY HH:mm')); | 					$('#historyStartDate').val(moment().startOf('day').format('DD-MM-YYYY HH:mm')); | ||||||
| 					$('#historyEndDate').val(moment().endOf('day').format('DD-MM-YYYY HH:mm')); | 					$('#historyEndDate').val(moment().endOf('day').format('DD-MM-YYYY HH:mm')); | ||||||
|  |  | ||||||
| 					// // for testing purpose | 					// for testing purpose | ||||||
|                     // State.historyStartDate = '1756054800'; |                     // State.historyStartDate = '1756054800'; | ||||||
|                     // State.historyEndDate = '1756141140'; |                     // State.historyEndDate = '1756141140'; | ||||||
| 					// $('#historyStartDate').val("25-08-2025 00:00"); | 					// $('#historyStartDate').val("25-08-2025 00:00"); | ||||||
| @ -2576,19 +2636,21 @@ | |||||||
|                 // last movement |                 // last movement | ||||||
|                 // $('#infoVehicles-infoMove').text('Last 24H from ' + moment.unix(truck?.lst_loc_crt).format('DD MMM YYYY HH:mm')); |                 // $('#infoVehicles-infoMove').text('Last 24H from ' + moment.unix(truck?.lst_loc_crt).format('DD MMM YYYY HH:mm')); | ||||||
|             }, |             }, | ||||||
|             bundleShowRouteTruck: async function(cache = false) { |             bundleShowRouteTruck: async function(cache = false, needUpdate = true) { | ||||||
|                 Menu.clearListMovements(); |                 Menu.clearListMovements(); | ||||||
|                 if (State.inShowVid != State.loadedLastMoveVid) $('#historyStartDate').trigger('clearFilterHistoryDate'); |                 if (State.inShowVid != State.loadedLastMoveVid)$('#historyStartDate').trigger('clearFilterHistoryDate'); | ||||||
|  |  | ||||||
| 				const getLastMove = await Trucks.getLastMove(State.inShowVid, cache); | 				if(needUpdate){ | ||||||
| 				Trucks.last_move = getLastMove.flat() | 					const getLastMove = await Trucks.getLastMove(State.inShowVid, cache); | ||||||
|  | 					Trucks.last_move = getLastMove.flat() | ||||||
| 					 | 					 | ||||||
|                 if (Trucks.last_move.length < 1) { | 					if (Trucks.last_move.length < 1) { | ||||||
|                     Helper.toast('Data Not Found', 'just now', 'There are no last data', 'bg-warning'); | 						Helper.toast('Data Not Found', 'just now', 'There are no last data', 'bg-warning'); | ||||||
|                     Trucks.last_move = null; | 						Trucks.last_move = null; | ||||||
|                     return false; | 						return false; | ||||||
|                 } | 					} | ||||||
|                 Trucks.showLastMoveToView(getLastMove); | 					Trucks.showLastMoveToView(getLastMove);					 | ||||||
|  | 				} | ||||||
|             }, |             }, | ||||||
|             getLastMove: async function(vid, cache = false) { |             getLastMove: async function(vid, cache = false) { | ||||||
|                 State.loadedLastMoveVid = vid; |                 State.loadedLastMoveVid = vid; | ||||||
| @ -2741,7 +2803,7 @@ | |||||||
|  |  | ||||||
| 				//     PgBar.setMinMax(0, truckRoutes.length - 1); | 				//     PgBar.setMinMax(0, truckRoutes.length - 1); | ||||||
|             // }, |             // }, | ||||||
|             routeStartEndGroupTrip:  function(truckRoutes) { |             routeStartEndGroupTrip: async function(truckRoutes) { | ||||||
| 				const colors = [ | 				const colors = [ | ||||||
| 					"#2980B9",  // Dark Blue | 					"#2980B9",  // Dark Blue | ||||||
| 					"#C0392B", // Dark Red | 					"#C0392B", // Dark Red | ||||||
| @ -2750,6 +2812,7 @@ | |||||||
| 					"#F39C12", // Dark Yellow/Gold | 					"#F39C12", // Dark Yellow/Gold | ||||||
| 				] | 				] | ||||||
| 				let i = 1 | 				let i = 1 | ||||||
|  | 				let allStartStop = [] | ||||||
| 				let polyTruckRoutes = truckRoutes.map((groupTrip, key0) => { | 				let polyTruckRoutes = truckRoutes.map((groupTrip, key0) => { | ||||||
| 					return groupTrip.map((obj, key) => { | 					return groupTrip.map((obj, key) => { | ||||||
| 						obj.key_index = i++ | 						obj.key_index = i++ | ||||||
| @ -2758,12 +2821,12 @@ | |||||||
| 						Menu.showToListMovement(obj) | 						Menu.showToListMovement(obj) | ||||||
| 						 | 						 | ||||||
| 						// add start end marker per group trip | 						// add start end marker per group trip | ||||||
| 						if(key == 0 || key == groupTrip.length - 1) | 						if(key == 0 || key == groupTrip.length - 1){ | ||||||
| 							Leaflet.addMarkers({ | 							const marker = Leaflet.addMarkers({ | ||||||
| 								lat: obj.latitude, | 								lat: obj.latitude, | ||||||
| 								lng: obj.longitude, | 								lng: obj.longitude, | ||||||
| 								label: ` | 								label: ` | ||||||
| 									<b>Start Poin</b><br>${obj.nopol1} ${obj.nopol2} ${obj.nopol3}<br> | 									<b>${(key == 0) ? 'Finish' : 'Start'} Poin</b><br>${obj.nopol1} ${obj.nopol2} ${obj.nopol3}<br> | ||||||
| 									${moment.unix(obj?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br> | 									${moment.unix(obj?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br> | ||||||
| 									Speed: ${(typeof obj.lst_speed != 'undefined') ? obj.lst_speed : '0'}<br> | 									Speed: ${(typeof obj.lst_speed != 'undefined') ? obj.lst_speed : '0'}<br> | ||||||
| 									${Number(obj.latitude).toFixed(5)},${Number(obj.longitude).toFixed(6)}<br> | 									${Number(obj.latitude).toFixed(5)},${Number(obj.longitude).toFixed(6)}<br> | ||||||
| @ -2775,6 +2838,10 @@ | |||||||
| 									// rotationAngle: 290 | 									// rotationAngle: 290 | ||||||
| 								} | 								} | ||||||
| 							}) | 							}) | ||||||
|  | 							allStartStop.push(marker) | ||||||
|  | 						} | ||||||
|  |  | ||||||
|  | 						const unix = parseInt(obj?.lst_loc_crt_d) + AppState.TIMEFIX; | ||||||
|  |  | ||||||
| 						return { | 						return { | ||||||
| 							lat: obj.latitude, | 							lat: obj.latitude, | ||||||
| @ -2789,84 +2856,73 @@ | |||||||
| 								radius: 5, | 								radius: 5, | ||||||
| 								markerOpacity: 0 | 								markerOpacity: 0 | ||||||
| 							}, | 							}, | ||||||
| 							label: `<b>${obj.nopol1} ${obj.nopol2} ${obj.nopol3}</b><br>${moment.unix(obj?.lst_loc_crt_d).format('DD MMM YYYY HH:mm:ss')}<br>${decodeURIComponent(obj?.fulladdress || 'address')}`, | 							label: `<b>${obj.nopol1} ${obj.nopol2} ${obj.nopol3}</b><br>${moment.unix(unix).format('DD MMM YYYY HH:mm:ss')}<br>${decodeURIComponent(obj?.fulladdress || 'address')}`, | ||||||
| 							// startLast : (key == 0) ? 0 : (key == (groupTrip.length - 1)) ? 1 : null, | 							// startLast : (key == 0) ? 0 : (key == (groupTrip.length - 1)) ? 1 : null, | ||||||
| 						} | 						} | ||||||
| 					})					 | 					})					 | ||||||
| 				}) | 				}) | ||||||
| 				// console.log("truckRoutes update", polyTruckRoutes); | 				// console.log("truckRoutes update", polyTruckRoutes); | ||||||
|  |  | ||||||
| 				polyTruckRoutes.forEach(async (poly, idx) => { | 				let circleCounter = 0 | ||||||
|  | 				for (const [idx, poly] of polyTruckRoutes.entries()) { | ||||||
| 					// console.log("poly", poly);					 | 					// console.log("poly", poly);					 | ||||||
| 					let polyline = Leaflet.addRoutes(poly) | 					function addCirclesAsync(polyCircle) { | ||||||
| 					// let polyline = Leaflet.addPolylines(poly) | 						// console.log("circleCounter", circleCounter); | ||||||
| 						 | 						 | ||||||
| 					// let circlesStartStop = [] | 						// return array of circles start end only | ||||||
| 					// Leaflet.addCircles(poly, function(circle, i) { |  | ||||||
| 					// 	if(i == 0 || i == (polyTruckRoutes.length - 1)) |  | ||||||
| 					// 		circlesStartStop.push(circle) |  | ||||||
|  |  | ||||||
| 					// 	window.addEventListener('eventRemoveRouteStartEnd', function handler(e) { |  | ||||||
| 					// 		circle.remove(); |  | ||||||
| 					// 	}); |  | ||||||
| 					// 	circle.on('click', function() { |  | ||||||
| 					// 		PgBar.syncToPlotTravelHistory(i); |  | ||||||
| 					// 	}) |  | ||||||
| 					// }); |  | ||||||
| 					function addCirclesAsync(poly) { |  | ||||||
| 						return new Promise(resolve => { | 						return new Promise(resolve => { | ||||||
| 							let circles = []; | 							let circles = []; | ||||||
|  |  | ||||||
| 							Leaflet.addCircles(poly, function(circle, i) { | 							Leaflet.addCircles(polyCircle, function(circle, i) { | ||||||
| 								if (i === 0 || i === (poly.length - 1)) { | 								// console.log("circle", i, circleCounter); | ||||||
|  | 								 | ||||||
|  | 								if (i === 0 || i === (polyCircle.length - 1)) { | ||||||
| 									circles.push(circle); | 									circles.push(circle); | ||||||
| 								} | 								} | ||||||
|  |  | ||||||
|  | 								const currentCounter = circleCounter; | ||||||
| 								circle.on('click', function() { | 								circle.on('click', function() { | ||||||
| 									PgBar.syncToPlotTravelHistory(i); | 									PgBar.syncToPlotTravelHistory(currentCounter); | ||||||
| 								}); | 								}); | ||||||
|  | 								circleCounter++ | ||||||
|  |  | ||||||
| 								// ✅ When last point processed, resolve | 								// ✅ When last point processed, resolve | ||||||
| 								if (i === poly.length - 1) { | 								if (i === polyCircle.length - 1) { | ||||||
| 									resolve(circles); | 									resolve(circles); | ||||||
| 								} | 								} | ||||||
| 							}); | 							}); | ||||||
| 						}); | 						}); | ||||||
| 					} | 					} | ||||||
|  |  | ||||||
| 					let circlesStartStop = []; | 					let polyline = await Leaflet.addRoutes(poly); | ||||||
| 					for (let [idx, poly] of polyTruckRoutes.entries()) { | 					// console.log("polyline", polyline); | ||||||
| 						let polyline = Leaflet.addRoutes(poly); | 					allStartStop.push(polyline) | ||||||
|  |  | ||||||
| 						// wait for all circles | 					// wait for all circles | ||||||
| 						circlesStartStop = await addCirclesAsync(poly); | 					const ssmarker = await addCirclesAsync(poly); | ||||||
|  | 					// console.log("ssmarker", ssmarker); | ||||||
|  | 					allStartStop.push(...ssmarker) | ||||||
|  | 				} | ||||||
| 					 | 					 | ||||||
| 						// keep event + removal logic here | 				console.log("allStartStop", allStartStop); | ||||||
| 						State.eventRemoveRouteStartEnd = new CustomEvent('eventRemoveRouteStartEnd', { | 				// console.log("circlesStartStop", circlesStartStop); | ||||||
| 							detail: { polyline, circlesStartStop } |  | ||||||
| 						}); |  | ||||||
| 				 | 				 | ||||||
| 						window.addEventListener('eventRemoveRouteStartEnd', function(e) { | 				// // remove marker, circle, event listener and all about this marker | ||||||
| 							polyline.remove(); | 				State.eventRemoveRouteStartEnd = new CustomEvent('eventRemoveRouteStartEnd', { | ||||||
| 							circlesStartStop.forEach(c => c.remove()); | 					// detail: { | ||||||
| 						}); | 						allStartStop, | ||||||
| 					} | 						// circlesStartStop | ||||||
|  | 					// } | ||||||
|  | 				}); | ||||||
|  |  | ||||||
| 					// // remove marker, circle, event listener and all about this marker | 				// window.addEventListener('eventRemoveRouteStartEnd', function (e) { | ||||||
| 					State.eventRemoveRouteStartEnd = new CustomEvent('eventRemoveRouteStartEnd', { | 				// 	// allStartStop.remove(); | ||||||
| 						detail: { | 				// 	allStartStop.forEach(c => c.remove()) | ||||||
| 							polyline, | 				// 	// circlesStartStop.forEach(c => c.remove()) | ||||||
| 							circlesStartStop | 				// }); | ||||||
| 						} |  | ||||||
| 					}); |  | ||||||
|  |  | ||||||
| 					window.addEventListener('eventRemoveRouteStartEnd', function (e) { |  | ||||||
| 						polyline.remove(); |  | ||||||
| 						circlesStartStop.forEach(c => c.remove()) |  | ||||||
| 					}); |  | ||||||
| 				}) |  | ||||||
|  |  | ||||||
|                 window.addEventListener('eventRemoveRouteStartEnd', function handler(e) { |                 window.addEventListener('eventRemoveRouteStartEnd', function handler(e) { | ||||||
|  | 					allStartStop.forEach(c => c.remove()) | ||||||
|                     e.currentTarget.removeEventListener(e.type, |                     e.currentTarget.removeEventListener(e.type, | ||||||
|                         handler); // window.removeEventListener('remove', this.handler, true); |                         handler); // window.removeEventListener('remove', this.handler, true); | ||||||
|                     State.eventRemoveRouteStartEnd = null; |                     State.eventRemoveRouteStartEnd = null; | ||||||
| @ -3436,6 +3492,8 @@ | |||||||
|                 } |                 } | ||||||
|             }, |             }, | ||||||
|             syncToPlotTravelHistory: function(arrIdx) { |             syncToPlotTravelHistory: function(arrIdx) { | ||||||
|  | 				console.log("syncToPlotTravelHistory", arrIdx); | ||||||
|  | 				 | ||||||
|                 const listPlotTravelHistory = document.querySelectorAll('#infoMove-plots .plotMove-item')[arrIdx]; |                 const listPlotTravelHistory = document.querySelectorAll('#infoMove-plots .plotMove-item')[arrIdx]; | ||||||
|                 listPlotTravelHistory.scrollIntoView(); |                 listPlotTravelHistory.scrollIntoView(); | ||||||
|                 listPlotTravelHistory.querySelector('li').classList.add('hover'); |                 listPlotTravelHistory.querySelector('li').classList.add('hover'); | ||||||
|  | |||||||
							
								
								
									
										347
									
								
								resources/views/menu_v1/reports/_trip_detail.blade.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										347
									
								
								resources/views/menu_v1/reports/_trip_detail.blade.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,347 @@ | |||||||
|  | <style> | ||||||
|  | 	#map { | ||||||
|  | 		height: 100%; | ||||||
|  | 		margin: 0; | ||||||
|  | 	} | ||||||
|  | 	.my-leaflet-map-container img { | ||||||
|  | 		max-height: none; | ||||||
|  | 	} | ||||||
|  | 	.dtl-text{ | ||||||
|  | 		font-size: 11px; | ||||||
|  | 	} | ||||||
|  | 	.head-text{ | ||||||
|  | 		font-size: 12px !important; | ||||||
|  | 	} | ||||||
|  | 	/* .leaflet-overlay-pane svg { | ||||||
|  | 		transform: none !important; | ||||||
|  | 	} */ | ||||||
|  | 	.leaflet-routing-container { | ||||||
|  | 		display: none !important; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	#viewPdf { | ||||||
|  | 		display: flex; | ||||||
|  | 		justify-content: center; | ||||||
|  | 		width: 794px; | ||||||
|  | 		/* height: auto; */ | ||||||
|  | 		max-width: 100%; /* Ensures it is responsive */ | ||||||
|  | 	} | ||||||
|  | 	/* .modal-dialog{ | ||||||
|  | 		width: 794px; | ||||||
|  | 	} */ | ||||||
|  |  | ||||||
|  | </style> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | <div class="modal-dialog modal-dialog modal-dialog-centered modal-dialog-scrollable modal-lg"> | ||||||
|  | 	<div class="modal-content"> | ||||||
|  | 		<div class="modal-header"> | ||||||
|  | 			<h5 class="modal-title" id="mdlDetailTripLabel">{{$nopol1}} Trip Detail</h5> | ||||||
|  | 			<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="modal-body"> | ||||||
|  | 			<div class="row head-text" id="viewPdf"> | ||||||
|  | 				<div class="col-12"> | ||||||
|  | 					<h4>{{$nopol1}}</h4> | ||||||
|  | 				</div> | ||||||
|  | 				<div class="col-4"> | ||||||
|  | 					<p class="text-bold mb-0">Start</p> | ||||||
|  | 					<p class="mb-0 time">{{ $start['time'] }}</p>	 | ||||||
|  | 					<p class="mb-0">Vehicle Mileage: {{number_format($start['mileage'], 2)}} km</p> | ||||||
|  | 					<p>{{$start['fulladdress']}}</p> | ||||||
|  | 				</div> | ||||||
|  | 				<div class="col-4"> | ||||||
|  | 					<p class="text-bold mb-0">Finish</p> | ||||||
|  | 					<p class="mb-0 time">{{ $finish['time'] }}</p>	 | ||||||
|  | 					<p class="mb-0">Vehicle Mileage: {{number_format($finish['mileage'], 2)}} km</p> | ||||||
|  | 					<p>{{$finish['fulladdress']}}</p> | ||||||
|  | 				</div> | ||||||
|  | 				<div class="col-2"> | ||||||
|  | 					<p class="text-bold mb-0">Distance</p> | ||||||
|  | 					<p class="mb-0">{{number_format($distance, 2)}} km</p>	 | ||||||
|  | 				</div> | ||||||
|  | 				<div class="col-2"> | ||||||
|  | 					<p class="text-bold mb-0">Duration</p> | ||||||
|  | 					<p class="mb-0">{{$duration}}</p>	 | ||||||
|  | 				</div> | ||||||
|  | 				<div class="col-12"> | ||||||
|  | 					<div id="leafMap" style="height: 400px;"></div> | ||||||
|  | 				</div> | ||||||
|  | 				<div class="col-12"> | ||||||
|  | 					<!-- <li class="list-group-item p-1 px-2"> | ||||||
|  | 						<p class="text-bold mb-0">Time: 25 Aug 2025 07:31:08</p> | ||||||
|  | 						<p class="text-muted mb-0 dtl-text">-8.55387 - 125.542409</p> | ||||||
|  | 						<p class="text-muted mb-0 dtl-text">Avenida Luro Mata, Praia dos Coqueiros, Bebunuk, Dom Aleixo, Dili, Timor-Leste;2066973</p> | ||||||
|  | 						<p class="mb-0 dtl-text">Current speed: 7km/h</p> | ||||||
|  | 					</li> --> | ||||||
|  | 					@foreach ($list as $item) | ||||||
|  | 						<!-- <li class="list-group-item p-1 px-2"> | ||||||
|  | 							<p class="text-bold mb-0">Time: {{date('d-m-Y H:i:s', $item->crt_d)}}</p> | ||||||
|  | 							<p class="text-muted mb-0 dtl-text">Vehicle Mileage: {{number_format($item->vhc_milleage, 2)}} km</p> | ||||||
|  | 							<p class="text-muted mb-0 dtl-text">{{number_format($item->latitude, 6)}} - {{number_format($item->longitude, 6)}}</p> | ||||||
|  | 							<p class="text-muted mb-0 dtl-text">{{urldecode($item->fulladdress)}}</p> | ||||||
|  | 							<p class="text-muted mb-0 dtl-text">Current speed: {{number_format($item->speed, 2)}} km/h</p> | ||||||
|  | 						</li> --> | ||||||
|  | 						<li class="list-group-item p-1 px-2"> | ||||||
|  | 							<div class="row"> | ||||||
|  | 								<div class="col-4"> | ||||||
|  | 									<p class="text-bold mb-0 dtl-text">Time: <span class="time">{{ $item->crt_d }}</span></p> | ||||||
|  | 									<p class="text-muted mb-0 dtl-text">Vehicle Mileage: {{number_format($item->vhc_milleage, 2)}} km</p> | ||||||
|  | 									<p class="text-muted mb-0 dtl-text">Current speed: {{$item->speed}} km/h</p> | ||||||
|  | 								</div> | ||||||
|  | 								<div class="col-8"> | ||||||
|  | 									<p class="text-muted mb-0 dtl-text">{{number_format($item->latitude, 6)}}, {{number_format($item->longitude, 6)}}</p> | ||||||
|  | 									<p class="text-muted mb-0 dtl-text">{{urldecode($item->fulladdress)}}</p> | ||||||
|  | 								</div> | ||||||
|  | 							</div> | ||||||
|  | 						</li>	 | ||||||
|  | 					@endforeach | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="modal-footer"> | ||||||
|  | 			<button class="btn btn-sm btn-danger ms-auto" id="btnDownloadReport">Download Report</button> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | <script> | ||||||
|  | 	$(document).ready(function(){ | ||||||
|  | 		$('.time').each(function () { | ||||||
|  | 			const unix = parseInt($(this).text().trim()) + AppState.TIMEFIX; | ||||||
|  | 			$(this).text(moment.unix(unix).format('DD MMM YYYY HH:mm:ss')); | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		let coords | ||||||
|  | 		setTimeout(async() => { | ||||||
|  | 			map.invalidateSize();            // force Leaflet to recalc | ||||||
|  | 			map.fitBounds(polyline.getBounds()); | ||||||
|  | 			// map.fitBounds(await coords.map(c => L.latLng(c[0], c[1]))); | ||||||
|  | 		}, 200); | ||||||
|  |  | ||||||
|  | 		const linesData = (@json($list)); | ||||||
|  |  | ||||||
|  | 		// 1) Initialize map | ||||||
|  | 		const map = L.map("leafMap").setView([-8.90507, 125.9945732], 10) | ||||||
|  |  | ||||||
|  | 		// 2) Add tile layer | ||||||
|  | 		L.tileLayer("https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png", { | ||||||
|  | 			maxZoom: 20, | ||||||
|  | 			crossOrigin: true | ||||||
|  | 		}).addTo(map); | ||||||
|  |  | ||||||
|  | 		// // // 3) Coordinates (Lat, Lng) for polyline | ||||||
|  | 		const points = linesData | ||||||
|  | 			.filter(p => p.latitude && p.longitude) | ||||||
|  | 			.map((point) => [point.latitude, point.longitude]) | ||||||
|  |  | ||||||
|  | 		// 4) Add polyline | ||||||
|  | 		const polyline = L.polyline(points,{ | ||||||
|  | 			color: 'red', | ||||||
|  | 			weight: 3, | ||||||
|  | 			opacity: 0.7, | ||||||
|  | 			smoothFactor: 1 | ||||||
|  | 		}) | ||||||
|  | 		// .addTo(map); | ||||||
|  |  | ||||||
|  | 		// const lines = L.Routing.control({ | ||||||
|  | 		// 	waypoints: points, | ||||||
|  | 		// 	// router: L.Routing.osrmv1(), | ||||||
|  | 		// 	router: L.Routing.osrmv1({ | ||||||
|  | 		// 		serviceUrl: "https://brilianapps.britimorleste.tl:5001/route/v1", | ||||||
|  | 		// 	}), | ||||||
|  | 		// 	show: false, | ||||||
|  | 		// 	itinerary: null, // 👈 completely removes the panel | ||||||
|  | 		// 	addWaypoints: false, // ❌ prevent adding points by clicking | ||||||
|  | 		// 	draggableWaypoints: false, // ❌ prevent dragging markers | ||||||
|  | 		// 	routeWhileDragging: false, // optional: don’t reroute while dragging | ||||||
|  | 		// 	createMarker: () => null, | ||||||
|  | 		// 	routingOptions: { | ||||||
|  | 		// 	radiuses: [100, 100] // tolerance in meters per waypoint | ||||||
|  | 		// 	}, | ||||||
|  | 		// 	lineOptions:{ | ||||||
|  | 		// 		styles: [{ color: "#C0392B", weight: 4, opacity: 0.7 }], | ||||||
|  | 		// 	}, | ||||||
|  | 		// }).addTo(map) | ||||||
|  |  | ||||||
|  | 		function chunkArray(arr, size) { | ||||||
|  | 			const result = []; | ||||||
|  | 			for (let i = 0; i < arr.length; i += size) { | ||||||
|  | 				result.push(arr.slice(i, i + size)); | ||||||
|  | 			} | ||||||
|  | 			return result; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		function fetchOsrm(points) { | ||||||
|  | 			const coords = points; | ||||||
|  | 			const hints = ";".repeat(points.length - 1); | ||||||
|  |  | ||||||
|  | 			const body = { | ||||||
|  | 				coordinates: coords, | ||||||
|  | 				overview: "false", | ||||||
|  | 				alternatives: "false", | ||||||
|  | 				steps: "true", | ||||||
|  | 				hints: hints | ||||||
|  | 			}; | ||||||
|  |  | ||||||
|  | 			let config = { | ||||||
|  | 				method: 'post', | ||||||
|  | 				maxBodyLength: Infinity, | ||||||
|  | 				url: 'https://brilianapps.britimorleste.tl/osrm-backend/post-route/v1/driving/', | ||||||
|  | 				headers: { | ||||||
|  | 					'Content-Type': 'application/json' | ||||||
|  | 				}, | ||||||
|  | 				data: body | ||||||
|  | 			}; | ||||||
|  |  | ||||||
|  | 			return axios.request(config) | ||||||
|  | 				.then((response) => { | ||||||
|  | 					return response.data; | ||||||
|  | 				}) | ||||||
|  | 				.catch((error) => { | ||||||
|  | 					console.error("Error:", error.message); | ||||||
|  | 					return null; | ||||||
|  | 				}); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		function decodeOSRMGeometry(encoded) { | ||||||
|  | 			const coordinates = []; | ||||||
|  | 			let index = 0, | ||||||
|  | 				lat = 0, | ||||||
|  | 				lng = 0; | ||||||
|  |  | ||||||
|  | 			while (index < encoded.length) { | ||||||
|  | 				let result = 1, | ||||||
|  | 					shift = 0, | ||||||
|  | 					b; | ||||||
|  | 				do { | ||||||
|  | 					b = encoded.charCodeAt(index++) - 63 - 1; | ||||||
|  | 					result += b << shift; | ||||||
|  | 					shift += 5; | ||||||
|  | 				} while (b >= 0x1f); | ||||||
|  | 				lat += (result & 1 ? ~(result >> 1) : result >> 1); | ||||||
|  |  | ||||||
|  | 				result = 1; | ||||||
|  | 				shift = 0; | ||||||
|  | 				do { | ||||||
|  | 					b = encoded.charCodeAt(index++) - 63 - 1; | ||||||
|  | 					result += b << shift; | ||||||
|  | 					shift += 5; | ||||||
|  | 				} while (b >= 0x1f); | ||||||
|  | 				lng += (result & 1 ? ~(result >> 1) : result >> 1); | ||||||
|  |  | ||||||
|  | 				coordinates.push([lat / 1e5, lng / 1e5]); | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			return coordinates; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		async function getCoordinates(points) { | ||||||
|  | 			const chunkSize = 500; | ||||||
|  | 			const chunks = chunkArray(points, chunkSize); // Split the points array into chunks of 500 | ||||||
|  | 			let allCoords = []; | ||||||
|  |  | ||||||
|  | 			for (const chunk of chunks) { | ||||||
|  | 				const osrm = await fetchOsrm(chunk); // Fetch OSRM data for each chunk | ||||||
|  |  | ||||||
|  | 				if (!osrm) { | ||||||
|  | 					console.log("OSRM failed for chunk"); | ||||||
|  | 					return; | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				const coords = osrm.routes[0].legs.flatMap(leg => | ||||||
|  | 					leg.steps.flatMap(step => | ||||||
|  | 						decodeOSRMGeometry(step.geometry) | ||||||
|  | 					) | ||||||
|  | 				); | ||||||
|  |  | ||||||
|  | 				allCoords = allCoords.concat(coords); // Combine the result | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			// Now add the polyline to the map | ||||||
|  | 			L.polyline(allCoords, { | ||||||
|  | 				color: "#2980B9", | ||||||
|  | 				weight: 3, | ||||||
|  | 				opacity: 0.8 | ||||||
|  | 			}).addTo(map); | ||||||
|  | 			// map.fitBounds(allCoords.map(c => L.latLng(c[0], c[1]))); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Usage: Pass the array of coordinates to the function | ||||||
|  | 		getCoordinates(points); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 		// start and finish point | ||||||
|  | 		const startIcon = L.icon({ | ||||||
|  | 			iconUrl: "{{ asset('images/start.png') }}", | ||||||
|  | 			iconSize: [30, 30], | ||||||
|  | 			iconAnchor: [15, 28], // lb, rt, bottom, rb. Positive | ||||||
|  | 		}) | ||||||
|  | 		L.marker(points[0], {icon: startIcon}).addTo(map) | ||||||
|  |  | ||||||
|  | 		const finishIcon = L.icon({ | ||||||
|  | 			iconUrl: "{{ asset('images/finish.png') }}", | ||||||
|  | 			iconSize: [30, 30], | ||||||
|  | 			iconAnchor: [15, 28], // lb, rt, bottom, rb. Positive | ||||||
|  | 		}) | ||||||
|  | 		L.marker(points[points.length - 1], {icon: finishIcon}).addTo(map) | ||||||
|  |  | ||||||
|  | 		// // 5) Auto-fit map to polyline bounds | ||||||
|  | 		// map.fitBounds(polyline.getBounds()) | ||||||
|  |  | ||||||
|  | 		// download pdf | ||||||
|  | 		window._downloadReportBound ||= ( | ||||||
|  | 			$(document).on('click', '#btnDownloadReport', function () { | ||||||
|  | 				$('#viewPdf').printThis({ | ||||||
|  | 					debug: false,               // show the iframe for debugging | ||||||
|  | 					importCSS: true,            // copy linked styles | ||||||
|  | 					importStyle: true,          // copy inline styles | ||||||
|  | 				}); | ||||||
|  |  | ||||||
|  | 				// const viewPdf = document.getElementById("viewPdf"); | ||||||
|  |  | ||||||
|  | 				// // find overlay svg (the one holding polylines) | ||||||
|  | 				// const overlaySvg = document.querySelector('.leaflet-overlay-pane svg'); | ||||||
|  | 				// let originalTransform = ''; | ||||||
|  | 				// if (overlaySvg) { | ||||||
|  | 				// 	originalTransform = overlaySvg.style.transform; | ||||||
|  | 				// 	overlaySvg.style.transform = 'none'; | ||||||
|  | 				// } | ||||||
|  | 			 | ||||||
|  | 				// html2canvas(viewPdf, {  | ||||||
|  | 				// 	scale: 2,  | ||||||
|  | 				// 	useCORS: true,  | ||||||
|  | 				// 	logging: true | ||||||
|  | 				// }).then(canvas => { | ||||||
|  | 				// 	const imgData = canvas.toDataURL('image/png'); | ||||||
|  | 				// 	const { jsPDF } = window.jspdf; | ||||||
|  |  | ||||||
|  | 				// 	const pdf = new jsPDF('p', 'mm', 'a4'); | ||||||
|  | 				// 	const pageWidth = pdf.internal.pageSize.getWidth(); | ||||||
|  | 				// 	const pageHeight = pdf.internal.pageSize.getHeight(); | ||||||
|  |  | ||||||
|  | 				// 	const imgWidth = pageWidth - 20; // margin | ||||||
|  | 				// 	const imgHeight = canvas.height * imgWidth / canvas.width; | ||||||
|  |  | ||||||
|  | 				// 	let position = 10; | ||||||
|  |  | ||||||
|  | 				// 	// 👉 Handle multipage content | ||||||
|  | 				// 	let heightLeft = imgHeight; | ||||||
|  | 				// 	while (heightLeft > 0) { | ||||||
|  | 				// 		pdf.addImage(imgData, 'PNG', 10, position, imgWidth, imgHeight); | ||||||
|  | 				// 		heightLeft -= pageHeight; | ||||||
|  | 				// 		if (heightLeft > 0) { | ||||||
|  | 				// 			pdf.addPage(); | ||||||
|  | 				// 			position = 0; | ||||||
|  | 				// 		} | ||||||
|  | 				// 	} | ||||||
|  | 			 | ||||||
|  | 				// 	pdf.save(`{{$nopol1}} Trip Report {{$start['time']}}.pdf`); | ||||||
|  |  | ||||||
|  | 				// }); | ||||||
|  | 			}), | ||||||
|  | 			true | ||||||
|  | 		); | ||||||
|  |  | ||||||
|  | 	}); | ||||||
|  | </script> | ||||||
| @ -43,7 +43,7 @@ | |||||||
| 							<div class="form-group"> | 							<div class="form-group"> | ||||||
| 								<label class="text-muted">From</label> | 								<label class="text-muted">From</label> | ||||||
| 								<!-- default today --> | 								<!-- default today --> | ||||||
| 								<!-- <input type="date" class="form-control" id="tgl0" value="{{ date('Y-m-d') }}"> --> | 								<!-- <input class="form-control" id="tgl0" value="20-08-2025 00:00"> --> | ||||||
| 								<input class="form-control" id="tgl0" value="{{ date('d-m-Y 00:00') }}"> | 								<input class="form-control" id="tgl0" value="{{ date('d-m-Y 00:00') }}"> | ||||||
| 							</div> | 							</div> | ||||||
| 						</div> | 						</div> | ||||||
| @ -80,6 +80,8 @@ | |||||||
|                                     <!-- <th class="text-center">Action</th> --> |                                     <!-- <th class="text-center">Action</th> --> | ||||||
|                                     <!-- <th class="">Vehicle Name</th> --> |                                     <!-- <th class="">Vehicle Name</th> --> | ||||||
|                                     <th class="">License Plate Number</th> |                                     <th class="">License Plate Number</th> | ||||||
|  | 									<th class="">Dist. Cat.</th> | ||||||
|  | 									<th class="">Pool</th> | ||||||
|                                     <th class="">Time</th> |                                     <th class="">Time</th> | ||||||
|                                     <th class="">Speed (kph)</th> |                                     <th class="">Speed (kph)</th> | ||||||
|                                     <th class="">Location</th> |                                     <th class="">Location</th> | ||||||
| @ -200,11 +202,19 @@ | |||||||
| 							data: 'nopol1', | 							data: 'nopol1', | ||||||
| 							className: 'text-start', | 							className: 'text-start', | ||||||
| 						}, | 						}, | ||||||
|  | 						{ | ||||||
|  | 							data: 'dc_code', | ||||||
|  | 							className: 'text-start', | ||||||
|  | 						}, | ||||||
|  | 						{ | ||||||
|  | 							data: 'pool_code', | ||||||
|  | 							className: 'text-start', | ||||||
|  | 						}, | ||||||
| 						{  | 						{  | ||||||
| 							data: "crt_d", | 							data: "crt_d", | ||||||
| 							className: 'text-nowrap', | 							className: 'text-nowrap', | ||||||
| 							render: (data, type, row, meta) => { | 							render: (data, type, row, meta) => { | ||||||
| 								return moment.unix(data).format('DD MMM YYYY HH:mm:ss'); | 								return moment.unix(data + AppState.TIMEFIX).format('DD MMM YYYY HH:mm:ss'); | ||||||
| 							} | 							} | ||||||
| 						}, | 						}, | ||||||
| 						{ data: "speed", className: 'text-end'}, | 						{ data: "speed", className: 'text-end'}, | ||||||
|  | |||||||
| @ -5,10 +5,16 @@ | |||||||
| @extends('app.app') | @extends('app.app') | ||||||
|  |  | ||||||
| @section('title') | @section('title') | ||||||
|     Vehicles |     Vehicles Trip Report | ||||||
| @endsection | @endsection | ||||||
|  |  | ||||||
| @section('customcss') | @section('customcss') | ||||||
|  | 	<link | ||||||
|  | 		rel="stylesheet" | ||||||
|  | 		href="https://unpkg.com/leaflet@1.9.4/dist/leaflet.css" | ||||||
|  | 		integrity="sha256-p4NxAoJBhIIN+hmNHrzRCf9tD/miZyoHS5obTRR9BMY=" | ||||||
|  | 		crossorigin="" | ||||||
|  | 	/> | ||||||
| @endsection | @endsection | ||||||
|  |  | ||||||
| @section('content') | @section('content') | ||||||
| @ -43,13 +49,14 @@ | |||||||
| 							<div class="form-group"> | 							<div class="form-group"> | ||||||
| 								<label class="text-muted">From</label> | 								<label class="text-muted">From</label> | ||||||
| 								<!-- default today --> | 								<!-- default today --> | ||||||
| 								<!-- <input class="form-control" id="tgl0" value="25-08-2025 00:00"> --> | 								<!-- <input class="form-control" id="tgl0" value="02-09-2025 00:00"> --> | ||||||
| 								<input class="form-control" id="tgl0" value="{{ date('d-m-Y 00:00') }}"> | 								<input class="form-control" id="tgl0" value="{{ date('d-m-Y 00:00') }}"> | ||||||
| 							</div> | 							</div> | ||||||
| 						</div> | 						</div> | ||||||
| 						<div class="col-2"> | 						<div class="col-2"> | ||||||
| 							<div class="form-group"> | 							<div class="form-group"> | ||||||
| 								<label class="text-muted">To</label> | 								<label class="text-muted">To</label> | ||||||
|  | 								<!-- <input class="form-control" id="tgl1" value="02-09-2025 23:00"> --> | ||||||
| 								<input class="form-control" id="tgl1" value="{{ date('d-m-Y 23:59') }}"> | 								<input class="form-control" id="tgl1" value="{{ date('d-m-Y 23:59') }}"> | ||||||
| 							</div> | 							</div> | ||||||
| 						</div> | 						</div> | ||||||
| @ -76,14 +83,20 @@ | |||||||
|                         <table id="tVehicleTrips" class="table table-hover dataTable w-100"> |                         <table id="tVehicleTrips" class="table table-hover dataTable w-100"> | ||||||
|                             <thead> |                             <thead> | ||||||
|                                 <tr class=""> |                                 <tr class=""> | ||||||
|                                     <th class="">Vehicle ID</th> |                                     <th class="" hidden>Vehicle ID</th> | ||||||
|                                     <th class="">License Plate Number</th> |                                     <th class="">License Plate Number</th> | ||||||
|  | 									<th class="">Dist. Cat.</th> | ||||||
|  | 									<th class="">Pool</th> | ||||||
|                                     <th class="">Number of Trip</th> |                                     <th class="">Number of Trip</th> | ||||||
|                                     <th class="">Total Mileage (km)</th> |                                     <th class="">Total Mileage (km)</th> | ||||||
|                                     <th class="">Trip #</th> |                                     <th class="">Trip #</th> | ||||||
|                                     <th class="">Start</th> |                                     <th class="">Start</th> | ||||||
|                                     <th class="">Finish</th> |                                     <th class="">Finish</th> | ||||||
|                                     <th class="">Mileage (km)</th> | 									<th class="">Duration</th> | ||||||
|  |                                     <th class="">Start (km)</th> | ||||||
|  |                                     <th class="">Finish (km)</th> | ||||||
|  |                                     <th class="">Distance (km)</th> | ||||||
|  |                                     <th class="">Detail</th> | ||||||
|                                 </tr> |                                 </tr> | ||||||
|                             </thead> |                             </thead> | ||||||
|                             <!-- <tbody> |                             <!-- <tbody> | ||||||
| @ -162,6 +175,20 @@ | |||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
| 	<!-- modal --> | 	<!-- modal --> | ||||||
|  |     <div class="modal fade" data-bs-backdrop="static" data-bs-keyboard="false" id="mdlDetailTrip" aria-labelledby="mdlDetailTripLabel" aria-hidden="true"> | ||||||
|  |         <!-- <div class="modal-dialog modal-dialog modal-dialog-centered modal-dialog-scrollable modal-xl"> | ||||||
|  |             <div class="modal-content"> | ||||||
|  |                 <div class="modal-header"> | ||||||
|  |                     <h5 class="modal-title" id="mdlDetailTripLabel">-----</h5> | ||||||
|  |                     <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="modal-body"> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="modal-footer"> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |         </div> --> | ||||||
|  |     </div> | ||||||
| @endsection | @endsection | ||||||
|  |  | ||||||
| @section('customjs') | @section('customjs') | ||||||
| @ -170,6 +197,14 @@ | |||||||
| 	<script src="https://cdn.datatables.net/buttons/2.4.2/js/dataTables.buttons.min.js"></script> | 	<script src="https://cdn.datatables.net/buttons/2.4.2/js/dataTables.buttons.min.js"></script> | ||||||
| 	<script src="https://cdnjs.cloudflare.com/ajax/libs/jszip/3.10.1/jszip.min.js"></script> | 	<script src="https://cdnjs.cloudflare.com/ajax/libs/jszip/3.10.1/jszip.min.js"></script> | ||||||
| 	<script src="https://cdn.datatables.net/buttons/2.4.2/js/buttons.html5.min.js"></script> | 	<script src="https://cdn.datatables.net/buttons/2.4.2/js/buttons.html5.min.js"></script> | ||||||
|  |     <!-- <script src="https://cdnjs.cloudflare.com/ajax/libs/leaflet.draw/1.0.4/leaflet.draw.js" integrity="sha512-ozq8xQKq6urvuU6jNgkfqAmT7jKN2XumbrX1JiB3TnF7tI48DPI4Gy1GXKD/V3EExgAs1V+pRO7vwtS1LHg0Gw==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> --> | ||||||
|  | 	<script src="https://unpkg.com/leaflet@1.9.4/dist/leaflet.js" integrity="sha256-20nQCchB9co0qIjJZRGuk2/Z9VM+kNiyxNV1lvTlZBo=" crossorigin=""></script> | ||||||
|  |     <script src="https://unpkg.com/leaflet-routing-machine@3.2.12/dist/leaflet-routing-machine.js"></script> | ||||||
|  | 	<script src="https://cdnjs.cloudflare.com/ajax/libs/html2canvas/1.4.1/html2canvas.min.js"></script> | ||||||
|  | 	<script src="https://cdnjs.cloudflare.com/ajax/libs/jspdf/2.5.1/jspdf.umd.min.js"></script> | ||||||
|  | 	<script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script> | ||||||
|  | 	<script src="{{ asset('assets/vendor/printThis.js') }}"></script> | ||||||
|  |  | ||||||
| 	<script> | 	<script> | ||||||
|         'use strict'; |         'use strict'; | ||||||
|         const State = { |         const State = { | ||||||
| @ -185,6 +220,7 @@ | |||||||
|             activate: function() { |             activate: function() { | ||||||
|                 Wrapper.init(); |                 Wrapper.init(); | ||||||
|                 Wrapper.event(); |                 Wrapper.event(); | ||||||
|  | 				// DTable.table = $('#tVehicleTrips').DataTable() | ||||||
|                 // DTable.activate(); |                 // DTable.activate(); | ||||||
| 			}, | 			}, | ||||||
| 			init: ()=>{ | 			init: ()=>{ | ||||||
| @ -204,6 +240,21 @@ | |||||||
| 				$('#btnDownloadReport').on('click', function() { | 				$('#btnDownloadReport').on('click', function() { | ||||||
| 					DTable.table.button('.buttons-excel').trigger(); | 					DTable.table.button('.buttons-excel').trigger(); | ||||||
| 				}); | 				}); | ||||||
|  |  | ||||||
|  | 				// detail trip | ||||||
|  | 				$('#tVehicleTrips').on('click', '.btnDetailTrip', function(e) { | ||||||
|  | 					e.preventDefault(); | ||||||
|  | 					const vid = $(this).data('vid'); | ||||||
|  | 					const tgl0 = $(this).data('tgl0'); | ||||||
|  | 					const tgl1 = $(this).data('tgl1'); | ||||||
|  | 					const nopol1 = $(this).data('nopol1'); | ||||||
|  |  | ||||||
|  | 					// $("#mdlDetailTrip").load("{{ route('view_report_trip_detail') }}"); | ||||||
|  | 					$('#mdlDetailTrip').empty().load("{{ route('view_report_trip_detail') }}", `vid=${vid}&tgl0=${tgl0}&tgl1=${tgl1}&nopol1=${nopol1}`, () => { | ||||||
|  | 						$('#mdlDetailTrip').modal('show') | ||||||
|  | 					}) | ||||||
|  | 					 | ||||||
|  | 				}); | ||||||
|             }, |             }, | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
| @ -244,7 +295,8 @@ | |||||||
| 								cptid=${AppState.current_company} | 								cptid=${AppState.current_company} | ||||||
| 								&from_date=${moment($('#tgl0').val(), "DD-MM-YYYY HH:mm").unix()} | 								&from_date=${moment($('#tgl0').val(), "DD-MM-YYYY HH:mm").unix()} | ||||||
| 								&to_date=${moment($('#tgl1').val(), "DD-MM-YYYY HH:mm").unix()} | 								&to_date=${moment($('#tgl1').val(), "DD-MM-YYYY HH:mm").unix()} | ||||||
| 								&vid=${$('#filterNopol').val() || ''}`, | 								&vid=${$('#filterNopol').val() || ''} | ||||||
|  | 							`, | ||||||
| 							type: 'GET', | 							type: 'GET', | ||||||
| 							success: function(json) { | 							success: function(json) { | ||||||
| 								callback(json); | 								callback(json); | ||||||
| @ -269,7 +321,9 @@ | |||||||
| 							orderable: true, | 							orderable: true, | ||||||
| 							searchable: true, | 							searchable: true, | ||||||
| 						}, | 						}, | ||||||
|  						{ data: "total_trip", className: 'text-end'}, | 						{ data: "dc_code", className: 'text-start'}, | ||||||
|  | 						{ data: "pool_code", className: 'text-start'}, | ||||||
|  | 						{ data: "total_trip", className: 'text-end'}, | ||||||
| 						{ data: "total_mileage", className: 'text-end', render: (data, type, row, meta) => parseFloat(data).toFixed(2)}, | 						{ data: "total_mileage", className: 'text-end', render: (data, type, row, meta) => parseFloat(data).toFixed(2)}, | ||||||
| 						{ data: "trip_id", className: 'text-end'}, | 						{ data: "trip_id", className: 'text-end'}, | ||||||
| 						{  | 						{  | ||||||
| @ -277,7 +331,7 @@ | |||||||
| 							render: (data, type, row, meta) => { | 							render: (data, type, row, meta) => { | ||||||
| 								// let addr = row | 								// let addr = row | ||||||
| 								return ` | 								return ` | ||||||
| 									${moment.unix(data).format('DD MMM YYYY HH:mm:ss')}<br> | 									${moment.unix(data + AppState.TIMEFIX).format('DD MMM YYYY HH:mm:ss')}<br> | ||||||
| 									${Helper.shortenText(decodeURIComponent(row.startLoc || 'address'), 255) || "-"} | 									${Helper.shortenText(decodeURIComponent(row.startLoc || 'address'), 255) || "-"} | ||||||
| 								`; | 								`; | ||||||
| 							}  | 							}  | ||||||
| @ -287,17 +341,57 @@ | |||||||
| 							render: (data, type, row, meta) => { | 							render: (data, type, row, meta) => { | ||||||
| 								// let addr = row | 								// let addr = row | ||||||
| 								return ` | 								return ` | ||||||
| 									${moment.unix(data).format('DD MMM YYYY HH:mm:ss')}<br> | 									${moment.unix(data + AppState.TIMEFIX).format('DD MMM YYYY HH:mm:ss')}<br> | ||||||
| 									${Helper.shortenText(decodeURIComponent(row.finishLoc || 'address'), 255) || "-"} | 									${Helper.shortenText(decodeURIComponent(row.finishLoc || 'address'), 255) || "-"} | ||||||
| 								`; | 								`; | ||||||
| 							}  | 							}  | ||||||
| 						}, | 						}, | ||||||
| 						{  | 						{  | ||||||
| 							data: 'mileage',  | 							className: 'text-start', render: function(data, type, row, meta) { | ||||||
| 							className: 'text-end', render: function(data, type, row, meta) { | 								const start = moment.unix(row.start); | ||||||
| 								return (data === null) ? '0' : parseFloat(data).toFixed(2); | 								const finish = moment.unix(row.finish); | ||||||
|  | 		 | ||||||
|  | 								// calculate duration | ||||||
|  | 								const diff = moment.duration(finish.diff(start)); | ||||||
|  | 								const hours = Math.floor(diff.asHours()); | ||||||
|  | 								const minutes = diff.minutes(); | ||||||
|  |  | ||||||
|  | 								return `${hours}h ${minutes}m`;							 | ||||||
| 							} | 							} | ||||||
| 						}, | 						}, | ||||||
|  | 						{  | ||||||
|  | 							data: 'startMileage',  | ||||||
|  | 							className: 'text-end', render: function(data, type, row, meta) { | ||||||
|  | 								return (data === null) ? '0' : parseFloat(data).toFixed(0); | ||||||
|  | 							} | ||||||
|  | 						}, | ||||||
|  | 						{  | ||||||
|  | 							data: 'finishMileage',  | ||||||
|  | 							className: 'text-end', render: function(data, type, row, meta) { | ||||||
|  | 								return (data === null) ? '0' : parseFloat(data).toFixed(0); | ||||||
|  | 							} | ||||||
|  | 						}, | ||||||
|  | 						{  | ||||||
|  | 							data: 'mileage',  | ||||||
|  | 							className: 'text-end', render: function(data, type, row, meta) { | ||||||
|  | 								const mileage = (row.finishMileage !== null && row.startMileage !== null) ? (row.finishMileage - row.startMileage) : row.mileage; | ||||||
|  | 								return (parseFloat(mileage).toFixed(2)); | ||||||
|  | 							} | ||||||
|  | 						}, | ||||||
|  |                         { | ||||||
|  | 							// visible: false, | ||||||
|  |                             data: 'vhc_id', | ||||||
|  |                             className: 'text-center', | ||||||
|  |                             render: function(data, type, row, meta) { | ||||||
|  |                                 let action = ` | ||||||
|  | 									<a href="#" class="text-decoration-none me-1 btnDetailTrip"  | ||||||
|  | 										data-vid="${data}" data-tgl0="${row.start}" data-tgl1="${row.finish}" data-nopol1="${row.nopol1}"> | ||||||
|  | 										<span class="icon ion-eye fz-16"></span> | ||||||
|  | 									</a> | ||||||
|  | 								`; | ||||||
|  |                                 return action; | ||||||
|  |                             } | ||||||
|  |                         }, | ||||||
| 					], | 					], | ||||||
| 					paging: false, | 					paging: false, | ||||||
| 					drawCallback: function (settings) { | 					drawCallback: function (settings) { | ||||||
| @ -321,6 +415,9 @@ | |||||||
| 									$(rows).eq(i).find('td:eq(0)').attr('rowspan', rowspanCount); | 									$(rows).eq(i).find('td:eq(0)').attr('rowspan', rowspanCount); | ||||||
| 									$(rows).eq(i).find('td:eq(1)').attr('rowspan', rowspanCount); | 									$(rows).eq(i).find('td:eq(1)').attr('rowspan', rowspanCount); | ||||||
| 									$(rows).eq(i).find('td:eq(2)').attr('rowspan', rowspanCount); | 									$(rows).eq(i).find('td:eq(2)').attr('rowspan', rowspanCount); | ||||||
|  | 									$(rows).eq(i).find('td:eq(3)').attr('rowspan', rowspanCount); | ||||||
|  | 									$(rows).eq(i).find('td:eq(4)').attr('rowspan', rowspanCount); | ||||||
|  | 									 | ||||||
| 									// $(rows).eq(i).find('td:eq(3)').attr('rowspan', rowspanCount); | 									// $(rows).eq(i).find('td:eq(3)').attr('rowspan', rowspanCount); | ||||||
|  |  | ||||||
| 									last = group; | 									last = group; | ||||||
| @ -329,6 +426,8 @@ | |||||||
| 									$(rows).eq(i).find('td:eq(0)').remove(); | 									$(rows).eq(i).find('td:eq(0)').remove(); | ||||||
| 									$(rows).eq(i).find('td:eq(0)').remove(); | 									$(rows).eq(i).find('td:eq(0)').remove(); | ||||||
| 									$(rows).eq(i).find('td:eq(0)').remove(); | 									$(rows).eq(i).find('td:eq(0)').remove(); | ||||||
|  | 									$(rows).eq(i).find('td:eq(0)').remove(); | ||||||
|  | 									$(rows).eq(i).find('td:eq(0)').remove(); | ||||||
| 									// $(rows).eq(i).find('td:eq(0)').remove(); | 									// $(rows).eq(i).find('td:eq(0)').remove(); | ||||||
| 								} | 								} | ||||||
| 							}); | 							}); | ||||||
| @ -345,7 +444,7 @@ | |||||||
| 							}, | 							}, | ||||||
| 							className: 'd-none', // hide default button | 							className: 'd-none', // hide default button | ||||||
| 							exportOptions: { | 							exportOptions: { | ||||||
| 								columns: ':visible' // 🔹 exclude first column | 								columns: ':visible:not(:last-child)' //  | ||||||
| 							} | 							} | ||||||
| 						} | 						} | ||||||
| 					] | 					] | ||||||
| @ -358,16 +457,6 @@ | |||||||
|             return (val === undefined || val === null || val === '') ? null : val; |             return (val === undefined || val === null || val === '') ? null : val; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         const Filter = { |  | ||||||
|             activate: function() { |  | ||||||
|                 Filter.event(); |  | ||||||
|             }, |  | ||||||
|             event: function() {}, |  | ||||||
|             triggerFilterCompany: function() { |  | ||||||
|                 DTable.reload(); |  | ||||||
|             }, |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         async function convertImgHtmlToFile(imgHtml, imgWidth, imgHeight, mimeType, fileName) { |         async function convertImgHtmlToFile(imgHtml, imgWidth, imgHeight, mimeType, fileName) { | ||||||
|             return new Promise((resolve, reject) => { |             return new Promise((resolve, reject) => { | ||||||
|                 try { |                 try { | ||||||
|  | |||||||
| @ -77,7 +77,19 @@ | |||||||
|                         @foreach ($permission as $module => $perms) |                         @foreach ($permission as $module => $perms) | ||||||
|                             <div class="mb-3 row"> |                             <div class="mb-3 row"> | ||||||
|                                 <label for="add-first_name" class="form-label text-capitalize col-3"> |                                 <label for="add-first_name" class="form-label text-capitalize col-3"> | ||||||
|                                     {{ $module == 'transaction' ? 'job' : ($module == 'client' ? 'company' : ($module == 'config_truck_type' ? 'Config Vehicle Type' : ($module == 'config_master_device' ? 'Config Master Devices' : ($module == 'config_logs_gps' ? 'Config Logs GPS' : $module)))) }} |                                     {{  | ||||||
|  | 										$module == 'transaction' ? 'job' :  | ||||||
|  | 											($module == 'client' ? 'company' :  | ||||||
|  | 											($module == 'config_truck_type' ? 'Config Vehicle Type' :  | ||||||
|  | 											($module == 'config_master_device' ? 'Config Master Devices' :  | ||||||
|  | 											($module == 'config_logs_gps' ? 'Config Logs GPS' : | ||||||
|  | 											($module == 'config_distribution_category' ? 'Config Distribution Category' : | ||||||
|  | 											($module == 'config_pool' ? 'Config Pool' : | ||||||
|  | 											($module == 'report_vehicle_trip' ? 'Report Vehicle Trip' : | ||||||
|  | 											($module == 'report_abnormality' ? 'Report Abnormality' : | ||||||
|  | 											($module == 'user_logs' ? 'Config User Activity' : $module | ||||||
|  | 											))))))))) | ||||||
|  | 									}} | ||||||
|                                 </label> |                                 </label> | ||||||
|                                 <div class="col-9"> |                                 <div class="col-9"> | ||||||
|                                     <div class="d-flex flex-wrap align-items-center gap-2"> |                                     <div class="d-flex flex-wrap align-items-center gap-2"> | ||||||
| @ -125,7 +137,19 @@ | |||||||
|                         @foreach ($permission as $module => $perms) |                         @foreach ($permission as $module => $perms) | ||||||
|                             <div class="mb-3 row"> |                             <div class="mb-3 row"> | ||||||
|                                 <label for="add-first_name" class="form-label text-capitalize col-3"> |                                 <label for="add-first_name" class="form-label text-capitalize col-3"> | ||||||
|                                     {{ $module == 'transaction' ? 'job' : ($module == 'client' ? 'company' : ($module == 'config_truck_type' ? 'Config Vehicle Type' : ($module == 'config_master_device' ? 'Config Master Devices' : ($module == 'config_logs_gps' ? 'Config Logs GPS' : $module)))) }} |                                     {{  | ||||||
|  | 										$module == 'transaction' ? 'job' :  | ||||||
|  | 											($module == 'client' ? 'company' :  | ||||||
|  | 											($module == 'config_truck_type' ? 'Config Vehicle Type' :  | ||||||
|  | 											($module == 'config_master_device' ? 'Config Master Devices' :  | ||||||
|  | 											($module == 'config_logs_gps' ? 'Config Logs GPS' : | ||||||
|  | 											($module == 'config_distribution_category' ? 'Config Distribution Category' : | ||||||
|  | 											($module == 'config_pool' ? 'Config Pool' : | ||||||
|  | 											($module == 'report_vehicle_trip' ? 'Report Vehicle Trip' : | ||||||
|  | 											($module == 'report_abnormality' ? 'Report Abnormality' : | ||||||
|  | 											($module == 'user_logs' ? 'Config User Activity' : $module | ||||||
|  | 											))))))))) | ||||||
|  | 									}} | ||||||
|                                 </label> |                                 </label> | ||||||
|                                 <div class="col-9"> |                                 <div class="col-9"> | ||||||
|                                     <div class="d-flex flex-wrap align-items-center gap-2"> |                                     <div class="d-flex flex-wrap align-items-center gap-2"> | ||||||
|  | |||||||
							
								
								
									
										263
									
								
								resources/views/menu_v1/userLogs.blade.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										263
									
								
								resources/views/menu_v1/userLogs.blade.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,263 @@ | |||||||
|  | @php | ||||||
|  |     $user_role = Auth::user()->role; | ||||||
|  | @endphp | ||||||
|  |  | ||||||
|  | @extends('app.app') | ||||||
|  |  | ||||||
|  | @section('title') | ||||||
|  |     User Activity | ||||||
|  | @endsection | ||||||
|  |  | ||||||
|  | @section('customcss') | ||||||
|  | 	<link | ||||||
|  | 		rel="stylesheet" | ||||||
|  | 		href="https://unpkg.com/leaflet@1.9.4/dist/leaflet.css" | ||||||
|  | 		integrity="sha256-p4NxAoJBhIIN+hmNHrzRCf9tD/miZyoHS5obTRR9BMY=" | ||||||
|  | 		crossorigin="" | ||||||
|  | 	/> | ||||||
|  | @endsection | ||||||
|  |  | ||||||
|  | @section('content') | ||||||
|  |     <div class="container-fluid"> | ||||||
|  |         <div class="content"> | ||||||
|  |             <div class="card"> | ||||||
|  |                 <div class="card-header"> | ||||||
|  |                     <div class="row d-flex align-items-center"> | ||||||
|  |                         <div class="col-3"> | ||||||
|  |                             <p class="card-title text-bold mb-0">User Activity</p> | ||||||
|  |                         </div> | ||||||
|  |                         <div class="col-auto text-end ps-0"> | ||||||
|  |                             <!-- <button class="btn btn-sm btn-danger">Download</button> --> | ||||||
|  |                         </div> | ||||||
|  |                     </div> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="card-body"> | ||||||
|  | 					<!-- filter --> | ||||||
|  | 					<div class="row"> | ||||||
|  | 						<div class="col-2"> | ||||||
|  | 							<div class="form-group"> | ||||||
|  | 								<label class="text-muted">From</label> | ||||||
|  | 								<!-- default today --> | ||||||
|  | 								<!-- <input class="form-control" id="tgl0" value="02-09-2025 00:00"> --> | ||||||
|  | 								<input class="form-control" id="tgl0" value="{{ date('d-m-Y 00:00') }}"> | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 						<div class="col-2"> | ||||||
|  | 							<div class="form-group"> | ||||||
|  | 								<label class="text-muted">To</label> | ||||||
|  | 								<!-- <input class="form-control" id="tgl1" value="02-09-2025 23:00"> --> | ||||||
|  | 								<input class="form-control" id="tgl1" value="{{ date('d-m-Y 23:59') }}"> | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 						<div class="col-2"> | ||||||
|  | 							<div class="form-group"> | ||||||
|  | 								<label class="text-muted">User</label> | ||||||
|  | 								<select name="user" class="form-control" id="filterUser"> | ||||||
|  | 									<option value="">-- All --</option> | ||||||
|  | 									@foreach ($users as $user) | ||||||
|  | 										<option value="{{ $user->id }}">{{ $user->email }}</option> | ||||||
|  | 									@endforeach | ||||||
|  | 								</select> | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 						<div class="col-1 d-flex align-items-end"> | ||||||
|  | 							<button class="btn btn-primary" id="submitFilter">Submit</button> | ||||||
|  | 						</div> | ||||||
|  | 						<div class="col-5 d-flex align-items-end"> | ||||||
|  | 							<button class="btn btn-sm btn-danger ms-auto" id="btnDownloadReport">Download Report</button> | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
|  | 					</div> | ||||||
|  |                     <div class="table-responsive p-3"> | ||||||
|  |                         <table id="tLogs" class="table table-hover dataTable w-100"> | ||||||
|  |                             <thead> | ||||||
|  |                                 <tr class=""> | ||||||
|  |                                     <th class="">Time</th> | ||||||
|  | 									<th class="">User</th> | ||||||
|  | 									<th class="">Module</th> | ||||||
|  | 									<th class="">Description</th> | ||||||
|  | 									<th class="">Action</th> | ||||||
|  |                                 </tr> | ||||||
|  |                             </thead> | ||||||
|  |                         </table> | ||||||
|  |                     </div> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |     </div> | ||||||
|  |  | ||||||
|  | @endsection | ||||||
|  |  | ||||||
|  | @section('customjs') | ||||||
|  |     <script src="{{ asset('assets/js/load-image.all.min.js') }}"></script> | ||||||
|  | 	<!-- DataTables Buttons + JSZip (for Excel export) --> | ||||||
|  | 	<script src="https://cdn.datatables.net/buttons/2.4.2/js/dataTables.buttons.min.js"></script> | ||||||
|  | 	<script src="https://cdnjs.cloudflare.com/ajax/libs/jszip/3.10.1/jszip.min.js"></script> | ||||||
|  | 	<script src="https://cdn.datatables.net/buttons/2.4.2/js/buttons.html5.min.js"></script> | ||||||
|  |     <!-- <script src="https://cdnjs.cloudflare.com/ajax/libs/leaflet.draw/1.0.4/leaflet.draw.js" integrity="sha512-ozq8xQKq6urvuU6jNgkfqAmT7jKN2XumbrX1JiB3TnF7tI48DPI4Gy1GXKD/V3EExgAs1V+pRO7vwtS1LHg0Gw==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> --> | ||||||
|  | 	<script src="https://unpkg.com/leaflet@1.9.4/dist/leaflet.js" integrity="sha256-20nQCchB9co0qIjJZRGuk2/Z9VM+kNiyxNV1lvTlZBo=" crossorigin=""></script> | ||||||
|  |     <script src="https://unpkg.com/leaflet-routing-machine@3.2.12/dist/leaflet-routing-machine.js"></script> | ||||||
|  | 	<script src="https://cdnjs.cloudflare.com/ajax/libs/html2canvas/1.4.1/html2canvas.min.js"></script> | ||||||
|  | 	<script src="https://cdnjs.cloudflare.com/ajax/libs/jspdf/2.5.1/jspdf.umd.min.js"></script> | ||||||
|  | 	<script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script> | ||||||
|  | 	<script src="{{ asset('assets/vendor/printThis.js') }}"></script> | ||||||
|  |  | ||||||
|  | 	<script> | ||||||
|  |         'use strict'; | ||||||
|  |  | ||||||
|  |         const Wrapper = { | ||||||
|  |             activate: function() { | ||||||
|  |                 Wrapper.init(); | ||||||
|  |                 Wrapper.event(); | ||||||
|  |                 DTable.activate(); | ||||||
|  | 			}, | ||||||
|  | 			init: ()=>{ | ||||||
|  | 				$('.time').each(function () { | ||||||
|  | 					const unix = parseInt($(this).text().trim()) + AppState.TIMEFIX; | ||||||
|  | 					$(this).text(moment.unix(unix).format('DD MMM YYYY HH:mm:ss')); | ||||||
|  | 				}); | ||||||
|  | 				$('#tgl0, #tgl1').datetimepicker({ | ||||||
|  | 					format:'d-m-Y H:i', | ||||||
|  | 					defaultTime:'00:00', | ||||||
|  | 					closeOnDateSelect: true, | ||||||
|  | 					// mask:true | ||||||
|  | 				}); | ||||||
|  | 			}, | ||||||
|  |             event: function() { | ||||||
|  | 				$('#submitFilter').on('click', function() { | ||||||
|  | 					DTable.reload(); | ||||||
|  | 				}); | ||||||
|  | 				// Trigger export on external button click | ||||||
|  | 				$('#btnDownloadReport').on('click', function() { | ||||||
|  | 					DTable.table.button('.buttons-excel').trigger(); | ||||||
|  | 				}); | ||||||
|  |             }, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         const DTable = { | ||||||
|  | 			table: null, | ||||||
|  |     		lastAjax: null, // keep track of the last ajax request | ||||||
|  |             activate: function() { | ||||||
|  |                 DTable.reload(); | ||||||
|  |             }, | ||||||
|  |             reload: function() { | ||||||
|  | 				// Abort the last request if it's still running | ||||||
|  | 				if (DTable.lastAjax) { | ||||||
|  | 					DTable.lastAjax.abort(); | ||||||
|  | 				} | ||||||
|  | 				if (DTable.table) { | ||||||
|  | 					// If table already exists → reload | ||||||
|  | 					DTable.table.ajax.reload(); | ||||||
|  | 					return; | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				DTable.table = $('#tLogs').DataTable({ | ||||||
|  | 					searching: false,   // 🔹 remove search box | ||||||
|  | 					ordering: false,    // 🔹 disable sorting for all columns | ||||||
|  | 					processing: true, | ||||||
|  | 					serverSide: false, | ||||||
|  | 					bLengthChange: true, | ||||||
|  | 					deferRender: true, | ||||||
|  | 					destroy: true, | ||||||
|  | 					ajax: function(data, callback, settings) { | ||||||
|  | 						// Abort previous | ||||||
|  | 						if (DTable.lastAjax) { | ||||||
|  | 							DTable.lastAjax.abort(); | ||||||
|  | 						} | ||||||
|  |  | ||||||
|  | 						// Fire new request | ||||||
|  | 						DTable.lastAjax = $.ajax({ | ||||||
|  | 							url: `{{ route('api_user_logs') }}? | ||||||
|  | 								tgl0=${moment($('#tgl0').val(), "DD-MM-YYYY HH:mm").unix()} | ||||||
|  | 								&tgl1=${moment($('#tgl1').val(), "DD-MM-YYYY HH:mm").unix()} | ||||||
|  | 								&userId=${$('#filterUser').val() || ''} | ||||||
|  | 							`, | ||||||
|  | 							type: 'GET', | ||||||
|  | 							success: function(json) { | ||||||
|  | 								callback(json); | ||||||
|  | 							}, | ||||||
|  | 							error: function(xhr, status, error) { | ||||||
|  | 								if (status !== 'abort') { | ||||||
|  | 									console.error("AJAX error:", error); | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							complete: function() { | ||||||
|  | 								DTable.lastAjax = null; | ||||||
|  | 							} | ||||||
|  | 						}); | ||||||
|  | 					}, | ||||||
|  |                     deferRender: true, | ||||||
|  | 					columns: [ | ||||||
|  | 						{ | ||||||
|  | 							data: 'crt', | ||||||
|  | 							render: (data, type, row, meta) => { | ||||||
|  | 								// let addr = row | ||||||
|  | 								return ` | ||||||
|  | 									${moment.unix(data).format('DD MMM YYYY HH:mm:ss')}<br> | ||||||
|  | 								`; | ||||||
|  | 							}  | ||||||
|  |  | ||||||
|  | 						}, | ||||||
|  | 						{ data: 'email' }, | ||||||
|  | 						{  | ||||||
|  | 							data: 'log', | ||||||
|  | 							render: (data, type, row, meta) => { | ||||||
|  | 								let log = JSON.parse(data); | ||||||
|  | 								let module = log.module || '-'; | ||||||
|  | 								return module; | ||||||
|  | 							} | ||||||
|  | 						}, | ||||||
|  | 						{  | ||||||
|  | 							data: 'log', | ||||||
|  | 							render: (data, type, row, meta) => { | ||||||
|  | 								let log = JSON.parse(data); | ||||||
|  | 								let desc = log.desc || '-'; | ||||||
|  | 								return desc; | ||||||
|  | 							} | ||||||
|  | 						}, | ||||||
|  | 						{  | ||||||
|  | 							data: 'log', | ||||||
|  | 							render: (data, type, row, meta) => { | ||||||
|  | 								let log = JSON.parse(data); | ||||||
|  | 								let action = log.action || '-'; | ||||||
|  |  | ||||||
|  | 								// use badge for action | ||||||
|  | 								// action = `<span class="badge bg-secondary">${action}</span>`; | ||||||
|  | 								let badge = ''; | ||||||
|  | 								if (action.toLowerCase() === 'create') { | ||||||
|  | 									badge = `<span class="badge bg-success">${action}</span>`; | ||||||
|  | 								} else if (action.toLowerCase() === 'update') { | ||||||
|  | 									badge = `<span class="badge bg-warning">${action}</span>`; | ||||||
|  | 								} else if (action.toLowerCase() === 'delete') { | ||||||
|  | 									badge = `<span class="badge bg-danger">${action}</span>`; | ||||||
|  | 								} else { | ||||||
|  | 									badge = `<span class="badge bg-secondary">${action}</span>`; | ||||||
|  | 								} | ||||||
|  | 								return badge; | ||||||
|  | 							} | ||||||
|  | 						}, | ||||||
|  | 					], | ||||||
|  | 					paging: false, | ||||||
|  | 					buttons: [ | ||||||
|  | 						{ | ||||||
|  | 							extend: 'excelHtml5', | ||||||
|  | 							title: () => { | ||||||
|  | 								return ` | ||||||
|  | 								User Activity Report -  | ||||||
|  | 								${moment($('#tgl0').val(), "DD-MM-YYYY HH:mm").format('DD MMM YYYY HH:mm')}  | ||||||
|  | 								to  | ||||||
|  | 								${moment($('#tgl1').val(), "DD-MM-YYYY HH:mm").format('DD MMM YYYY HH:mm')}` | ||||||
|  | 							}, | ||||||
|  | 							className: 'd-none', // hide default button | ||||||
|  | 							// exportOptions: { | ||||||
|  | 							// 	columns: ':visible:not(:last-child)' //  | ||||||
|  | 							// } | ||||||
|  | 						} | ||||||
|  | 					] | ||||||
|  | 				}); | ||||||
|  |             }, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         Wrapper.activate(); | ||||||
|  |     </script> | ||||||
|  | @endsection | ||||||
| @ -437,6 +437,11 @@ | |||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|  |     <div class="modal fade" data-bs-backdrop="static" data-bs-keyboard="false" id="mdlUserLogs" aria-labelledby="mdlUserLogsLabel" aria-hidden="true"> | ||||||
|  | 		<!-- ajax --> | ||||||
|  |     </div> | ||||||
|  |  | ||||||
| @endsection | @endsection | ||||||
|  |  | ||||||
| @section('customjs') | @section('customjs') | ||||||
| @ -635,10 +640,14 @@ | |||||||
|                                 // 	<span class="icon ion-eye fz-16"></span> |                                 // 	<span class="icon ion-eye fz-16"></span> | ||||||
|                                 // </a> |                                 // </a> | ||||||
|                                 let action = ` |                                 let action = ` | ||||||
|                                     <a href="#" class="text-decoration-none me-1 btnEdtUser" data-bs-toggle="tooltip" |                                     <a href="#" class="text-decoration-none me-2 btnEdtUser" data-bs-toggle="tooltip" | ||||||
|                                         data-bs-placement="bottom" title="Edit"> |                                         data-bs-placement="bottom" title="Edit"> | ||||||
|                                         <span class="icon ion-eye fz-16"></span> |                                         <span class="icon ion-eye fz-16"></span> | ||||||
|                                     </a> |                                     </a> | ||||||
|  | 									<a href="#" class="text-decoration-none me-2 btnLogUser" data-bs-toggle="tooltip" data-id="${row.id}" | ||||||
|  | 										data-bs-placement="bottom" title="Logs"> | ||||||
|  | 										<span class="icon ion-android-laptop fz-16"></span> | ||||||
|  | 									</a> | ||||||
| 								`; | 								`; | ||||||
|                                 // <a href="#" class="text-decoration-none text-danger btnDelUser" |                                 // <a href="#" class="text-decoration-none text-danger btnDelUser" | ||||||
|                                 //     data-bs-toggle="tooltip" data-bs-placement="bottom" title="Delete"> |                                 //     data-bs-toggle="tooltip" data-bs-placement="bottom" title="Delete"> | ||||||
| @ -877,6 +886,26 @@ | |||||||
|                     } |                     } | ||||||
|                     UUpdate.passDataToView(resp.data); |                     UUpdate.passDataToView(resp.data); | ||||||
|                 }); |                 }); | ||||||
|  |                 $('#tUsers').on('click', '.btnLogUser', async function(e) { | ||||||
|  |                     // let uid = $(e.target).closest('tr').find('td[data-id]').data('id'); | ||||||
|  |                     // UUpdate.clearInput(); | ||||||
|  |                     // let resp = await UUpdate.reqData({ | ||||||
|  |                     //     uid | ||||||
|  |                     // }); | ||||||
|  |                     // if (resp.type != 'success') { | ||||||
|  |                     //     Helper.toast('User Not Found', 'just now', 'please try again'); | ||||||
|  |                     //     return false; | ||||||
|  |                     // } | ||||||
|  |                     // UUpdate.passDataToView(resp.data); | ||||||
|  |  | ||||||
|  | 					e.preventDefault(); | ||||||
|  | 					const id = $(this).data('id'); | ||||||
|  | 					$('#mdlUserLogs').empty().load("{{ route('view_user_logs1') }}", `id=${id}`, () => { | ||||||
|  | 						$('#mdlUserLogs').modal('show') | ||||||
|  | 					}) | ||||||
|  |  | ||||||
|  |                 }); | ||||||
|  | 				 | ||||||
|                 $('#updtUserModal').on('shown.bs.modal', function() { |                 $('#updtUserModal').on('shown.bs.modal', function() { | ||||||
|                     // initiate select2 if there |                     // initiate select2 if there | ||||||
|                 }); |                 }); | ||||||
|  | |||||||
| @ -1317,6 +1317,9 @@ | |||||||
|  |  | ||||||
|                 $('#mdlEdtVhc').data('id', data.vid); |                 $('#mdlEdtVhc').data('id', data.vid); | ||||||
|                 $('#mdlEdtVhc').modal('show'); |                 $('#mdlEdtVhc').modal('show'); | ||||||
|  |  | ||||||
|  | 				$('#edt-dc_code').val(data?.dc_code).trigger('change'); | ||||||
|  | 				$('#edt-pool_code').val(data?.pool_code).trigger('change'); | ||||||
|             }, |             }, | ||||||
|             removeOptionDevice: function(type) { |             removeOptionDevice: function(type) { | ||||||
|                 $(`#edt-deviceid option[data-type='${type}']`).remove(); |                 $(`#edt-deviceid option[data-type='${type}']`).remove(); | ||||||
|  | |||||||
| @ -61,7 +61,14 @@ | |||||||
|                         </ul> |                         </ul> | ||||||
|                     </li> |                     </li> | ||||||
|                 @endif |                 @endif | ||||||
|                 @if (auth()->user()->can('config_truck_type.view') || auth()->user()->can('config_master_device.view') || auth()->user()->can('config_logs_gps.view')) |                 @if ( | ||||||
|  | 					auth()->user()->can('config_truck_type.view') ||  | ||||||
|  | 					auth()->user()->can('config_master_device.view') ||  | ||||||
|  | 					auth()->user()->can('config_logs_gps.view') || | ||||||
|  | 					auth()->user()->can('config_distribution_category.view') || | ||||||
|  | 					auth()->user()->can('config_pool.view') || | ||||||
|  | 					auth()->user()->can('user_logs.view') | ||||||
|  | 				) | ||||||
|                     <li class="nav-item dropdown {{ Request::segment(1) == 'config' ? 'active' : '' }}"> |                     <li class="nav-item dropdown {{ Request::segment(1) == 'config' ? 'active' : '' }}"> | ||||||
|                         <a class="nav-link dropdown-toggle" href="#" id="dropdownConfig" role="button" data-bs-toggle="dropdown" aria-expanded="false"> |                         <a class="nav-link dropdown-toggle" href="#" id="dropdownConfig" role="button" data-bs-toggle="dropdown" aria-expanded="false"> | ||||||
|                             Configuration |                             Configuration | ||||||
| @ -92,6 +99,17 @@ | |||||||
|                                     <a class="dropdown-item {{ Request::segment(2) == 'pool' ? 'active' : '' }}" href="{{ route('view_config_pool') }}" title="">Pool</a> |                                     <a class="dropdown-item {{ Request::segment(2) == 'pool' ? 'active' : '' }}" href="{{ route('view_config_pool') }}" title="">Pool</a> | ||||||
|                                 </li> |                                 </li> | ||||||
| 							@endcan | 							@endcan | ||||||
|  | 							@can('user_logs.view') | ||||||
|  | 								<li> | ||||||
|  |                                     <a class="dropdown-item {{ Request::segment(1) == 'user_logs' ? 'active' : '' }}" href="{{ route('view_user_logs') }}" title="">User Activity</a> | ||||||
|  |                                 </li> | ||||||
|  |  | ||||||
|  | 							<!-- <li class="nav-item {{ Request::segment(1) == 'user_logs' ? 'active' : '' }}"> | ||||||
|  | 									<a class="nav-link d-flex align-items-center text-capitalize" aria-current="page" href="{{ route('view_user_logs') }}"> | ||||||
|  | 										User Logs | ||||||
|  | 									</a> | ||||||
|  | 								</li> --> | ||||||
|  | 							@endcan | ||||||
|                         </ul> |                         </ul> | ||||||
|                     </li> |                     </li> | ||||||
|                 @endif |                 @endif | ||||||
|  | |||||||
| @ -120,6 +120,11 @@ Route::middleware(["auth", "auth.user"])->group(function () { | |||||||
|     Route::get("/users", "UsersController@view_users") |     Route::get("/users", "UsersController@view_users") | ||||||
|         ->name("view_users") |         ->name("view_users") | ||||||
|         ->middleware("permission:user.view"); |         ->middleware("permission:user.view"); | ||||||
|  |     Route::get("/users/logs", "UsersController@view_user_logs1")->name("view_user_logs1"); | ||||||
|  |     Route::get("/api/users/logs", "UsersController@api_user_logs1")->name("api_user_logs1"); | ||||||
|  |  | ||||||
|  |     Route::get("/user-logs", "UserLogsController@view_user_logs")->name("view_user_logs"); | ||||||
|  |     Route::get("/api/user-logs", "UserLogsController@api_user_logs")->name("api_user_logs"); | ||||||
|  |  | ||||||
|     Route::get("/roles", "RolesController@view") |     Route::get("/roles", "RolesController@view") | ||||||
|         ->name("view_roles") |         ->name("view_roles") | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user
	