Files
gps-frontend/resources/views/menu_v1/dashboard.blade.php
Pringgosutono 136db1a158 time fix
2025-10-08 12:15:14 +07:00

3509 lines
175 KiB
PHP
Executable File

@extends('app.app')
@section('title')
Dashboard
@endsection
@section('customcss')
{{-- <link rel="stylesheet" href="https://unpkg.com/leaflet-routing-machine@3.2.12/dist/leaflet-routing-machine.css" /> --}}
{{-- <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/leaflet.draw/1.0.4/leaflet.draw.css" integrity="sha512-gc3xjCmIy673V6MyOAZhIW93xhM9ei1I+gLbmFjUHIjocENRsLX/QUE1htk5q1XV2D/iie/VQ8DXI6Vu8bexvQ==" crossorigin="anonymous" referrerpolicy="no-referrer" /> --}}
<link rel="stylesheet" href="{{ asset('assets/css/r.css') }}" />
<style>
.infoMove-plots li:hover {
/* background: #ffd25c; */
background: rgba(229, 41, 32, .25);
}
.infoMove-plots li.hover {
/* background: #ffd25c; */
background: rgba(229, 41, 32, .25);
}
.leaflet-control-container .leaflet-left {
display: none;
}
.leaflet-routing-container {
display: none !important;
}
#openLeftSideBar1Mobile {
display: none;
}
.ptName {
display: none;
}
@media (max-width: 425px) {
#openLeftSideBar1 {
display: none !important;
}
#openLeftSideBar1Mobile {
display: block !important;
}
.ptName {
display: block;
}
}
</style>
@endsection
@section('content')
<div class="container-fluid bg-white">
<div class="row">
{{-- list zones --}}
<div class="col-sm-2 bg-white overflow-auto panel-right border-end d-none" id="leftSideBar1">
<div class="row d-flex align-items-center mb-3">
<div class="col text-start">
<a href="javascript:void(0);" class="text-decoration-none text-dark" id="closeLeftSideBar1">
<span class="icon ion-ios-arrow-left fz-18"></span>
</a>
</div>
<div class="col text-end">
<p class="mb-0 text-bold">Zone List (<span id="c_list_zone">2</span>)</p>
</div>
</div>
<div class="row d-flex align-items-center mb-3 ptName">
<div class="col text-dark text-bold"><span id="ptName">{{ $client_group->c_name }}</span></div>
</div>
<div class="row d-flex align-items-center mb-3">
<div class="col text-danger"><a id="clearZone" href="javascript:void(0);"><small>hide zone</small></a></div>
</div>
<div class="form-group mb-3">
<input id="filterZoneSearch" type="text" class="form-control form-control-sm" placeholder="Search Zone">
</div>
<ul id="listZone" class="list-group">
<li class="list-group-item vehicles-list-wrapper zone-item p-1 px-2">
<a href="#" class="text-dark">
<div class="row d-flex align-items-center">
<div class="col-3 d-flex justify-content-center">
<div class="align-items-center d-flex justify-content-center zone-icon" style="background: #B600FF">
<span class="ion-android-pin text-white"></span>
</div>
</div>
<div class="col ps-0">
<p class="text-bold mb-0">BRK001</p>
<p class="text-muted mb-0">Warehouse</p>
</div>
</div>
</a>
</li>
<li class="list-group-item vehicles-list-wrapper p-1 px-2">
<a href="#" class="text-dark">
<div class="row d-flex align-items-center">
<div class="col-3 d-flex justify-content-center">
<div class="align-items-center d-flex justify-content-center zone-icon" style="background: #B600FF">
<span class="ion-android-pin text-white"></span>
</div>
</div>
<div class="col ps-0">
<p class="text-bold mb-0">BRK002</p>
<p class="text-muted mb-0">Warehouse</p>
</div>
</div>
</a>
</li>
</ul>
</div>
{{-- zone detail --}}
<div class="col-sm-2 bg-white overflow-auto panel-right border-end d-none" id="leftSideBar2">
<div class="text-end">
<a href="javascript:void(0);" class="text-decoration-none text-danger" id="closeLeftSideBar2">
<span class="icon ion-android-close fz-18"></span>
</a>
</div>
<div class="card shadow-none border mb-3">
<div class="card-body p-2">
<div class="row d-flex align-items-center">
<div class="col-3 d-flex justify-content-center">
<div id="zd-boundary_hex_color" class="align-items-center d-flex justify-content-center zone-icon" style="background: #B600FF">
<span class="ion-android-pin text-white"></span>
</div>
</div>
<div class="col ps-0">
<p id="zd-name" class="text-bold mb-0">BRK001</p>
<p id="zd-type_name" class="text-muted mb-0">Warehouse</p>
</div>
</div>
</div>
</div>
<ul class="list-group">
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Client</p>
<p id="zd-client" class="text-muted mb-0">JNE</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Tipe</p>
<p id="zd-type_name1" class="text-muted mb-0">Warehouse</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Jenis Zona</p>
<p id="zd-workflow_type_name" class="text-muted mb-0">Pickup</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Alamat</p>
<p id="zd-fulladdress" class="text-muted mb-0">Jl. Letjen Mt. Haryono No.Kav. 20, RW.1, Cawang, Kec.
Kramat jati, Kota Jakarta Timur, Daerah Khusus Ibukota Jakarta</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Kode Pos</p>
<p id="zd-shiptocode" class="text-muted mb-0">13630</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Pembaharuan Terakhir</p>
<p class="text-muted mb-0">
<span id="zd-updt_at">22-Sep-2020 04:43:43</span>
<span id="zd-updt_by">RTadmin</span>
</p>
</li>
</ul>
</div>
<div class="col px-0">
<div id="map" style="{{ Auth::user()->role != 2 ? 'height: calc(100vh - 90.52px) !important' : '' }}">
{{-- client logo --}}
{{-- <div class="card floating-menu-left mb-0" id="openLeftSideBar1">
<div class="card-body">
<div class="cNameGroup row d-flex align-items-center justify-content-between mb-1">
<div class="col-auto">
<p id="c_name" class="mb-0 fz-14">{{ $client_group->c_name }}</p>
</div>
<div class="col-1 me-1">
<a href="javascript:void(0);" class="text-decoration-none text-dark fz-18">
<span class="icon ion-ios-arrow-right"></span>
</a>
</div>
</div>
<div class="cLogoGroup row">
<div class="col-11 text-center ms-2">
@if ($client_group)
<img id="c_logo" src="{{ asset('storage/' . $client_group->c_logo) ?? asset('images/swa-nusa-logo.png') }}" alt="Client logo" class="img-fluid thumb-img-landscape-med">
@else
<img id="c_logo" src="{{ asset('images/swa-nusa-logo.png') }}" alt="Client logo" class="img-fluid" style="height: 40px">
@endif
</div>
</div>
</div>
</div> --}}
<a href="javascript:void(0);" class="btn btn-sm bg-white text-decoration-none text-dark fz-18 floating-menu-left" id="openLeftSideBar1" style="width:38px;">
<span class="icon ion-ios-arrow-right"></span>
</a>
<a href="javascript:void(0);" class="btn btn-sm bg-white text-decoration-none text-dark fz-18 floating-menu-left" id="openLeftSideBar1Mobile" style="width:38px;">
<span class="icon ion-ios-arrow-right"></span>
</a>
<a href="javascript:void(0);" class="btn btn-sm bg-white text-decoration-none text-dark fz-18 floating-menu-right d-none" id="openRightSideBar1">
<span class="icon ion-ios-arrow-left"></span>
</a>
<div id="menuPlayback" class="floating-menu-bottom bg-light mb-0 disablePanning d-none">
<div class="row">
<div id="pgplay" class="col-auto ps-4 pe-0" style="margin-top: 0.15rem;">
<i id="icon-pg" class="ion ion-play icon-pg"></i>
</div>
<div class="col">
<div id="pgcontainer" class="slidecontainer">
<input type="range" min="0" max="1000" value="0" class="slider" id="pgbar">
</div>
{{-- <div id="pgcontainer" class="pgcontainer">
<div id="pgbar" class="pgbar"></div>
</div> --}}
</div>
</div>
</div>
</div>
</div>
{{-- vehicle detail --}}
<div class="col-sm-2 bg-white overflow-auto panel-right border-end d-none" style="{{ Auth::user()->role != 2 ? 'height: calc(100vh - 90.52px) !important' : '' }}" id="rightSideBar2">
<div class="text-end">
<a href="javascript:void(0);" class="text-decoration-none text-danger" id="closeRightSideBar2"><span class="icon ion-android-close fz-18"></span></a>
</div>
<div class="card shadow-none border">
<div class="card-body p-2">
<div class="align-items-center d-flex row">
<div class="col-4">
<img id="infoVehicles-thumb-sm" src="https://products.unitedtractors.com/wp-content/uploads/2021/03/udtruck-GWE370.png" class="img-fluid thumb-img-small" alt="">
</div>
<div class="col ps-0">
<p id="infoVehicles-platno" class="text-bold mb-0">B 1201 SYX</p>
<p id="infoVehicles-addr-title" class="text-muted mb-0">Kebon Baru - Tebet</p>
</div>
</div>
</div>
</div>
<div class="form-group mb-3 w-100">
<select name="" class="form-control select2" style="width: 100%;" id="selectFiter">
<option value="1">Vehicle</option>
<option value="2">Driver</option>
<option value="3">Job</option>
<option value="4">Travel History</option>
</select>
</div>
<ul class="list-group" id="infoVehicles">
<li class="list-group-item p-1">
<img id="infoVehicles-thumb-md" src="https://products.unitedtractors.com/wp-content/uploads/2021/03/udtruck-GWE370.png" class="img-fluid thumb-img-landscape" alt="">
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Timestamp</p>
<p id="infoVehicles-crt" class="text-muted mb-0">20-Jan-2022 23:16:18</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Current Location</p>
<p id="infoVehicles-addr" class="text-muted mb-0">Kebon Baru - Tebet</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Latitude, Longitude</p>
<p id="infoVehicles-lat_lng" class="text-muted mb-0">-6.27013, 106.731371</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Vehicle Mileage (km)</p>
<p id="infoVehicles-mileage" class="text-muted mb-0">45080.83</p>
</li>
{{-- <li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Batas kecepatan (kph)</p>
<p id="infoVehicles-speedLimit" class="text-muted mb-0">100</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Keccepatan Saat Ini (kph)</p>
<p id="infoVehicles-currentSpeed" class="text-muted mb-0">60</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Pengapian</p>
<p id="infoVehicles-ignition" class="text-muted mb-0">ON</p>
</li> --}}
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Engine Status</p>
<p id="infoVehicles-engineStatus" class="text-muted mb-0">Idling</p>
</li>
{{-- <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 id="infoVehicles-idlingDuration" class="text-muted mb-0">2946</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Zona Peringatan</p>
<p id="infoVehicles-alertZones" class="text-muted mb-0 d-block">Pasar Induk Cibitung, RAPID Office,
Rumah Ranko KBB, toll
Jorr</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Lacak di Hari Libur</p>
<p id="infoVehicles-isTrackHoliday" class="text-muted mb-0">Enabled</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Jadwal Pelacakan</p>
<p id="infoVehicles-trackSch" class="text-muted mb-0">24/7</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Perusahaan</p>
<p id="infoVehicles-company" class="text-muted mb-0">Swa Nusa Multimedia</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Tanggal Bergabung</p>
<p id="infoVehicles-serviceStart" class="text-muted mb-0">23-Aug-2018</p>
</li> --}}
</ul>
<ul class="list-group" id="infoDriver">
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Driver Information</p>
<p id="infoDrv-updt" class="text-muted mb-0">20-Jan-2022 23:16:18</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Driver Name</p>
<p id="infoDrv-name" class="text-muted mb-0">Emrsyf</p>
</li>
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Phone</p>
<p class="mb-0">
<a href="tel:0" id="infoDrv-phone1-tel">
<i class="text-dark ion-ios-telephone"></i>&nbsp;
<span class="infoDrv-phone1-text"></span>
</a>
</p>
<p class="mb-0">
<a href="https://api.whatsapp.com/send/?phone=62&text=Halo&app_absent=0" id="infoDrv-phone1-wa" target="_blank"><i class="text-success ion-social-whatsapp"></i>
<span class="infoDrv-phone1-text"></span>
</a>
</p>
</li>
{{-- <li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Phone 2</p>
<p class="mb-0">
<a href="tel:0" id="infoDrv-phone2-tel">
<i class="text-dark ion-ios-telephone"></i>&nbsp;
<span class="infoDrv-phone2-text"></span>
</a>
</p>
<p class="mb-0">
<a href="https://api.whatsapp.com/send/?phone=62&text=Halo&app_absent=0" id="infoDrv-phone2-wa" target="_blank"><i class="text-success ion-social-whatsapp"></i>
<span class="infoDrv-phone2-text"></span>
</a>
</p>
</li> --}}
</ul>
<div id="infoJob">
{{-- @for ($i = 0; $i < 20; $i++)
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">#902192102910</p>
<div class="row d-flex align-items-center justify-content-between">
<div class="col-auto">
<small class="text-muted">Client name</small>
<p id="" class="mb-0">JNE</p>
</div>
<div class="col-auto">
<small class="text-muted">Destination</small>
<p id="" class="mb-0">JKT-01</p></div>
</div>
</li>
@endfor --}}
{{-- <ul class="list-group mb-3">
<li class="list-group-item p-1 px-2">
<div class="row d-flex align-items-center justify-content-between">
<div class="col-12">
<small class="text-muted">Client</small>
<p id="infoJob-ord_client_name" class="mb-0">PT</p>
</div>
</div>
</li>
</ul> --}}
<ul class="list-group mb-3">
<li class="list-group-item p-1 px-2">
{{-- <p class="text-bold mb-0">#902192102910</p> --}}
<div class="row d-flex align-items-center justify-content-between">
<div class="col-12">
<small class="text-muted">Origin</small>
{{-- <p id="infoJob-pck_city" class="">Jakarta</p> --}}
<p id="infoJob-pck_name" class="mb-1 text-bold">JKT-01</p>
<p id="infoJob-pck_addr" class="mb-0">Jl. Pancoran Timur Raya No.9, RT.8/RW.9, Pancoran, Kec. Pancoran, Kota Jakarta Selatan, Daerah Khusus Ibukota Jakarta 12780</p>
</div>
</div>
</li>
<li class="list-group-item p-1 px-2">
{{-- <p class="text-bold mb-0">#902192102910</p> --}}
<div class="row d-flex align-items-center justify-content-between">
<div class="col-12">
<small class="text-muted">Destination</small>
{{-- <p id="infoJob-drop_city" class="">Indramayu</p> --}}
<p id="infoJob-drop_name" class="mb-1 text-bold">IND-01</p>
<p id="infoJob-drop_addr" class="mb-0">Jl. Jend. Sudirman, Karanganyar, Kec. Indramayu, Kabupaten Indramayu, Jawa Barat</p>
</div>
</div>
</li>
</ul>
<ul class="list-group">
<li class="list-group-item p-1 px-2">
{{-- <p class="text-bold mb-0">#902192102910</p> --}}
<div class="row d-flex align-items-center justify-content-between">
<div class="col-12">
<small class="text-muted">Status</small>
<p id="infoJob-ord_stts" class="mb-0">On Going</p>
</div>
</div>
</li>
</ul>
</div>
<ul class="list-group" id="infoMovement">
<li class="list-group-item p-1 px-2 mb-2" style="border-radius: 0.25rem;">
<p class="text-bold mb-0">Trip History</p>
<p id="infoVehicles-infoMove" class="text-muted mb-0">Most Recent</p>
</li>
<div>
<div class="mb-2">
<label class="text-muted">From</label>
<input class="form-control" id="historyStartDate">
<!-- <input class="form-control" id="historyStartDate" value="11-08-2025 00:00"> -->
</div>
<div class="mb-2">
<label class="text-muted">To</label>
<input class="form-control" id="historyEndDate">
<!-- <input class="form-control" id="historyEndDate" value="11-08-2025 23:00"> -->
</div>
<button class="btn btn-primary btn-sm w-100 mb-3" id="historySearch">Search</button>
</div>
<div id="infoMove-plots" class="infoMove-plots">
{{-- <a href="#" class="plotMove-item">
<li class="list-group-item p-1 px-2">
<p class="text-bold mb-0">Number</p>
<p class="text-muted mb-0">${Number(obj.latitude).toFixed(6)} - ${Number(obj.longitude).toFixed(6)}</p>
<p class="text-muted mb-0">${moment.unix(obj?.lst_loc_crt).format('DD MMM YYYY HH:mm')}</p>
</li>
</a> --}}
</div>
</ul>
</div>
{{-- vehicle list & filter --}}
<div class="col-sm-2 bg-white overflow-auto panel-right" style="{{ Auth::user()->role != 2 ? 'height: calc(100vh - 90.52px) !important' : '' }}" id="rightSideBar1">
{{-- filter --}}
<div class="row d-flex align-items-center mb-3 justify-content-between">
<div class="col-auto">
<div class="row">
<div class="col-auto text-start align-items-center d-flex">
<a href="javascript:void(0);" class="mb-0 text-bold text-dark" id="filterVhcZone" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Filter Vehicles Within and Outside the Zone">
<span class="ion-android-car me-1" style="font-size: 17px">
</span>
</a>
<span id="c_vhc_zone">-</span>
</div>
<div class="col-auto text-start align-items-center d-flex">
<a href="javascript:void(0);" class="mb-0 text-bold text-dark me-2" id="filterVhcMarker" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Marker Info">
<span class="ion-android-pin me-2 text-muted" style="font-size: 17px"></span>
</a>
</div>
{{-- <div class="col-auto text-start align-items-center d-flex">
<a href="javascript:void(0);" class="mb-0 text-bold text-dark" id="filterVhcGps" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Vehicle GPS Filter">
<span class="ion-pinpoint me-1 text-muted" style="font-size: 17px"></span>
</a>
<span id="c_vhc_gps">-</span>
</div>
<div class="col-auto text-start align-items-center d-flex">
<a href="javascript:void(0);" class="mb-0 text-bold text-dark" id="filterVhcGsm" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Vehicle GSM Filter">
<span class="ion-android-wifi me-1 text-muted" style="font-size: 17px"></span>
</a>
<span id="c_vhc_gsm">-</span>
</div> --}}
</div>
</div>
<div class="col-auto">
<div class="">
<div class="col text-end align-items-center d-flex">
{{-- <a href="javascript:void(0);" class="mb-0 text-bold text-dark me-2" id="filterVhcMarker" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Marker Info">
<span class="ion-android-pin me-2 text-muted" style="font-size: 17px"></span>
</a> --}}
<a href="javascript:void(0);" class="text-decoration-none text-dark fz-18" id="closeRightSideBar1">
<span class="icon ion-ios-arrow-right"></span>
</a>
</div>
</div>
</div>
</div>
<div class="form-group mb-3">
<input type="text" id="filterVhcSearch" class="form-control form-control-sm" placeholder="Search Vehicle">
</div>
{{-- vehicle list --}}
<ul id="listTrucks" class="list-group">
{{-- <li class="list-group-item vehicles-list-wrapper p-1 px-2">
<a href="#" class="text-dark">
<div class="row d-flex align-items-center">
<div class="col-3">
<img src="https://products.unitedtractors.com/wp-content/uploads/2021/03/udtruck-GWE370.png"
class="img-fluid" alt="">
</div>
<div class="col ps-0">
<p class="text-bold mb-0">B 1201 SYX</p>
<p class="text-muted mb-0">Kebon Baru - Tebet</p>
</div>
</div>
</a>
</li>
<li class="list-group-item vehicles-list-wrapper p-1 px-2">
<a href="#" class="text-dark">
<div class="row d-flex align-items-center">
<div class="col-3">
<img src="https://products.unitedtractors.com/wp-content/uploads/2021/03/udtruck-GWE370.png"
class="img-fluid" alt="">
</div>
<div class="col ps-0">
<p class="text-bold mb-0">B 1202 SYX</p>
<p class="text-muted mb-0">Kebon Baru - Tebet</p>
</div>
</div>
</a>
</li> --}}
</ul>
</div>
</div>
</div>
<div class="footer-dashboard">
<div class="container-fluid">
<div class="row">
<div class="col d-flex align-content-center">
<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-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>
@endsection
@section('customjs')
<script src="{{ asset('assets/vendor/leaflet-1.7.1/leaflet-src.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://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://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script>
<script>
'use strict';
const State = {
isDrawPolygon: 0,
timingRouteAnimation: 100,
storage_lara: "{{ asset('storage') }}/",
ord_code: 0,
zone_boundary_type: {
circle: "{{ App\Models\Zone::ZONE_BOUNDARY_CIRCLE }}",
polygon: "{{ App\Models\Zone::ZONE_BOUNDARY_POLYGON }}",
rectangle: "{{ App\Models\Zone::ZONE_BOUNDARY_RECTANGLE }}",
},
zone_status: {
active: "{{ App\Models\Zone::STATUS_ACTIVE }}",
inactive: "{{ App\Models\Zone::STATUS_INACTIVE }}",
},
stts_engine: {
idling: "{{ App\Models\Tracks::STTS_EN_IDLING }}",
moving: "{{ App\Models\Tracks::STTS_EN_MOVING }}",
stoping: "{{ App\Models\Tracks::STTS_EN_STOPING }}",
},
stts_gps: {
active: "{{ App\Models\Tracks::STTS_GPS_ON }}",
inactive: "{{ App\Models\Tracks::STTS_GPS_OFF }}",
},
stts_gsm: {
no_signal: "{{ App\Models\Tracks::STTS_GSM_NO_SIGNAL }}",
bad_signal: "{{ App\Models\Tracks::STTS_GSM_BAD_SIGNAL }}",
weak_signal: "{{ App\Models\Tracks::STTS_GSM_WEAK_SIGNAL }}",
good_signal: "{{ App\Models\Tracks::STTS_GSM_GOOD_SIGNAL }}",
strong_signal: "{{ App\Models\Tracks::STTS_GSM_STRONG_SIGNAL }}",
},
stts_ignition: {
active: "{{ App\Models\Tracks::STTS_IGNITION_ON }}",
inactive: "{{ App\Models\Tracks::STTS_IGNITION_OFF }}",
low: "{{ App\Models\Tracks::STTS_IGNITION_LOW }}",
high: "{{ App\Models\Tracks::STTS_IGNITION_HIGH }}",
},
stts_filterDetail: {
vehicles: 1,
driver: 2,
assignJob: 3,
lastMove: 4,
},
stts_order: {
wait: "{{ App\Models\Orders::STTS_WAIT }}",
confirm: "{{ App\Models\Orders::STTS_CONFIRM }}",
have_get_vhc: "{{ App\Models\Orders::STTS_HAVE_GET_VHC }}",
pck: "{{ App\Models\Orders::STTS_PCK }}",
go: "{{ App\Models\Orders::STTS_GO }}",
arv: "{{ App\Models\Orders::STTS_ARV }}",
drop: "{{ App\Models\Orders::STTS_DROP }}",
client_pay: "{{ App\Models\Orders::STTS_CLIENT_PAY }}",
vendor_payed: "{{ App\Models\Orders::STTS_VENDOR_PAYED }}",
close: "{{ App\Models\Orders::STTS_CLOSE }}",
cancel: "{{ App\Models\Orders::STTS_CANCEL }}",
},
statusFilterPopupVhcMarker: 1, // 0=>hidden,1=>nopol,2=>speed,3=>address
eventFilterPopupVhcMarker: null,
statusFilterVhcGps: 0, // 0=>all,1=>active,2=>not
lastStatusFilterVhcGps: 0,
isShowToltipFilterVhcGps: 0,
eventFilterVhcGps: null,
statusFilterVhcGsm: 0, // 0=>all,1=>active,2=>not
lastStatusFilterVhcGsm: 0,
isShowToltipFilterVhcGsm: 0,
eventFilterVhcGsm: null,
statusFilterVhcZone: 0, // 0=>all,1=>inside,2=>outside
lastStatusFilterVhcZone: 0,
isShowToltipFilterVhcZone: 1,
eventFilterVhcZone: null,
eventRemoveListTrucks: null,
eventRemoveRouteStartEnd: null,
eventRemoveDetailPlotMovement: null,
eventHideTruckNotSelected: null,
eventHideAllTruck: null,
inShowVid: null,
inShowLastMove: null,
loadedLastMoveVid: null,
lastShowVids: [],
inShowZid: null,
delay_typing_front: 1000,
delay_hideTruckNotSelected: 1000,
delay_auto_refresh: 1 * (60 * 1000), // per n minute
historyStartDate: '',
historyEndDate: '',
isPanning: true,
playback: {
stts: 0, // 0=>not,1=>play,2=>pause,3=>resume,4=>stop
crntIndicator: 0,
minIndicator: 0,
maxIndicator: 0,
intrvl: null, // setInterval()
intrvlTime: 1000, // 200 per ms
},
};
const Icon = {
hub: function() {
return L.icon({
iconUrl: "{{ asset('assets/icons/pin-resto.png') }}",
iconSize: [38, 48],
iconAnchor: [20, 46], // lb, rt, bottom, rb. Positive
popupAnchor: [0, -32],
// shadowUrl: 'my-icon-shadow.png',
// shadowSize: [68, 95],
// shadowAnchor: [22, 94]
});
},
destination: function() {
return L.icon({
iconUrl: "{{ asset('assets/icons/pin-tujuan.png') }}",
iconSize: [38, 48],
iconAnchor: [20, 46], // lb, rt, bottom, rb. Positive
popupAnchor: [0, -32],
// shadowUrl: 'my-icon-shadow.png',
// shadowSize: [68, 95],
// shadowAnchor: [22, 94]
});
},
point: function() {
return L.icon({
// iconUrl: 'https://www.nicepng.com/png/full/101-1015767_map-marker-circle-png.png',
iconUrl: "{{ asset('assets/icons/maps-marker-circle-blue-radius.png') }}",
iconSize: [38, 38],
// iconAnchor: [20, 46], // lb, rt, bottom, rb. Positive
// popupAnchor: [0, -32],
// shadowUrl: 'my-icon-shadow.png',
// shadowSize: [68, 95],
// shadowAnchor: [22, 94]
});
},
startNav: function() {
return L.icon({
iconUrl: 'https://image.flaticon.com/icons/png/512/803/803559.png',
iconSize: [38, 38],
iconAnchor: [20, 16], // lb, rt, bottom, rb. Positive
// popupAnchor: [0, -32],
// shadowUrl: 'my-icon-shadow.png',
// shadowSize: [68, 95],
// shadowAnchor: [22, 94]
});
},
titikAwal: function() {
return L.icon({
iconUrl: "{{ asset('images/start.png') }}",
iconSize: [30, 30],
iconAnchor: [15, 28], // lb, rt, bottom, rb. Positive
// popupAnchor: [0, -32],
// shadowUrl: 'my-icon-shadow.png',
// shadowSize: [68, 95],
// shadowAnchor: [22, 94]
});
},
titikAkhir: function() {
return L.icon({
iconUrl: "{{ asset('images/finish.png') }}",
iconSize: [30, 30],
iconAnchor: [15, 28], // lb, rt, bottom, rb. Positive
// popupAnchor: [0, -32],
// shadowUrl: 'my-icon-shadow.png',
// shadowSize: [68, 95],
// shadowAnchor: [22, 94]
});
},
vechicleIcon: function() {
return L.icon({
iconUrl: "{{ asset('images/2active.png') }}",
iconSize: [30, 30],
iconAnchor: [15, 28], // lb, rt, bottom, rb. Positive
popupAnchor: [-3, -35],
// shadowUrl: 'my-icon-shadow.png',
// shadowSize: [68, 95],
// shadowAnchor: [22, 94]
});
},
vhcIdleIcon: function() {
return L.icon({
iconUrl: "{{ asset('images/2idle.png') }}",
iconSize: [30, 30],
iconAnchor: [15, 28], // lb, rt, bottom, rb. Positive
popupAnchor: [-3, -35],
// shadowUrl: 'my-icon-shadow.png',
// shadowSize: [68, 95],
// shadowAnchor: [22, 94]
});
},
vhcStopIcon: function() {
return L.icon({
iconUrl: "{{ asset('images/2stop.png') }}",
iconSize: [30, 30],
iconAnchor: [15, 28], // lb, rt, bottom, rb. Positive
popupAnchor: [-3, -35],
// shadowUrl: 'my-icon-shadow.png',
// shadowSize: [68, 95],
// shadowAnchor: [22, 94]
});
},
};
const Wrapper = {
activate: function() {
Wrapper.init();
Leaflet.activate();
Menu.activate();
Trucks.activate();
Filter.activate();
PgBar.activate();
},
init: function() {
const queryString = Helper.initQueryString();
State.ord_code = queryString.ord_code;
if (Wrapper.checkIsMobileDevice()) {
$('#rightSideBar1').addClass('d-none');
$('#openRightSideBar1').removeClass('d-none');
}
$('#historyStartDate, #historyEndDate').datetimepicker({
format:'d-m-Y H:i',
defaultTime:'00:00',
closeOnDateSelect: true,
// mask:true
});
},
calcIdlingDur: function(lst_idle_unix, lst_speed) {
let now = moment();
let idlingDur = 0;
let last_idle = moment.unix(lst_idle_unix);
let unit = 'min';
if (lst_speed <= 0 || lst_speed == null || lst_speed == '' || typeof lst_speed == 'undefined') {
if (last_idle.isValid()) {
let diff = moment.duration(now.diff(last_idle));
idlingDur = diff.asMinutes();
// parsing to hour
if (idlingDur > 60) {
idlingDur = idlingDur / 60;
unit = 'hour';
}
}
}
return {
unit,
dur: idlingDur.toFixed(0),
};
},
calcPeriodeDate: function(startDate, endDate) {
if (startDate.format('YYYY-MM-DD HH:mm') === 'Invalid date') {
Helper.toast('Warning', 'just now', 'Start date of the period is not valid');
return {
isValid: false
};
}
if (endDate.format('YYYY-MM-DD HH:mm') === 'Invalid date') {
Helper.toast('Warning', 'just now', 'End date of the period is not valid');
return {
isValid: false
};
}
// endDate.add(23, 'hour').add(59, 'minutes').add(59, 'seconds');
const duration = moment.duration(endDate.diff(startDate));
const diffDays = duration.days();
if (diffDays > 30) {
Helper.toast('Warning', 'just now', 'Maximum period is 1 month');
return {
isValid: false
};
}
if (diffDays < 0) {
Helper.toast('Warning', 'just now', 'Invalid period range');
return {
isValid: false
};
}
return {
isValid: true,
data: {
startDate,
endDate
}
};
},
checkIsMobileDevice: function() {
return (/android|webos|iphone|ipad|ipod|blackberry|iemobile|opera mini/i.test(navigator.userAgent.toLowerCase()));
},
};
// ## centering + zoom
// Leaflet.map.fitBounds([
// [40.712, -74.227],
// [40.774, -74.125]
// ]);
// ## centering with animation
// Leaflet.map.panTo([40.712, -74.227], { duration: 5 });
const Leaflet = {
map: null,
activate: function() {
// centering indonesia country => .setView([-1.38116, 117.6168817], 5.4);
// Leaflet.map = L.map('map').setView([-7.1451449, 109.9776078], 8); // centering javanese province
// Leaflet.map = L.map('map').setView([-7.8241413, 112.9071746], 9); // centering east java province
// Leaflet.map = L.map('map').setView([-6.2106272, 106.8477106], 9); // centering east java province
Leaflet.map = L.map('map').setView([-8.90507, 125.9945732], 10); // centering timor leste
L.tileLayer('https://api.mapbox.com/styles/v1/{id}/tiles/{z}/{x}/{y}?access_token={accessToken}', {
attribution: 'Map data &copy; <a href="https://www.mapbox.com/feedback/">Mapbox</a> © <a href="http://www.openstreetmap.org/copyright">OpenStreetMap</a>',
// maxZoom: 20,
// minZoom: 4,
id: 'mapbox/streets-v11',
tileSize: 512,
zoomOffset: -1,
accessToken: 'pk.eyJ1IjoibWV1c2luZmlybWFyeSIsImEiOiJja3lsd2xveDAydGhqMnVxaHJsZ2ZncG8yIn0.f7MJAyawHUdCegw7sWjrww',
}).addTo(Leaflet.map);
Leaflet.events();
L.control.scale({
position: 'bottomright'
}).addTo(Leaflet.map);
L.control.zoom({
position: 'bottomright'
}).addTo(Leaflet.map);
},
events: function() {
Leaflet.map.on('click', function(e) {
// e.latlng.lat, e.latlng.lng
});
// https://gis.stackexchange.com/questions/104507/disable-panning-dragging-on-leaflet-map-for-div-within-map // number 2 from bottom
$('.disablePanning').on('mousedown', function() {
if (State.isPanning) {
State.isPanning = false;
Leaflet.map.dragging.disable();
}
});
$('.disablePanning').on('mouseup', function() {
if (!State.isPanning) {
State.isPanning = true;
Leaflet.map.dragging.enable();
}
});
},
addMarkers: function(locs = [], cb = null) {
if (locs.constructor === Array) {
let markers = [];
for (let i = 0; i < locs.length; i++) {
let label = locs[i]?.label || '';
let marker = new L.marker([locs[i].lat, locs[i].lng], locs[i]?.options);
let popup = new L.Popup({
autoClose: false,
closeOnClick: true,
}).setContent(locs[i]?.label || '').setLatLng([locs[i].lat, locs[i].lng]);
// marker.bindPopup(locs[i]?.label || '');
marker.bindPopup(popup);
marker.bindTooltip(locs[i]?.label || '');
markers.push(marker);
if (cb) cb(marker);
marker.addTo(Leaflet.map);
}
return markers;
} else if (locs.constructor === Object) {
let label = locs.label;
let marker = new L.marker([locs.lat, locs.lng], locs?.options);
let popup = new L.Popup({
autoClose: false,
closeOnClick: true,
}).setContent(locs?.label || '').setLatLng([locs.lat, locs.lng]);
// marker.bindPopup(locs?.label || '');
marker.bindPopup(popup);
marker.bindTooltip(locs?.label || '');
if (cb) cb(marker);
marker.addTo(Leaflet.map);
return marker;
}
},
addCircles: function(locs = [], cb = null) {
if (locs.constructor === Array) {
let circles = [];
for (let i = 0; i < locs.length; i++) {
// invisible circle
if(locs[i].options?.markerOpacity == 0){
locs[i].options.opacity = 0
locs[i].options.fill = false
}
let circle = L.circle([locs[i].lat, locs[i].lng], locs[i]?.options);
circle.bindPopup(locs[i]?.label || '');
circle.bindTooltip(locs[i]?.label || '');
circles.push(circle);
if (cb) cb(circle, i);
circle.addTo(Leaflet.map);
}
return circles;
} else if (locs.constructor === Object) {
let circle = L.circle([locs.lat, locs.lng], locs?.options);
circle.bindPopup(locs?.label || '');
circle.bindTooltip(locs?.label || '');
if (cb) cb(circle);
circle.addTo(Leaflet.map);
return circle;
}
},
addPolylines: function(locs = [], cb = null) {
let latLngs = [];
for (let i = 0; i < locs.length; i++) {
latLngs.push([locs[i].lat, locs[i].lng]);
}
let polyline = L.polyline(latLngs, locs[0]?.options).addTo(Leaflet.map);
if (cb) cb(polyline);
return polyline;
},
addRoutes: async function(locs = [], cb = null) {
let latLngs = [];
for (let i = 0; i < locs.length; i++) {
latLngs.push([locs[i].lat, locs[i].lng]);
}
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: "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
addCustomPolylines: function(locs = [], cb = null) {
const radius = 0.5 /* corner smooth radius, keep value in range 0 - 0.5 to avoid artifacts */
let latLngs = [];
for (let i = 0; i < locs.length; i++) {
latLngs.push([locs[i].lat, locs[i].lng]);
}
let SmoothPoly = L.Polyline.extend({
// override default method to use custom points-to-path method
_updatePath: function() {
let path = Leaflet.roundPathCorners(this._parts, radius)
this._renderer._setPath(this, path);
}
})
let polyline = new SmoothPoly(latLngs, locs[0]?.options).addTo(Leaflet.map);
if (cb) cb(polyline);
return polyline;
},
roundPathCorners: function(rings, radius) {
function moveTowardsFractional(movingPoint, targetPoint, fraction) {
return {
x: movingPoint.x + (targetPoint.x - movingPoint.x) * fraction,
y: movingPoint.y + (targetPoint.y - movingPoint.y) * fraction
};
}
function pointForCommand(cmd) {
return {
x: parseFloat(cmd[cmd.length - 2]),
y: parseFloat(cmd[cmd.length - 1])
};
}
let resultCommands = [];
if (+radius) {
// negative numbers create artifacts
radius = Math.abs(radius);
} else {
radius = 0.15;
}
for (let i = 0, len = rings.length; i < len; i++) {
let commands = rings[i];
// start point
resultCommands.push(["M", commands[0].x, commands[0].y]);
for (let cmdIndex = 1; cmdIndex < commands.length; cmdIndex++) {
let prevCmd = resultCommands[resultCommands.length - 1];
let curCmd = commands[cmdIndex];
let nextCmd = commands[cmdIndex + 1];
if (nextCmd && prevCmd) {
// Calc the points we're dealing with
let prevPoint = pointForCommand(prevCmd); // convert to Object
let curPoint = curCmd;
let nextPoint = nextCmd;
// The start and end of the cuve are just our point moved towards the previous and next points, respectivly
let curveStart, curveEnd;
curveStart = moveTowardsFractional(
curPoint,
prevCmd.origPoint || prevPoint,
radius
);
curveEnd = moveTowardsFractional(
curPoint,
nextCmd.origPoint || nextPoint,
radius
);
// Adjust the current command and add it
curCmd = Object.values(curveStart);
curCmd.origPoint = curPoint;
curCmd.unshift("L");
resultCommands.push(curCmd);
// The curve control points are halfway between the start/end of the curve and
// calculate curve, if radius is different than 0
if (radius) {
let startControl = moveTowardsFractional(curveStart, curPoint, 0.5);
let endControl = moveTowardsFractional(curPoint, curveEnd, 0.5);
// Create the curve
let curveCmd = [
"C",
startControl.x,
startControl.y,
endControl.x,
endControl.y,
curveEnd.x,
curveEnd.y
];
// Save the original point for fractional calculations
curveCmd.origPoint = curPoint;
resultCommands.push(curveCmd);
}
} else {
// Pass through commands that don't qualify
let el = Object.values(curCmd);
el.unshift("L");
resultCommands.push(el);
}
}
}
return (
resultCommands.reduce(function(str, c) {
return str + c.join(" ") + " ";
}, "") || "M0 0"
);
},
// end custom polylines
addPolygons: function(locs = [], cb = null) {
let latLngs = [];
for (let i = 0; i < locs.length; i++) {
latLngs.push([locs[i].lat, locs[i].lng]);
}
let polygon = L.polygon(latLngs, locs[0]?.options).addTo(Leaflet.map);
polygon.bindPopup(locs[0]?.label || '');
polygon.bindTooltip(locs[0]?.label || '');
if (cb) cb(polygon);
return polygon;
},
addWaypoints: function(locs = [], cb = null) {
let latLngs = [];
for (let i = 0; i < locs.length; i++) {
latLngs.push(L.latLng(locs[i].lat, locs[i].lng));
}
let control = L.Routing.control({
waypoints: latLngs,
show: false,
waypointMode: 'snap',
createMarker: function() {}
}).addTo(Leaflet.map);
if (cb) cb(control);
return control;
},
drawPolygon: function(lat, lng, options = {}) {
let polygon = L.polygon([
[lat, lng]
], options).addTo(Leaflet.map);
return polygon;
},
eventMarkerClick: function(marker, cb = null) {
marker.on('click', function(e) {
// console.log(e.target.options); // get passing data from set marker: https://stackoverflow.com/questions/17423261/how-to-pass-data-with-marker-in-leaflet-js
Leaflet.map.invalidateSize();
if (Leaflet.map.getZoom() <= 14) {
Leaflet.map.setView(e.target.getLatLng(), 14);
}
if (cb) cb(e);
});
},
clearLayer: function(type = 'all', cb = null) {
if (type == 'all') {
if (State.eventRemoveListTrucks) window.dispatchEvent(State.eventRemoveListTrucks);
if (State.eventRemoveRouteStartEnd) window.dispatchEvent(State.eventRemoveRouteStartEnd);
if (State.eventRemoveDetailPlotMovement) window.dispatchEvent(State
.eventRemoveDetailPlotMovement);
if (State.eventRemoveListZones) window.dispatchEvent(State.eventRemoveListZones);
if (cb) cb();
} else if (type == 'eventRemoveListTrucks') {
if (State.eventRemoveListTrucks) {
window.dispatchEvent(State.eventRemoveListTrucks);
if (cb) cb();
}
} else if (type == 'eventRemoveRouteStartEnd') {
// console.log('clear route eventRemoveRouteStartEnd');
if (State.eventRemoveRouteStartEnd) {
window.dispatchEvent(State.eventRemoveRouteStartEnd);
if (cb) cb();
}
} else if (type == 'eventRemoveDetailPlotMovement') {
if (State.eventRemoveDetailPlotMovement) {
window.dispatchEvent(State.eventRemoveDetailPlotMovement);
if (cb) cb();
}
} else if (type == 'eventRemoveListZones') {
if (State.eventRemoveListZones) {
window.dispatchEvent(State.eventRemoveListZones);
if (cb) cb();
}
} else if (type == 'eventAboutTruck') {
if (State.eventRemoveListTrucks) window.dispatchEvent(State.eventRemoveListTrucks);
if (State.eventRemoveRouteStartEnd) window.dispatchEvent(State.eventRemoveRouteStartEnd);
if (State.eventRemoveDetailPlotMovement) window.dispatchEvent(State
.eventRemoveDetailPlotMovement);
if (cb) cb();
}
},
filterLayer: function(type = 'all', cb = null) {
if (type == 'all') {
if (State.eventFilterPopupVhcMarker) window.dispatchEvent(State.eventFilterPopupVhcMarker);
if (State.eventFilterVhcGps) window.dispatchEvent(State.eventFilterVhcGps);
if (State.eventFilterVhcGsm) window.dispatchEvent(State.eventFilterVhcGsm);
if (State.eventFilterVhcZone) window.dispatchEvent(State.eventFilterVhcZone);
if (cb) cb();
} else if (type == 'eventFilterPopupVhcMarker') {
if (State.eventFilterPopupVhcMarker) {
window.dispatchEvent(State.eventFilterPopupVhcMarker);
if (cb) cb();
}
} else if (type == 'eventFilterVhcGps') {
if (State.eventFilterVhcGps) {
window.dispatchEvent(State.eventFilterVhcGps);
if (cb) cb();
}
} else if (type == 'eventFilterVhcGsm') {
if (State.eventFilterVhcGsm) {
window.dispatchEvent(State.eventFilterVhcGsm);
if (cb) cb();
}
} else if (type == 'eventFilterVhcZone') {
if (State.eventFilterVhcZone) {
window.dispatchEvent(State.eventFilterVhcZone);
if (cb) cb();
}
}
},
hideLayer: function(type = 'all', cb = null) {
if (type == 'all') {
if (State.eventHideAllTruck) window.dispatchEvent(State.eventHideAllTruck);
if (cb) cb();
} else if (type == 'eventHideTruckNotSelected') {
if (State.eventHideTruckNotSelected) {
if (State.eventHideTruckNotSelected) {
window.dispatchEvent(State.eventHideTruckNotSelected);
if (cb) cb();
}
}
} else if (type == 'eventHideAllTruck') {
if (State.eventHideAllTruck) {
if (State.eventHideAllTruck) {
window.dispatchEvent(State.eventHideAllTruck);
if (cb) cb();
}
}
}
},
};
const Menu = {
activate: function() {
$('.select2').select2();
$('#openRightSideBar1').on('click', function() {
$('#rightSideBar1').removeClass('d-none');
$('#rightSideBar1').addClass('d-block');
$(this).removeClass('d-block');
$(this).addClass('d-none');
Leaflet.map.invalidateSize();
})
$('#closeRightSideBar1').on('click', function() {
$('#rightSideBar1').addClass('d-none');
// if (Wrapper.checkIsMobileDevice()) {
// $('#openLeftSideBar1Mobile').removeClass('d-none');
// $('#openLeftSideBar1Mobile').addClass('d-block');
// }
$('#openRightSideBar1').removeClass('d-none');
$('#openRightSideBar1').addClass('d-block');
$('#rightSideBar2').addClass('d-none');
Leaflet.map.invalidateSize();
})
$('#openLeftSideBar1Mobile').on('click', function() {
Zone.bundleGetListZones(true);
$('#leftSideBar1').removeClass('d-none');
$('#leftSideBar1').addClass('d-block');
$(this).removeClass('d-block');
$(this).addClass('d-none');
Leaflet.map.invalidateSize();
});
$('#closeRightSideBar2').on('click', function() {
$('#rightSideBar2').addClass('d-none');
Leaflet.map.invalidateSize();
State.inShowVid = null;
Leaflet.clearLayer('eventAboutTruck');
Trucks.bundleGetListTrucks(true);
})
$('#openLeftSideBar1').on('click', function() {
Zone.bundleGetListZones(true);
$('#leftSideBar1').removeClass('d-none');
$('#leftSideBar1').addClass('d-block');
$(this).removeClass('d-block');
$(this).addClass('d-none');
Leaflet.map.invalidateSize();
})
$('#closeLeftSideBar1').on('click', function() {
Zone.bundleGetListZones(false);
$('#leftSideBar1').addClass('d-none');
if (Wrapper.checkIsMobileDevice()) {
$('#openLeftSideBar1Mobile').removeClass('d-none');
$('#openLeftSideBar1Mobile').addClass('d-block');
} else {
$('#openLeftSideBar1').removeClass('d-none');
$('#openLeftSideBar1').addClass('d-block');
}
Leaflet.map.invalidateSize();
})
$('#listZone').on('click', '.zone-item', function(e) {
$('#leftSideBar2').removeClass('d-none');
$('#leftSideBar2').addClass('d-block');
});
$('#closeLeftSideBar2').on('click', function() {
$('#leftSideBar2').addClass('d-none');
Leaflet.map.invalidateSize();
})
$('#selectFiter').on('select2:select', function(e) {
Menu.showViewDetailVehicle($(this).val());
});
Menu.showViewDetailVehicle(State.stts_filterDetail.vehicles);
Menu.eventListVehicle();
Menu.eventListMovement();
Menu.eventListZone();
Menu.eventFilter();
Menu.eventFilterHistoryDate();
},
showViewDetailVehicle: async function(type) {
if (type == State.stts_filterDetail.vehicles) { // vehicles
$('#infoVehicles').removeClass('d-none');
$('#infoVehicles').addClass('d-block');
$('#infoDriver').addClass('d-none');
$('#infoJob').addClass('d-none');
$('#infoMovement').addClass('d-none');
} else if (type == State.stts_filterDetail.driver) { // driver
$('#infoVehicles').addClass('d-none');
$('#infoDriver').removeClass('d-none');
$('#infoDriver').addClass('d-block');
$('#infoJob').addClass('d-none');
$('#infoMovement').addClass('d-none');
} else if (type == State.stts_filterDetail.assignJob) { // assign job
$('#infoVehicles').addClass('d-none');
$('#infoDriver').addClass('d-none');
$('#infoJob').removeClass('d-none');
$('#infoJob').addClass('d-block');
$('#infoMovement').addClass('d-none');
} else if (type == State.stts_filterDetail.lastMove) { // last movement
$('#infoVehicles').addClass('d-none');
// $('#infoDriver').addClass('d-none');
$('#infoJob').addClass('d-none');
$('#infoMovement').removeClass('d-none');
$('#infoMovement').addClass('d-block');
if (State.inShowVid) {
let cache = !(State.inShowVid != State.loadedLastMoveVid);
Trucks.bundleShowRouteTruck(cache, false); // jika data yang diload sebelumnya beda, maka tidak mengambil dari cache
}
// Menu.clearListMovements();
}
},
eventListVehicle: function() {
$('#listTrucks').on('click', '.vhc-item', async function(e) {
let clicked = $(e.target)
let vid = clicked.closest('.vhc-item').data('vid');
if (Trucks.lists) {
for (let i = 0; i < Trucks.lists.length; i++) {
if (Trucks.lists[i].vid == vid) {
Trucks.showDetailGeneral(Trucks.lists[i]);
Leaflet.hideLayer('eventHideTruckNotSelected');
Leaflet.clearLayer('eventRemoveRouteStartEnd', async function() {
await Trucks.bundleGetListTrucks(true);
setTimeout(() => {
Leaflet.hideLayer('eventHideTruckNotSelected');
}, State.delay_hideTruckNotSelected);
});
Leaflet.clearLayer('eventRemoveDetailPlotMovement');
$('#selectFiter').val(State.stts_filterDetail.vehicles).trigger('change');
$('#selectFiter').val(State.stts_filterDetail.vehicles).trigger('select2:select');
$('#rightSideBar2').removeClass('d-none');
if (Trucks.lists[i]?.lst_lat == null || Trucks.lists[i]?.lst_lng == null) {
Helper.toast('Warning', 'just now',
`Last location for this vehicle not found\n${Trucks.lists[i].nopol1} ${Trucks.lists[i].nopol2} ${Trucks.lists[i].nopol3}`
);
continue;
}
// Leaflet.map.fitBounds([
// [Trucks.lists[i].lst_lat, Trucks.lists[i].lst_lng],
// ]);
if (Leaflet.map.getZoom() <= 14) {
Leaflet.map.setView([Trucks.lists[i].lst_lat, Trucks.lists[i].lst_lng],
14);
} else {
Leaflet.map.setView([Trucks.lists[i].lst_lat, Trucks.lists[i].lst_lng]);
}
}
}
}
});
},
showToListMovement: function(obj) {
let arrIdx = Helper.getIndexReversedSequence(obj.key_index - 1, Trucks.last_move.length - 1);
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(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-2">${Helper.shortenText(decodeURIComponent(obj?.fulladdress || 'address'), 255)}</p>
<p class="mb-0">Current speed: ${Number(obj.speed)}km/h</p>
</li>
</a>
`);
},
eventListMovement: function() {
$('#infoMove-plots').on('click', '.plotMove-item', function(e) {
let tr = JSON.parse(decodeURIComponent(window.atob($(e.target).closest('.plotMove-item').data('obj'))));
// console.log("tr", tr);
let arrIdx = Helper.getIndexReversedSequence(tr.key_index - 1, Trucks.last_move.length - 1);
// console.log("arrIdx", arrIdx);
PgBar.setCrntVal(arrIdx);
Menu.createMarkerDetailPlotMovement(tr);
})
},
createMarkerDetailPlotMovement: function(tr, opt = {}) {
Leaflet.clearLayer('eventRemoveDetailPlotMovement');
const unix = parseInt(tr?.lst_loc_crt_d) + AppState.TIMEFIX;
let marker = Leaflet.addMarkers({
lat: tr.latitude,
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.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`,
options: {
// icon: Icon.destination()
}
});
marker.openPopup();
let curr_zoom = Leaflet.map.getZoom();
if (opt.nozoom) {
Leaflet.map.setView([tr.latitude, tr.longitude], curr_zoom);
} else {
if (curr_zoom <= 16) {
Leaflet.map.setView([tr.latitude, tr.longitude], 16);
} else {
Leaflet.map.setView([tr.latitude, tr.longitude], curr_zoom);
}
}
// remove marker, circle, event listener and all about this marker
State.eventRemoveDetailPlotMovement = new CustomEvent('eventRemoveDetailPlotMovement', {
marker,
});
window.addEventListener('eventRemoveDetailPlotMovement', function handler(e) {
marker.removeEventListener('click');
marker.remove();
e.currentTarget.removeEventListener(e.type,
handler); // window.removeEventListener('remove', this.handler, true);
State.eventRemoveDetailPlotMovement = null;
});
},
clearListMovements: function() {
$('#infoMove-plots').html('');
},
eventListZone: function() {
$('#listZone').on('click', '.zone-item', function(e) {
let clicked = $(e.target)
let zid = clicked.closest('.zone-item').data('zid');
if (Zone.lists) {
for (let i = 0; i < Zone.lists.length; i++) {
if (Zone.lists[i].id == zid) {
Zone.showDetailGeneral(Zone.lists[i]);
let boundary_latlngs = JSON.parse(Zone.lists[i].boundary_latlngs);
let fitBounds = [];
for (let row of boundary_latlngs) {
let corner = L.latLng(row.lat, row.lng);
fitBounds.push(L.latLngBounds(corner, corner));
}
Leaflet.map.fitBounds(fitBounds);
// if circle
// if (Leaflet.map.getZoom() <= 14) {
// Leaflet.map.setView([Zone.lists[i].lst_lat, Zone.lists[i].lst_lng],
// 14)
// }
}
}
}
});
$('#clearZone').on('click', function() {
Leaflet.clearLayer('eventRemoveListZones', null);
});
},
eventFilter: function() {
$('#filterVhcMarker').on('click', function() {
Leaflet.filterLayer('eventFilterPopupVhcMarker');
});
$('#filterVhcGps').on('click', function() {
Leaflet.filterLayer('eventFilterVhcGps');
});
$('#filterVhcGsm').on('click', function() {
Leaflet.filterLayer('eventFilterVhcGsm');
});
$('#filterVhcZone').on('click', function() {
Leaflet.filterLayer('eventFilterVhcZone');
});
// searching
let timeoutFilterVhcSearch = null;
$('#filterVhcSearch').on('keyup', function() {
clearTimeout(timeoutFilterVhcSearch);
timeoutFilterVhcSearch = setTimeout(() => {
Menu.handlerFilterVhcSearch($('#filterVhcSearch').val());
}, State.delay_typing_front);
});
let timeoutFilterZoneSearch = null;
$('#filterZoneSearch').on('keyup', function() {
clearTimeout(timeoutFilterZoneSearch);
timeoutFilterZoneSearch = setTimeout(() => {
Menu.handlerFilterZoneSearch($('#filterZoneSearch').val());
}, State.delay_typing_front);
});
},
eventFilterHistoryDate: function() {
$('#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 date1 = moment($('#historyEndDate').val(), "DD-MM-YYYY HH:mm")
const {
isValid,
data
} = Wrapper.calcPeriodeDate(
date0,
date1
)
if (!isValid) return false;
// console.log("isValid", isValid, data);
State.historyStartDate = date0.unix();
State.historyEndDate = date1.unix();
Leaflet.clearLayer('eventRemoveRouteStartEnd');
Trucks.bundleShowRouteTruck(false);
});
// $('#historyEndDate').on('change', function(e) {
// const {
// isValid,
// data
// } = Wrapper.calcPeriodeDate(moment($('#historyStartDate').val()), moment($('#historyEndDate').val()))
// if (!isValid) return false;
// State.historyStartDate = data.startDate.unix();
// State.historyEndDate = data.endDate.unix();
// Leaflet.clearLayer('eventRemoveRouteStartEnd');
// Trucks.bundleShowRouteTruck(false);
// });
$('#historyStartDate').on('clearFilterHistoryDate', function(e) {
// default date today
State.historyStartDate = moment().startOf('day').unix();
State.historyEndDate = moment().endOf('day').unix();
$('#historyStartDate').val(moment().startOf('day').format('DD-MM-YYYY HH:mm'));
$('#historyEndDate').val(moment().endOf('day').format('DD-MM-YYYY HH:mm'));
// for testing purpose
// State.historyStartDate = '1756054800';
// State.historyEndDate = '1756141140';
// $('#historyStartDate').val("25-08-2025 00:00");
// $('#historyEndDate').val("25-08-2025 23:59");
});
},
handlerFilterPopupVhcMarker: function(markers) {
if (State.inShowLastMove) {
Helper.toast('Warning', 'just now', 'Cannot show marker info when see last movement');
return false;
}
// 0=>hidden,1=>nopol,2=>speed,3=>address
let latLng = Leaflet.map.getCenter();
for (let marker of markers) {
let mdata = marker.options.mdata;
if (!State.lastShowVids.includes(mdata.vid)) {
continue;
}
if (State.statusFilterPopupVhcMarker === 0) {
// let content =
// `<div class="bg-color"><b>${mdata.nopol1} ${mdata.nopol2} ${mdata.nopol3}</b></div>`;
// marker.bindTooltip(content);
// marker.closePopup();
} else if (State.statusFilterPopupVhcMarker === 1) {
// let content =
// `<div class="bg-color"><b>${mdata.nopol1} ${mdata.nopol2} ${mdata.nopol3}</b></div>`;
let content = `<b>${mdata.nopol1} ${mdata.nopol2} ${mdata.nopol3}</b>`;
if (mdata?.ord_pck_ktname && mdata?.ord_drop_ktname) {
content += `<div style="margin-bottom:10px;"><strong>${mdata?.ord_c_pt_name}</strong></div>`;
content += `<div style="margin-bottom:0;">Origin</div>`;
content += `<div style="margin-bottom:0.25rem;"><strong>${mdata?.ord_pck_ktname}</strong></div>`;
content += `<div style="margin-bottom:0;">Destination</div>`;
content += `<div style="margin-bottom:0;"><strong>${mdata?.ord_drop_ktname}</strong></div>`;
}
marker.setPopupContent(content);
marker.bindTooltip(content);
marker.openPopup();
} else if (State.statusFilterPopupVhcMarker === 2) {
// cara lama
// let speedOrIdle = (typeof mdata.lst_speed != 'undefined') ? mdata.lst_speed : 0;
// if (mdata.stts_engine == State.stts_engine.idling) speedOrIdle = 0;
// if (speedOrIdle === 0) {
// let idlingDur = Wrapper.calcIdlingDur(mdata?.lst_idle_at, mdata?.lst_speed);
// if (idlingDur.unit == 'hour') {
// let min = idlingDur.dur * 60;
// if (idlingDur.dur <= 1) {
// speedOrIdle = ` <= ${min.toFixed(0)} min`;
// } else if (idlingDur.dur > 6) {
// speedOrIdle = ' > 6 hr';
// } else if (idlingDur.dur > 4) {
// speedOrIdle = ' > 4 hr';
// } else if (idlingDur.dur > 2) {
// speedOrIdle = ' > 2 hr';
// } else if (idlingDur.dur > 1) {
// speedOrIdle = ' > 1 hr';
// }
// } else if (idlingDur.unit == 'min') {
// let hr = idlingDur.dur / 60;
// if (hr <= 1) {
// speedOrIdle = ` <= ${idlingDur.dur} min`;
// } else if (hr > 6) {
// speedOrIdle = ' > 6 hr';
// } else if (hr > 4) {
// speedOrIdle = ' > 4 hr';
// } else if (hr > 2) {
// speedOrIdle = ' > 2 hr';
// } else if (hr > 1) {
// speedOrIdle = ' > 1 hr';
// }
// }
// } else {
// speedOrIdle += ' Kph';
// }
// differentation minute / hour
function getDiff(last) {
let now = moment(); // as start
let end = moment.unix(last ?? now.unix()); // as end
let duration = moment.duration(end.diff(now));
let diff = duration.asMinutes() * -1;
let unit = 'min';
// parsing to hour
if (diff > 60) {
diff = diff / 60;
unit = 'hr';
}
return {
duration,
diff,
unit,
}
}
let speedOrIdle = '-';
let a = getDiff(mdata.lst_idle_at);
let diff = a.diff.toFixed(0);
let unit = a.unit;
if (mdata.stts_engine == State.stts_engine.moving) {
speedOrIdle = `${mdata.lst_speed} Kph`;
if (typeof mdata.lst_speed == 'undefined' || mdata.lst_speed == 0 || mdata.lst_speed == null) {
speedOrIdle = `> ${diff} ${unit}`;
}
} else if (mdata.stts_engine == State.stts_engine.idling) {
speedOrIdle = `> ${diff} ${unit}`;
} else if (mdata.stts_engine == State.stts_engine.stoping) {
a = getDiff(mdata.lst_stop_at);
diff = a.diff.toFixed(0);
unit = a.unit;
speedOrIdle = `> ${diff} ${unit}`;
}
let content = `<div class="bg-color"><b>${speedOrIdle}</b></div>`;
marker.setPopupContent(content);
marker.bindTooltip(content);
marker.openPopup();
} else if (State.statusFilterPopupVhcMarker === 3) {
let content =
`<div class="bg-color"><b>${Helper.shortenText(decodeURIComponent(mdata?.lst_fulladdress || 'address'), 100)}</b></div>`;
marker.setPopupContent(content);
marker.bindTooltip(content);
marker.openPopup();
}
Leaflet.map.setView(latLng);
}
if (State.statusFilterPopupVhcMarker === 3) {
State.statusFilterPopupVhcMarker = 1;
} else {
State.statusFilterPopupVhcMarker += 1;
}
},
handlerFilterVhcGps: function(markers) {
if (State.inShowVid) {
Helper.toast('Warning', 'just now', 'Cannot filter when see detail vehicle');
return false;
}
// 0=>all,1=>active,2=>not
if (State.statusFilterVhcGps === 2) {
State.statusFilterVhcGps = 0;
} else {
State.statusFilterVhcGps += 1;
}
State.lastShowVids = [];
function show(marker, mdata) {
marker.setOpacity(1);
$(`.vhc-item[data-vid="${mdata.vid}"]`).removeClass('d-none');
State.lastShowVids.push(mdata.vid);
}
function hide(marker, mdata) {
marker.setOpacity(0);
$(`.vhc-item[data-vid="${mdata.vid}"]`).addClass('d-none');
}
function validate(marker, mdata) {
if ((State.statusFilterVhcGsm === 1 || State.statusFilterVhcGsm === 2) && (State.statusFilterVhcZone === 1 || State.statusFilterVhcZone === 2)) {
if (Menu.isFilterVhc_Gsm(mdata) && Menu.isFilterVhc_Zone(mdata)) {
show(marker, mdata);
}
} else if (State.statusFilterVhcGsm === 1 || State.statusFilterVhcGsm === 2) {
if (Menu.isFilterVhc_Gsm(mdata)) {
show(marker, mdata);
}
} else if (State.statusFilterVhcZone === 1 || State.statusFilterVhcZone === 2) {
if (Menu.isFilterVhc_Zone(mdata)) {
show(marker, mdata);
}
} else {
show(marker, mdata);
}
}
let latLng = Leaflet.map.getCenter();
let c_vhc_gps = 0;
for (let marker of markers) {
let mdata = marker.options.mdata;
if (State.statusFilterVhcGps === 0) {
c_vhc_gps += 1;
validate(marker, mdata);
$($('#filterVhcGps > span')[0]).removeClass('text-danger text-success').addClass(
'text-muted');
$('#filterVhcGps').attr('data-bs-original-title', 'Filter vehicle GPS');
if (State.isShowToltipFilterVhcGps) $('#filterVhcGps').tooltip('show');
} else if (State.statusFilterVhcGps === 1) {
if (mdata.stts_gps == State.stts_gps.active) {
c_vhc_gps += 1;
validate(marker, mdata);
} else {
hide(marker, mdata);
}
$($('#filterVhcGps > span')[0]).removeClass('text-muted text-danger').addClass(
'text-success');
$('#filterVhcGps').attr('data-bs-original-title', 'Filter vehicle GPS Active');
if (State.isShowToltipFilterVhcGps) $('#filterVhcGps').tooltip('show');
} else if (State.statusFilterVhcGps === 2) {
if (mdata.stts_gps != State.stts_gps.active) {
c_vhc_gps += 1;
validate(marker, mdata);
} else {
hide(marker, mdata);
}
$($('#filterVhcGps > span')[0]).removeClass('text-success text-muted').addClass(
'text-danger');
$('#filterVhcGps').attr('data-bs-original-title', 'Filter vehicle GPS Inactive');
if (State.isShowToltipFilterVhcGps) $('#filterVhcGps').tooltip('show');
}
Leaflet.map.setView(latLng);
}
$('#c_vhc_gps').text(c_vhc_gps);
State.lastStatusFilterVhcGps = State.statusFilterVhcGps;
},
handlerFilterVhcGsm: function(markers) {
if (State.inShowVid) {
Helper.toast('Warning', 'just now', 'Cannot filter when see detail vehicle');
return false;
}
// 0=>all,1=>active,2=>not
if (State.statusFilterVhcGsm === 2) {
State.statusFilterVhcGsm = 0;
} else {
State.statusFilterVhcGsm += 1;
}
State.lastShowVids = [];
function show(marker, mdata) {
marker.setOpacity(1);
$(`.vhc-item[data-vid="${mdata.vid}"]`).removeClass('d-none');
State.lastShowVids.push(mdata.vid);
}
function hide(marker, mdata) {
marker.setOpacity(0);
$(`.vhc-item[data-vid="${mdata.vid}"]`).addClass('d-none');
}
function validate(marker, mdata) {
if ((State.statusFilterVhcGps === 1 || State.statusFilterVhcGps === 2) && (State.statusFilterVhcZone === 1 || State.statusFilterVhcZone === 2)) {
if (Menu.isFilterVhc_Gps(mdata) && Menu.isFilterVhc_Zone(mdata)) {
show(marker, mdata);
}
} else if (State.statusFilterVhcGps === 1 || State.statusFilterVhcGps === 2) {
if (Menu.isFilterVhc_Gps(mdata)) {
show(marker, mdata);
}
} else if (State.statusFilterVhcZone === 1 || State.statusFilterVhcZone === 2) {
if (Menu.isFilterVhc_Zone(mdata)) {
show(marker, mdata);
}
} else {
show(marker, mdata);
}
}
let latLng = Leaflet.map.getCenter();
let c_vhc_gsm = 0;
for (let marker of markers) {
let mdata = marker.options.mdata;
if (State.statusFilterVhcGsm === 0) {
c_vhc_gsm += 1;
validate(marker, mdata);
$($('#filterVhcGsm > span')[0]).removeClass('text-danger text-success').addClass(
'text-muted');
$('#filterVhcGsm').attr('data-bs-original-title', 'Filter vehicle GSM');
if (State.isShowToltipFilterVhcGsm) $('#filterVhcGsm').tooltip('show');
} else if (State.statusFilterVhcGsm === 1) {
if (mdata.stts_gsm >= State.stts_gsm.bad_signal) {
if (mdata.lst_heartbeat > 0) {
c_vhc_gsm += 1;
validate(marker, mdata);
} else {
hide(marker, mdata);
}
} else {
hide(marker, mdata);
}
$($('#filterVhcGsm > span')[0]).removeClass('text-muted text-danger').addClass(
'text-success');
$('#filterVhcGsm').attr('data-bs-original-title', 'Filter vehicle GSM Active');
if (State.isShowToltipFilterVhcGsm) $('#filterVhcGsm').tooltip('show');
} else if (State.statusFilterVhcGsm === 2) {
if (mdata.stts_gsm <= State.stts_gsm.no_signal) {
c_vhc_gsm += 1;
validate(marker, mdata);
} else {
if (mdata.lst_heartbeat < 1) {
c_vhc_gsm += 1;
validate(marker, mdata);
} else {
hide(marker, mdata);
}
}
$($('#filterVhcGsm > span')[0]).removeClass('text-success text-muted').addClass(
'text-danger');
$('#filterVhcGsm').attr('data-bs-original-title', 'Filter vehicle GSM No Signal');
if (State.isShowToltipFilterVhcGsm) $('#filterVhcGsm').tooltip('show');
}
Leaflet.map.setView(latLng);
}
$('#c_vhc_gsm').text(c_vhc_gsm);
State.lastStatusFilterVhcGsm = State.statusFilterVhcGsm;
},
handlerFilterVhcZone: function(markers) {
if (State.inShowVid) {
Helper.toast('Warning', 'just now', 'Cannot filter when see detail vehicle');
return false;
}
// 0=>all,1=>inside,2=>outside
if (State.statusFilterVhcZone === 2) {
State.statusFilterVhcZone = 0;
} else {
State.statusFilterVhcZone += 1;
}
State.lastShowVids = [];
function show(marker, mdata) {
marker.setOpacity(1);
$(`.vhc-item[data-vid="${mdata.vid}"]`).removeClass('d-none');
State.lastShowVids.push(mdata.vid);
}
function hide(marker, mdata) {
marker.setOpacity(0);
$(`.vhc-item[data-vid="${mdata.vid}"]`).addClass('d-none');
}
function validate(marker, mdata) {
if ((State.statusFilterVhcGps === 1 || State.statusFilterVhcGps === 2) && (State.statusFilterVhcGsm === 1 || State.statusFilterVhcGsm === 2)) {
if (Menu.isFilterVhc_Gps(mdata) && Menu.isFilterVhc_Gsm(mdata)) {
show(marker, mdata);
}
} else if (State.statusFilterVhcGps === 1 || State.statusFilterVhcGps === 2) {
if (Menu.isFilterVhc_Gps(mdata)) {
show(marker, mdata);
}
} else if (State.statusFilterVhcGsm === 1 || State.statusFilterVhcGsm === 2) {
if (Menu.isFilterVhc_Gsm(mdata)) {
show(marker, mdata);
}
} else {
show(marker, mdata);
}
}
let latLng = Leaflet.map.getCenter();
let c_vhc_zone = 0;
for (let marker of markers) {
let mdata = marker.options.mdata;
if (State.statusFilterVhcZone === 0) {
c_vhc_zone += 1;
validate(marker, mdata);
$($('#filterVhcZone > span')[0]).removeClass('text-danger text-success').addClass(
'text-muted');
$('#filterVhcZone').attr('data-bs-original-title', 'Filter vehicle inside & outside zone');
if (State.isShowToltipFilterVhcZone) $('#filterVhcZone').tooltip('show');
} else if (State.statusFilterVhcZone === 1) {
if (mdata.inside_zones != null) {
c_vhc_zone += 1;
validate(marker, mdata);
} else {
hide(marker, mdata);
}
$($('#filterVhcZone > span')[0]).removeClass('text-muted text-danger').addClass(
'text-success');
$('#filterVhcZone').attr('data-bs-original-title', 'Filter vehicle inside zone');
if (State.isShowToltipFilterVhcZone) $('#filterVhcZone').tooltip('show');
} else if (State.statusFilterVhcZone === 2) {
if (mdata.inside_zones == null) {
c_vhc_zone += 1;
validate(marker, mdata);
} else {
hide(marker, mdata);
}
$($('#filterVhcZone > span')[0]).removeClass('text-success text-muted').addClass(
'text-danger');
$('#filterVhcZone').attr('data-bs-original-title', 'Filter vehicle outside zone');
if (State.isShowToltipFilterVhcZone) $('#filterVhcZone').tooltip('show');
}
Leaflet.map.setView(latLng);
}
$('#c_vhc_zone').text(c_vhc_zone);
State.lastStatusFilterVhcZone = State.statusFilterVhcZone;
},
handlerFilterVhcSearch: function(search) {
let vhcItems = $('#listTrucks').find('.vhc-item');
for (let item of vhcItems) {
let vid = $(item).data('vid');
let textNopol = $($(item).find('.textNopol')[0]).text();
let textAddr = $($(item).find('.textAddr')[0]).text();
if (textNopol.toLowerCase().indexOf(search.toLowerCase()) > -1) {
$(item).removeClass('d-none');
} else if (textAddr.toLowerCase().indexOf(search.toLowerCase()) > -1) {
$(item).removeClass('d-none');
} else {
$(item).addClass('d-none');
}
}
},
handlerFilterZoneSearch: function(search) {
let zoneItems = $('#listZone').find('.zone-item');
for (let item of zoneItems) {
let zid = $(item).data('zid');
let textZoneName = $($(item).find('.textZoneName')[0]).text();
let textZoneType = $($(item).find('.textZoneType')[0]).text();
if (textZoneName.toLowerCase().indexOf(search.toLowerCase()) > -1) {
$(item).removeClass('d-none');
} else if (textZoneType.toLowerCase().indexOf(search.toLowerCase()) > -1) {
$(item).removeClass('d-none');
} else {
$(item).addClass('d-none');
}
}
},
isFilterVhc_GpsGsmZone: function(truck) {
// if return false no filter, true there filter
if (Menu.isFilterVhc_Gps(truck)) return true;
if (Menu.isFilterVhc_Gsm(truck)) return true;
if (Menu.isFilterVhc_Zone(truck)) return true;
return false;
},
isFilterVhc_Gps: function(truck) {
// if return false no filter, true there filter
if (State.statusFilterVhcGps === 0) {} else if (State.statusFilterVhcGps === 1) {
if (truck.stts_gps == State.stts_gps.active) return true;
} else if (State.statusFilterVhcGps === 2) {
if (truck.stts_gps != State.stts_gps.active) return true;
}
return false;
},
isFilterVhc_Gsm: function(truck) {
// if return false no filter, true there filter
if (State.statusFilterVhcGsm === 0) {} else if (State.statusFilterVhcGsm === 1) {
if (truck.stts_gsm >= State.stts_gsm.bad_signal) return true;
} else if (State.statusFilterVhcGsm === 2) {
if (truck.stts_gsm <= State.stts_gsm.no_signal) return true;
}
return false;
},
isFilterVhc_Zone: function(truck) {
// if return false no filter, true there filter
if (State.statusFilterVhcZone === 0) {} else if (State.statusFilterVhcZone === 1) {
if (truck.inside_zones != null) return true;
} else if (State.statusFilterVhcZone === 2) {
if (truck.inside_zones == null) return true;
}
return false;
},
isLastFilterVhc_Gps: function(truck) {
// if return false no filter, true there filter
if (State.lastStatusFilterVhcGps === 0) {} else if (State.lastStatusFilterVhcGps === 1) {
if (truck.stts_gps == State.stts_gps.active) return true;
} else if (State.lastStatusFilterVhcGps === 2) {
if (truck.stts_gps != State.stts_gps.active) return true;
}
return false;
},
isLastFilterVhc_Gsm: function(truck) {
// if return false no filter, true there filter
if (State.lastStatusFilterVhcGsm === 0) {} else if (State.lastStatusFilterVhcGsm === 1) {
if (truck.stts_gsm >= State.stts_gsm.bad_signal) return true;
} else if (State.lastStatusFilterVhcGsm === 2) {
if (truck.stts_gsm <= State.stts_gsm.no_signal) return true;
}
return false;
},
isLastFilterVhc_Zone: function(truck) {
// if return false no filter, true there filter
if (State.lastStatusFilterVhcZone === 0) {} else if (State.lastStatusFilterVhcZone === 1) {
if (truck.inside_zones != null) return true;
} else if (State.lastStatusFilterVhcZone === 2) {
if (truck.inside_zones == null) return true;
}
return false;
},
}
const Trucks = {
lists: null,
last_move: null,
activate: function() {
Trucks.bundleGetListTrucks(false);
Trucks.event();
},
event: function() {
setInterval(async () => {
if (State.inShowLastMove) return false;
await Trucks.bundleGetListTrucks(false);
if (State.inShowVid) {
setTimeout(() => {
Leaflet.hideLayer('eventHideTruckNotSelected');
}, State.delay_hideTruckNotSelected);
}
State.loadedLastMoveVid = null;
if ($('#selectFiter').val() == State.stts_filterDetail.vehicles) {
setTimeout(() => {
if (Trucks.lists) {
for (let i = 0; i < Trucks.lists.length; i++) {
if (Trucks.lists[i].vid == State.inShowVid) {
Trucks.showDetailGeneral(Trucks.lists[i]);
}
}
}
}, 1000);
}
}, State.delay_auto_refresh);
},
bundleGetListTrucks: async function(cache = false) {
Leaflet.clearLayer('eventRemoveListTrucks');
Trucks.lists = await Trucks.getListTrucks(cache);
Trucks.showToLists(Trucks.lists);
Trucks.showListsToMap(Trucks.lists);
},
getListTrucks: async function(cache = false) {
if (cache) {
if (Trucks.lists == null) {
return (await Trucks.reqListTrucks()).data;
} else {
return Trucks.lists;
}
}
return (await Trucks.reqListTrucks()).data;
},
reqListTrucks: function(params) {
return new Promise((resolve, reject) => {
let url = "{{ route('api_list_currect_track_vhc') }}?cptid=" + AppState.current_company;
if (State.ord_code) {
url += "&ord_code=" + State.ord_code;
}
$.ajax({
url,
method: 'GET',
crossDomain: true,
processData: true,
headers: {
'x-api-key': Helper.getCookie('_trtk'),
},
data: params,
success: (data, textStatus, jqXHR) => {
if (data.meta.type != 'success') {
resolve({
type: 'fail',
data: null,
});
Helper.toast('Warning', 'just now', data.meta.message);
return false;
}
resolve({
type: 'success',
data: data.data,
});
},
error: (jqXHR, textStatus, error) => {
if (jqXHR.status >= 500) {
Helper.toast('Error', 'just now', 'Cannot get lists vehicles');
} else {
Helper.toast('Error', 'just now', jqXHR.responseJSON.meta
.message);
}
resolve({
type: 'error',
data: null,
});
}
})
});
},
showToLists: function(trucks) {
$('#listTrucks').html('');
let str = '',
refilter = '';
let c_vhc_gps = 0,
c_vhc_gsm = 0,
c_vhc_zone = 0;
State.statusFilterVhcGps = (State.lastStatusFilterVhcGps) ? State.lastStatusFilterVhcGps : State.statusFilterVhcGps;
State.statusFilterVhcGsm = (State.lastStatusFilterVhcGsm) ? State.lastStatusFilterVhcGsm : State.statusFilterVhcGsm;
State.statusFilterVhcZone = (State.lastStatusFilterVhcZone) ? State.lastStatusFilterVhcZone : State.statusFilterVhcZone;
for (let i = 0; i < trucks.length; i++) {
// refilter list trucks
refilter = 'd-none';
if (State.lastStatusFilterVhcGps !== 0 || State.lastStatusFilterVhcGsm !== 0 || State.lastStatusFilterVhcZone !== 0) {
c_vhc_gps = Number($('#c_vhc_gps').text());
c_vhc_gsm = Number($('#c_vhc_gsm').text());
c_vhc_zone = Number($('#c_vhc_zone').text());
if (State.lastStatusFilterVhcGps !== 0) {
if ((State.lastStatusFilterVhcGsm === 1 || State.lastStatusFilterVhcGsm === 2) && (State.lastStatusFilterVhcZone === 1 || State.lastStatusFilterVhcZone === 2)) {
if (Menu.isLastFilterVhc_Gps(trucks[i]) && Menu.isLastFilterVhc_Gsm(trucks[i]) && Menu.isLastFilterVhc_Zone(trucks[i])) {
refilter = '';
// c_vhc_gps += 1;
// c_vhc_gsm += 1;
// c_vhc_zone += 1;
}
} else if (State.lastStatusFilterVhcGsm === 1 || State.lastStatusFilterVhcGsm === 2) {
if (Menu.isLastFilterVhc_Gps(trucks[i]) && Menu.isLastFilterVhc_Gsm(trucks[i])) {
refilter = '';
// c_vhc_gps += 1;
// c_vhc_gsm += 1;
}
} else if (State.lastStatusFilterVhcZone === 1 || State.lastStatusFilterVhcZone === 2) {
if (Menu.isLastFilterVhc_Gps(trucks[i]) && Menu.isLastFilterVhc_Zone(trucks[i])) {
refilter = '';
// c_vhc_gps += 1;
// c_vhc_zone += 1;
}
} else {
if (Menu.isLastFilterVhc_Gps(trucks[i])) {
refilter = '';
// c_vhc_gps += 1;
// c_vhc_gsm += 1;
// c_vhc_zone += 1;
} else {
// c_vhc_gsm += 1;
// c_vhc_zone += 1;
}
}
} else if (State.lastStatusFilterVhcGsm !== 0) {
if ((State.lastStatusFilterVhcGps === 1 || State.lastStatusFilterVhcGps === 2) && (State.lastStatusFilterVhcZone === 1 || State.lastStatusFilterVhcZone === 2)) {
if (Menu.isLastFilterVhc_Gsm(trucks[i]) && Menu.isLastFilterVhc_Gps(trucks[i]) && Menu.isLastFilterVhc_Zone(trucks[i])) {
refilter = '';
// c_vhc_gsm += 1;
// c_vhc_gps += 1;
// c_vhc_zone += 1;
}
} else if (State.lastStatusFilterVhcGps === 1 || State.lastStatusFilterVhcGps === 2) {
if (Menu.isLastFilterVhc_Gsm(trucks[i]) && Menu.isLastFilterVhc_Gps(trucks[i])) {
refilter = '';
// c_vhc_gsm += 1;
// c_vhc_gps += 1;
}
} else if (State.lastStatusFilterVhcZone === 1 || State.lastStatusFilterVhcZone === 2) {
if (Menu.isLastFilterVhc_Gsm(trucks[i]) && Menu.isLastFilterVhc_Zone(trucks[i])) {
refilter = '';
// c_vhc_gsm += 1;
// c_vhc_zone += 1;
}
} else {
if (Menu.isLastFilterVhc_Gsm(trucks[i])) {
refilter = '';
// c_vhc_gsm += 1;
// c_vhc_gps += 1;
// c_vhc_zone += 1;
} else {
// c_vhc_gps += 1;
// c_vhc_zone += 1;
}
}
} else if (State.lastStatusFilterVhcZone !== 0) {
if ((State.lastStatusFilterVhcGps === 1 || State.lastStatusFilterVhcGps === 2) && (State.lastStatusFilterVhcGsm === 1 || State.lastStatusFilterVhcGsm === 2)) {
if (Menu.isLastFilterVhc_Zone(trucks[i]) && Menu.isLastFilterVhc_Gps(trucks[i]) && Menu.isLastFilterVhc_Gsm(trucks[i])) {
refilter = '';
// c_vhc_zone += 1;
// c_vhc_gps += 1;
// c_vhc_gsm += 1;
}
} else if (State.lastStatusFilterVhcGps === 1 || State.lastStatusFilterVhcGps === 2) {
if (Menu.isLastFilterVhc_Zone(trucks[i]) && Menu.isLastFilterVhc_Gps(trucks[i])) {
refilter = '';
// c_vhc_zone += 1;
// c_vhc_gps += 1;
}
} else if (State.lastStatusFilterVhcGsm === 1 || State.lastStatusFilterVhcGsm === 2) {
if (Menu.isLastFilterVhc_Zone(trucks[i]) && Menu.isLastFilterVhc_Gsm(trucks[i])) {
refilter = '';
// c_vhc_zone += 1;
// c_vhc_gsm += 1;
}
} else {
if (Menu.isLastFilterVhc_Zone(trucks[i])) {
refilter = '';
// c_vhc_zone += 1;
// c_vhc_gps += 1;
// c_vhc_gsm += 1;
} else {
// c_vhc_gps += 1;
// c_vhc_gsm += 1;
}
}
}
} else {
// without filter
refilter = '';
c_vhc_gps += 1;
c_vhc_gsm += 1;
c_vhc_zone += 1;
State.lastShowVids.push(trucks[i].vid);
}
// list trucks
// <img src="${State.storage_lara}${trucks[i].fvhc_img}" class="img-fluid" alt="">
// <img src="https://products.unitedtractors.com/wp-content/uploads/2021/03/udtruck-GWE370.png" class="img-fluid" alt="">
// <p class="text-muted mb-0"> ${(trucks[i]?.lst_city_text || trucks[i]?.lst_state_text || 'address')} - ${(trucks[i]?.lst_postcode || 'postcode')}</p>
str += `
<li class="list-group-item vehicles-list-wrapper p-1 px-2 vhc-item ${refilter}" data-vid="${trucks[i].vid}">
<a href="#" class="text-dark">
<div class="row d-flex align-items-center">
<div class="col-4">
<img src="${State.storage_lara}${trucks[i].fvhc_img}" class="img-fluid thumb-img-small" alt="">
</div>
<div class="col ps-0">
<p class="text-bold mb-0 textNopol">${trucks[i].nopol1} ${trucks[i].nopol2} ${trucks[i].nopol3}</p>
<p class="text-muted mb-0 textAddr">${Helper.shortenText(decodeURIComponent(trucks[i]?.lst_fulladdress || 'address'), 25)}</p>
</div>
</div>
</a>
</li>
`;
}
$('#c_vhc_zone').text(c_vhc_zone);
$('#c_vhc_gsm').text(c_vhc_gsm);
$('#c_vhc_gps').text(c_vhc_gps);
$('#listTrucks').html(str);
},
showListsToMap: function(trucks) {
// let locations = trucks.map((tr) => {
// let at = moment.unix(tr?.lst_loc_crt).format('DD MMM YYYY HH:mm')
// return {
// lat: tr.lst_lat,
// lng: tr.lst_lng,
// label: `<b>${tr.nopol1} ${tr.nopol2} ${tr.nopol3}</b><br>${at}<br>${tr.lst_lat},${tr.lst_lng}`,
// options: {
// mdata: tr,
// },
// };
// });
let locations = [];
for (let i = 0; i < trucks.length; i++) {
if (trucks[i]?.lst_lat == null || trucks[i]?.lst_lng == null) {
continue;
}
let icon_stts_engine = Icon.vechicleIcon(); // moving
if (trucks[i].stts_engine == State.stts_engine.moving) {
icon_stts_engine = Icon.vechicleIcon();
if (trucks[i].lst_speed == 0 || trucks[i].lst_speed == null) {
icon_stts_engine = Icon.vhcIdleIcon();
}
} else if (trucks[i].stts_engine == State.stts_engine.idling) {
icon_stts_engine = Icon.vhcIdleIcon();
} else if (trucks[i].stts_engine == State.stts_engine.stoping) {
icon_stts_engine = Icon.vhcStopIcon();
}
// no updated data more than 1 day
if (trucks[i]?.lst_loc_crt < moment().subtract(24, 'hour').unix()) {
icon_stts_engine = Icon.vhcStopIcon();
}
let templateLabel = `<b>${trucks[i].nopol1} ${trucks[i].nopol2} ${trucks[i].nopol3}</b>`;
if (trucks[i]?.ord_pck_ktname && trucks[i]?.ord_drop_ktname) {
templateLabel += `<div style="margin-bottom:0;"><strong>${trucks[i]?.ord_c_pt_name}</strong></div>`;
templateLabel += `<div style="margin-bottom:0;">Dari</div>`;
templateLabel += `<div style="margin-bottom:0.25rem;"><strong>${trucks[i]?.ord_pck_ktname}</strong></div>`;
templateLabel += `<div style="margin-bottom:0;">Ke</div>`;
templateLabel += `<div style="margin-bottom:0;"><strong>${trucks[i]?.ord_drop_ktname}</strong></div>`;
}
locations.push({
lat: trucks[i].lst_lat,
lng: trucks[i].lst_lng,
// label: `<b>${trucks[i].nopol1} ${trucks[i].nopol2} ${trucks[i].nopol3}</b><br>${moment.unix(trucks[i]?.lst_loc_crt).format('DD MMM YYYY HH:mm')}<br>Speed: ${(typeof trucks[i].lst_speed != 'undefined') ? trucks[i].lst_speed : '0'}<br>${Number(trucks[i].lst_lat).toFixed(5)},${Number(trucks[i].lst_lng).toFixed(6)}<br>${Helper.shortenText(decodeURIComponent(trucks[i]?.lst_fulladdress || 'address'), 25)}`,
label: templateLabel,
options: {
mdata: trucks[i],
icon: icon_stts_engine,
// autoClose: false,
},
});
}
let markers = Leaflet.addMarkers(locations, function(marker) {
Leaflet.eventMarkerClick(marker, function(e) {
let tr = e.target.options.mdata;
$('#selectFiter').val(1).trigger('change');
$('#selectFiter').val(1).trigger('select2:select');
Trucks.showDetailGeneral(tr);
$('#rightSideBar2').removeClass('d-none');
});
});
for (let m of markers) {
if (!State.lastShowVids.includes(m.options.mdata.vid)) {
m.setOpacity(0);
}
}
// filter popup
State.eventFilterPopupVhcMarker = new CustomEvent('eventFilterPopupVhcMarker', {
detail: {
markers,
},
});
function handlerFilterPopupVhcMarker(e) {
Menu.handlerFilterPopupVhcMarker(e.detail.markers);
}
window.addEventListener('eventFilterPopupVhcMarker', handlerFilterPopupVhcMarker);
// filter vehicle gps
State.eventFilterVhcGps = new CustomEvent('eventFilterVhcGps', {
detail: {
markers,
},
});
function handlerFilterVhcGps(e) {
Menu.handlerFilterVhcGps(e.detail.markers);
}
window.addEventListener('eventFilterVhcGps', handlerFilterVhcGps);
// filter vehicle gsm
State.eventFilterVhcGsm = new CustomEvent('eventFilterVhcGsm', {
detail: {
markers,
},
});
function handlerFilterVhcGsm(e) {
Menu.handlerFilterVhcGsm(e.detail.markers);
}
window.addEventListener('eventFilterVhcGsm', handlerFilterVhcGsm);
// filter vehicle zone
State.eventFilterVhcZone = new CustomEvent('eventFilterVhcZone', {
detail: {
markers,
},
});
function handlerFilterVhcZone(e) {
Menu.handlerFilterVhcZone(e.detail.markers);
}
window.addEventListener('eventFilterVhcZone', handlerFilterVhcZone);
// hide all truck
State.eventHideAllTruck = new CustomEvent('eventHideAllTruck', {
detail: {
markers,
},
});
function handlerHideAllTruck(e) {
for (let marker of e.detail.markers) {
marker.setOpacity(0);
}
}
window.addEventListener('eventHideAllTruck', handlerHideAllTruck);
// hide truck not selected
State.eventHideTruckNotSelected = new CustomEvent('eventHideTruckNotSelected', {
detail: {
markers,
},
});
function handlerHideTruckNotSelected(e) {
for (let marker of e.detail.markers) {
if (State.inShowVid == marker.options.mdata.vid) {
marker.setOpacity(1);
continue;
}
marker.setOpacity(0);
}
}
window.addEventListener('eventHideTruckNotSelected', handlerHideTruckNotSelected);
// remove marker, circle, event listener and all about this marker
State.eventRemoveListTrucks = new CustomEvent('eventRemoveListTrucks', {
detail: {
markers,
},
});
window.addEventListener('eventRemoveListTrucks', function handler(e) {
for (let marker of e.detail.markers) {
marker.removeEventListener('click');
marker.removeEventListener('moveend');
marker.remove();
}
e.currentTarget.removeEventListener(e.type,
handler); // window.removeEventListener('remove', this.handler, true);
window.removeEventListener('eventFilterPopupVhcMarker', handlerFilterPopupVhcMarker);
window.removeEventListener('eventFilterVhcGps', handlerFilterVhcGps);
window.removeEventListener('eventFilterVhcGsm', handlerFilterVhcGsm);
window.removeEventListener('eventFilterVhcZone', handlerFilterVhcZone);
window.removeEventListener('eventHideTruckNotSelected', handlerHideTruckNotSelected);
window.removeEventListener('eventHideAllTruck', handlerHideAllTruck);
State.eventRemoveListTrucks = null;
State.statusFilterPopupVhcMarker = 1;
State.eventFilterPopupVhcMarker = null;
State.statusFilterVhcGps = 0;
State.eventFilterVhcGps = null;
State.statusFilterVhcGsm = 0;
State.eventFilterVhcGsm = null;
State.statusFilterVhcZone = 0;
State.eventFilterVhcZone = null;
State.eventHideTruckNotSelected = null;
State.eventHideAllTruck = null;
});
},
showDetailGeneral: function(truck) {
State.inShowVid = truck.vid;
// title
// <img src="https://products.unitedtractors.com/wp-content/uploads/2021/03/udtruck-GWE370.png" class="img-fluid" alt="">
// $('#infoVehicles-thumb-md').attr('src',`${State.storage_lara}${truck?.fvhc_img}`);
$('#infoVehicles-thumb-md').attr('src',
`${State.storage_lara}${truck?.fvhc_img}`);
// $('#infoVehicles-thumb-sm').attr('src',`${State.storage_lara}${truck?.fvhc_img}`);
$('#infoVehicles-thumb-sm').attr('src',
`${State.storage_lara}${truck?.fvhc_img}`);
$('#infoVehicles-platno').text(`${truck?.nopol1} ${truck?.nopol2} ${truck?.nopol3}`);
// vehicles
$('#infoVehicles-crt').text(moment.unix(truck?.lst_loc_crt).format('DD MMM YYYY HH:mm'));
// $('#infoVehicles-crt').text(moment.unix(truck?.lst_loc_crt_d).utcOffset(9 * 60).format('DD MMM YYYY HH:mm:ss'));
// ${(truck?.city_text || truck?.state_text || 'address')} - ${(truck?.postcode || 'postcode')}
if (truck?.ignition == State.stts_ignition.high) {
$('#infoVehicles-ignition').text('ON');
} else {
$('#infoVehicles-ignition').text('OFF');
}
if (truck?.stts_engine == State.stts_engine.moving) {
$('#infoVehicles-engineStatus').text('MOVING');
if (truck?.lst_speed == 0 || truck?.lst_speed == null) {
$('#infoVehicles-engineStatus').text('IDLING');
}
} else if (truck?.stts_engine == State.stts_engine.idling) {
$('#infoVehicles-engineStatus').text('IDLING');
} else if (truck?.stts_engine == State.stts_engine.stoping) {
$('#infoVehicles-engineStatus').text('STOPPED');
}
// no updated data more than 1 day
if (truck?.lst_loc_crt < moment().subtract(24, 'hour').unix()) {
$('#infoVehicles-ignition').text('OFF');
$('#infoVehicles-engineStatus').text('STOPPED');
}
$('#infoVehicles-addr-title').text(
`${Helper.shortenText(decodeURIComponent(truck?.lst_fulladdress || 'address'), 25)}`);
$('#infoVehicles-addr').text(
`${Helper.shortenText(decodeURIComponent(truck?.lst_fulladdress || 'address'), 25)}`);
$('#infoVehicles-lat_lng').text(Number(truck?.lst_lat).toFixed(5) + ', ' + Number(truck?.lst_lng).toFixed(6));
$('#infoVehicles-mileage').text(Number(truck?.vhc_sum_milleage_1).toFixed(0));
$('#infoVehicles-speedLimit').text(truck?.speed_limit);
$('#infoVehicles-currentSpeed').text(truck?.lst_speed);
let idlingDur = Wrapper.calcIdlingDur(truck?.lst_idle_at, truck?.lst_speed);
$('#infoVehicles-idlingDuration').text(idlingDur.dur);
$('#infoVehicles-idlingUnit').text(idlingDur.unit);
$('#infoVehicles-alertZones').text('-');
$('#infoVehicles-isTrackHoliday').text(truck?.is_track_holiday_text);
$('#infoVehicles-trackSch').text(truck?.track_schedule);
$('#infoVehicles-company').text(truck?.client_group_name);
$('#infoVehicles-serviceStart').text(moment.unix(truck?.vhc_crt).format('DD MMM YYYY HH:mm:ss'));
// driver
$('#infoDrv-updt').text((truck?.ord_crt) ? moment.unix(truck?.ord_crt).format('DD MMM YYYY HH:mm:ss') : '-');
$('#infoDrv-name').text(truck?.ord_drv_name ?? 'Off Duty');
$('.infoDrv-phone1-text').text((truck?.ord_drv_phone_val) ? Helper.splitEvery4Char('0' + truck?.ord_drv_phone_val) : '-');
$('#infoDrv-phone1-tel').attr('tel:0' + (truck?.ord_drv_phone_val ?? ''));
$('#infoDrv-phone1-wa').attr('https://api.whatsapp.com/send/?phone=62' + (truck?.ord_drv_phone_val ?? '') + '&text=Halo&app_absent=0');
$('.infoDrv-phone2-text').text((truck?.ord_drv_phone2_val) ? Helper.splitEvery4Char('0' + truck?.ord_drv_phone2_val) : '-');
$('#infoDrv-phone2-tel').attr('tel:0' + (truck?.ord_drv_phone2_val ?? ''));
$('#infoDrv-phone2-wa').attr('https://api.whatsapp.com/send/?phone=62' + (truck?.ord_drv_phone2_val ?? '') + '&text=Halo&app_absent=0');
// job
// $('#infoJob').html('');
// for (let i = 0; i < 20; i++) {
// $('#infoJob').append(`
// <li class="list-group-item p-1 px-2">
// <p class="text-bold mb-0">#902192${truck.vid}${Math.floor(Math.random()*90000) + 10000}</p>
// <div class="row d-flex align-items-center justify-content-between">
// <div class="col-auto">
// <small class="text-muted">Client name</small>
// <p id="" class="mb-0">JNE</p>
// </div>
// <div class="col-auto">
// <small class="text-muted">Destination</small>
// <p id="" class="mb-0">JKT-01</p></div>
// </div>
// </li>
// `);
// }
$('#infoJob-ord_client_name').text(truck?.ord_c_pt_name ?? '-');
$('#infoJob-pck_city').text(truck?.ord_pck_ktname ?? '-')
$('#infoJob-pck_name').text(truck?.ord_pck_name ?? '-');
$('#infoJob-pck_addr').text(truck?.ord_pck_addr ?? '-');
$('#infoJob-drop_city').text(truck?.ord_drop_ktname ?? '-')
$('#infoJob-drop_name').text(truck?.ord_drop_name ?? '-');
$('#infoJob-drop_addr').text(truck?.ord_drop_addr ?? '-');
let txtOrdStts = 'Not in order';
if (State.stts_order.wait == truck?.ord_stts) {
txtOrdStts = 'Waiting for Confirmation';
} else if (State.stts_order.confirm == truck?.ord_stts) {
txtOrdStts = 'Searching for Vehicle';
} else if (State.stts_order.have_get_vhc == truck?.ord_stts) {
txtOrdStts = 'Vehicle Heading to Pickup Location';
} else if (State.stts_order.pck == truck?.ord_stts) {
txtOrdStts = 'Order Successfully Picked Up';
} else if (State.stts_order.go == truck?.ord_stts) {
txtOrdStts = 'Heading to Delivery Location';
} else if (State.stts_order.arv == truck?.ord_stts) {
txtOrdStts = 'Arrived at Delivery Location';
} else if (State.stts_order.drop == truck?.ord_stts) {
txtOrdStts = 'Delivery Completed';
} else if (State.stts_order.client_pay == truck?.ord_stts) {
// txtOrdStts = 'Invoicing to Client';
txtOrdStts = 'Completed (Waiting for Payment)';
} else if (State.stts_order.vendor_payed == truck?.ord_stts) {
// txtOrdStts = 'Payment to Vendor';
txtOrdStts = 'Completed (Awaiting Vendor Payment)';
} else if (State.stts_order.close == truck?.ord_stts) {
txtOrdStts = 'Completed';
} else if (State.stts_order.cancel == truck?.ord_stts) {
txtOrdStts = 'Cancelled';
}
$('#infoJob-ord_stts').text(txtOrdStts);
// last movement
// $('#infoVehicles-infoMove').text('Last 24H from ' + moment.unix(truck?.lst_loc_crt).format('DD MMM YYYY HH:mm'));
},
bundleShowRouteTruck: async function(cache = false, needUpdate = true) {
Menu.clearListMovements();
if (State.inShowVid != State.loadedLastMoveVid)$('#historyStartDate').trigger('clearFilterHistoryDate');
if(needUpdate){
const getLastMove = await Trucks.getLastMove(State.inShowVid, cache);
Trucks.last_move = getLastMove.flat()
if (Trucks.last_move.length < 1) {
Helper.toast('Data Not Found', 'just now', 'There are no last data', 'bg-warning');
Trucks.last_move = null;
return false;
}
Trucks.showLastMoveToView(getLastMove);
}
},
getLastMove: async function(vid, cache = false) {
State.loadedLastMoveVid = vid;
State.inShowLastMove = vid;
if (cache) {
if (Trucks.last_move == null) {
return (await Trucks.reqLastMove({
vid
})).data;
} else {
return Trucks.last_move;
}
}
return (await Trucks.reqLastMove({
vid
})).data;
},
reqLastMove: function(params) {
return new Promise((resolve, reject) => {
$.ajax({
url: "{{ route('api_last_move_track_vhc') }}?start_date=" + `${State.historyStartDate}&end_date=${State.historyEndDate}`,
method: 'GET',
crossDomain: true,
processData: true,
headers: {
'x-api-key': Helper.getCookie('_trtk'),
},
data: params,
success: (data, textStatus, jqXHR) => {
if (data.meta.type != 'success') {
resolve({
type: 'fail',
data: null,
});
Helper.toast('Warning', 'just now', data.meta.message);
return false;
}
resolve({
type: 'success',
data: data.data,
});
},
error: (jqXHR, textStatus, error) => {
if (jqXHR.status >= 500) {
Helper.toast('Error', 'just now', 'Cannot get lists trucks');
} else {
Helper.toast('Error', 'just now', jqXHR.responseJSON.meta
.message);
}
resolve({
type: 'error',
data: null,
});
}
})
});
},
showLastMoveToView: function(truckRoutes) {
// Leaflet.clearLayer('eventAboutTruck');
Leaflet.hideLayer('eventHideAllTruck');
// Trucks.routeStartEnd(truckRoutes);
Trucks.routeStartEndGroupTrip(truckRoutes);
},
// routeStartEnd: function(truckRoutes) {
// let key_length = truckRoutes.length;
// let polyTruckRoutes = truckRoutes.map((obj, key) => {
// obj.key_index = key + 1 // key_length - key;
// // lists per detail info movement
// Menu.showToListMovement(obj);
// return {
// lat: obj.latitude,
// lng: obj.longitude,
// options: {
// // polyline
// smoothFactor: 1.0,
// noClip: true,
// bubblingMouseEvents: false,
// // circle
// radius: 5,
// // markerOpacity: 0
// },
// // 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.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')}`,
// }
// });
// let polyline = Leaflet.addPolylines(polyTruckRoutes);
// Leaflet.map.fitBounds(polyline.getBounds());
// // let ctrWaypoint = Leaflet.addWaypoints(polyTruckRoutes.slice(0, 100))
// Leaflet.addCircles(polyTruckRoutes, function(circle, i) {
// window.addEventListener('eventRemoveRouteStartEnd', function handler(e) {
// circle.remove();
// });
// circle.on('click', function() {
// PgBar.syncToPlotTravelHistory(i);
// })
// });
// let start = truckRoutes.at(-1);
// let finish = truckRoutes.at(0);
// // ${(start?.city_text || start?.state_tex || 'address't)} - ${(start?.postcode || 'postcode')}
// // ${(finish?.city_text || finish?.state_text || 'address')} - ${(finish?.postcode || 'postcode')}
// let startMarker = Leaflet.addMarkers({
// lat: start.latitude,
// 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_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: {
// icon: Icon.titikAwal(),
// // rotationAngle: 290
// }
// });
// let finishMarker = Leaflet.addMarkers({
// lat: finish.latitude,
// 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_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: {
// icon: Icon.titikAkhir()
// }
// });
// // remove marker, circle, event listener and all about this marker
// State.eventRemoveRouteStartEnd = new CustomEvent('eventRemoveRouteStartEnd', {
// startMarker,
// finishMarker,
// polyline,
// polyline,
// });
// 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,
// handler); // window.removeEventListener('remove', this.handler, true);
// State.eventRemoveRouteStartEnd = null;
// State.inShowLastMove = null;
// PgBar.tglMenuPlayback(false);
// PgBar.reset();
// });
// PgBar.setMinMax(0, truckRoutes.length - 1);
// },
routeStartEndGroupTrip: async function(truckRoutes) {
const colors = [
"#2980B9", // Dark Blue
"#C0392B", // Dark Red
"#27AE60", // Dark Green
"#D35400", // Dark Orange
"#F39C12", // Dark Yellow/Gold
]
let i = 1
let allStartStop = []
let polyTruckRoutes = truckRoutes.map((groupTrip, key0) => {
return groupTrip.map((obj, key) => {
obj.key_index = i++
// console.log("obj", obj.key_index);
Menu.showToListMovement(obj)
// add start end marker per group trip
if(key == 0 || key == groupTrip.length - 1){
const marker = Leaflet.addMarkers({
lat: obj.latitude,
lng: obj.longitude,
label: `
<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>
Speed: ${(typeof obj.lst_speed != 'undefined') ? obj.lst_speed : '0'}<br>
${Number(obj.latitude).toFixed(5)},${Number(obj.longitude).toFixed(6)}<br>
${decodeURIComponent(obj.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: {
icon: (key == 0) ? Icon.titikAkhir() : (key == (groupTrip.length - 1)) ? Icon.titikAwal() : null,
// rotationAngle: 290
}
})
allStartStop.push(marker)
}
const unix = parseInt(obj?.lst_loc_crt_d) + AppState.TIMEFIX;
return {
lat: obj.latitude,
lng: obj.longitude,
options: {
// polyline
color : colors[key0 % colors.length],
smoothFactor : 0,
noClip : true,
bubblingMouseEvents : false,
// circle
radius: 5,
markerOpacity: 0
},
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,
}
})
})
// console.log("truckRoutes update", polyTruckRoutes);
let circleCounter = 0
for (const [idx, poly] of polyTruckRoutes.entries()) {
// console.log("poly", poly);
function addCirclesAsync(polyCircle) {
// console.log("circleCounter", circleCounter);
// return array of circles start end only
return new Promise(resolve => {
let circles = [];
Leaflet.addCircles(polyCircle, function(circle, i) {
// console.log("circle", i, circleCounter);
if (i === 0 || i === (polyCircle.length - 1)) {
circles.push(circle);
}
const currentCounter = circleCounter;
circle.on('click', function() {
PgBar.syncToPlotTravelHistory(currentCounter);
});
circleCounter++
// ✅ When last point processed, resolve
if (i === polyCircle.length - 1) {
resolve(circles);
}
});
});
}
let polyline = await Leaflet.addRoutes(poly);
// console.log("polyline", polyline);
allStartStop.push(polyline)
// wait for all circles
const ssmarker = await addCirclesAsync(poly);
// console.log("ssmarker", ssmarker);
allStartStop.push(...ssmarker)
}
console.log("allStartStop", allStartStop);
// console.log("circlesStartStop", circlesStartStop);
// // remove marker, circle, event listener and all about this marker
State.eventRemoveRouteStartEnd = new CustomEvent('eventRemoveRouteStartEnd', {
// detail: {
allStartStop,
// circlesStartStop
// }
});
// window.addEventListener('eventRemoveRouteStartEnd', function (e) {
// // allStartStop.remove();
// allStartStop.forEach(c => c.remove())
// // circlesStartStop.forEach(c => c.remove())
// });
window.addEventListener('eventRemoveRouteStartEnd', function handler(e) {
allStartStop.forEach(c => c.remove())
e.currentTarget.removeEventListener(e.type,
handler); // window.removeEventListener('remove', this.handler, true);
State.eventRemoveRouteStartEnd = null;
State.inShowLastMove = null;
PgBar.tglMenuPlayback(false);
PgBar.reset();
});
PgBar.setMinMax(0, Trucks.last_move.length - 1);
},
routePerPoint: function() {
Leaflet.addMarkers(truckRoutes.routes.map((obj) => {
return {
lat: obj.lat,
lng: obj.lng,
options: {
icon: Icon.point(),
vhc_id: truckRoutes.vhc_id,
gps_id: truckRoutes.gps_id
}
};
}));
},
routeStartEndWithStops: function() {
Leaflet.addPolylines(truckRoutes.routes.map((obj) => {
return {
lat: obj.lat,
lng: obj.lng,
options: {}
}
}), function(polyline) {
// zoom the map to the polyline
// Leaflet.map.fitBounds(polyline.getBounds());
});
Leaflet.addMarkers([{
...truckRoutes.routes.at(0),
label: 'Lokasi Mulai Perjalanan Truk',
options: {
icon: Icon.startNav(),
rotationAngle: 290
}
}]);
Leaflet.addMarkers([{
...truckRoutes.routes.at(20),
label: 'Lokasi Berhenti Truk',
options: {
icon: Icon.point()
}
}]);
Leaflet.addMarkers([{
...truckRoutes.routes.at(40),
label: 'Lokasi Berhenti Truk',
options: {
icon: Icon.point()
}
}]);
Leaflet.addMarkers([{
...truckRoutes.routes.at(60),
label: 'Lokasi Berhenti Truk',
options: {
icon: Icon.point()
}
}]);
Leaflet.addMarkers([{
...truckRoutes.routes.at(-1),
label: 'Lokasi Tujuan Truk',
options: {
icon: Icon.destination()
}
}]);
},
routeAnimationStartShringkingToEnd: function() {
let polyline = null,
marker = null,
timeouts = [];
for (let i = 0; i < truckRoutes.routes.length; i++) {
timeouts.push(setTimeout(() => {
if (polyline) polyline.remove();
if (marker) marker.remove();
// jika typeof variable !== 'undefined'
if (!!truckRoutes.routes.at(i + 1)) {
polyline = Leaflet.addPolylines(truckRoutes.routes.slice((i + 1)).map((
obj) => {
return {
lat: obj.lat,
lng: obj.lng,
options: {}
}
}), function(polyline) {
// zoom the map to the polyline
Leaflet.map.fitBounds(polyline.getBounds());
});
}
marker = Leaflet.addMarkers({
...truckRoutes.routes.at(i),
label: 'Lokasi Truk',
options: {
icon: Icon.startNav(),
rotationAngle: 0
}
});
}, (i + 1) * State.timingRouteAnimation));
}
// remove marker, polyline, event listener and all about this marker
State.eventRemoveRouteAnimationStartShrinkingToEnd = new CustomEvent(
'eventRemoveRouteAnimationStartShrinkingToEnd', {
marker: marker,
polyline: polyline
});
window.addEventListener('eventRemoveRouteAnimationStartShrinkingToEnd', function handler(e) {
for (let i = 0; i < timeouts.length; i++) {
clearTimeout(timeouts[i]);
}
if (marker) marker.remove();
if (polyline) polyline.remove();
e.currentTarget.removeEventListener(e.type,
handler); // window.removeEventListener('remove', this.handler, true);
State.eventRemoveRouteAnimationStartShrinkingToEnd = null;
});
},
routeAnimationStartFootprintsTillEnd: function() {
let polyline = null,
marker = null,
timeouts = [];
for (let i = 0; i < truckRoutes.routes.length; i++) {
timeouts.push(setTimeout(() => {
if (polyline) polyline.remove();
if (marker) marker.remove();
polyline = Leaflet.addPolylines(truckRoutes.routes.slice(0, (i + 1)).map((
obj) => {
return {
lat: obj.lat,
lng: obj.lng,
options: {}
}
}), function(polyline) {
// zoom the map to the polyline
Leaflet.map.fitBounds(polyline.getBounds());
});
marker = Leaflet.addMarkers({
...truckRoutes.routes.at(i),
label: 'Lokasi Truk',
options: {
icon: Icon.startNav(),
rotationAngle: 0
}
});
}, (i + 1) * State.timingRouteAnimation));
}
// remove marker, polyline, event listener and all about this marker
State.eventRemoveRouteAnimationStartFootprintsTillEnd = new CustomEvent(
'eventRemoveRouteAnimationStartFootprintsTillEnd', {
marker: marker,
polyline: polyline
});
window.addEventListener('eventRemoveRouteAnimationStartFootprintsTillEnd', function handler(e) {
for (let i = 0; i < timeouts.length; i++) {
clearTimeout(timeouts[i]);
}
if (marker) marker.remove();
if (polyline) polyline.remove();
e.currentTarget.removeEventListener(e.type,
handler); // window.removeEventListener('remove', this.handler, true);
State.eventRemoveRouteAnimationStartFootprintsTillEnd = null;
});
},
}
const Zone = {
lists: null,
activate: function() {
Zone.bundleGetListZones(false);
},
bundleGetListZones: async function(cache = false) {
Zone.lists = await Zone.getListZones(cache);
Zone.showToLists(Zone.lists);
Zone.showListsToMap(Zone.lists);
},
getListZones: async function(cache = false) {
if (cache) {
if (Zone.lists == null) {
return (await Zone.reqListZones()).data;
} else {
return Zone.lists;
}
}
return (await Zone.reqListZones()).data;
},
reqListZones: function(params) {
return new Promise((resolve, reject) => {
$.ajax({
url: "{{ route('api_universal_list_zones') }}?status=" + State.zone_status.active + "&cptid=" + AppState.current_company,
method: 'GET',
crossDomain: true,
processData: true,
headers: {
'x-api-key': Helper.getCookie('_trtk'),
},
data: params,
success: (data, textStatus, jqXHR) => {
if (data.meta.type != 'success') {
resolve({
type: 'fail',
data: null,
});
Helper.toast('Warning', 'just now', data.meta.message);
return false;
}
resolve({
type: 'success',
data: data.data,
});
},
error: (jqXHR, textStatus, error) => {
if (jqXHR.status >= 500) {
Helper.toast('Error', 'just now', 'Cannot get lists trucks');
} else {
Helper.toast('Error', 'just now', jqXHR.responseJSON.meta
.message);
}
resolve({
type: 'error',
data: null,
});
}
})
});
},
showToLists: function(zones) {
$('#listZone').html('');
let str = '';
for (let i = 0; i < zones.length; i++) {
str += `
<li class="list-group-item vehicles-list-wrapper zone-item p-1 px-2" data-zid="${zones[i].id}">
<a href="#" class="text-dark">
<div class="row d-flex align-items-center">
<div class="col-3 d-flex justify-content-center">
<div class="align-items-center d-flex justify-content-center zone-icon" style="background: ${zones[i].boundary_hex_color}">
<span class="ion-android-pin text-white"></span>
</div>
</div>
<div class="col ps-0">
<p class="text-bold mb-0 textZoneName">${zones[i].name}</p>
<p class="text-muted mb-0 textZoneType">${zones[i].type_name}</p>
</div>
</div>
</a>
</li>
`;
}
$('#c_list_zone').text(zones.length);
$('#listZone').html(str);
},
showListsToMap: function(zones) {
let locations = []; // [{ lat,lng,label,options:{} }]
let circles = [],
polygons = [];
for (let i = 0; i < zones.length; i++) {
let boundary_latlngs = JSON.parse(zones[i].boundary_latlngs);
boundary_latlngs = boundary_latlngs.map((obj) => {
return {
lat: obj.lat,
lng: obj.lng,
label: `<b>${zones[i].name}</b><br>${decodeURIComponent(zones[i].fulladdress || 'address')}`,
options: {
mdata: zones[i],
color: zones[i].boundary_hex_color,
fillColor: zones[i].boundary_hex_color,
radius: zones[i].boundary_radius,
},
};
});
if (zones[i].boundary_type == State.zone_boundary_type.circle) {
let circle = Leaflet.addCircles(boundary_latlngs[0]);
circles.push(circle);
} else if (zones[i].boundary_type == State.zone_boundary_type.polygon || zones[i]
.boundary_type == State.zone_boundary_type.rectangle) {
let polygon = Leaflet.addPolygons(boundary_latlngs);
polygons.push(polygon);
} else {
Helper.toast('Error', 'just now', 'unknown boundary type');
continue;
}
}
// remove marker, circle, event listener and all about this marker
State.eventRemoveListZones = new CustomEvent('eventRemoveListZones', {
circles,
polygons
});
window.addEventListener('eventRemoveListZones', function handler(e) {
for (let circle of circles) {
circle.removeEventListener('click');
circle.remove();
}
for (let polygon of polygons) {
polygon.removeEventListener('click');
polygon.remove();
}
e.currentTarget.removeEventListener(e.type,
handler); // window.removeEventListener('remove', this.handler, true);
State.eventRemoveListZones = null;
});
},
showDetailGeneral: function(zone) {
State.inShowZid = zone.zid;
$('#zd-name').text(zone.name);
$('#zd-boundary_hex_color').css('background-color', zone.boundary_hex_color || '#000000')
$('#zd-type_name').text(zone.type_name);
$('#zd-type_name1').text(zone.type_name);
$('#zd-client').text(zone.client_group_name);
$('#zd-workflow_type_name').text(zone.workflow_type_name);
$('#zd-fulladdress').text(decodeURIComponent(zone.fulladdress || 'address'));
$('#zd-shiptocode').text(zone.shiptocode);
$('#zd-updt_at').text(moment(zone.updt).format('DD MMM YYYY HH:mm'));
$('#zd-updt_by').text(zone.updt_name);
},
}
const Client = {
client: null,
activate: function() {
Client.bundleGetClient(false);
},
bundleGetClient: async function(cache = false) {
Client.client = await Client.getClient(cache);
Client.showToView(Client.client);
},
getClient: async function(cache = false) {
if (cache) {
if (Client.client == null) {
return (await Client.reqClient()).data;
} else {
return Client.client;
}
}
return (await Client.reqClient()).data;
},
reqClient: function(params) {
return new Promise((resolve, reject) => {
$.ajax({
url: "{{ route('api_universal_show_client_pt') }}?cptid=" + AppState.current_company,
method: 'GET',
crossDomain: true,
processData: true,
headers: {
'x-api-key': Helper.getCookie('_trtk'),
},
data: params,
success: (data, textStatus, jqXHR) => {
if (data.meta.type != 'success') {
resolve({
type: 'fail',
data: null,
});
Helper.toast('Warning', 'just now', data.meta.message);
return false;
}
resolve({
type: 'success',
data: data.data,
});
},
error: (jqXHR, textStatus, error) => {
if (jqXHR.status >= 500) {
Helper.toast('Error', 'just now', 'Cannot get company information');
} else {
Helper.toast('Error', 'just now', jqXHR.responseJSON.meta
.message);
}
resolve({
type: 'error',
data: null,
});
}
})
});
},
showToView: function(client) {
$('#c_name').text(client.c_name);
$('#ptName').text(client.c_name);
$('#c_logo').attr('src', State.storage_lara + '/' + client.c_logo);
},
}
const Filter = {
activate: function() {
Filter.event();
},
event: function() {},
triggerFilterCompany: async function() {
Leaflet.clearLayer('all');
// client
await Client.bundleGetClient(false);
// trucks
State.loadedLastMoveVid = null;
State.inShowZid = null;
State.lastShowVids = [];
await Trucks.bundleGetListTrucks(false);
// zones
State.inShowZid = null;
await Zone.bundleGetListZones(false);
},
}
const PgBar = {
activate: function() {
PgBar.event();
},
event: function() {
$('#pgplay').on('click', function(e) {
const click = $(e.target);
if (State.playback.stts === 0 || State.playback.stts === 2 || State.playback.stts === 4) {
PgBar.resume();
} else if (State.playback.stts === 1 || State.playback.stts === 3) {
PgBar.pause();
}
});
document.getElementById('pgbar').oninput = function() {
this.value = Number(this.value);
this.min = Number(this.min);
this.max = Number(this.max);
State.playback.crntPgIndicator = this.value;
let indicator = Helper.getPercentRange(this.value, this.min, this.max);
this.style.background = `linear-gradient(to right, var(--bs-blue) 0%, var(--bs-blue) ${indicator}%, #d3d3d3 0%)`;
let arrIdx = Helper.getIndexReversedSequence(this.value, this.max);
if (Trucks.last_move === null || Trucks.last_move.length < 1) {} else {
let tr = Trucks.last_move[arrIdx];
setTimeout(() => {
Menu.createMarkerDetailPlotMovement(tr);
}, 1);
}
PgBar.syncToPlotTravelHistory(arrIdx);
}
},
setMinMax: function(min, max) {
min = Number(min);
max = Number(max);
PgBar.reset();
PgBar.tglMenuPlayback(true);
const bar = document.getElementById('pgbar');
bar.min = min;
bar.max = max;
State.playback.minIndicator = min;
State.playback.maxIndicator = max;
},
setCrntVal: function(val) {
val = Number(val);
State.playback.crntIndicator = val;
const bar = document.getElementById('pgbar');
bar.value = State.playback.crntIndicator;
let indicator = Helper.getPercentRange(bar.value, bar.min, bar.max);
bar.style.background = `linear-gradient(to right, var(--bs-blue) 0%, var(--bs-blue) ${indicator}%, #d3d3d3 0%)`;
},
reset: function() {
State.playback.stts = 0;
State.playback.crntIndicator = 0;
State.playback.minIndicator = 0;
State.playback.maxIndicator = 1000;
clearInterval(State.playback.intrvl);
const bar = document.getElementById('pgbar');
bar.min = State.playback.minIndicator;
bar.max = State.playback.maxIndicator;
bar.value = State.playback.crntIndicator;
bar.style.background = `linear-gradient(to right, var(--bs-blue) 0%, var(--bs-blue) ${0}%, #d3d3d3 0%)`;
PgBar.tglIconPlayback(State.playback.stts);
},
resume: function() {
// not allowed
if (State.playback.stts === 1 || State.playback.stts === 3) return false;
if (Trucks.last_move === null || Trucks.last_move.length < 1) {
Helper.toast('Warning', 'just now', 'Data riwayat tidak ada');
return false;
}
State.playback.stts = 1;
PgBar.tglIconPlayback(State.playback.stts);
State.playback.intrvl = setInterval(frame, State.playback.intrvlTime);
function frame() {
const bar = document.getElementById('pgbar');
let value = Number(bar.value);
let min = Number(bar.min);
let max = Number(bar.max);
let arrIdx = Helper.getIndexReversedSequence(value, max);
if (value >= max) {
clearInterval(State.playback.intrvl);
State.playback.stts = 4;
PgBar.tglIconPlayback(State.playback.stts);
} else {
++value;
bar.value = value;
let indicator = Helper.getPercentRange(value, min, max);
bar.style.background = `linear-gradient(to right, var(--bs-blue) 0%, var(--bs-blue) ${indicator}%, #d3d3d3 0%)`;
let tr = Trucks.last_move[arrIdx];
Menu.createMarkerDetailPlotMovement(tr, {
nozoom: true
});
}
PgBar.syncToPlotTravelHistory(arrIdx);
}
},
pause: function() {
// not allowed
if (State.playback.stts === 0 || State.playback.stts === 2 || State.playback.stts === 4) return false;
State.playback.stts = 2;
const bar = document.getElementById('pgbar');
PgBar.tglIconPlayback(State.playback.stts);
clearInterval(State.playback.intrvl);
// just show markers with openPopup
let value = Number(bar.value);
let min = Number(bar.min);
let max = Number(bar.max);
let arrIdx = Helper.getIndexReversedSequence(value, max);
let tr = Trucks.last_move[arrIdx];
setTimeout(() => {
Menu.createMarkerDetailPlotMovement(tr, {
nozoom: true
});
}, 1);
},
play1: function() {
let indicator = 0; // percent: double
const bar = document.getElementById('pgbar');
let frameProgress = setInterval(frame, 10);
function frame() {
if (indicator >= 100) {
clearInterval(frameProgress);
} else {
++indicator;
bar.style.width = indicator + '%';
}
}
},
tglMenuPlayback: function(show = true) {
if (show) document.getElementById('menuPlayback').classList.remove('d-none');
else document.getElementById('menuPlayback').classList.add('d-none');
},
tglIconPlayback: function(stts) {
const iconPg = document.getElementById('icon-pg');
if (stts === 0 || stts === 2 || stts === 4) {
iconPg.classList.remove('ion-pause');
iconPg.classList.add('ion-play');
} else if (stts === 1 || stts === 3) {
iconPg.classList.remove('ion-play');
iconPg.classList.add('ion-pause');
}
},
syncToPlotTravelHistory: function(arrIdx) {
console.log("syncToPlotTravelHistory", arrIdx);
const listPlotTravelHistory = document.querySelectorAll('#infoMove-plots .plotMove-item')[arrIdx];
listPlotTravelHistory.scrollIntoView();
listPlotTravelHistory.querySelector('li').classList.add('hover');
setTimeout(() => {
listPlotTravelHistory.querySelector('li').classList.remove('hover');
}, 500);
}
}
Wrapper.activate();
</script>
@endsection