Compare commits
	
		
			2 Commits
		
	
	
		
			3ad22e0cf5
			...
			ccd7b19ff5
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ccd7b19ff5 | |||
| c492711d83 | 
| @ -22,11 +22,11 @@ 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"; | ||||||
|  | 		$listNopol = DB::select($q); | ||||||
|  |  | ||||||
| 		$data = [ | 		$data = [ | ||||||
| 			// 'lanes' => ConfRates::getLanesActive(), | 			'listNopol' => $listNopol, | ||||||
| 			// 'provs' => Region::listProv(), |  | ||||||
| 			// 'vendors' => Users::getUsersActiveByRole(Users::ROLE_VENDOR), |  | ||||||
| 			// 'truck_types' => ConfTruckTypes::listTruckTypes(ConfTruckTypes::IS_ACTIVE), |  | ||||||
| 		]; | 		]; | ||||||
|  |  | ||||||
| 		return view('menu_v1.reports.vehicle_trips', $data); | 		return view('menu_v1.reports.vehicle_trips', $data); | ||||||
| @ -49,10 +49,9 @@ 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'); | ||||||
|  | 		$vid = $req->input('vid'); | ||||||
| 		// $from_date = 1756054800; | 		// $from_date = 1756054800; | ||||||
| 		// $to_date = 1756745940; | 		// $to_date = 1756745940; | ||||||
|  |  | ||||||
| @ -81,19 +80,25 @@ class ReportsController extends Controller | |||||||
| 						SUM(trip_start) OVER (PARTITION BY vhc_id ORDER BY crt_d) AS trip_id | 						SUM(trip_start) OVER (PARTITION BY vhc_id ORDER BY crt_d) AS trip_id | ||||||
| 					FROM trips | 					FROM trips | ||||||
| 				) | 				) | ||||||
| 				SELECT | 				select | ||||||
| 					v.id, | 					v.id, | ||||||
| 					coalesce(max(a.trip_id), 0) numOfTrip, | 					-- 	coalesce(max(a.trip_id), 0) numOfTrip, | ||||||
| 					SUM(pre_milleage) AS total_milleage, | 					-- 	SUM(pre_milleage) AS total_milleage, | ||||||
| 					v.name, v.nopol1  | 					v.name, v.nopol1, | ||||||
|  | 					vhc_id, trip_id, | ||||||
|  | 					sum(pre_milleage) milleage, min(a.crt_d ) start, max(a.crt_d ) finish, | ||||||
|  | 					(select fulladdress from t_gps_tracks_address where master_id = min(a.id) limit 1) startLoc, | ||||||
|  | 					(select fulladdress from t_gps_tracks_address where master_id = max(a.id) limit 1) finishLoc | ||||||
| 				FROM   | 				FROM   | ||||||
| 					t_vehicles v | 					t_vehicles v | ||||||
| 					left join numbered a on a.vhc_id = v.id | 					left join numbered a on a.vhc_id = v.id | ||||||
| 				WHERE v.dlt is null | 				WHERE  | ||||||
| 				GROUP BY v.id | 					v.dlt is null and trip_id != 0  | ||||||
| 				ORDER BY v.id; | 					and if(? , v.id = ? , 1=1) | ||||||
| 			"; | 				GROUP BY v.id, a.trip_id  | ||||||
| 			$d = [$from_date, $to_date]; | 				ORDER BY v.id | ||||||
|  | 				"; | ||||||
|  | 			$d = [$from_date, $to_date, $vid, $vid]; | ||||||
|  |  | ||||||
| 			$list = DB::select($q, $d); | 			$list = DB::select($q, $d); | ||||||
|  |  | ||||||
| @ -177,5 +182,72 @@ class ReportsController extends Controller | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	public function view_report_abnormalities(Request $req) | ||||||
|  | 	{ | ||||||
|  | 		$q = "select id, nopol1 from t_vehicles WHERE dlt is null order by nopol1"; | ||||||
|  | 		$listNopol = DB::select($q); | ||||||
|  |  | ||||||
|  | 		$data = [ | ||||||
|  | 			'listNopol' => $listNopol, | ||||||
|  | 		]; | ||||||
|  |  | ||||||
|  | 		return view('menu_v1.reports.abnormalities', $data); | ||||||
|  | 	} | ||||||
|  | 	public function api_report_abnormalities_list(Request $req) | ||||||
|  | 	{ | ||||||
|  | 		// Validate input | ||||||
|  | 		$rules = [ | ||||||
|  | 			// 'from_date' => 'required|date', | ||||||
|  | 			// 'to_date' => 'required|date|after_or_equal:from_date', | ||||||
|  | 			'type' => 'nullable|in:report,list', // enum "report", "list". nullable default "list" | ||||||
|  | 		]; | ||||||
|  |  | ||||||
|  | 		$isValidInput = Validator::make($req->all(), $rules); | ||||||
|  | 		if (!$isValidInput->passes()) { | ||||||
|  | 			$apiResp = Responses::bad_input($isValidInput->messages()->first()); | ||||||
|  | 			return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		$from_date = $req->input('from_date'); | ||||||
|  | 		$to_date = $req->input('to_date'); | ||||||
|  | 		$vid = $req->input('vid'); | ||||||
|  |  | ||||||
|  | 		try { | ||||||
|  | 			$q = " | ||||||
|  | 				select | ||||||
|  | 					tv.name, tv.nopol1,  | ||||||
|  | 					t.crt_d, t.speed, tgta.fulladdress, | ||||||
|  | 					tvd.speed_limit | ||||||
|  | 				from | ||||||
|  | 					t_gps_tracks t | ||||||
|  | 					left join t_vehicles tv on tv.id = t.vhc_id  | ||||||
|  | 					left join t_vehicles_detail tvd on tvd.vid = tv.id | ||||||
|  | 					left join t_gps_tracks_address tgta on tgta.master_id = t.id | ||||||
|  | 				WHERE  | ||||||
|  | 					t.action = 'location' | ||||||
|  | 					and t.speed != 0  | ||||||
|  | 					AND t.crt_d BETWEEN ? AND ? | ||||||
|  | 					and if(? , tv.id = ? , 1=1) | ||||||
|  | 					-- and t.speed > tvd.speed_limit  | ||||||
|  | 				having t.speed > tvd.speed_limit | ||||||
|  | 				ORDER BY t.crt_d | ||||||
|  | 			"; | ||||||
|  | 			$d = [$from_date, $to_date, $vid, $vid]; | ||||||
|  |  | ||||||
|  | 			$list = DB::select($q, $d); | ||||||
|  |  | ||||||
|  | 			// // RETURN 1 - LIST | ||||||
|  | 			// if($req->type != 'report'){ | ||||||
|  | 				$apiResp = Responses::success("success list abnormalities report"); | ||||||
|  | 				$apiResp["data"] = $list; | ||||||
|  | 				return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|  | 			// } | ||||||
|  |  | ||||||
|  | 		} catch (\Exception $e) { | ||||||
|  |             $apiResp = Responses::error($e->getMessage()); | ||||||
|  |             return new Response($apiResp, $apiResp["meta"]["code"]); | ||||||
|  | 			// return Responses::json(Responses::SERVER_ERROR, 'An error occurred while generating the report.', (object)[]); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -272,6 +272,7 @@ | |||||||
|                         <p class="text-bold mb-0">Engine Status</p> |                         <p class="text-bold mb-0">Engine Status</p> | ||||||
|                         <p id="infoVehicles-engineStatus" class="text-muted mb-0">Idling</p> |                         <p id="infoVehicles-engineStatus" class="text-muted mb-0">Idling</p> | ||||||
|                     </li> |                     </li> | ||||||
|  | 					 | ||||||
|                     {{-- <li class="list-group-item p-1 px-2"> |                     {{-- <li class="list-group-item p-1 px-2"> | ||||||
|                         <p class="text-bold mb-0">Durasi Berhenti (<span id="infoVehicles-idlingUnit">min</span>)</p> |                         <p class="text-bold mb-0">Durasi Berhenti (<span id="infoVehicles-idlingUnit">min</span>)</p> | ||||||
|                         <p id="infoVehicles-idlingDuration" class="text-muted mb-0">2946</p> |                         <p id="infoVehicles-idlingDuration" class="text-muted mb-0">2946</p> | ||||||
| @ -408,10 +409,11 @@ | |||||||
|                             <label class="text-muted">From</label> |                             <label class="text-muted">From</label> | ||||||
|                             <input class="form-control" id="historyStartDate"> |                             <input class="form-control" id="historyStartDate"> | ||||||
|                         </div> |                         </div> | ||||||
|                         <div class="mb-3"> |                         <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"> | ||||||
|                         </div> |                         </div> | ||||||
|  | 						<button class="btn btn-primary btn-sm w-100 mb-3" id="historySearch">Search</button> | ||||||
|                     </div> |                     </div> | ||||||
|                     <div id="infoMove-plots" class="infoMove-plots"> |                     <div id="infoMove-plots" class="infoMove-plots"> | ||||||
|                         {{-- <a href="#" class="plotMove-item"> |                         {{-- <a href="#" class="plotMove-item"> | ||||||
| @ -513,6 +515,13 @@ | |||||||
|                     <span class="badge p-1 ps-3 pe-3 bg-danger me-2">Stopped</span> |                     <span class="badge p-1 ps-3 pe-3 bg-danger me-2">Stopped</span> | ||||||
|                     <span class="badge p-1 ps-3 pe-3 bg-warning me-2">Idling</span> |                     <span class="badge p-1 ps-3 pe-3 bg-warning me-2">Idling</span> | ||||||
|                     <span class="badge p-1 ps-3 pe-3 bg-success">Moving</span> |                     <span class="badge p-1 ps-3 pe-3 bg-success">Moving</span> | ||||||
|  | 				</div> | ||||||
|  | 				<div class="col text-end" id="tripInfo"> | ||||||
|  | 					<span class="badge p-1 ps-3 pe-3 bg-info me-2" style="background-color:#C0392B !important;">Trip 1</span> | ||||||
|  | 					<span class="badge p-1 ps-3 pe-3 bg-info me-2" style="background-color:#2980B9 !important;">Trip 2</span> | ||||||
|  | 					<span class="badge p-1 ps-3 pe-3 bg-info me-2" style="background-color:#27AE60 !important;">Trip 3</span> | ||||||
|  | 					<span class="badge p-1 ps-3 pe-3 bg-info me-2" style="background-color:#D35400 !important;">Trip 4</span> | ||||||
|  | 					<span class="badge p-1 ps-3 pe-3 bg-info me-2" style="background-color:#F39C12 !important;">Trip 5</span> | ||||||
|                 </div> |                 </div> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
| @ -941,7 +950,7 @@ | |||||||
| 					waypoints: latLngs.reverse(), // reverse to make first point as start | 					waypoints: latLngs.reverse(), // reverse to make first point as start | ||||||
| 					// router: L.Routing.osrmv1(), | 					// router: L.Routing.osrmv1(), | ||||||
| 					router: L.Routing.osrmv1({ | 					router: L.Routing.osrmv1({ | ||||||
| 						serviceUrl: "https://brilianapps.britimorleste.tl/osrm-backend/route/v1", | 						serviceUrl: "https://brilianapps.britimorleste.tl:5001/route/v1", | ||||||
| 					}), | 					}), | ||||||
| 					show: false, | 					show: false, | ||||||
| 					itinerary: null, // 👈 completely removes the panel | 					itinerary: null, // 👈 completely removes the panel | ||||||
| @ -950,7 +959,7 @@ | |||||||
| 					routeWhileDragging: false, // optional: don’t reroute while dragging | 					routeWhileDragging: false, // optional: don’t reroute while dragging | ||||||
| 					createMarker: () => null, | 					createMarker: () => null, | ||||||
| 					lineOptions:{ | 					lineOptions:{ | ||||||
| 						styles: [{ color: locs[0]?.options?.color, weight: 5, opacity: 0.7 }], | 						styles: [{ color: locs[0]?.options?.color, weight: 3, opacity: 0.7 }], | ||||||
| 					}, | 					}, | ||||||
| 				}).addTo(Leaflet.map) | 				}).addTo(Leaflet.map) | ||||||
|  |  | ||||||
| @ -1505,7 +1514,14 @@ | |||||||
|                 }); |                 }); | ||||||
|             }, |             }, | ||||||
|             eventFilterHistoryDate: function() { |             eventFilterHistoryDate: function() { | ||||||
|                 $('#historyStartDate, #historyEndDate').on('change', function(e) { |                 $('#historySearch').on('click', function(e) { | ||||||
|  | 					e.preventDefault(); | ||||||
|  | 					// console.log("change history date", this.id, this.value); | ||||||
|  | 					const value = this.value; | ||||||
|  | 					const id = this.id; | ||||||
|  | 					if(State[id] == moment(value, "DD-MM-YYYY HH:mm").unix()) | ||||||
|  | 						return false; | ||||||
|  |  | ||||||
| 					const date0 = moment($('#historyStartDate').val(), "DD-MM-YYYY HH:mm") | 					const date0 = moment($('#historyStartDate').val(), "DD-MM-YYYY HH:mm") | ||||||
| 					const date1 = moment($('#historyEndDate').val(), "DD-MM-YYYY HH:mm") | 					const date1 = moment($('#historyEndDate').val(), "DD-MM-YYYY HH:mm") | ||||||
|                     const { |                     const { | ||||||
| @ -2637,95 +2653,95 @@ | |||||||
|                 Trucks.routeStartEndGroupTrip(truckRoutes); |                 Trucks.routeStartEndGroupTrip(truckRoutes); | ||||||
|             }, |             }, | ||||||
|             // routeStartEnd: function(truckRoutes) { |             // routeStartEnd: function(truckRoutes) { | ||||||
|             //     let key_length = truckRoutes.length; | 				//     let key_length = truckRoutes.length; | ||||||
|             //     let polyTruckRoutes = truckRoutes.map((obj, key) => { | 				//     let polyTruckRoutes = truckRoutes.map((obj, key) => { | ||||||
| 						 | 						 | ||||||
|             //         obj.key_index = key + 1 // key_length - key; | 				//         obj.key_index = key + 1 // key_length - key; | ||||||
|             //         // lists per detail info movement | 				//         // lists per detail info movement | ||||||
|             //         Menu.showToListMovement(obj); | 				//         Menu.showToListMovement(obj); | ||||||
|             //         return { | 				//         return { | ||||||
|             //             lat: obj.latitude, | 				//             lat: obj.latitude, | ||||||
|             //             lng: obj.longitude, | 				//             lng: obj.longitude, | ||||||
|             //             options: { | 				//             options: { | ||||||
|             //                 // polyline | 				//                 // polyline | ||||||
|             //                 smoothFactor: 1.0, | 				//                 smoothFactor: 1.0, | ||||||
|             //                 noClip: true, | 				//                 noClip: true, | ||||||
|             //                 bubblingMouseEvents: false, | 				//                 bubblingMouseEvents: false, | ||||||
|             //                 // circle | 				//                 // circle | ||||||
|             //                 radius: 5, | 				//                 radius: 5, | ||||||
| 			// 				// markerOpacity: 0 | 				// 				// markerOpacity: 0 | ||||||
|             //             }, | 				//             }, | ||||||
|             //             // circle | 				//             // circle | ||||||
|             //             // label: `<b>${obj.key_index}</b><br>${obj.nopol1} ${obj.nopol2} ${obj.nopol3}<br>${moment.unix(obj?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br>Speed: ${(typeof obj.speed != 'undefined') ? obj.speed : '0'}<br>${Number(obj.latitude).toFixed(5)} - ${Number(obj.longitude).toFixed(6)}<br>${decodeURIComponent(obj?.fulladdress || 'address')}`, | 				//             // label: `<b>${obj.key_index}</b><br>${obj.nopol1} ${obj.nopol2} ${obj.nopol3}<br>${moment.unix(obj?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br>Speed: ${(typeof obj.speed != 'undefined') ? obj.speed : '0'}<br>${Number(obj.latitude).toFixed(5)} - ${Number(obj.longitude).toFixed(6)}<br>${decodeURIComponent(obj?.fulladdress || 'address')}`, | ||||||
|             //             // label: `<b>${obj.nopol1} ${obj.nopol2} ${obj.nopol3}</b><br>${moment.unix(obj?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br>${decodeURIComponent(obj?.fulladdress || 'address')}`, | 				//             // label: `<b>${obj.nopol1} ${obj.nopol2} ${obj.nopol3}</b><br>${moment.unix(obj?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br>${decodeURIComponent(obj?.fulladdress || 'address')}`, | ||||||
|             //             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(obj?.lst_loc_crt_d).format('DD MMM YYYY HH:mm:ss')}<br>${decodeURIComponent(obj?.fulladdress || 'address')}`, | ||||||
|             //         } | 				//         } | ||||||
|             //     }); | 				//     }); | ||||||
|  |  | ||||||
|             //     let polyline = Leaflet.addPolylines(polyTruckRoutes); | 				//     let polyline = Leaflet.addPolylines(polyTruckRoutes); | ||||||
|             //     Leaflet.map.fitBounds(polyline.getBounds()); | 				//     Leaflet.map.fitBounds(polyline.getBounds()); | ||||||
|             //     // let ctrWaypoint = Leaflet.addWaypoints(polyTruckRoutes.slice(0, 100)) | 				//     // let ctrWaypoint = Leaflet.addWaypoints(polyTruckRoutes.slice(0, 100)) | ||||||
|  |  | ||||||
|             //     Leaflet.addCircles(polyTruckRoutes, function(circle, i) { | 				//     Leaflet.addCircles(polyTruckRoutes, function(circle, i) { | ||||||
|             //         window.addEventListener('eventRemoveRouteStartEnd', function handler(e) { | 				//         window.addEventListener('eventRemoveRouteStartEnd', function handler(e) { | ||||||
|             //             circle.remove(); | 				//             circle.remove(); | ||||||
|             //         }); | 				//         }); | ||||||
|             //         circle.on('click', function() { | 				//         circle.on('click', function() { | ||||||
|             //             PgBar.syncToPlotTravelHistory(i); | 				//             PgBar.syncToPlotTravelHistory(i); | ||||||
|             //         }) | 				//         }) | ||||||
|             //     }); | 				//     }); | ||||||
|  |  | ||||||
|             //     let start = truckRoutes.at(-1); | 				//     let start = truckRoutes.at(-1); | ||||||
|             //     let finish = truckRoutes.at(0); | 				//     let finish = truckRoutes.at(0); | ||||||
|             //     // ${(start?.city_text || start?.state_tex || 'address't)} - ${(start?.postcode || 'postcode')} | 				//     // ${(start?.city_text || start?.state_tex || 'address't)} - ${(start?.postcode || 'postcode')} | ||||||
|             //     // ${(finish?.city_text || finish?.state_text || 'address')} - ${(finish?.postcode || 'postcode')} | 				//     // ${(finish?.city_text || finish?.state_text || 'address')} - ${(finish?.postcode || 'postcode')} | ||||||
|             //     let startMarker = Leaflet.addMarkers({ | 				//     let startMarker = Leaflet.addMarkers({ | ||||||
|             //         lat: start.latitude, | 				//         lat: start.latitude, | ||||||
|             //         lng: start.longitude, | 				//         lng: start.longitude, | ||||||
|             //         label: `<b>Start Poin</b><br>${start.nopol1} ${start.nopol2} ${start.nopol3}<br>${moment.unix(start?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br>Speed: ${(typeof start.lst_speed != 'undefined') ? start.lst_speed : '0'}<br>${Number(start.latitude).toFixed(5)},${Number(start.longitude).toFixed(6)}<br>${decodeURIComponent(start.fulladdress || 'address')}`, | 				//         label: `<b>Start Poin</b><br>${start.nopol1} ${start.nopol2} ${start.nopol3}<br>${moment.unix(start?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br>Speed: ${(typeof start.lst_speed != 'undefined') ? start.lst_speed : '0'}<br>${Number(start.latitude).toFixed(5)},${Number(start.longitude).toFixed(6)}<br>${decodeURIComponent(start.fulladdress || 'address')}`, | ||||||
|             //         // label: `<b>Start Poin</b><br>${start.nopol1} ${start.nopol2} ${start.nopol3}<br>${moment.unix(start?.lst_loc_crt_d).utcOffset(9 * 60).format('DD MMM YYYY HH:mm:ss')}<br>Speed: ${(typeof start.lst_speed != 'undefined') ? start.lst_speed : '0'}<br>${Number(start.latitude).toFixed(5)},${Number(start.longitude).toFixed(6)}<br>${decodeURIComponent(start.fulladdress || 'address')}`, | 				//         // label: `<b>Start Poin</b><br>${start.nopol1} ${start.nopol2} ${start.nopol3}<br>${moment.unix(start?.lst_loc_crt_d).utcOffset(9 * 60).format('DD MMM YYYY HH:mm:ss')}<br>Speed: ${(typeof start.lst_speed != 'undefined') ? start.lst_speed : '0'}<br>${Number(start.latitude).toFixed(5)},${Number(start.longitude).toFixed(6)}<br>${decodeURIComponent(start.fulladdress || 'address')}`, | ||||||
|             //         options: { | 				//         options: { | ||||||
|             //             icon: Icon.titikAwal(), | 				//             icon: Icon.titikAwal(), | ||||||
|             //             // rotationAngle: 290 | 				//             // rotationAngle: 290 | ||||||
|             //         } | 				//         } | ||||||
|             //     }); | 				//     }); | ||||||
|             //     let finishMarker = Leaflet.addMarkers({ | 				//     let finishMarker = Leaflet.addMarkers({ | ||||||
|             //         lat: finish.latitude, | 				//         lat: finish.latitude, | ||||||
|             //         lng: finish.longitude, | 				//         lng: finish.longitude, | ||||||
|             //         label: `<b>End Poin</b><br>${finish.nopol1} ${finish.nopol2} ${finish.nopol3}<br>${moment.unix(finish?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br>Speed: ${(typeof finish.lst_speed != 'undefined') ? finish.lst_speed : '0'}<br>${Number(finish.latitude).toFixed(5)},${Number(finish.longitude).toFixed(6)}<br>${decodeURIComponent(finish.fulladdress || 'address')}`, | 				//         label: `<b>End Poin</b><br>${finish.nopol1} ${finish.nopol2} ${finish.nopol3}<br>${moment.unix(finish?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br>Speed: ${(typeof finish.lst_speed != 'undefined') ? finish.lst_speed : '0'}<br>${Number(finish.latitude).toFixed(5)},${Number(finish.longitude).toFixed(6)}<br>${decodeURIComponent(finish.fulladdress || 'address')}`, | ||||||
|             //         //label: `<b>End Poin</b><br>${finish.nopol1} ${finish.nopol2} ${finish.nopol3}<br>${moment.unix(finish?.lst_loc_crt_d).utcOffset(9 * 60).format('DD MMM YYYY HH:mm:ss')}<br>Speed: ${(typeof finish.lst_speed != 'undefined') ? finish.lst_speed : '0'}<br>${Number(finish.latitude).toFixed(5)},${Number(finish.longitude).toFixed(6)}<br>${decodeURIComponent(finish.fulladdress || 'address')}`, | 				//         //label: `<b>End Poin</b><br>${finish.nopol1} ${finish.nopol2} ${finish.nopol3}<br>${moment.unix(finish?.lst_loc_crt_d).utcOffset(9 * 60).format('DD MMM YYYY HH:mm:ss')}<br>Speed: ${(typeof finish.lst_speed != 'undefined') ? finish.lst_speed : '0'}<br>${Number(finish.latitude).toFixed(5)},${Number(finish.longitude).toFixed(6)}<br>${decodeURIComponent(finish.fulladdress || 'address')}`, | ||||||
|             //         options: { | 				//         options: { | ||||||
|             //             icon: Icon.titikAkhir() | 				//             icon: Icon.titikAkhir() | ||||||
|             //         } | 				//         } | ||||||
|             //     }); | 				//     }); | ||||||
|  |  | ||||||
|             //     // remove marker, circle, event listener and all about this marker | 				//     // remove marker, circle, event listener and all about this marker | ||||||
|             //     State.eventRemoveRouteStartEnd = new CustomEvent('eventRemoveRouteStartEnd', { | 				//     State.eventRemoveRouteStartEnd = new CustomEvent('eventRemoveRouteStartEnd', { | ||||||
|             //         startMarker, | 				//         startMarker, | ||||||
|             //         finishMarker, | 				//         finishMarker, | ||||||
|             //         polyline, | 				//         polyline, | ||||||
|             //         polyline, | 				//         polyline, | ||||||
|             //     }); | 				//     }); | ||||||
|             //     window.addEventListener('eventRemoveRouteStartEnd', function handler(e) { | 				//     window.addEventListener('eventRemoveRouteStartEnd', function handler(e) { | ||||||
|             //         startMarker.removeEventListener('click'); | 				//         startMarker.removeEventListener('click'); | ||||||
|             //         startMarker.removeEventListener('moveend'); | 				//         startMarker.removeEventListener('moveend'); | ||||||
|             //         startMarker.remove(); | 				//         startMarker.remove(); | ||||||
|             //         finishMarker.removeEventListener('click'); | 				//         finishMarker.removeEventListener('click'); | ||||||
|             //         finishMarker.removeEventListener('moveend'); | 				//         finishMarker.removeEventListener('moveend'); | ||||||
|             //         finishMarker.remove(); | 				//         finishMarker.remove(); | ||||||
|             //         polyline.remove(); | 				//         polyline.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; | ||||||
|             //         State.inShowLastMove = null; | 				//         State.inShowLastMove = null; | ||||||
|  |  | ||||||
|             //         PgBar.tglMenuPlayback(false); | 				//         PgBar.tglMenuPlayback(false); | ||||||
|             //         PgBar.reset(); | 				//         PgBar.reset(); | ||||||
|             //     }); | 				//     }); | ||||||
|  |  | ||||||
|             //     PgBar.setMinMax(0, truckRoutes.length - 1); | 				//     PgBar.setMinMax(0, truckRoutes.length - 1); | ||||||
|             // }, |             // }, | ||||||
|             routeStartEndGroupTrip: function(truckRoutes) { |             routeStartEndGroupTrip:  function(truckRoutes) { | ||||||
| 				const colors = [ | 				const colors = [ | ||||||
| 					"#2980B9",  // Dark Blue | 					"#2980B9",  // Dark Blue | ||||||
| 					"#C0392B", // Dark Red | 					"#C0392B", // Dark Red | ||||||
| @ -2780,29 +2796,61 @@ | |||||||
| 				}) | 				}) | ||||||
| 				// console.log("truckRoutes update", polyTruckRoutes); | 				// console.log("truckRoutes update", polyTruckRoutes); | ||||||
|  |  | ||||||
| 				polyTruckRoutes.forEach((poly, idx) => { | 				polyTruckRoutes.forEach(async (poly, idx) => { | ||||||
| 					// console.log("poly", poly);					 | 					// console.log("poly", poly);					 | ||||||
| 					let polyline = Leaflet.addRoutes(poly) | 					let polyline = Leaflet.addRoutes(poly) | ||||||
| 					// let polyline = Leaflet.addPolylines(poly) | 					// let polyline = Leaflet.addPolylines(poly) | ||||||
| 					// if(idx == 0) |  | ||||||
| 						// Leaflet.map.fitBounds(polyline.getBounds()); |  | ||||||
|  |  | ||||||
|  | 					// let circlesStartStop = [] | ||||||
|  | 					// Leaflet.addCircles(poly, function(circle, i) { | ||||||
|  | 					// 	if(i == 0 || i == (polyTruckRoutes.length - 1)) | ||||||
|  | 					// 		circlesStartStop.push(circle) | ||||||
|  |  | ||||||
| 					// let circlesStartStop = [...poly.filter((p, i) => i == 0 || i == (poly.length - 1))] | 					// 	window.addEventListener('eventRemoveRouteStartEnd', function handler(e) { | ||||||
| 					let circlesStartStop = [] | 					// 		circle.remove(); | ||||||
| 					// console.log("circlesStartStop", circlesStartStop); | 					// 	}); | ||||||
|  | 					// 	circle.on('click', function() { | ||||||
|  | 					// 		PgBar.syncToPlotTravelHistory(i); | ||||||
|  | 					// 	}) | ||||||
|  | 					// }); | ||||||
|  | 					function addCirclesAsync(poly) { | ||||||
|  | 						return new Promise(resolve => { | ||||||
|  | 							let circles = []; | ||||||
|  |  | ||||||
| 					Leaflet.addCircles(poly, function(circle, i) { | 							Leaflet.addCircles(poly, function(circle, i) { | ||||||
| 						if(i == 0 || i == (polyTruckRoutes.length - 1)) | 								if (i === 0 || i === (poly.length - 1)) { | ||||||
| 							circlesStartStop.push(circle) | 									circles.push(circle); | ||||||
|  | 								} | ||||||
|  |  | ||||||
| 						window.addEventListener('eventRemoveRouteStartEnd', function handler(e) { | 								circle.on('click', function() { | ||||||
| 							circle.remove(); | 									PgBar.syncToPlotTravelHistory(i); | ||||||
|  | 								}); | ||||||
|  |  | ||||||
|  | 								// ✅ When last point processed, resolve | ||||||
|  | 								if (i === poly.length - 1) { | ||||||
|  | 									resolve(circles); | ||||||
|  | 								} | ||||||
|  | 							}); | ||||||
| 						}); | 						}); | ||||||
| 						circle.on('click', function() { | 					} | ||||||
| 							PgBar.syncToPlotTravelHistory(i); |  | ||||||
| 						}) | 					let circlesStartStop = []; | ||||||
| 					}); | 					for (let [idx, poly] of polyTruckRoutes.entries()) { | ||||||
|  | 						let polyline = Leaflet.addRoutes(poly); | ||||||
|  |  | ||||||
|  | 						// wait for all circles | ||||||
|  | 						circlesStartStop = await addCirclesAsync(poly); | ||||||
|  |  | ||||||
|  | 						// keep event + removal logic here | ||||||
|  | 						State.eventRemoveRouteStartEnd = new CustomEvent('eventRemoveRouteStartEnd', { | ||||||
|  | 							detail: { polyline, circlesStartStop } | ||||||
|  | 						}); | ||||||
|  |  | ||||||
|  | 						window.addEventListener('eventRemoveRouteStartEnd', function(e) { | ||||||
|  | 							polyline.remove(); | ||||||
|  | 							circlesStartStop.forEach(c => c.remove()); | ||||||
|  | 						}); | ||||||
|  | 					} | ||||||
| 					 | 					 | ||||||
| 					// // remove marker, circle, event listener and all about this marker | 					// // remove marker, circle, event listener and all about this marker | ||||||
| 					State.eventRemoveRouteStartEnd = new CustomEvent('eventRemoveRouteStartEnd', { | 					State.eventRemoveRouteStartEnd = new CustomEvent('eventRemoveRouteStartEnd', { | ||||||
| @ -2813,28 +2861,12 @@ | |||||||
| 					}); | 					}); | ||||||
|  |  | ||||||
| 					window.addEventListener('eventRemoveRouteStartEnd', function (e) { | 					window.addEventListener('eventRemoveRouteStartEnd', function (e) { | ||||||
| 					// console.log("circlesStartStop", circlesStartStop); |  | ||||||
| 						polyline.remove(); | 						polyline.remove(); | ||||||
| 						circlesStartStop.forEach(c => c.remove()) | 						circlesStartStop.forEach(c => c.remove()) | ||||||
| 					}); | 					}); | ||||||
| 					// State.eventRemoveRouteStartEnd = new CustomEvent('eventRemoveRouteStartEnd', { |  | ||||||
| 					// 	polyline, |  | ||||||
| 					// 	circlesStartStop, |  | ||||||
| 					// }); |  | ||||||
| 					// console.log("polyline", polyline); |  | ||||||
| 					// console.log("circlesStartStop", circlesStartStop); |  | ||||||
| 					// console.log("State.eventRemoveRouteStartEnd", State.eventRemoveRouteStartEnd); |  | ||||||
| 					 |  | ||||||
| 				}) | 				}) | ||||||
|  |  | ||||||
|                 window.addEventListener('eventRemoveRouteStartEnd', function handler(e) { |                 window.addEventListener('eventRemoveRouteStartEnd', function handler(e) { | ||||||
|                     // startMarker.removeEventListener('click'); |  | ||||||
|                     // startMarker.removeEventListener('moveend'); |  | ||||||
|                     // startMarker.remove(); |  | ||||||
|                     // finishMarker.removeEventListener('click'); |  | ||||||
|                     // finishMarker.removeEventListener('moveend'); |  | ||||||
|                     // finishMarker.remove(); |  | ||||||
|                     // polyline.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; | ||||||
|  | |||||||
							
								
								
									
										241
									
								
								resources/views/menu_v1/reports/abnormalities.blade.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										241
									
								
								resources/views/menu_v1/reports/abnormalities.blade.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,241 @@ | |||||||
|  | @php | ||||||
|  |     $user_role = Auth::user()->role; | ||||||
|  | @endphp | ||||||
|  |  | ||||||
|  | @extends('app.app') | ||||||
|  |  | ||||||
|  | @section('title') | ||||||
|  |     Vehicles | ||||||
|  | @endsection | ||||||
|  |  | ||||||
|  | @section('customcss') | ||||||
|  | @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">Abnormality Report</p> | ||||||
|  |                         </div> | ||||||
|  |                         @if ($user_role == \App\Models\Users::ROLE_VENDOR || $user_role == \App\Models\Users::ROLE_ADMIN) | ||||||
|  |                             @can('vehicle.create') | ||||||
|  |                                 <!-- <div class="col text-end"> | ||||||
|  |                                     <button id="btnMdlNewVhc" class="btn btn-sm btn-danger">Add New Vehicle</button> | ||||||
|  |                                 </div> --> | ||||||
|  |                             @endcan | ||||||
|  |  | ||||||
|  |                             {{-- <div class="col-auto text-end ps-0"> | ||||||
|  |                                 <button class="btn btn-sm btn-danger">Upload</button> | ||||||
|  |                             </div> --}} | ||||||
|  |                         @endif | ||||||
|  |                         <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 type="date" class="form-control" id="tgl0" value="{{ date('Y-m-d') }}"> --> | ||||||
|  | 								<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="{{ date('d-m-Y 23:59') }}"> | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 						<div class="col-2"> | ||||||
|  | 							<div class="form-group"> | ||||||
|  | 								<label class="text-muted">License Plate</label> | ||||||
|  | 								<select name="license_plate" class="form-control" id="filterNopol"> | ||||||
|  | 									<option value="">-- All --</option> | ||||||
|  | 									@foreach ($listNopol as $nopol) | ||||||
|  | 										<option value="{{ $nopol->id }}">{{ $nopol->nopol1 }}</option> | ||||||
|  | 									@endforeach | ||||||
|  | 								</select> | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 						<div class="col-1 d-flex align-items-end"> | ||||||
|  | 							<button class="btn btn-sm 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="tVehicleTrips" class="table table-hover dataTable w-100"> | ||||||
|  |                             <thead> | ||||||
|  |                                 <tr class=""> | ||||||
|  |                                     <!-- <th class="">Vehicle ID</th> --> | ||||||
|  |                                     <!-- <th class="text-center">Action</th> --> | ||||||
|  |                                     <th class="">Vehicle Name</th> | ||||||
|  |                                     <th class="">License Plate Number</th> | ||||||
|  |                                     <th class="">Time</th> | ||||||
|  |                                     <th class="">Speed (kph)</th> | ||||||
|  |                                     <th class="">Location</th> | ||||||
|  |                                 </tr> | ||||||
|  |                             </thead> | ||||||
|  |                         </table> | ||||||
|  |                     </div> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |     </div> | ||||||
|  |  | ||||||
|  | 	<!-- modal --> | ||||||
|  | @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> | ||||||
|  |         'use strict'; | ||||||
|  |         const State = { | ||||||
|  |             file_jimp_worker: "{{ asset('assets/js/worker/jimp.js') }}", | ||||||
|  |             storage_lara: "{{ asset('storage') }}/", | ||||||
|  |             dvc_type: { | ||||||
|  |                 built_in: "{{ \App\Models\Devices::TYPE_BUILT_IN }}", | ||||||
|  |                 portable: "{{ \App\Models\Devices::TYPE_PORTABLE }}", | ||||||
|  |             }, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         const Wrapper = { | ||||||
|  |             activate: function() { | ||||||
|  |                 Wrapper.init(); | ||||||
|  |                 Wrapper.event(); | ||||||
|  |                 DTable.activate(); | ||||||
|  |  | ||||||
|  | 			}, | ||||||
|  | 			init: ()=>{ | ||||||
|  | 				$('#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 = $('#tVehicleTrips').DataTable({ | ||||||
|  | 					searching: false,   // 🔹 remove search box | ||||||
|  | 					ordering: false,    // 🔹 disable sorting for all columns | ||||||
|  | 					processing: true, | ||||||
|  | 					serverSide: false, | ||||||
|  | 					bLengthChange: true, | ||||||
|  | 					deferRender: true, | ||||||
|  | 					ajax: function(data, callback, settings) { | ||||||
|  | 						// Abort previous | ||||||
|  | 						if (DTable.lastAjax) { | ||||||
|  | 							DTable.lastAjax.abort(); | ||||||
|  | 						} | ||||||
|  |  | ||||||
|  | 						// Fire new request | ||||||
|  | 						DTable.lastAjax = $.ajax({ | ||||||
|  | 							url: `{{ route('api_report_abnormalities_list') }}? | ||||||
|  | 								cptid=${AppState.current_company} | ||||||
|  | 								&from_date=${moment($('#tgl0').val(), "DD-MM-YYYY HH:mm").unix()} | ||||||
|  | 								&to_date=${moment($('#tgl1').val(), "DD-MM-YYYY HH:mm").unix()} | ||||||
|  | 								&vid=${$('#filterNopol').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: 'id', visible: false }, // vhc_id | ||||||
|  | 						{ | ||||||
|  | 							data: 'name', | ||||||
|  | 							className: 'text-start text-nowrap', | ||||||
|  | 						}, | ||||||
|  | 						{ | ||||||
|  | 							data: 'nopol1', | ||||||
|  | 							className: 'text-start', | ||||||
|  | 						}, | ||||||
|  | 						{  | ||||||
|  | 							data: "crt_d", | ||||||
|  | 							className: 'text-nowrap', | ||||||
|  | 							render: (data, type, row, meta) => { | ||||||
|  | 								return moment.unix(data).format('DD MMM YYYY HH:mm:ss'); | ||||||
|  | 							} | ||||||
|  | 						}, | ||||||
|  | 						{ data: "speed", className: 'text-end'}, | ||||||
|  | 						{  | ||||||
|  | 							data: "fulladdress", | ||||||
|  | 							render: (data, type, row, meta) => { | ||||||
|  | 								return Helper.shortenText(decodeURIComponent(data || ''), 255) | ||||||
|  | 							}, | ||||||
|  | 						}, | ||||||
|  | 					], | ||||||
|  | 					paging: false, | ||||||
|  | 					buttons: [ | ||||||
|  | 						{ | ||||||
|  | 							extend: 'excelHtml5', | ||||||
|  | 							title: `Abnormality Report - ${moment(safeVal('#tgl0')).format('DD MMM YYYY')} to ${moment(safeVal('#tgl1')).format('DD MMM YYYY')}`, | ||||||
|  | 							className: 'd-none', // hide default button | ||||||
|  | 							exportOptions: { | ||||||
|  | 								columns: ':visible:not(:first-child)' // 🔹 exclude first column | ||||||
|  | 							} | ||||||
|  | 						} | ||||||
|  | 					] | ||||||
|  | 				}); | ||||||
|  |             }, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         function safeVal(selector) { | ||||||
|  |             const val = $(selector).val(); | ||||||
|  |             return (val === undefined || val === null || val === '') ? null : val; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         Wrapper.activate(); | ||||||
|  |     </script> | ||||||
|  | @endsection | ||||||
| @ -43,30 +43,49 @@ | |||||||
| 							<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 type="date" class="form-control" id="tgl0" value="{{ date('Y-m-d') }}"> --> | ||||||
|  | 								<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 type="date" class="form-control" id="tgl1" value="{{ date('Y-m-d') }}"> | 								<input class="form-control" id="tgl1" value="{{ date('d-m-Y 23:59') }}"> | ||||||
| 							</div> | 							</div> | ||||||
| 						</div> | 						</div> | ||||||
| 						<div class="col-8 d-flex align-items-end"> | 						<div class="col-2"> | ||||||
|  | 							<div class="form-group"> | ||||||
|  | 								<label class="text-muted">License Plate</label> | ||||||
|  | 								<select name="license_plate" class="form-control" id="filterNopol"> | ||||||
|  | 									<option value="">-- All --</option> | ||||||
|  | 									@foreach ($listNopol as $nopol) | ||||||
|  | 										<option value="{{ $nopol->id }}">{{ $nopol->nopol1 }}</option> | ||||||
|  | 									@endforeach | ||||||
|  | 								</select> | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 						<div class="col-1 d-flex align-items-end"> | ||||||
|  | 							<button class="btn btn-sm 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> | 							<button class="btn btn-sm btn-danger ms-auto" id="btnDownloadReport">Download Report</button> | ||||||
| 						</div> | 						</div> | ||||||
| 					</div> | 					</div> | ||||||
| 					</div> | 					</div> | ||||||
|                     <div class="table-responsive p-3"> |                     <div class="table-responsive p-3"> | ||||||
|                         <table id="tVehicleTrips" class="table table-hover dataTable "> |                         <table id="tVehicleTrips" class="table table-hover dataTable w-100"> | ||||||
|                             <thead> |                             <thead> | ||||||
|                                 <tr class=""> |                                 <tr class=""> | ||||||
|                                     <!-- <th class="">#</th> --> |                                     <th class="">Vehicle ID</th> | ||||||
|                                     <!-- <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="">Number of Trip</th> |                                     <th class="">Number of Trip</th> | ||||||
|                                     <th class="">Total Milage (km)</th> |                                     <th class="">Total Mileage (km)</th> | ||||||
|  |                                     <th class="">Trip #</th> | ||||||
|  |                                     <th class="">Start</th> | ||||||
|  |                                     <th class="">Finish</th> | ||||||
|  |                                     <th class="">Mileage (km)</th> | ||||||
|                                 </tr> |                                 </tr> | ||||||
|                             </thead> |                             </thead> | ||||||
|                             <!-- <tbody> |                             <!-- <tbody> | ||||||
| @ -166,12 +185,22 @@ | |||||||
|  |  | ||||||
|         const Wrapper = { |         const Wrapper = { | ||||||
|             activate: function() { |             activate: function() { | ||||||
|  |                 Wrapper.init(); | ||||||
|                 Wrapper.event(); |                 Wrapper.event(); | ||||||
|                 DTable.activate(); |                 DTable.activate(); | ||||||
|  |  | ||||||
| 			}, | 			}, | ||||||
|  | 			init: ()=>{ | ||||||
|  | 				$('#tgl0, #tgl1').datetimepicker({ | ||||||
|  | 					format:'d-m-Y H:i', | ||||||
|  | 					defaultTime:'00:00', | ||||||
|  | 					closeOnDateSelect: true, | ||||||
|  | 					// mask:true | ||||||
|  | 				}); | ||||||
|  |  | ||||||
|  | 			}, | ||||||
|             event: function() { |             event: function() { | ||||||
|                 $('#tgl0, #tgl1').on('change', function() { |                 $('#submitFilter').on('click', function() { | ||||||
| 					DTable.reload(); | 					DTable.reload(); | ||||||
| 				}); | 				}); | ||||||
| 				// Trigger export on external button click | 				// Trigger export on external button click | ||||||
| @ -192,8 +221,19 @@ | |||||||
| 				if (DTable.lastAjax) { | 				if (DTable.lastAjax) { | ||||||
| 					DTable.lastAjax.abort(); | 					DTable.lastAjax.abort(); | ||||||
| 				} | 				} | ||||||
|  | 				if (DTable.table) { | ||||||
|  | 					// If table already exists → reload | ||||||
|  | 					DTable.table.ajax.reload(); | ||||||
|  | 					return; | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				let mileageSum = {}; | ||||||
|  | 				let tripCount = {}; | ||||||
|  |  | ||||||
|  |  | ||||||
| 				DTable.table = $('#tVehicleTrips').DataTable({ | 				DTable.table = $('#tVehicleTrips').DataTable({ | ||||||
|  | 					searching: false,   // 🔹 remove search box | ||||||
|  | 					ordering: false,    // 🔹 disable sorting for all columns | ||||||
| 					processing: true, | 					processing: true, | ||||||
| 					serverSide: false, | 					serverSide: false, | ||||||
| 					bLengthChange: true, | 					bLengthChange: true, | ||||||
| @ -209,10 +249,22 @@ | |||||||
| 						DTable.lastAjax = $.ajax({ | 						DTable.lastAjax = $.ajax({ | ||||||
| 							url: `{{ route('api_report_vehicle_trips_list') }}? | 							url: `{{ route('api_report_vehicle_trips_list') }}? | ||||||
| 								cptid=${AppState.current_company} | 								cptid=${AppState.current_company} | ||||||
| 								&from_date=${moment(safeVal('#tgl0')).startOf('day').unix()} | 								&from_date=${moment($('#tgl0').val(), "DD-MM-YYYY HH:mm").unix()} | ||||||
| 								&to_date=${moment(safeVal('#tgl1')).endOf('day').unix()}`, | 								&to_date=${moment($('#tgl1').val(), "DD-MM-YYYY HH:mm").unix()} | ||||||
|  | 								&vid=${$('#filterNopol').val() || ''}`, | ||||||
| 							type: 'GET', | 							type: 'GET', | ||||||
| 							success: function(json) { | 							success: function(json) { | ||||||
|  | 								// Precompute totals by vhc_id | ||||||
|  |  | ||||||
|  | 								json.data.forEach(row => { | ||||||
|  | 									const id = row.id; // vhc_id | ||||||
|  | 									if (!mileageSum[id]) { | ||||||
|  | 										mileageSum[id] = 0; | ||||||
|  | 										tripCount[id] = 0; | ||||||
|  | 									} | ||||||
|  | 									mileageSum[id] += parseFloat(row.milleage || 0); | ||||||
|  | 									tripCount[id] += 1; | ||||||
|  | 								}); | ||||||
| 								callback(json); | 								callback(json); | ||||||
| 							}, | 							}, | ||||||
| 							error: function(xhr, status, error) { | 							error: function(xhr, status, error) { | ||||||
| @ -227,6 +279,7 @@ | |||||||
| 					}, | 					}, | ||||||
|                     deferRender: true, |                     deferRender: true, | ||||||
| 					columns: [ | 					columns: [ | ||||||
|  | 						{ data: 'id', visible: false }, // vhc_id | ||||||
| 						{ | 						{ | ||||||
| 							data: 'name', | 							data: 'name', | ||||||
| 							className: 'text-start text-nowrap', | 							className: 'text-start text-nowrap', | ||||||
| @ -241,29 +294,88 @@ | |||||||
| 							orderable: true, | 							orderable: true, | ||||||
| 							searchable: true, | 							searchable: true, | ||||||
| 						}, | 						}, | ||||||
|  |  						{ data: "id", className: 'text-end'}, | ||||||
|  | 						{ data: "id", className: 'text-end'}, | ||||||
|  | 						{ data: "trip_id", className: 'text-end'}, | ||||||
| 						{  | 						{  | ||||||
| 							data: 'numOfTrip', | 							data: 'start',  | ||||||
| 							className: 'text-end', | 							render: (data, type, row, meta) => { | ||||||
| 							visible: true, | 								// let addr = row | ||||||
| 							orderable: true, | 								return ` | ||||||
| 							searchable: true, | 									${moment.unix(data).format('DD MMM YYYY HH:mm:ss')}<br> | ||||||
|  | 									${Helper.shortenText(decodeURIComponent(row.startLoc || 'address'), 255) || "-"} | ||||||
|  | 								`; | ||||||
|  | 							}  | ||||||
| 						}, | 						}, | ||||||
| 						{  | 						{  | ||||||
| 							data: 'total_milleage', | 							data: 'finish',  | ||||||
| 							className: 'text-end', | 							render: (data, type, row, meta) => { | ||||||
| 							visible: true, | 								// let addr = row | ||||||
| 							orderable: true, | 								return ` | ||||||
| 							searchable: false, | 									${moment.unix(data).format('DD MMM YYYY HH:mm:ss')}<br> | ||||||
| 							render: function(data, type, row, meta) { | 									${Helper.shortenText(decodeURIComponent(row.finishLoc || 'address'), 255) || "-"} | ||||||
|  | 								`; | ||||||
|  | 							}  | ||||||
|  | 						}, | ||||||
|  | 						{  | ||||||
|  | 							data: 'milleage',  | ||||||
|  | 							className: 'text-end', render: function(data, type, row, meta) { | ||||||
| 								return (data === null) ? '0' : parseFloat(data).toFixed(2); | 								return (data === null) ? '0' : parseFloat(data).toFixed(2); | ||||||
| 							} | 							} | ||||||
| 						}, | 						}, | ||||||
| 					], | 					], | ||||||
|  | 					paging: false, | ||||||
|  | 					drawCallback: function (settings) { | ||||||
|  | 						const api = this.api(); | ||||||
|  | 						const rows = api.rows({ page: 'current' }).nodes(); | ||||||
|  | 						let last = null; | ||||||
|  |  | ||||||
|  | 						api.column(0, { page: 'current' }) // vhc_id col | ||||||
|  | 							.data() | ||||||
|  | 							.each(function (group, i) { | ||||||
|  | 								if (last !== group) { | ||||||
|  | 									// count how many rows this vhc_id spans | ||||||
|  | 									const rowspanCount = api | ||||||
|  | 										.column(0, { page: 'current' }) | ||||||
|  | 										.data() | ||||||
|  | 										.filter(function (d) { | ||||||
|  | 											return d === group; | ||||||
|  | 										}).length; | ||||||
|  |  | ||||||
|  | 									// merge vhc_id | ||||||
|  | 									$(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(2)').attr('rowspan', rowspanCount); | ||||||
|  | 									$(rows).eq(i).find('td:eq(3)').attr('rowspan', rowspanCount); | ||||||
|  |  | ||||||
|  | 									// fill trip count | ||||||
|  | 									$(rows).eq(i).find('td:eq(2)') | ||||||
|  | 										.attr('rowspan', rowspanCount) | ||||||
|  | 										.text(tripCount[group]); | ||||||
|  |  | ||||||
|  | 									// fill mileage sum | ||||||
|  | 									$(rows).eq(i).find('td:eq(3)') | ||||||
|  | 										.attr('rowspan', rowspanCount) | ||||||
|  | 										.text(mileageSum[group].toFixed(2)); | ||||||
|  |  | ||||||
|  | 									last = group; | ||||||
|  | 								} else { | ||||||
|  | 									// remove duplicate cells | ||||||
|  | 									$(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(); | ||||||
|  | 								} | ||||||
|  | 							}); | ||||||
|  | 						}, | ||||||
| 					buttons: [ | 					buttons: [ | ||||||
| 						{ | 						{ | ||||||
| 							extend: 'excelHtml5', | 							extend: 'excelHtml5', | ||||||
| 							title: `Vehicle Trip Report - ${moment(safeVal('#tgl0')).format('DD MMM YYYY')} to ${moment(safeVal('#tgl1')).format('DD MMM YYYY')}`, | 							title: `Vehicle Trip Report - ${moment(safeVal('#tgl0')).format('DD MMM YYYY')} to ${moment(safeVal('#tgl1')).format('DD MMM YYYY')}`, | ||||||
| 							className: 'd-none' // hide default button | 							className: 'd-none', // hide default button | ||||||
|  | 							exportOptions: { | ||||||
|  | 								columns: ':visible:not(:first-child)' // 🔹 exclude first column | ||||||
|  | 							} | ||||||
| 						} | 						} | ||||||
| 					] | 					] | ||||||
| 				}); | 				}); | ||||||
|  | |||||||
| @ -51,6 +51,9 @@ | |||||||
| 							<li> | 							<li> | ||||||
| 								<a class="dropdown-item {{ Request::segment(2) == 'vehicle-trips' ? 'active' : '' }}" href="{{ route('view_report_vehicle_trips') }}" title="">Vehicle Trips</a> | 								<a class="dropdown-item {{ Request::segment(2) == 'vehicle-trips' ? 'active' : '' }}" href="{{ route('view_report_vehicle_trips') }}" title="">Vehicle Trips</a> | ||||||
| 							</li> | 							</li> | ||||||
|  | 							<li> | ||||||
|  | 								<a class="dropdown-item {{ Request::segment(2) == 'abnormalities' ? 'active' : '' }}" href="{{ route('view_report_abnormalities') }}" title="">Abnormalities</a> | ||||||
|  | 							</li> | ||||||
|                         </ul> |                         </ul> | ||||||
|                     </li> |                     </li> | ||||||
|                 @endif |                 @endif | ||||||
|  | |||||||
| @ -156,6 +156,8 @@ Route::middleware(["auth", "auth.user"])->group(function () { | |||||||
| 	// reports | 	// reports | ||||||
|     Route::get("/reports/vehicle-trips", "ReportsController@view_report_vehicle_trips")->name("view_report_vehicle_trips"); |     Route::get("/reports/vehicle-trips", "ReportsController@view_report_vehicle_trips")->name("view_report_vehicle_trips"); | ||||||
|     Route::get("/reports/vehicle-trips-list", "ReportsController@api_report_vehicle_trips_list")->name("api_report_vehicle_trips_list"); |     Route::get("/reports/vehicle-trips-list", "ReportsController@api_report_vehicle_trips_list")->name("api_report_vehicle_trips_list"); | ||||||
|  |     Route::get("/reports/abnormalities", "ReportsController@view_report_abnormalities")->name("view_report_abnormalities"); | ||||||
|  |     Route::get("/reports/abnormalities-list", "ReportsController@api_report_abnormalities_list")->name("api_report_abnormalities_list"); | ||||||
|  |  | ||||||
|  |  | ||||||
|     Route::get("/user/vendor/transactions", "MenuController@view_user_vendor_transaction")->name( |     Route::get("/user/vendor/transactions", "MenuController@view_user_vendor_transaction")->name( | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user
	