Compare commits
52 Commits
feat/load-
...
fix/sync-t
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
22ee7502ad | ||
|
|
2f3bc172eb | ||
|
|
bba932b2ba | ||
|
|
3f5d0eb1cd | ||
|
|
1f33d0de4e | ||
|
|
86d694bcac | ||
|
|
cb5a3243fc | ||
|
|
15210a56ee | ||
|
|
a08f2cb2b7 | ||
|
|
632433c496 | ||
|
|
5b4780495e | ||
|
|
0a7012a57c | ||
|
|
435a19346b | ||
|
|
8fcf8859d6 | ||
|
|
43a246d234 | ||
|
|
d6d0acf8fb | ||
|
|
b4ec7a9d25 | ||
|
|
5203babe11 | ||
|
|
c0faafdbd7 | ||
|
|
c5e3fdd915 | ||
|
|
572b86299c | ||
|
|
cdd84d02da | ||
|
|
ee1a395c75 | ||
|
|
3bfcaddba4 | ||
|
|
9ea7e96af1 | ||
|
|
e0d11af7d2 | ||
|
|
fefef609ac | ||
|
|
f5790cda94 | ||
|
|
f3db3783f9 | ||
|
|
101e76c0fa | ||
|
|
4cc698a623 | ||
|
|
544ad1db46 | ||
|
|
30ca819aa1 | ||
|
|
b0bab784d1 | ||
|
|
01fda22c89 | ||
|
|
de300c2c32 | ||
|
|
7f8a2e4936 | ||
|
|
38948b6633 | ||
|
|
aa9943ba45 | ||
|
|
9e55ea0dbb | ||
|
|
93f0ac5ef7 | ||
|
|
8bae33519f | ||
|
|
6865e353e6 | ||
|
|
bb4ab5c769 | ||
|
|
e743b82087 | ||
|
|
c3c7b8e3ec | ||
|
|
d49035ce8d | ||
|
|
ffd9d3514c | ||
|
|
675477c734 | ||
|
|
4350c466e3 | ||
|
|
5ab407d672 | ||
|
|
2aaa487746 |
85
README.md
85
README.md
@@ -1,66 +1,37 @@
|
|||||||
<p align="center"><a href="https://laravel.com" target="_blank"><img src="https://raw.githubusercontent.com/laravel/art/master/logo-lockup/5%20SVG/2%20CMYK/1%20Full%20Color/laravel-logolockup-cmyk-red.svg" width="400" alt="Laravel Logo"></a></p>
|
# Usage icon
|
||||||
|
|
||||||
<p align="center">
|
search or pick icon in <a href="https://icon-sets.iconify.design/mingcute/?keyword=mingcute">here</a>
|
||||||
<a href="https://github.com/laravel/framework/actions"><img src="https://github.com/laravel/framework/workflows/tests/badge.svg" alt="Build Status"></a>
|
|
||||||
<a href="https://packagist.org/packages/laravel/framework"><img src="https://img.shields.io/packagist/dt/laravel/framework" alt="Total Downloads"></a>
|
|
||||||
<a href="https://packagist.org/packages/laravel/framework"><img src="https://img.shields.io/packagist/v/laravel/framework" alt="Latest Stable Version"></a>
|
|
||||||
<a href="https://packagist.org/packages/laravel/framework"><img src="https://img.shields.io/packagist/l/laravel/framework" alt="License"></a>
|
|
||||||
</p>
|
|
||||||
|
|
||||||
## About Laravel
|
# Set up queue for running automatically
|
||||||
|
|
||||||
Laravel is a web application framework with expressive, elegant syntax. We believe development must be an enjoyable and creative experience to be truly fulfilling. Laravel takes the pain out of development by easing common tasks used in many web projects, such as:
|
- Install Supervisor
|
||||||
|
|
||||||
- [Simple, fast routing engine](https://laravel.com/docs/routing).
|
```
|
||||||
- [Powerful dependency injection container](https://laravel.com/docs/container).
|
sudo apt update && sudo apt install supervisor -y
|
||||||
- Multiple back-ends for [session](https://laravel.com/docs/session) and [cache](https://laravel.com/docs/cache) storage.
|
```
|
||||||
- Expressive, intuitive [database ORM](https://laravel.com/docs/eloquent).
|
|
||||||
- Database agnostic [schema migrations](https://laravel.com/docs/migrations).
|
|
||||||
- [Robust background job processing](https://laravel.com/docs/queues).
|
|
||||||
- [Real-time event broadcasting](https://laravel.com/docs/broadcasting).
|
|
||||||
|
|
||||||
Laravel is accessible, powerful, and provides tools required for large, robust applications.
|
- Create Supervisor Config
|
||||||
|
|
||||||
## Learning Laravel
|
```
|
||||||
|
sudo nano /etc/supervisor/conf.d/laravel-worker.conf
|
||||||
|
|
||||||
Laravel has the most extensive and thorough [documentation](https://laravel.com/docs) and video tutorial library of all modern web application frameworks, making it a breeze to get started with the framework.
|
[program:laravel-worker]
|
||||||
|
process_name=%(program_name)s_%(process_num)02d
|
||||||
|
command=php /path-to-your-project/artisan queue:work --tries=3 --timeout=600
|
||||||
|
autostart=true
|
||||||
|
autorestart=true
|
||||||
|
numprocs=1
|
||||||
|
user=www-data
|
||||||
|
redirect_stderr=true
|
||||||
|
stdout_logfile=/var/log/supervisor/laravel-worker.log
|
||||||
|
```
|
||||||
|
|
||||||
You may also try the [Laravel Bootcamp](https://bootcamp.laravel.com), where you will be guided through building a modern Laravel application from scratch.
|
- Reload Supervisor
|
||||||
|
|
||||||
If you don't feel like reading, [Laracasts](https://laracasts.com) can help. Laracasts contains thousands of video tutorials on a range of topics including Laravel, modern PHP, unit testing, and JavaScript. Boost your skills by digging into our comprehensive video library.
|
```
|
||||||
|
sudo supervisorctl reread
|
||||||
## Laravel Sponsors
|
sudo supervisorctl update
|
||||||
|
sudo supervisorctl start laravel-worker
|
||||||
We would like to extend our thanks to the following sponsors for funding Laravel development. If you are interested in becoming a sponsor, please visit the [Laravel Partners program](https://partners.laravel.com).
|
sudo supervisorctl restart laravel-worker
|
||||||
|
sudo supervisorctl status
|
||||||
### Premium Partners
|
```
|
||||||
|
|
||||||
- **[Vehikl](https://vehikl.com/)**
|
|
||||||
- **[Tighten Co.](https://tighten.co)**
|
|
||||||
- **[WebReinvent](https://webreinvent.com/)**
|
|
||||||
- **[Kirschbaum Development Group](https://kirschbaumdevelopment.com)**
|
|
||||||
- **[64 Robots](https://64robots.com)**
|
|
||||||
- **[Curotec](https://www.curotec.com/services/technologies/laravel/)**
|
|
||||||
- **[Cyber-Duck](https://cyber-duck.co.uk)**
|
|
||||||
- **[DevSquad](https://devsquad.com/hire-laravel-developers)**
|
|
||||||
- **[Jump24](https://jump24.co.uk)**
|
|
||||||
- **[Redberry](https://redberry.international/laravel/)**
|
|
||||||
- **[Active Logic](https://activelogic.com)**
|
|
||||||
- **[byte5](https://byte5.de)**
|
|
||||||
- **[OP.GG](https://op.gg)**
|
|
||||||
|
|
||||||
## Contributing
|
|
||||||
|
|
||||||
Thank you for considering contributing to the Laravel framework! The contribution guide can be found in the [Laravel documentation](https://laravel.com/docs/contributions).
|
|
||||||
|
|
||||||
## Code of Conduct
|
|
||||||
|
|
||||||
In order to ensure that the Laravel community is welcoming to all, please review and abide by the [Code of Conduct](https://laravel.com/docs/contributions#code-of-conduct).
|
|
||||||
|
|
||||||
## Security Vulnerabilities
|
|
||||||
|
|
||||||
If you discover a security vulnerability within Laravel, please send an e-mail to Taylor Otwell via [taylor@laravel.com](mailto:taylor@laravel.com). All security vulnerabilities will be promptly addressed.
|
|
||||||
|
|
||||||
## License
|
|
||||||
|
|
||||||
The Laravel framework is open-sourced software licensed under the [MIT license](https://opensource.org/licenses/MIT).
|
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Console\Commands;
|
namespace App\Console\Commands;
|
||||||
|
|
||||||
|
use App\Jobs\SyncronizeSIMBG;
|
||||||
use App\Services\ServiceSIMBG;
|
use App\Services\ServiceSIMBG;
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
use \Illuminate\Support\Facades\Log;
|
use \Illuminate\Support\Facades\Log;
|
||||||
@@ -28,13 +29,12 @@ class ExecuteScraping extends Command
|
|||||||
|
|
||||||
private $service_simbg;
|
private $service_simbg;
|
||||||
|
|
||||||
public function __construct(ServiceSIMBG $service_simbg){
|
public function __construct(){
|
||||||
$this->service_simbg = $service_simbg;
|
|
||||||
parent::__construct();
|
parent::__construct();
|
||||||
}
|
}
|
||||||
public function handle()
|
public function handle()
|
||||||
{
|
{
|
||||||
|
SyncronizeSIMBG::dispatch();
|
||||||
Log::info("running scheduler daily scraping");
|
Log::info("running scheduler daily scraping");
|
||||||
$this->service_simbg->syncTaskList();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
234
app/Http/Controllers/Api/BigDataResumeController.php
Normal file
234
app/Http/Controllers/Api/BigDataResumeController.php
Normal file
@@ -0,0 +1,234 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Http\Controllers\Api;
|
||||||
|
|
||||||
|
use App\Http\Controllers\Controller;
|
||||||
|
use App\Http\Resources\BigdataResumeResource;
|
||||||
|
use App\Models\BigdataResume;
|
||||||
|
use App\Models\DataSetting;
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
use Illuminate\Support\Facades\Log;
|
||||||
|
|
||||||
|
class BigDataResumeController extends Controller
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Display a listing of the resource.
|
||||||
|
*/
|
||||||
|
public function index(Request $request)
|
||||||
|
{
|
||||||
|
try{
|
||||||
|
$filterDate = $request->get("filterByDate");
|
||||||
|
|
||||||
|
if (!$filterDate || $filterDate === "latest") {
|
||||||
|
$big_data_resume = BigdataResume::where('year', now()->year)->latest()->first();
|
||||||
|
if (!$big_data_resume) {
|
||||||
|
return $this->response_empty_resume();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$big_data_resume = BigdataResume::whereDate('created_at', $filterDate)
|
||||||
|
->orderBy('id', 'desc')
|
||||||
|
->first();
|
||||||
|
|
||||||
|
if (!$big_data_resume) {
|
||||||
|
return $this->response_empty_resume();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$data_settings = DataSetting::all();
|
||||||
|
if($data_settings->isEmpty()){
|
||||||
|
return response()->json(['message' => 'No data setting found']);
|
||||||
|
}
|
||||||
|
|
||||||
|
function cleanNumber($value) {
|
||||||
|
return floatval(str_replace('.', '', $value));
|
||||||
|
}
|
||||||
|
|
||||||
|
$target_pad = floatval(optional($data_settings->where('key', 'TARGET_PAD')->first())->value);
|
||||||
|
$realisasi_terbit_pbg_sum = cleanNumber(optional($data_settings->where('key', 'REALISASI_TERBIT_PBG_SUM')->first())->value);
|
||||||
|
$realisasi_terbit_pbg_count = cleanNumber(optional($data_settings->where('key', 'REALISASI_TERBIT_PBG_COUNT')->first())->value);
|
||||||
|
$menunggu_klik_dpmptsp_sum = cleanNumber(optional($data_settings->where('key', 'MENUNGGU_KLIK_DPMPTSP_SUM')->first())->value);
|
||||||
|
$menunggu_klik_dpmptsp_count = cleanNumber(optional($data_settings->where('key', 'MENUNGGU_KLIK_DPMPTSP_COUNT')->first())->value);
|
||||||
|
$proses_dinas_teknis_sum = cleanNumber(optional($data_settings->where('key', 'PROSES_DINAS_TEKNIS_SUM')->first())->value);
|
||||||
|
$proses_dinas_teknis_count = cleanNumber(optional($data_settings->where('key', 'PROSES_DINAS_TEKNIS_COUNT')->first())->value);
|
||||||
|
|
||||||
|
$tata_ruang = $big_data_resume->spatial_sum;
|
||||||
|
$kekurangan_potensi = $target_pad - $big_data_resume->potention_sum;
|
||||||
|
|
||||||
|
// percentage kekurangan potensi
|
||||||
|
$kekurangan_potensi_percentage = $target_pad > 0 && $target_pad > 0
|
||||||
|
? round(($kekurangan_potensi / $target_pad) * 100, 2) : 0;
|
||||||
|
|
||||||
|
// percentage total potensi
|
||||||
|
$total_potensi_percentage = $big_data_resume->potention_sum > 0 && $target_pad > 0
|
||||||
|
? round(($big_data_resume->potention_sum / $target_pad) * 100, 2) : 0;
|
||||||
|
|
||||||
|
// percentage verified document
|
||||||
|
$verified_percentage = $big_data_resume->verified_sum > 0 && $big_data_resume->potention_sum > 0
|
||||||
|
? round(($big_data_resume->verified_sum / $big_data_resume->potention_sum) * 100, 2) : 0;
|
||||||
|
|
||||||
|
// percentage non-verified document
|
||||||
|
$non_verified_percentage = $big_data_resume->non_verified_sum > 0 && $big_data_resume->potention_sum > 0
|
||||||
|
? round(($big_data_resume->non_verified_sum / $big_data_resume->potention_sum) * 100, 2) : 0;
|
||||||
|
|
||||||
|
// percentage business document
|
||||||
|
$business_percentage = $big_data_resume->business_sum > 0 && $big_data_resume->non_verified_sum > 0
|
||||||
|
? round(($big_data_resume->business_sum / $big_data_resume->non_verified_sum) * 100, 2) : 0;
|
||||||
|
|
||||||
|
// percentage non-business document
|
||||||
|
$non_business_percentage = $big_data_resume->non_business_sum > 0 && $big_data_resume->potention_sum > 0
|
||||||
|
? round(($big_data_resume->non_business_sum / $big_data_resume->potention_sum) * 100, 2) : 0;
|
||||||
|
|
||||||
|
// percentage tata ruang
|
||||||
|
$tata_ruang_percentage = $tata_ruang > 0 && $big_data_resume->potention_sum > 0
|
||||||
|
? round(($tata_ruang / $big_data_resume->potention_sum) * 100, 2) : 0;
|
||||||
|
|
||||||
|
// percentage realisasi terbit pbg
|
||||||
|
$realisasi_terbit_percentage = $big_data_resume->verified_sum > 0 && $realisasi_terbit_pbg_sum > 0
|
||||||
|
? round(($realisasi_terbit_pbg_sum / $big_data_resume->verified_sum) * 100, 2) : 0;
|
||||||
|
|
||||||
|
// percentage menunggu klik dpmptsp
|
||||||
|
$menunggu_klik_dpmptsp_percentage = $big_data_resume->verified_sum > 0 && $menunggu_klik_dpmptsp_sum > 0
|
||||||
|
? round(($menunggu_klik_dpmptsp_sum / $big_data_resume->verified_sum) * 100, 2) : 0;
|
||||||
|
|
||||||
|
// percentage proses_dinas_teknis
|
||||||
|
$proses_dinas_teknis_percentage = $big_data_resume->verified_sum > 0 && $proses_dinas_teknis_sum > 0
|
||||||
|
? round(($proses_dinas_teknis_sum / $big_data_resume->verified_sum) * 100, 2) : 0;
|
||||||
|
|
||||||
|
$result = [
|
||||||
|
'target_pad' => [
|
||||||
|
'sum' => $target_pad,
|
||||||
|
'percentage' => 100,
|
||||||
|
],
|
||||||
|
'tata_ruang' => [
|
||||||
|
'sum' => $big_data_resume->spatial_sum,
|
||||||
|
'count' => $big_data_resume->spatial_count,
|
||||||
|
'percentage' => $tata_ruang_percentage,
|
||||||
|
],
|
||||||
|
'kekurangan_potensi' => [
|
||||||
|
'sum' => $kekurangan_potensi,
|
||||||
|
'percentage' => $kekurangan_potensi_percentage
|
||||||
|
],
|
||||||
|
'total_potensi' => [
|
||||||
|
'sum' => (float) $big_data_resume->potention_sum,
|
||||||
|
'count' => $big_data_resume->potention_count,
|
||||||
|
'percentage' => $total_potensi_percentage
|
||||||
|
],
|
||||||
|
'verified_document' => [
|
||||||
|
'sum' => (float) $big_data_resume->verified_sum,
|
||||||
|
'count' => $big_data_resume->verified_count,
|
||||||
|
'percentage' => $verified_percentage
|
||||||
|
],
|
||||||
|
'non_verified_document' => [
|
||||||
|
'sum' => (float) $big_data_resume->non_verified_sum,
|
||||||
|
'count' => $big_data_resume->non_verified_count,
|
||||||
|
'percentage' => $non_verified_percentage
|
||||||
|
],
|
||||||
|
'business_document' => [
|
||||||
|
'sum' => (float) $big_data_resume->business_sum,
|
||||||
|
'count' => $big_data_resume->business_count,
|
||||||
|
'percentage' => $business_percentage
|
||||||
|
],
|
||||||
|
'non_business_document' => [
|
||||||
|
'sum' => (float) $big_data_resume->non_business_sum,
|
||||||
|
'count' => $big_data_resume->non_business_count,
|
||||||
|
'percentage' => $non_business_percentage
|
||||||
|
],
|
||||||
|
'realisasi_terbit' => [
|
||||||
|
'sum' => $realisasi_terbit_pbg_sum,
|
||||||
|
'count' => $realisasi_terbit_pbg_count,
|
||||||
|
'percentage' => $realisasi_terbit_percentage
|
||||||
|
],
|
||||||
|
'menunggu_klik_dpmptsp' => [
|
||||||
|
'sum' => $menunggu_klik_dpmptsp_sum,
|
||||||
|
'count' => $menunggu_klik_dpmptsp_count,
|
||||||
|
'percentage' => $menunggu_klik_dpmptsp_percentage
|
||||||
|
],
|
||||||
|
'proses_dinas_teknis' => [
|
||||||
|
'sum' => $proses_dinas_teknis_sum,
|
||||||
|
'count' => $proses_dinas_teknis_count,
|
||||||
|
'percentage' => $proses_dinas_teknis_percentage
|
||||||
|
]
|
||||||
|
];
|
||||||
|
return response()->json($result);
|
||||||
|
}catch(\Exception $e){
|
||||||
|
return response()->json(['message' => 'Error when fetching data'], 500);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public function bigdata_report(Request $request){
|
||||||
|
try{
|
||||||
|
$query = BigdataResume::query()->orderBy('id', 'desc');
|
||||||
|
|
||||||
|
if($request->filled('search')){
|
||||||
|
$query->where('name', 'LIKE', '%'.$request->input('search').'%');
|
||||||
|
}
|
||||||
|
|
||||||
|
$query = $query->paginate(config('app.paginate_per_page', 50));
|
||||||
|
return BigdataResumeResource::collection($query)->response()->getData(true);
|
||||||
|
}catch(\Exception $e){
|
||||||
|
Log::error($e->getMessage());
|
||||||
|
return response()->json(['message' => 'Error when fetching data'], 500);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function response_empty_resume(){
|
||||||
|
$result = [
|
||||||
|
'target_pad' => [
|
||||||
|
'sum' => 0,
|
||||||
|
'percentage' => 100,
|
||||||
|
],
|
||||||
|
'tata_ruang' => [
|
||||||
|
'sum' => 0,
|
||||||
|
'percentage' => 0,
|
||||||
|
],
|
||||||
|
'kekurangan_potensi' => [
|
||||||
|
'sum' => 0,
|
||||||
|
'percentage' => 0
|
||||||
|
],
|
||||||
|
'total_potensi' => [
|
||||||
|
'sum' => 0,
|
||||||
|
'count' => 0,
|
||||||
|
'percentage' => 0
|
||||||
|
],
|
||||||
|
'verified_document' => [
|
||||||
|
'sum' => 0,
|
||||||
|
'count' => 0,
|
||||||
|
'percentage' => 0
|
||||||
|
],
|
||||||
|
'non_verified_document' => [
|
||||||
|
'sum' => 0,
|
||||||
|
'count' => 0,
|
||||||
|
'percentage' => 0
|
||||||
|
],
|
||||||
|
'business_document' => [
|
||||||
|
'sum' => 0,
|
||||||
|
'count' => 0,
|
||||||
|
'percentage' => 0
|
||||||
|
],
|
||||||
|
'non_business_document' => [
|
||||||
|
'sum' => 0,
|
||||||
|
'count' => 0,
|
||||||
|
'percentage' => 0
|
||||||
|
],
|
||||||
|
'realisasi_terbit' => [
|
||||||
|
'sum' => 0,
|
||||||
|
'count' => 0,
|
||||||
|
'percentage' => 0
|
||||||
|
],
|
||||||
|
'menunggu_klik_dpmptsp' => [
|
||||||
|
'sum' => 0,
|
||||||
|
'count' => 0,
|
||||||
|
'percentage' => 0
|
||||||
|
],
|
||||||
|
'proses_dinas_teknis' => [
|
||||||
|
'sum' => 0,
|
||||||
|
'count' => 0,
|
||||||
|
'percentage' => 0
|
||||||
|
]
|
||||||
|
];
|
||||||
|
|
||||||
|
return response()->json($result);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -9,6 +9,7 @@ use App\Models\BusinessOrIndustry;
|
|||||||
use Illuminate\Http\Request;
|
use Illuminate\Http\Request;
|
||||||
use Maatwebsite\Excel\Facades\Excel;
|
use Maatwebsite\Excel\Facades\Excel;
|
||||||
use \Illuminate\Support\Facades\Validator;
|
use \Illuminate\Support\Facades\Validator;
|
||||||
|
use App\Http\Requests\ExcelUploadRequest;
|
||||||
class BusinessOrIndustriesController extends Controller
|
class BusinessOrIndustriesController extends Controller
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
@@ -30,7 +31,7 @@ class BusinessOrIndustriesController extends Controller
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
return response()->json($query->paginate());
|
return response()->json($query->paginate(config('app.paginate_per_page', 50)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -79,29 +80,15 @@ class BusinessOrIndustriesController extends Controller
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function upload(Request $request){
|
public function upload(ExcelUploadRequest $request){
|
||||||
|
try {
|
||||||
if ($request->hasFile('file')) {
|
if(!$request->hasFile('file')){
|
||||||
$file = $request->file('file');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Validasi file
|
|
||||||
$validator = Validator::make($request->all(), [
|
|
||||||
'file' => 'required|mimes:xlsx,xls|max:102400', // Max 100MB
|
|
||||||
]);
|
|
||||||
|
|
||||||
if ($validator->fails()) {
|
|
||||||
return response()->json([
|
return response()->json([
|
||||||
'message' => 'File validation failed.',
|
'error' => 'No file provided'
|
||||||
'errors' => $validator->errors()
|
|
||||||
], 400);
|
], 400);
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
|
||||||
// Ambil file dari request
|
|
||||||
$file = $request->file('file');
|
$file = $request->file('file');
|
||||||
|
|
||||||
// Menggunakan Laravel Excel untuk mengimpor file
|
|
||||||
Excel::import(new BusinessIndustriesImport, $file);
|
Excel::import(new BusinessIndustriesImport, $file);
|
||||||
|
|
||||||
// Jika sukses, kembalikan respons sukses
|
// Jika sukses, kembalikan respons sukses
|
||||||
|
|||||||
119
app/Http/Controllers/Api/ChatbotController.php
Normal file
119
app/Http/Controllers/Api/ChatbotController.php
Normal file
@@ -0,0 +1,119 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Http\Controllers\Api;
|
||||||
|
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
use App\Services\OpenAIService;
|
||||||
|
use App\Http\Controllers\Controller;
|
||||||
|
use Illuminate\Http\Response;
|
||||||
|
use Illuminate\Support\Facades\DB;
|
||||||
|
use Illuminate\Support\Facades\Log;
|
||||||
|
|
||||||
|
class ChatbotController extends Controller
|
||||||
|
{
|
||||||
|
protected $openAIService;
|
||||||
|
|
||||||
|
public function __construct(OpenAIService $openAIService)
|
||||||
|
{
|
||||||
|
$this->openAIService = $openAIService;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function generateText(Request $request)
|
||||||
|
{
|
||||||
|
$request->validate([
|
||||||
|
'tab_active' => 'required|string',
|
||||||
|
'prompt' => 'required|string',
|
||||||
|
]);
|
||||||
|
|
||||||
|
$tab_active = $request->input('tab_active');
|
||||||
|
$main_content = match ($tab_active) {
|
||||||
|
"count-retribusi" => "RETRIBUTION",
|
||||||
|
"document-validation" => "DOCUMENT VALIDATION",
|
||||||
|
"data-information" => "DATA SUMMARY",
|
||||||
|
default => "UNKNOWN",
|
||||||
|
};
|
||||||
|
|
||||||
|
$chatHistory = $request->input('chatHistory');
|
||||||
|
// Log::info('Chat history sebelum disimpan:', ['history' => $chatHistory]);
|
||||||
|
|
||||||
|
if ($main_content === "UNKNOWN") {
|
||||||
|
return response()->json(['response' => 'Invalid tab_active value.'], 400);
|
||||||
|
}
|
||||||
|
|
||||||
|
// info($main_content);
|
||||||
|
|
||||||
|
$queryResponse = $this->openAIService->generateQueryBasedMainContent($request->input('prompt'), $main_content, $chatHistory);
|
||||||
|
|
||||||
|
$firstValidation = $this->openAIService->validateSyntaxQuery($queryResponse);
|
||||||
|
$secondValidation = $this->openAIService->validateSyntaxQuery($queryResponse);
|
||||||
|
|
||||||
|
$formattedResultQuery = "[]";
|
||||||
|
$queryResponse = str_replace(['```sql', '```'], '', $queryResponse);
|
||||||
|
$resultQuery = DB::select($queryResponse);
|
||||||
|
$formattedResultQuery = json_encode($resultQuery, JSON_PRETTY_PRINT);
|
||||||
|
// info($formattedResultQuery);
|
||||||
|
|
||||||
|
$nlpResult = $this->openAIService->generateNLPFromQuery($request->input('prompt'), $formattedResultQuery);
|
||||||
|
$finalGeneratedText =$this->openAIService->generateFinalText($nlpResult);
|
||||||
|
return response()->json(['response' => $finalGeneratedText, 'nlpResponse' => $queryResponse]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function mainGenerateText(Request $request)
|
||||||
|
{
|
||||||
|
// Log hanya data yang relevan
|
||||||
|
info("Received prompt: " . $request->input('prompt'));
|
||||||
|
|
||||||
|
// Validasi input
|
||||||
|
$request->validate([
|
||||||
|
'prompt' => 'required|string',
|
||||||
|
]);
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Panggil service untuk generate text
|
||||||
|
$classifyResponse = $this->openAIService->classifyMainGenerateText($request->input('prompt'));
|
||||||
|
info($classifyResponse);
|
||||||
|
|
||||||
|
// Pastikan hasil klasifikasi valid sebelum melanjutkan
|
||||||
|
$validCategories = [
|
||||||
|
'reklame', 'business_or_industries', 'customers',
|
||||||
|
'pbg', 'retribusi', 'spatial_plannings',
|
||||||
|
'tourisms', 'umkms'
|
||||||
|
];
|
||||||
|
|
||||||
|
if (!in_array($classifyResponse, $validCategories)) {
|
||||||
|
return response()->json([
|
||||||
|
'error' => ''
|
||||||
|
], 400);
|
||||||
|
}
|
||||||
|
|
||||||
|
$chatHistory = $request->input('chatHistory');
|
||||||
|
Log::info('Chat history sebelum disimpan:', ['history' => $chatHistory]);
|
||||||
|
|
||||||
|
$queryResponse = $this->openAIService->createMainQuery($classifyResponse, $request->input('prompt'), $chatHistory);
|
||||||
|
info($queryResponse);
|
||||||
|
|
||||||
|
$firstValidation = $this->openAIService->validateSyntaxQuery($queryResponse);
|
||||||
|
$secondValidation = $this->openAIService->validateSyntaxQuery($queryResponse);
|
||||||
|
|
||||||
|
$formattedResultQuery = "[]";
|
||||||
|
|
||||||
|
$queryResponse = str_replace(['```sql', '```'], '', $queryResponse);
|
||||||
|
$queryResult = DB::select($queryResponse);
|
||||||
|
|
||||||
|
$formattedResultQuery = json_encode($queryResult, JSON_PRETTY_PRINT);
|
||||||
|
|
||||||
|
$nlpResult = $this->openAIService->generateNLPFromQuery($request->input('prompt'), $formattedResultQuery);
|
||||||
|
$finalGeneratedText =$this->openAIService->generateFinalText($nlpResult);
|
||||||
|
|
||||||
|
return response()->json(['response' => $finalGeneratedText, 'nlpResponse' => $queryResponse]);
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
// Tangani error dan log exception
|
||||||
|
\Log::error("Error generating text: " . $e->getMessage());
|
||||||
|
|
||||||
|
return response()->json([
|
||||||
|
'error' => ''
|
||||||
|
], 500);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -22,9 +22,9 @@ class CustomersController extends Controller
|
|||||||
if ($request->has('search') &&!empty($request->get('search'))) {
|
if ($request->has('search') &&!empty($request->get('search'))) {
|
||||||
$query = $query->where('nomor_pelanggan', 'LIKE', '%'.$request->get('search').'%')
|
$query = $query->where('nomor_pelanggan', 'LIKE', '%'.$request->get('search').'%')
|
||||||
->orWhere('nama', 'LIKE', '%'.$request->get('search').'%')
|
->orWhere('nama', 'LIKE', '%'.$request->get('search').'%')
|
||||||
->orWhere('kota_palayanan', 'LIKE', '%'.$request->get('search').'%');
|
->orWhere('kota_pelayanan', 'LIKE', '%'.$request->get('search').'%');
|
||||||
}
|
}
|
||||||
return CustomersResource::collection($query->paginate());
|
return CustomersResource::collection($query->paginate(config('app.paginate_per_page', 50)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ class ImportDatasourceController extends Controller
|
|||||||
$search = $request->get("search");
|
$search = $request->get("search");
|
||||||
$query->where('status', 'like', "%".$search."%");
|
$query->where('status', 'like', "%".$search."%");
|
||||||
}
|
}
|
||||||
return ImportDatasourceResource::collection($query->paginate());
|
return ImportDatasourceResource::collection($query->paginate(config('app.paginate_per_page', 50)));
|
||||||
}
|
}
|
||||||
|
|
||||||
public function checkImportDatasource(){
|
public function checkImportDatasource(){
|
||||||
|
|||||||
34
app/Http/Controllers/Api/LackOfPotentialController.php
Normal file
34
app/Http/Controllers/Api/LackOfPotentialController.php
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Http\Controllers\Api;
|
||||||
|
|
||||||
|
use App\Http\Controllers\Controller;
|
||||||
|
use App\Models\Advertisement;
|
||||||
|
use App\Models\Customer;
|
||||||
|
use App\Models\SpatialPlanning;
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
use App\Models\TourismBasedKBLI;
|
||||||
|
|
||||||
|
class LackOfPotentialController extends Controller
|
||||||
|
{
|
||||||
|
|
||||||
|
public function count_lack_of_potential(){
|
||||||
|
try{
|
||||||
|
$total_reklame = Advertisement::count();
|
||||||
|
$total_pdam = Customer::count();
|
||||||
|
$total_tata_ruang = SpatialPlanning::count();
|
||||||
|
$data_report_tourism = TourismBasedKBLI::all();
|
||||||
|
|
||||||
|
return response()->json([
|
||||||
|
'total_reklame' => $total_reklame,
|
||||||
|
'total_pdam' => $total_pdam,
|
||||||
|
'total_tata_ruang' => $total_tata_ruang,
|
||||||
|
'data_report' => $data_report_tourism,
|
||||||
|
], 200);
|
||||||
|
}catch(\Exception $e){
|
||||||
|
return response()->json([
|
||||||
|
'message' => 'Error: '.$e->getMessage()
|
||||||
|
], 500);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -3,8 +3,11 @@
|
|||||||
namespace App\Http\Controllers\Api;
|
namespace App\Http\Controllers\Api;
|
||||||
|
|
||||||
use App\Http\Controllers\Controller;
|
use App\Http\Controllers\Controller;
|
||||||
|
use App\Http\Requests\MenuRequest;
|
||||||
|
use App\Http\Resources\MenuResource;
|
||||||
use App\Models\Menu;
|
use App\Models\Menu;
|
||||||
use Illuminate\Http\Request;
|
use Illuminate\Http\Request;
|
||||||
|
use Illuminate\Support\Facades\Log;
|
||||||
|
|
||||||
class MenusController extends Controller
|
class MenusController extends Controller
|
||||||
{
|
{
|
||||||
@@ -13,21 +16,27 @@ class MenusController extends Controller
|
|||||||
*/
|
*/
|
||||||
public function index(Request $request)
|
public function index(Request $request)
|
||||||
{
|
{
|
||||||
$query = Menu::query();
|
$query = Menu::query()->orderBy('id', 'desc');
|
||||||
|
|
||||||
if($request->has("search") && !empty($request->get("search"))){
|
if($request->has("search") && !empty($request->get("search"))){
|
||||||
$query = $query->where("name", "like", "%".$request->get("search")."%");
|
$query = $query->where("name", "like", "%".$request->get("search")."%");
|
||||||
}
|
}
|
||||||
|
|
||||||
return response()->json($query->paginate());
|
return response()->json($query->paginate(config('app.paginate_per_page', 50)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Store a newly created resource in storage.
|
* Store a newly created resource in storage.
|
||||||
*/
|
*/
|
||||||
public function store(Request $request)
|
public function store(MenuRequest $request)
|
||||||
{
|
{
|
||||||
//
|
try{
|
||||||
|
$menu = Menu::create($request->validated());
|
||||||
|
return response()->json(['message' => 'Menu created successfully', 'data' => new MenuResource($menu)]);
|
||||||
|
}catch(\Exception $e){
|
||||||
|
Log::error($e);
|
||||||
|
return response()->json(['message' => 'Error when creating menu'], 500);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -35,15 +44,37 @@ class MenusController extends Controller
|
|||||||
*/
|
*/
|
||||||
public function show(string $id)
|
public function show(string $id)
|
||||||
{
|
{
|
||||||
//
|
try{
|
||||||
|
$menu = Menu::find($id);
|
||||||
|
if($menu){
|
||||||
|
return response()->json(['message' => 'Menu found', 'data' => new MenuResource($menu)]);
|
||||||
|
} else {
|
||||||
|
return response()->json(['message' => 'Menu not found'], 404);
|
||||||
|
}
|
||||||
|
}catch(\Exception $e){
|
||||||
|
Log::error($e);
|
||||||
|
Log::error($e->getMessage());
|
||||||
|
return response()->json(['message' => 'Error when finding menu'], 500);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Update the specified resource in storage.
|
* Update the specified resource in storage.
|
||||||
*/
|
*/
|
||||||
public function update(Request $request, string $id)
|
public function update(MenuRequest $request, string $id)
|
||||||
{
|
{
|
||||||
//
|
try{
|
||||||
|
$menu = Menu::findOrFail($id);
|
||||||
|
if($menu){
|
||||||
|
$menu->update($request->validated());
|
||||||
|
return response()->json(['message' => 'Menu updated successfully', 'data' => new MenuResource($menu)]);
|
||||||
|
} else {
|
||||||
|
return response()->json(['message' => 'Menu not found'], 404);
|
||||||
|
}
|
||||||
|
}catch(\Exception $e){
|
||||||
|
Log::error($e);
|
||||||
|
return response()->json(['message' => 'Error when updating menu'], 500);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -51,6 +82,28 @@ class MenusController extends Controller
|
|||||||
*/
|
*/
|
||||||
public function destroy(string $id)
|
public function destroy(string $id)
|
||||||
{
|
{
|
||||||
//
|
try{
|
||||||
|
$menu = Menu::findOrFail($id);
|
||||||
|
if($menu){
|
||||||
|
$this->deleteChildren($menu);
|
||||||
|
$menu->roles()->detach();
|
||||||
|
$menu->delete();
|
||||||
|
return response()->json(['message' => 'Menu deleted successfully']);
|
||||||
|
} else {
|
||||||
|
return response()->json(['message' => 'Menu not found'], 404);
|
||||||
|
}
|
||||||
|
}catch(\Exception $e){
|
||||||
|
Log::error($e);
|
||||||
|
return response()->json(['message' => 'Error when deleting menu'], 500);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function deleteChildren($menu)
|
||||||
|
{
|
||||||
|
foreach ($menu->children as $child) {
|
||||||
|
$this->deleteChildren($child); // Recursively delete its children
|
||||||
|
$child->roles()->detach(); // Detach roles before deleting
|
||||||
|
$child->delete();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,9 +21,35 @@ class PbgTaskController extends Controller
|
|||||||
public function __construct(GoogleSheetService $googleSheetService){
|
public function __construct(GoogleSheetService $googleSheetService){
|
||||||
$this->googleSheetService = $googleSheetService;
|
$this->googleSheetService = $googleSheetService;
|
||||||
}
|
}
|
||||||
public function index()
|
public function index(Request $request)
|
||||||
{
|
{
|
||||||
//
|
info($request);
|
||||||
|
|
||||||
|
$isLastUpdated = filter_var($request->query('isLastUpdated', false), FILTER_VALIDATE_BOOLEAN);
|
||||||
|
|
||||||
|
$query = PbgTask::query();
|
||||||
|
|
||||||
|
if ($isLastUpdated) {
|
||||||
|
$query->orderBy('updated_at', 'desc');
|
||||||
|
} else {
|
||||||
|
$query->where('status', 20);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ambil maksimal 10 data
|
||||||
|
$pbg_task = $query->limit(10)->get();
|
||||||
|
$totalData = $pbg_task->count();
|
||||||
|
|
||||||
|
// Tambahkan nomor urut
|
||||||
|
$data = $pbg_task->map(function ($item, $index) {
|
||||||
|
return array_merge($item->toArray(), ['no' => $index + 1]);
|
||||||
|
});
|
||||||
|
|
||||||
|
return response()->json([
|
||||||
|
'data' => $data,
|
||||||
|
'meta' => [
|
||||||
|
'total' => $totalData
|
||||||
|
]
|
||||||
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -17,7 +17,8 @@ class RequestAssignmentController extends Controller
|
|||||||
$query = PbgTask::query()->orderBy('id', 'desc');
|
$query = PbgTask::query()->orderBy('id', 'desc');
|
||||||
if($request->has('search') && !empty($request->get("search"))){
|
if($request->has('search') && !empty($request->get("search"))){
|
||||||
$query->where('name', 'LIKE', '%'.$request->get('search').'%')
|
$query->where('name', 'LIKE', '%'.$request->get('search').'%')
|
||||||
->orWhere('registration_number', 'LIKE', '%'.$request->get('search').'%');
|
->orWhere('registration_number', 'LIKE', '%'.$request->get('search').'%')
|
||||||
|
->orWhere('document_number', 'LIKE', '%'.$request->get('search').'%');
|
||||||
}
|
}
|
||||||
return RequestAssignmentResouce::collection($query->paginate());
|
return RequestAssignmentResouce::collection($query->paginate());
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
namespace App\Http\Controllers\Api;
|
namespace App\Http\Controllers\Api;
|
||||||
|
|
||||||
use App\Http\Controllers\Controller;
|
use App\Http\Controllers\Controller;
|
||||||
|
use App\Http\Requests\RoleRequest;
|
||||||
use App\Models\Role;
|
use App\Models\Role;
|
||||||
use Illuminate\Http\Request;
|
use Illuminate\Http\Request;
|
||||||
|
|
||||||
@@ -13,21 +14,26 @@ class RolesController extends Controller
|
|||||||
*/
|
*/
|
||||||
public function index(Request $request)
|
public function index(Request $request)
|
||||||
{
|
{
|
||||||
$query = Role::query();
|
$query = Role::query()->orderBy('id', 'desc');
|
||||||
|
|
||||||
if($request->has('search') && !empty($request->get('search'))){
|
if($request->has('search') && !empty($request->get('search'))){
|
||||||
$query = $query->where('name', 'like', '%'. $request->get('search') . '%');
|
$query = $query->where('name', 'like', '%'. $request->get('search') . '%');
|
||||||
}
|
}
|
||||||
|
|
||||||
return response()->json($query->paginate());
|
return response()->json($query->paginate(config('app.paginate_per_page', 50)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Store a newly created resource in storage.
|
* Store a newly created resource in storage.
|
||||||
*/
|
*/
|
||||||
public function store(Request $request)
|
public function store(RoleRequest $request)
|
||||||
{
|
{
|
||||||
//
|
try{
|
||||||
|
$role = Role::create($request->validated());
|
||||||
|
return response()->json(['message' => 'Successfully created', 'data' => $role]);
|
||||||
|
}catch(\Exception $e){
|
||||||
|
return response()->json(['message' => 'Error when creating role', 'error' => $e->getMessage()], 500);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -35,15 +41,34 @@ class RolesController extends Controller
|
|||||||
*/
|
*/
|
||||||
public function show(string $id)
|
public function show(string $id)
|
||||||
{
|
{
|
||||||
//
|
try{
|
||||||
|
$role = Role::find($id);
|
||||||
|
if($role){
|
||||||
|
return response()->json(['data' => $role]);
|
||||||
|
} else {
|
||||||
|
return response()->json(['message' => 'Role not found'], 404);
|
||||||
|
}
|
||||||
|
}catch(\Exception $e){
|
||||||
|
return response()->json(['message' => 'Error when getting role', 'error' => $e->getMessage()], 500);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Update the specified resource in storage.
|
* Update the specified resource in storage.
|
||||||
*/
|
*/
|
||||||
public function update(Request $request, string $id)
|
public function update(RoleRequest $request, string $id)
|
||||||
{
|
{
|
||||||
//
|
try{
|
||||||
|
$role = Role::find($id);
|
||||||
|
if($role){
|
||||||
|
$role->update($request->validated());
|
||||||
|
return response()->json(['message' => 'Successfully updated', 'data' => $role]);
|
||||||
|
} else {
|
||||||
|
return response()->json(['message' => 'Role not found'], 404);
|
||||||
|
}
|
||||||
|
}catch(\Exception $e){
|
||||||
|
return response()->json(['message' => 'Error when updating role', 'error' => $e->getMessage()], 500);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -51,6 +76,16 @@ class RolesController extends Controller
|
|||||||
*/
|
*/
|
||||||
public function destroy(string $id)
|
public function destroy(string $id)
|
||||||
{
|
{
|
||||||
//
|
try{
|
||||||
|
$role = Role::find($id);
|
||||||
|
if($role){
|
||||||
|
$role->delete();
|
||||||
|
return response()->json(['message' => 'Successfully deleted']);
|
||||||
|
} else {
|
||||||
|
return response()->json(['message' => 'Role not found'], 404);
|
||||||
|
}
|
||||||
|
}catch(\Exception $e){
|
||||||
|
return response()->json(['message' => 'Error when deleting role', 'error' => $e->getMessage()], 500);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ namespace App\Http\Controllers\Api;
|
|||||||
|
|
||||||
use App\Enums\ImportDatasourceStatus;
|
use App\Enums\ImportDatasourceStatus;
|
||||||
use App\Http\Controllers\Controller;
|
use App\Http\Controllers\Controller;
|
||||||
|
use App\Jobs\SyncronizeSIMBG;
|
||||||
use App\Models\ImportDatasource;
|
use App\Models\ImportDatasource;
|
||||||
use App\Traits\GlobalApiResponse;
|
use App\Traits\GlobalApiResponse;
|
||||||
use Illuminate\Support\Facades\Artisan;
|
use Illuminate\Support\Facades\Artisan;
|
||||||
@@ -23,8 +24,8 @@ class ScrapingController extends Controller
|
|||||||
}
|
}
|
||||||
|
|
||||||
// run service artisan command
|
// run service artisan command
|
||||||
Artisan::call("app:execute-scraping");
|
SyncronizeSIMBG::dispatch();
|
||||||
return $this->resSuccess("Success execute scraping service please wait");
|
return $this->resSuccess(["message" => "Success execute scraping service on background, check status for more"]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -106,6 +106,18 @@ class TourismController extends Controller
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getAllLocation()
|
||||||
|
{
|
||||||
|
$locations = Tourism::whereNotNull('longitude')
|
||||||
|
->whereNotNull('latitude')
|
||||||
|
->select('project_name', 'longitude', 'latitude')
|
||||||
|
->get();
|
||||||
|
|
||||||
|
return response()->json([
|
||||||
|
'data' => $locations
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Display the specified resource.
|
* Display the specified resource.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ class UsersController extends Controller
|
|||||||
if($request->has('search') && !empty($request->get("search"))){
|
if($request->has('search') && !empty($request->get("search"))){
|
||||||
$query->where('name', 'LIKE', '%'.$request->get('search').'%');
|
$query->where('name', 'LIKE', '%'.$request->get('search').'%');
|
||||||
}
|
}
|
||||||
return UserResource::collection($query->paginate());
|
return UserResource::collection($query->paginate(config('app.paginate_per_page', 50)));
|
||||||
}
|
}
|
||||||
public function logout(Request $request){
|
public function logout(Request $request){
|
||||||
$request->user()->tokens()->delete();
|
$request->user()->tokens()->delete();
|
||||||
|
|||||||
48
app/Http/Controllers/BigdataResumesController.php
Normal file
48
app/Http/Controllers/BigdataResumesController.php
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Http\Controllers;
|
||||||
|
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
|
||||||
|
class BigdataResumesController extends Controller
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Display a listing of the resource.
|
||||||
|
*/
|
||||||
|
public function index()
|
||||||
|
{
|
||||||
|
return view('bigdata-resumes.index');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Show the form for creating a new resource.
|
||||||
|
*/
|
||||||
|
public function create()
|
||||||
|
{
|
||||||
|
//
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Display the specified resource.
|
||||||
|
*/
|
||||||
|
public function show(string $id)
|
||||||
|
{
|
||||||
|
//
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Show the form for editing the specified resource.
|
||||||
|
*/
|
||||||
|
public function edit(string $id)
|
||||||
|
{
|
||||||
|
//
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove the specified resource from storage.
|
||||||
|
*/
|
||||||
|
public function destroy(string $id)
|
||||||
|
{
|
||||||
|
//
|
||||||
|
}
|
||||||
|
}
|
||||||
17
app/Http/Controllers/Chatbot/ChatbotController.php
Normal file
17
app/Http/Controllers/Chatbot/ChatbotController.php
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Http\Controllers\Chatbot;
|
||||||
|
|
||||||
|
use App\Http\Controllers\Controller;
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
|
||||||
|
class ChatbotController extends Controller
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Displya a listing of the resource
|
||||||
|
*/
|
||||||
|
public function index()
|
||||||
|
{
|
||||||
|
return view('chatbot.index');
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Http\Controllers\ChatbotPimpinan;
|
||||||
|
|
||||||
|
use App\Http\Controllers\Controller;
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
|
||||||
|
class ChatbotPimpinanController extends Controller
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Display a listing of the resource
|
||||||
|
*/
|
||||||
|
public function index()
|
||||||
|
{
|
||||||
|
return view('chatbot-pimpinan.index');
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -15,7 +15,8 @@ class BigDataController extends Controller
|
|||||||
return view('dashboards.bigdata', compact('latest_created'));
|
return view('dashboards.bigdata', compact('latest_created'));
|
||||||
}
|
}
|
||||||
|
|
||||||
public function pbg(){
|
public function pbg()
|
||||||
return view('index');
|
{
|
||||||
|
return view('dashboards.pbg');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -62,7 +62,7 @@ class AdvertisementController extends Controller
|
|||||||
// Pastikan model ditemukan
|
// Pastikan model ditemukan
|
||||||
if (!$modelInstance) {
|
if (!$modelInstance) {
|
||||||
info("AdvertisementController@edit: Model tidak ditemukan.");
|
info("AdvertisementController@edit: Model tidak ditemukan.");
|
||||||
return redirect()->route('advertisements.index')->with('error', 'Advertisement not found');
|
return redirect()->route('web.advertisements.index')->with('error', 'Advertisement not found');
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mengambil dan memetakan village_name dan district_name
|
// Mengambil dan memetakan village_name dan district_name
|
||||||
|
|||||||
@@ -58,7 +58,7 @@ class TourismController extends Controller
|
|||||||
$modelInstance = Tourism::find($id);
|
$modelInstance = Tourism::find($id);
|
||||||
// Pastikan model ditemukan
|
// Pastikan model ditemukan
|
||||||
if (!$modelInstance) {
|
if (!$modelInstance) {
|
||||||
return redirect()->route('tourisms.index') ->with('error', 'Pariwisata tidak ditemukan');
|
return redirect()->route('web-tourisms.index') ->with('error', 'Pariwisata tidak ditemukan');
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mengambil dan memetakan village_name dan district_name
|
// Mengambil dan memetakan village_name dan district_name
|
||||||
|
|||||||
@@ -60,7 +60,7 @@ class UmkmController extends Controller
|
|||||||
$modelInstance = Umkm::find($id);
|
$modelInstance = Umkm::find($id);
|
||||||
// Pastikan model ditemukan
|
// Pastikan model ditemukan
|
||||||
if (!$modelInstance) {
|
if (!$modelInstance) {
|
||||||
return redirect()->route('umkm.index')->with('error', 'Umkm not found');
|
return redirect()->route('web-umkm.index')->with('error', 'Umkm not found');
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mengambil dan memetakan village_name dan district_name
|
// Mengambil dan memetakan village_name dan district_name
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ class PbgTaskController extends Controller
|
|||||||
*/
|
*/
|
||||||
public function show(string $id)
|
public function show(string $id)
|
||||||
{
|
{
|
||||||
$data = PbgTask::with(['pbg_task_retributions','pbg_task_index_integrations', 'pbg_task_retributions.pbg_task_prasarana'])->findOrFail($id);
|
$data = PbgTask::with(['pbg_task_retributions','pbg_task_index_integrations', 'pbg_task_retributions.pbg_task_prasarana', 'taskAssignments'])->findOrFail($id);
|
||||||
return view("pbg_task.show", compact("data"));
|
return view("pbg_task.show", compact("data"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -17,12 +17,12 @@ class SyncronizeController extends Controller
|
|||||||
}
|
}
|
||||||
|
|
||||||
public function syncPbgTask(){
|
public function syncPbgTask(){
|
||||||
$res = $this->service_simbg->syncTaskList();
|
$res = $this->service_simbg->syncTaskPBG();
|
||||||
return $res;
|
return $res;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function syncronizeTask(Request $request){
|
public function syncronizeTask(Request $request){
|
||||||
$res = $this->service_simbg->syncTaskList();
|
$res = $this->service_simbg->syncTaskPBG();
|
||||||
return redirect()->back()->with('success', 'Processing completed successfully');
|
return redirect()->back()->with('success', 'Processing completed successfully');
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -33,7 +33,7 @@ class SyncronizeController extends Controller
|
|||||||
|
|
||||||
public function syncIndexIntegration(Request $request, $uuid){
|
public function syncIndexIntegration(Request $request, $uuid){
|
||||||
$token = $request->get('token');
|
$token = $request->get('token');
|
||||||
$res = $this->service_simbg->syncIndexIntegration($uuid, $token);
|
$res = $this->service_simbg->syncIndexIntegration($uuid);
|
||||||
return $res;
|
return $res;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -42,4 +42,9 @@ class SyncronizeController extends Controller
|
|||||||
$res = $this->service_simbg->syncTaskDetailSubmit($uuid, $token);
|
$res = $this->service_simbg->syncTaskDetailSubmit($uuid, $token);
|
||||||
return $res;
|
return $res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function syncTaskAssignments($uuid){
|
||||||
|
$res = $this->service_simbg->syncTaskAssignments($uuid);
|
||||||
|
return $res;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ class DataSettingRequest extends FormRequest
|
|||||||
*/
|
*/
|
||||||
public function rules(): array
|
public function rules(): array
|
||||||
{
|
{
|
||||||
$id = $this->route('data_setting');
|
$id = $this->route('data_setting_id');
|
||||||
return [
|
return [
|
||||||
"key" => "required|unique:data_settings,key," . $id,
|
"key" => "required|unique:data_settings,key," . $id,
|
||||||
"value" => "required",
|
"value" => "required",
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
namespace App\Http\Requests;
|
namespace App\Http\Requests;
|
||||||
|
|
||||||
use Illuminate\Foundation\Http\FormRequest;
|
use Illuminate\Foundation\Http\FormRequest;
|
||||||
|
use Illuminate\Validation\Rule;
|
||||||
|
|
||||||
class MenuRequest extends FormRequest
|
class MenuRequest extends FormRequest
|
||||||
{
|
{
|
||||||
@@ -21,12 +22,14 @@ class MenuRequest extends FormRequest
|
|||||||
*/
|
*/
|
||||||
public function rules(): array
|
public function rules(): array
|
||||||
{
|
{
|
||||||
|
$menuId = $this->route('menu_id'); // Get the menu ID if updating
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'name' => ['required','string','max:255'],
|
'name' => ['required', 'string', 'max:255', Rule::unique('menus', 'name')->ignore($menuId)],
|
||||||
'url' => ['nullable','string','max:255'],
|
'url' => ['nullable', 'string', 'max:255'],
|
||||||
'icon' => ['nullable','string','max:255'],
|
'icon' => ['nullable', 'string', 'max:255'],
|
||||||
'parent_id' => ['nullable','exists:menus,id'],
|
'parent_id' => ['nullable', 'exists:menus,id'],
|
||||||
'sort_order' => ['required','integer'],
|
'sort_order' => ['required', 'integer'],
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ class RoleRequest extends FormRequest
|
|||||||
*/
|
*/
|
||||||
public function rules(): array
|
public function rules(): array
|
||||||
{
|
{
|
||||||
$roleId = $this->route('role');
|
$roleId = $this->route('role_id');
|
||||||
return [
|
return [
|
||||||
'name' => 'required|string|max:255|unique:roles,name,' . ($roleId ?? 'NULL') . ',id',
|
'name' => 'required|string|max:255|unique:roles,name,' . ($roleId ?? 'NULL') . ',id',
|
||||||
'description' => 'nullable|string',
|
'description' => 'nullable|string',
|
||||||
|
|||||||
51
app/Http/Resources/BigdataResumeResource.php
Normal file
51
app/Http/Resources/BigdataResumeResource.php
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Http\Resources;
|
||||||
|
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
use Illuminate\Http\Resources\Json\JsonResource;
|
||||||
|
|
||||||
|
class BigdataResumeResource extends JsonResource
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Transform the resource into an array.
|
||||||
|
*
|
||||||
|
* @return array<string, mixed>
|
||||||
|
*/
|
||||||
|
public function toArray(Request $request): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'id' => $this->id,
|
||||||
|
'import_datasource_id' => $this->import_datasource_id,
|
||||||
|
'potention_count' => (int) $this->potention_count,
|
||||||
|
'potention_sum' => number_format((float) $this->potention_sum, 2, ',', '.'),
|
||||||
|
|
||||||
|
'non_verified_count' => (int) $this->non_verified_count,
|
||||||
|
'non_verified_sum' => number_format((float) $this->non_verified_sum, 2, ',', '.'),
|
||||||
|
|
||||||
|
'verified_count' => (int) $this->verified_count,
|
||||||
|
'verified_sum' => number_format((float) $this->verified_sum, 2, ',', '.'),
|
||||||
|
|
||||||
|
'business_count' => (int) $this->business_count,
|
||||||
|
'business_sum' => number_format((float) $this->business_sum, 2, ',', '.'),
|
||||||
|
|
||||||
|
'non_business_count' => (int) $this->non_business_count,
|
||||||
|
'non_business_sum' => number_format((float) $this->non_business_sum, 2, ',', '.'),
|
||||||
|
|
||||||
|
'spatial_count' => (int) $this->spatial_count,
|
||||||
|
'spatial_sum' => number_format((float) $this->spatial_sum, 2, ',', '.'),
|
||||||
|
|
||||||
|
'issuance_realization_pbg_count' => (int) $this->issuance_realization_pbg_count,
|
||||||
|
'issuance_realization_pbg_sum' => number_format((float) $this->issuance_realization_pbg_sum, 2, ',', '.'),
|
||||||
|
|
||||||
|
'waiting_click_dpmptsp_count' => (int) $this->waiting_click_dpmptsp_count,
|
||||||
|
'waiting_click_dpmptsp_sum' => number_format((float) $this->waiting_click_dpmptsp_sum, 2, ',', '.'),
|
||||||
|
|
||||||
|
'process_in_technical_office_count' => (int) $this->process_in_technical_office_count,
|
||||||
|
'process_in_technical_office_sum' => number_format((float) $this->process_in_technical_office_sum, 2, ',', '.'),
|
||||||
|
|
||||||
|
'year' => $this->year,
|
||||||
|
'created_at' => $this->created_at->toDateTimeString(),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
19
app/Http/Resources/MenuResource.php
Normal file
19
app/Http/Resources/MenuResource.php
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Http\Resources;
|
||||||
|
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
use Illuminate\Http\Resources\Json\JsonResource;
|
||||||
|
|
||||||
|
class MenuResource extends JsonResource
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Transform the resource into an array.
|
||||||
|
*
|
||||||
|
* @return array<string, mixed>
|
||||||
|
*/
|
||||||
|
public function toArray(Request $request): array
|
||||||
|
{
|
||||||
|
return parent::toArray($request);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -3,37 +3,105 @@
|
|||||||
namespace App\Imports;
|
namespace App\Imports;
|
||||||
|
|
||||||
use App\Models\BusinessOrIndustry;
|
use App\Models\BusinessOrIndustry;
|
||||||
use Maatwebsite\Excel\Concerns\ToModel;
|
|
||||||
use Maatwebsite\Excel\Concerns\ToCollection;
|
|
||||||
use Illuminate\Support\Collection;
|
use Illuminate\Support\Collection;
|
||||||
|
use Maatwebsite\Excel\Concerns\ToCollection;
|
||||||
|
use Maatwebsite\Excel\Concerns\WithChunkReading;
|
||||||
|
use Maatwebsite\Excel\Concerns\WithHeadingRow;
|
||||||
|
use Maatwebsite\Excel\Concerns\WithMultipleSheets;
|
||||||
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||||
|
use Maatwebsite\Excel\Concerns\WithBatchInserts;
|
||||||
|
use Illuminate\Support\Facades\Log;
|
||||||
|
|
||||||
class BusinessIndustriesImport implements ToCollection
|
class BusinessIndustriesImport implements ToCollection, WithMultipleSheets, WithChunkReading, WithBatchInserts, ShouldQueue, WithHeadingRow
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @param array $row
|
* @param Collection $collection
|
||||||
*
|
|
||||||
* @return \Illuminate\Database\Eloquent\Model|null
|
|
||||||
*/
|
*/
|
||||||
public function collection(Collection $rows)
|
public function collection(Collection $collection)
|
||||||
{
|
{
|
||||||
foreach ($rows->skip(1) as $row){
|
try{
|
||||||
$clean_nop = preg_replace('/[^A-Za-z0-9]/', '', $row[2]);
|
$batchData = [];
|
||||||
if (!BusinessOrIndustry::where('nop', $clean_nop)->exists()) {
|
$batchSize = 1000;
|
||||||
BusinessOrIndustry::create([
|
|
||||||
'nama_kecamatan' => $row[0],
|
foreach ($collection as $row){
|
||||||
'nama_kelurahan' => $row[1],
|
if(!isset($row['nop']) || empty($row['nop'])){
|
||||||
'nop' => $clean_nop, // Store cleaned 'nop'
|
continue;
|
||||||
'nama_wajib_pajak' => $row[3],
|
}
|
||||||
'alamat_wajib_pajak' => $row[4],
|
|
||||||
'alamat_objek_pajak' => $row[5],
|
|
||||||
'luas_bumi' => $row[6],
|
$clean_nop = preg_replace('/[^A-Za-z0-9]/', '', $row['nop']);
|
||||||
'luas_bangunan' => $row[7],
|
|
||||||
'njop_bumi' => $row[8],
|
$batchData[] = [
|
||||||
'njop_bangunan' => $row[9],
|
'nama_kecamatan' => $row['nama_kecamatan'],
|
||||||
'ketetapan' => $row[10],
|
'nama_kelurahan' => $row['nama_kelurahan'],
|
||||||
'tahun_pajak' => $row[11],
|
'nop' => $clean_nop,
|
||||||
|
'nama_wajib_pajak' => $row['nama_wajib_pajak'],
|
||||||
|
'alamat_wajib_pajak' => $row['alamat_wajib_pajak'],
|
||||||
|
'alamat_objek_pajak' => $row['alamat_objek_pajak'],
|
||||||
|
'luas_bumi' => $row['luas_bumi'],
|
||||||
|
'luas_bangunan' => $row['luas_bangunan'],
|
||||||
|
'njop_bumi' => $row['njop_bumi'],
|
||||||
|
'njop_bangunan' => $row['njop_bangunan'],
|
||||||
|
'ketetapan' => $row['ketetapan'],
|
||||||
|
'tahun_pajak' => $row['tahun_pajak'],
|
||||||
|
];
|
||||||
|
|
||||||
|
if(count($batchData) >= $batchSize){
|
||||||
|
BusinessOrIndustry::upsert($batchData, ['nop'], [
|
||||||
|
'nama_kecamatan',
|
||||||
|
'nama_kelurahan',
|
||||||
|
'nama_wajib_pajak',
|
||||||
|
'alamat_wajib_pajak',
|
||||||
|
'alamat_objek_pajak',
|
||||||
|
'luas_bumi',
|
||||||
|
'luas_bangunan',
|
||||||
|
'njop_bumi',
|
||||||
|
'njop_bangunan',
|
||||||
|
'ketetapan',
|
||||||
|
'tahun_pajak',
|
||||||
|
]);
|
||||||
|
$batchData = [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(!empty($batchData)){
|
||||||
|
BusinessOrIndustry::upsert($batchData, ['nop'], [
|
||||||
|
'nama_kecamatan',
|
||||||
|
'nama_kelurahan',
|
||||||
|
'nama_wajib_pajak',
|
||||||
|
'alamat_wajib_pajak',
|
||||||
|
'alamat_objek_pajak',
|
||||||
|
'luas_bumi',
|
||||||
|
'luas_bangunan',
|
||||||
|
'njop_bumi',
|
||||||
|
'njop_bangunan',
|
||||||
|
'ketetapan',
|
||||||
|
'tahun_pajak',
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
}catch(\Exception $exception){
|
||||||
|
Log::error('Error while importing Business Industries data:', ['error' => $exception->getMessage()]);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function sheets(): array {
|
||||||
|
return [
|
||||||
|
0 => $this
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function headingRow(): int
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function chunkSize(): int
|
||||||
|
{
|
||||||
|
return 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function batchSize(): int
|
||||||
|
{
|
||||||
|
return 1000;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,8 +11,9 @@ use Maatwebsite\Excel\Concerns\WithHeadingRow;
|
|||||||
use Maatwebsite\Excel\Concerns\WithMultipleSheets;
|
use Maatwebsite\Excel\Concerns\WithMultipleSheets;
|
||||||
use Illuminate\Contracts\Queue\ShouldQueue;
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||||
use Maatwebsite\Excel\Concerns\WithBatchInserts;
|
use Maatwebsite\Excel\Concerns\WithBatchInserts;
|
||||||
|
use Illuminate\Support\Facades\Log;
|
||||||
|
|
||||||
class CustomersImport implements ToCollection, WithMultipleSheets
|
class CustomersImport implements ToCollection, WithMultipleSheets, WithChunkReading, WithBatchInserts, ShouldQueue, WithHeadingRow
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @param Collection $collection
|
* @param Collection $collection
|
||||||
@@ -20,23 +21,51 @@ class CustomersImport implements ToCollection, WithMultipleSheets
|
|||||||
public function collection(Collection $collection)
|
public function collection(Collection $collection)
|
||||||
{
|
{
|
||||||
$batchData = [];
|
$batchData = [];
|
||||||
|
$batchSize = 1000;
|
||||||
|
|
||||||
foreach ($collection->skip(1) as $row) {
|
foreach ($collection as $row) {
|
||||||
if (!isset($row[0]) || empty($row[0])) {
|
if (!isset($row['nomor_pelanggan']) || empty($row['nomor_pelanggan'])) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
$latitude = filter_var($row[4], FILTER_VALIDATE_FLOAT) ? bcadd($row[4], '0', 18) : null;
|
$latitude = '0';
|
||||||
$longitude = filter_var($row[5], FILTER_VALIDATE_FLOAT) ? bcadd($row[5], '0', 18) : null;
|
$longitude = '0';
|
||||||
|
|
||||||
|
if (isset($row['latkor']) && !empty(trim($row['latkor']))) {
|
||||||
|
$latitude = str_replace(',', '.', trim($row['latkor']));
|
||||||
|
if (is_numeric($latitude)) {
|
||||||
|
$latitude = bcadd($latitude, '0', 18);
|
||||||
|
} else {
|
||||||
|
$latitude = '0';
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$latitude = '0';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isset($row['lonkor']) && !empty(trim($row['lonkor']))) {
|
||||||
|
$longitude = str_replace(',', '.', trim($row['lonkor']));
|
||||||
|
if (is_numeric($longitude)) {
|
||||||
|
$longitude = bcadd($longitude, '0', 18);
|
||||||
|
} else {
|
||||||
|
$longitude = '0';
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$longitude = '0';
|
||||||
|
}
|
||||||
|
|
||||||
$batchData[] = [
|
$batchData[] = [
|
||||||
'nomor_pelanggan' => $row[0],
|
'nomor_pelanggan' => $row['nomor_pelanggan'] ?? '',
|
||||||
'kota_pelayanan' => $row[1],
|
'kota_pelayanan' => $row['kota_pelayanan'] ?? '',
|
||||||
'nama' => $row[2],
|
'nama' => $row['nama'] ?? '',
|
||||||
'alamat' => $row[3],
|
'alamat' => $row['alamat'] ?? '',
|
||||||
'latitude' => $latitude,
|
'latitude' => $latitude,
|
||||||
'longitude' => $longitude,
|
'longitude' => $longitude,
|
||||||
];
|
];
|
||||||
|
|
||||||
|
if (count($batchData) >= $batchSize) {
|
||||||
|
Customer::upsert($batchData, ['nomor_pelanggan'], ['kota_pelayanan', 'nama', 'alamat', 'latitude', 'longitude']);
|
||||||
|
$batchData = [];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!empty($batchData)) {
|
if (!empty($batchData)) {
|
||||||
@@ -44,9 +73,20 @@ class CustomersImport implements ToCollection, WithMultipleSheets
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public function sheets(): array {
|
public function sheets(): array {
|
||||||
return [
|
return [
|
||||||
0 => $this
|
0 => $this
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function chunkSize(): int
|
||||||
|
{
|
||||||
|
return 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function batchSize(): int
|
||||||
|
{
|
||||||
|
return 1000;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
28
app/Jobs/SyncronizeSIMBG.php
Normal file
28
app/Jobs/SyncronizeSIMBG.php
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Jobs;
|
||||||
|
|
||||||
|
use App\Services\GoogleSheetService;
|
||||||
|
use App\Services\ServiceSIMBG;
|
||||||
|
use Illuminate\Bus\Queueable;
|
||||||
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||||
|
use Illuminate\Foundation\Bus\Dispatchable;
|
||||||
|
use Illuminate\Queue\InteractsWithQueue;
|
||||||
|
use Illuminate\Queue\SerializesModels;
|
||||||
|
|
||||||
|
class SyncronizeSIMBG implements ShouldQueue
|
||||||
|
{
|
||||||
|
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
||||||
|
|
||||||
|
public $tries = 1;
|
||||||
|
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public function handle(): void
|
||||||
|
{
|
||||||
|
$serviceSIMBG = app(ServiceSIMBG::class);
|
||||||
|
$serviceSIMBG->syncTaskPBG();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -21,6 +21,15 @@ class BigdataResume extends Model
|
|||||||
'business_sum',
|
'business_sum',
|
||||||
'non_business_count',
|
'non_business_count',
|
||||||
'non_business_sum',
|
'non_business_sum',
|
||||||
|
'spatial_count',
|
||||||
|
'spatial_sum',
|
||||||
|
'year',
|
||||||
|
'waiting_click_dpmptsp_count',
|
||||||
|
'waiting_click_dpmptsp_sum',
|
||||||
|
'issuance_realization_pbg_count',
|
||||||
|
'issuance_realization_pbg_sum',
|
||||||
|
'process_in_technical_office_count',
|
||||||
|
'process_in_technical_office_sum',
|
||||||
];
|
];
|
||||||
|
|
||||||
public function importDatasource()
|
public function importDatasource()
|
||||||
@@ -28,88 +37,84 @@ class BigdataResume extends Model
|
|||||||
return $this->belongsTo(ImportDatasource::class, 'import_datasource_id');
|
return $this->belongsTo(ImportDatasource::class, 'import_datasource_id');
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function generateResumeData($import_datasource_id){
|
public static function generateResumeData($import_datasource_id, $year, $data_setting){
|
||||||
$query_verified = once( function () {
|
$stats = PbgTask::with(['googleSheet', 'pbg_task_retributions'])
|
||||||
return DB::table('pbg_task AS pt')
|
->leftJoin('pbg_task_retributions as ptr', 'pbg_task.uuid', '=', 'ptr.pbg_task_uid')
|
||||||
->leftJoin('pbg_task_google_sheet AS ptgs', 'pt.registration_number', '=', 'ptgs.no_registrasi')
|
->leftJoin('pbg_task_google_sheet as ptgs', 'pbg_task.registration_number', '=', 'ptgs.no_registrasi')
|
||||||
->leftJoin('pbg_task_retributions AS ptr', 'pt.uuid', '=', 'ptr.pbg_task_uid')
|
->when($year !== 'all', function ($query) use ($year) {
|
||||||
->whereRaw('LOWER(TRIM(ptgs.status_verifikasi)) = ?', [strtolower(trim('Selesai Verifikasi'))])
|
$query->whereYear('pbg_task.task_created_at', (int) $year);
|
||||||
->selectRaw('COUNT(pt.id) AS total_data,
|
})
|
||||||
SUM(ptr.nilai_retribusi_bangunan) AS total_retribution')
|
->selectRaw("
|
||||||
|
COUNT(CASE WHEN LOWER(TRIM(ptgs.status_verifikasi)) = 'selesai verifikasi' THEN 1 END) AS verified_count,
|
||||||
|
SUM(CASE WHEN LOWER(TRIM(ptgs.status_verifikasi)) = 'selesai verifikasi' THEN ptr.nilai_retribusi_bangunan ELSE 0 END) AS verified_total,
|
||||||
|
|
||||||
|
COUNT(CASE WHEN LOWER(TRIM(ptgs.status_verifikasi)) != 'selesai verifikasi' OR ptgs.status_verifikasi IS NULL THEN 1 END) AS non_verified_count,
|
||||||
|
SUM(CASE WHEN LOWER(TRIM(ptgs.status_verifikasi)) != 'selesai verifikasi' OR ptgs.status_verifikasi IS NULL THEN ptr.nilai_retribusi_bangunan ELSE 0 END) AS non_verified_total,
|
||||||
|
|
||||||
|
COUNT(CASE WHEN (LOWER(TRIM(ptgs.status_verifikasi)) != 'selesai verifikasi' OR ptgs.status_verifikasi IS NULL)
|
||||||
|
AND LOWER(TRIM(pbg_task.function_type)) = 'sebagai tempat usaha' THEN 1 END) AS business_count,
|
||||||
|
SUM(CASE WHEN (LOWER(TRIM(ptgs.status_verifikasi)) != 'selesai verifikasi' OR ptgs.status_verifikasi IS NULL)
|
||||||
|
AND LOWER(TRIM(pbg_task.function_type)) = 'sebagai tempat usaha' THEN ptr.nilai_retribusi_bangunan ELSE 0 END) AS business_total,
|
||||||
|
|
||||||
|
COUNT(CASE WHEN (LOWER(TRIM(ptgs.status_verifikasi)) != 'selesai verifikasi' OR ptgs.status_verifikasi IS NULL)
|
||||||
|
AND (LOWER(TRIM(pbg_task.function_type)) != 'sebagai tempat usaha' OR pbg_task.function_type IS NULL) THEN 1 END) AS non_business_count,
|
||||||
|
SUM(CASE WHEN (LOWER(TRIM(ptgs.status_verifikasi)) != 'selesai verifikasi' OR ptgs.status_verifikasi IS NULL)
|
||||||
|
AND (LOWER(TRIM(pbg_task.function_type)) != 'sebagai tempat usaha' OR pbg_task.function_type IS NULL) THEN ptr.nilai_retribusi_bangunan ELSE 0 END) AS non_business_total
|
||||||
|
")
|
||||||
->first();
|
->first();
|
||||||
|
|
||||||
|
// Assign Results
|
||||||
|
$verified_count = $stats->verified_count ?? 0;
|
||||||
|
$verified_total = $stats->verified_total ?? 0;
|
||||||
|
$non_verified_count = $stats->non_verified_count ?? 0;
|
||||||
|
$non_verified_total = $stats->non_verified_total ?? 0;
|
||||||
|
$business_count = $stats->business_count ?? 0;
|
||||||
|
$business_total = $stats->business_total ?? 0;
|
||||||
|
$non_business_count = $stats->non_business_count ?? 0;
|
||||||
|
$non_business_total = $stats->non_business_total ?? 0;
|
||||||
|
|
||||||
|
$query_potention = once(function () use ($year) {
|
||||||
|
$query = PbgTask::leftJoin('pbg_task_retributions as ptr', 'pbg_task.uuid', '=', 'ptr.pbg_task_uid')
|
||||||
|
->selectRaw('COUNT(DISTINCT pbg_task.id) as task_count, SUM(ptr.nilai_retribusi_bangunan) as total_retribution');
|
||||||
|
|
||||||
|
if ($year !== 'all') {
|
||||||
|
$query->whereYear('pbg_task.task_created_at', (int) $year);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $query->first();
|
||||||
});
|
});
|
||||||
|
|
||||||
$verified_count = $query_verified->total_data ?? 0;
|
|
||||||
$verified_total = $query_verified->total_retribution ?? 0;
|
|
||||||
|
|
||||||
$query_business = once(function () {
|
|
||||||
return DB::table('pbg_task AS pt')
|
|
||||||
->leftJoin('pbg_task_google_sheet AS ptgs', 'pt.registration_number', '=', 'ptgs.no_registrasi')
|
|
||||||
->leftJoin('pbg_task_retributions AS ptr', 'pt.uuid', '=', 'ptr.pbg_task_uid')
|
|
||||||
->where(function ($query) {
|
|
||||||
$query->whereRaw('LOWER(TRIM(ptgs.status_verifikasi)) != ?', [strtolower(trim('Selesai Verifikasi'))])
|
|
||||||
->orWhereNull('ptgs.status_verifikasi');
|
|
||||||
})
|
|
||||||
->where(function ($query) {
|
|
||||||
$query->whereRaw('LOWER(TRIM(pt.function_type)) = ?', [strtolower(trim('Sebagai Tempat Usaha'))]);
|
|
||||||
})
|
|
||||||
->selectRaw('COUNT(pt.id) AS total_data,
|
|
||||||
SUM(ptr.nilai_retribusi_bangunan) AS total_retribution')
|
|
||||||
->first();
|
|
||||||
});
|
|
||||||
|
|
||||||
$business_count = $query_business->total_data ?? 0;
|
|
||||||
$business_total = $query_business->total_retribution ?? 0;
|
|
||||||
|
|
||||||
$query_non_business = once( function () {
|
|
||||||
return DB::table('pbg_task AS pt')
|
|
||||||
->leftJoin('pbg_task_google_sheet AS ptgs', 'pt.registration_number', '=', 'ptgs.no_registrasi')
|
|
||||||
->leftJoin('pbg_task_retributions AS ptr', 'pt.uuid', '=', 'ptr.pbg_task_uid') // Join ke pbg_task_retributions
|
|
||||||
->where(function ($query) {
|
|
||||||
$query->whereRaw('LOWER(TRIM(ptgs.status_verifikasi)) != ?', [strtolower(trim('Selesai Verifikasi'))])
|
|
||||||
->orWhereNull('ptgs.status_verifikasi'); // Include NULL values
|
|
||||||
})
|
|
||||||
->where(function ($query) {
|
|
||||||
$query->whereRaw('LOWER(TRIM(pt.function_type)) != ?', [strtolower(trim('Sebagai Tempat Usaha'))])
|
|
||||||
->orWhereNull('pt.function_type'); // Include NULL values
|
|
||||||
})
|
|
||||||
->selectRaw('COUNT(pt.id) AS total_data,
|
|
||||||
SUM(ptr.nilai_retribusi_bangunan) AS total_retribution') // Menambahkan SUM dari pbg_task_retributions
|
|
||||||
->first();
|
|
||||||
});
|
|
||||||
$non_business_count = $query_non_business->total_data ?? 0;
|
|
||||||
$non_business_total = $query_non_business->total_retribution ?? 0;
|
|
||||||
|
|
||||||
$query_non_verified = once(function () {
|
|
||||||
return DB::table('pbg_task AS pt')
|
|
||||||
->leftJoin('pbg_task_google_sheet AS ptgs', 'pt.registration_number', '=', 'ptgs.no_registrasi')
|
|
||||||
->leftJoin('pbg_task_retributions AS ptr', 'pt.uuid', '=', 'ptr.pbg_task_uid') // Join tabel pbg_task_retributions
|
|
||||||
->where(function ($query) {
|
|
||||||
$query->whereRaw('LOWER(TRIM(ptgs.status_verifikasi)) != ?', [strtolower(trim('Selesai Verifikasi'))])
|
|
||||||
->orWhereNull('ptgs.status_verifikasi'); // Include NULL values
|
|
||||||
})
|
|
||||||
->selectRaw('COUNT(pt.id) AS total_data,
|
|
||||||
SUM(ptr.nilai_retribusi_bangunan) AS total_retribution') // Menambahkan SUM dari pbg_task_retributions
|
|
||||||
->first();
|
|
||||||
});
|
|
||||||
|
|
||||||
$non_verified_count = $query_non_verified->total_data ?? 0;
|
|
||||||
$non_verified_total = $query_non_verified->total_retribution ?? 0;
|
|
||||||
|
|
||||||
$query_potention = once( function () {
|
|
||||||
return DB::table('pbg_task as pt')
|
|
||||||
->leftJoin('pbg_task_retributions as ptr', 'pt.uuid', '=', 'ptr.pbg_task_uid')
|
|
||||||
->select(
|
|
||||||
DB::raw('COUNT(DISTINCT pt.id) as task_count'),
|
|
||||||
DB::raw('SUM(ptr.nilai_retribusi_bangunan) as total_retribution')
|
|
||||||
)
|
|
||||||
->first();
|
|
||||||
});
|
|
||||||
$potention_count = $query_potention->task_count ?? 0;
|
$potention_count = $query_potention->task_count ?? 0;
|
||||||
$potention_total = $query_potention->total_retribution ?? 0;
|
$potention_total = $query_potention->total_retribution ?? 0;
|
||||||
|
|
||||||
|
$query_spatial_plannings = once(function () use ($year) {
|
||||||
|
$query = PbgTask::leftJoin('spatial_plannings as sp', 'pbg_task.document_number', '=', 'sp.number')
|
||||||
|
->leftJoin('pbg_task_retributions as ptr', 'ptr.pbg_task_uid', '=', 'pbg_task.uuid')
|
||||||
|
->selectRaw('
|
||||||
|
CASE
|
||||||
|
WHEN COUNT(DISTINCT sp.id) > 0 THEN COUNT(DISTINCT sp.id)
|
||||||
|
ELSE (SELECT COUNT(*) FROM spatial_plannings)
|
||||||
|
END as task_count,
|
||||||
|
SUM(CASE WHEN sp.id IS NOT NULL AND ptr.id IS NOT NULL THEN ptr.nilai_retribusi_bangunan ELSE 0 END) as total_retribution
|
||||||
|
');
|
||||||
|
|
||||||
|
if ($year !== 'all') {
|
||||||
|
$query->whereYear('pbg_task.task_created_at', (int) $year);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $query->first();
|
||||||
|
});
|
||||||
|
|
||||||
|
$spatial_planning_count = $query_spatial_plannings->task_count ?? 0;
|
||||||
|
$spatial_planning_total = $query_spatial_plannings->total_retribution;
|
||||||
|
|
||||||
|
$potention_count -= $spatial_planning_count;
|
||||||
|
$potention_total -= $spatial_planning_total;
|
||||||
|
|
||||||
return self::create([
|
return self::create([
|
||||||
'import_datasource_id' => $import_datasource_id,
|
'import_datasource_id' => $import_datasource_id,
|
||||||
|
'spatial_count' => $spatial_planning_count,
|
||||||
|
'spatial_sum' => $spatial_planning_total ?? 0.00,
|
||||||
'potention_count' => $potention_count ?? 0,
|
'potention_count' => $potention_count ?? 0,
|
||||||
'potention_sum' => $potention_total ?? 0.00,
|
'potention_sum' => $potention_total ?? 0.00,
|
||||||
'non_verified_count' => $non_verified_count ?? 0,
|
'non_verified_count' => $non_verified_count ?? 0,
|
||||||
@@ -120,6 +125,13 @@ class BigdataResume extends Model
|
|||||||
'business_sum' => $business_total ?? 0.00,
|
'business_sum' => $business_total ?? 0.00,
|
||||||
'non_business_count' => $non_business_count ?? 0,
|
'non_business_count' => $non_business_count ?? 0,
|
||||||
'non_business_sum' => $non_business_total ?? 0.00,
|
'non_business_sum' => $non_business_total ?? 0.00,
|
||||||
|
'year' => $year,
|
||||||
|
'waiting_click_dpmptsp_count' => $data_setting['MENUNGGU_KLIK_DPMPTSP_COUNT'] ?? 0,
|
||||||
|
'waiting_click_dpmptsp_sum' => $data_setting['MENUNGGU_KLIK_DPMPTSP_SUM'] ?? 0.00,
|
||||||
|
'issuance_realization_pbg_count' => $data_setting['REALISASI_TERBIT_PBG_COUNT'] ?? 0,
|
||||||
|
'issuance_realization_pbg_sum' => $data_setting['REALISASI_TERBIT_PBG_SUM'] ?? 0.00,
|
||||||
|
'process_in_technical_office_count' => $data_setting['PROSES_DINAS_TEKNIS_COUNT'] ?? 0,
|
||||||
|
'process_in_technical_office_sum' => $data_setting['PROSES_DINAS_TEKNIS_SUM'] ??0.00,
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -37,4 +37,13 @@ class PbgTask extends Model
|
|||||||
public function pbg_task_index_integrations(){
|
public function pbg_task_index_integrations(){
|
||||||
return $this->hasOne(PbgTaskIndexIntegrations::class, 'pbg_task_uid', 'uuid');
|
return $this->hasOne(PbgTaskIndexIntegrations::class, 'pbg_task_uid', 'uuid');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function googleSheet(){
|
||||||
|
return $this->hasOne(PbgTaskGoogleSheet::class, 'no_registrasi', 'registration_number');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function taskAssignments()
|
||||||
|
{
|
||||||
|
return $this->hasMany(TaskAssignment::class, 'pbg_task_uid', 'uuid');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
28
app/Models/TaskAssignment.php
Normal file
28
app/Models/TaskAssignment.php
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Models;
|
||||||
|
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
||||||
|
|
||||||
|
class TaskAssignment extends Model
|
||||||
|
{
|
||||||
|
use HasFactory;
|
||||||
|
|
||||||
|
protected $fillable = [
|
||||||
|
'user_id', 'name', 'username', 'email', 'phone_number', 'role',
|
||||||
|
'role_name', 'is_active', 'file', 'expertise', 'experience',
|
||||||
|
'is_verif', 'uid', 'status', 'status_name', 'note', 'pbg_task_uid'
|
||||||
|
];
|
||||||
|
|
||||||
|
protected $casts = [
|
||||||
|
'is_active' => 'boolean',
|
||||||
|
'is_verif' => 'boolean',
|
||||||
|
'file' => 'array', // JSON field casting
|
||||||
|
];
|
||||||
|
|
||||||
|
public function pbgTask()
|
||||||
|
{
|
||||||
|
return $this->belongsTo(PbgTask::class, 'pbg_task_uid', 'uuid');
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -9,6 +9,8 @@ use Illuminate\Support\Facades\Blade;
|
|||||||
use Illuminate\Support\Facades\View;
|
use Illuminate\Support\Facades\View;
|
||||||
use Illuminate\Support\ServiceProvider;
|
use Illuminate\Support\ServiceProvider;
|
||||||
use Carbon\Carbon;
|
use Carbon\Carbon;
|
||||||
|
use App\Services\ServiceSIMBG;
|
||||||
|
use App\Services\GoogleSheetService;
|
||||||
|
|
||||||
class AppServiceProvider extends ServiceProvider
|
class AppServiceProvider extends ServiceProvider
|
||||||
{
|
{
|
||||||
@@ -17,7 +19,12 @@ class AppServiceProvider extends ServiceProvider
|
|||||||
*/
|
*/
|
||||||
public function register(): void
|
public function register(): void
|
||||||
{
|
{
|
||||||
//
|
$this->app->singleton(GoogleSheetService::class, function () {
|
||||||
|
return new GoogleSheetService();
|
||||||
|
});
|
||||||
|
$this->app->singleton(ServiceSIMBG::class, function ($app) {
|
||||||
|
return new ServiceSIMBG($app->make(GoogleSheetService::class));
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ class GoogleSheetService
|
|||||||
protected $client;
|
protected $client;
|
||||||
protected $service;
|
protected $service;
|
||||||
protected $spreadsheetID;
|
protected $spreadsheetID;
|
||||||
|
protected $service_sheets;
|
||||||
public function __construct()
|
public function __construct()
|
||||||
{
|
{
|
||||||
$this->client = new Google_Client();
|
$this->client = new Google_Client();
|
||||||
|
|||||||
276
app/Services/OpenAIService.php
Normal file
276
app/Services/OpenAIService.php
Normal file
@@ -0,0 +1,276 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Services;
|
||||||
|
|
||||||
|
use OpenAI;
|
||||||
|
use Illuminate\Support\Facades\Storage;
|
||||||
|
use Illuminate\Support\Facades\Log;
|
||||||
|
|
||||||
|
class OpenAIService
|
||||||
|
{
|
||||||
|
protected $client;
|
||||||
|
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
// $this->client = OpenAI::client(env('OPENAI_API_KEY'));
|
||||||
|
$this->client = OpenAI::client(env('OPENAI_API_KEY'));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function generateQueryBasedMainContent($prompt, $mainContent, $chatHistory)
|
||||||
|
{
|
||||||
|
// Load file JSON
|
||||||
|
$jsonPath = public_path('templates/contentTemplatePrompt.json'); // Sesuaikan path
|
||||||
|
$jsonData = json_decode(file_get_contents($jsonPath), true);
|
||||||
|
|
||||||
|
// Periksa apakah kategori ada dalam JSON
|
||||||
|
if (!isset($jsonData[$mainContent])) {
|
||||||
|
return "Template prompt tidak ditemukan.";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ambil template berdasarkan kategori
|
||||||
|
$promptTemplate = $jsonData[$mainContent]['prompt'];
|
||||||
|
|
||||||
|
// Menyusun pesan untuk OpenAI
|
||||||
|
$messages = [
|
||||||
|
['role' => 'system', 'content' => $promptTemplate],
|
||||||
|
];
|
||||||
|
|
||||||
|
// Menambahkan chat history sebagai konteks
|
||||||
|
foreach ($chatHistory as $chat) {
|
||||||
|
if (isset($chat['user'])) {
|
||||||
|
$messages[] = ['role' => 'user', 'content' => $chat['user']];
|
||||||
|
}
|
||||||
|
if (isset($chat['rawBotResponse'])) {
|
||||||
|
$messages[] = ['role' => 'assistant', 'content' => $chat['rawBotResponse']];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tambahkan prompt terbaru user
|
||||||
|
$messages[] = ['role' => 'user', 'content' => $prompt];
|
||||||
|
|
||||||
|
// Kirim request ke OpenAI API
|
||||||
|
$response = $this->client->chat()->create([
|
||||||
|
'model' => 'gpt-4o-mini',
|
||||||
|
'messages' => $messages,
|
||||||
|
]);
|
||||||
|
|
||||||
|
return trim($response['choices'][0]['message']['content'] ?? 'No response');
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// public function generateQueryBasedMainContent($prompt, $mainContent, $chatHistory)
|
||||||
|
// {
|
||||||
|
// // Load file JSON
|
||||||
|
// $jsonPath = public_path('templates/contentTemplatePrompt.json'); // Sesuaikan path
|
||||||
|
// $jsonData = json_decode(file_get_contents($jsonPath), true);
|
||||||
|
|
||||||
|
// // Periksa apakah kategori ada dalam JSON
|
||||||
|
// if (!isset($jsonData[$mainContent])) {
|
||||||
|
// return "Template prompt tidak ditemukan.";
|
||||||
|
// }
|
||||||
|
|
||||||
|
// // Ambil template berdasarkan kategori
|
||||||
|
// $promptTemplate = $jsonData[$mainContent]['prompt'];
|
||||||
|
|
||||||
|
// $response = $this->client->chat()->create([
|
||||||
|
// 'model' => 'gpt-4o-mini',
|
||||||
|
// 'messages' => [
|
||||||
|
// ['role' => 'system', 'content' => $promptTemplate],
|
||||||
|
// ['role' => 'user', 'content' => $prompt],
|
||||||
|
// ],
|
||||||
|
// ]);
|
||||||
|
|
||||||
|
// return trim($response['choices'][0]['message']['content'] ?? 'No response');
|
||||||
|
// }
|
||||||
|
|
||||||
|
|
||||||
|
public function validateSyntaxQuery($queryResponse)
|
||||||
|
{
|
||||||
|
$response = $this->client->chat()->create([
|
||||||
|
'model' => 'gpt-4o-mini',
|
||||||
|
'messages' => [
|
||||||
|
[
|
||||||
|
'role' => 'system',
|
||||||
|
'content' => "You are a MariaDB SQL expert. Your task is to validate the syntax of an SQL query to ensure it follows proper MariaDB syntax rules.
|
||||||
|
|
||||||
|
Guidelines:
|
||||||
|
- Check for any syntax errors, missing keywords, or incorrect clause usage.
|
||||||
|
- Ensure the query is well-structured and adheres to best practices.
|
||||||
|
- Verify that all SQL keywords are used correctly and in the right order.
|
||||||
|
- If the query is valid, respond with: \"VALID\".
|
||||||
|
- If the query has issues, respond with: \"INVALID\".
|
||||||
|
|
||||||
|
Always respond with either \"VALID\" or \"INVALID\"."
|
||||||
|
],
|
||||||
|
['role' => 'user', 'content' => $queryResponse],
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
|
||||||
|
return trim($response['choices'][0]['message']['content'] ?? 'No response');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function generateNLPFromQuery($inputUser, $resultQuery) {
|
||||||
|
$response = $this->client->chat()->create([
|
||||||
|
'model' => 'gpt-4o-mini',
|
||||||
|
'messages' => [
|
||||||
|
[
|
||||||
|
'role' => 'system',
|
||||||
|
'content' => "You are an expert assistant. Your task is to analyze the database query results and transform them into a human-readable answer based on the user's question.
|
||||||
|
|
||||||
|
Guidelines:
|
||||||
|
- Understand the user's question and extract the key intent.
|
||||||
|
- Summarize or format the query results to directly answer the user's question.
|
||||||
|
- Ensure the response is clear, concise, and relevant.
|
||||||
|
- If the query result is empty or does not match the question, provide a polite response indicating that no data is available.
|
||||||
|
|
||||||
|
Always provide a well-structured response that makes sense based on the input question."
|
||||||
|
],
|
||||||
|
['role' => 'user', 'content' => "User's question: $inputUser \nDatabase result: $resultQuery"],
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
|
||||||
|
return trim($response['choices'][0]['message']['content'] ?? 'No response');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function generateFinalText($nlpResult) {
|
||||||
|
$response = $this->client->chat()->create([
|
||||||
|
'model' => 'gpt-4o-mini',
|
||||||
|
'messages' => [
|
||||||
|
[
|
||||||
|
'role' => 'system',
|
||||||
|
'content' => "You are an expert text formatter. Your task is to take the given NLP result and format it into a structured, human-readable text suitable for rendering inside an HTML <div>.
|
||||||
|
|
||||||
|
Guidelines:
|
||||||
|
- Preserve the meaning and clarity of the content.
|
||||||
|
- Use proper line breaks for readability.
|
||||||
|
- If the text contains lists, convert them into bullet points.
|
||||||
|
- Emphasize important keywords using <strong> tags if necessary.
|
||||||
|
- Ensure the response remains clean and concise without extra explanations."
|
||||||
|
],
|
||||||
|
['role' => 'user', 'content' => "Here is the NLP result that needs formatting:\n\n$nlpResult"],
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
|
||||||
|
return trim($response['choices'][0]['message']['content'] ?? 'No response');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function classifyMainGenerateText($prompt) {
|
||||||
|
$response = $this->client->chat()->create([
|
||||||
|
'model' => 'gpt-4o-mini',
|
||||||
|
'messages' => [
|
||||||
|
[
|
||||||
|
'role' => 'system',
|
||||||
|
'content' => "You are an assistant that classifies text into one of the following categories:
|
||||||
|
- reklame (ads or product/service promotions)
|
||||||
|
- business_or_industries (business or industries in general)
|
||||||
|
- customers (customers, consumers, or service users)
|
||||||
|
- pbg (tasks related to Building Approval)
|
||||||
|
- retribusi (retributions related to PBG)
|
||||||
|
- spatial_plannings (spatial planning)
|
||||||
|
- tourisms (tourism and tourist destinations)
|
||||||
|
- umkms (Micro, Small, and Medium Enterprises)
|
||||||
|
|
||||||
|
Respond with only one of the categories above without any additional explanation."
|
||||||
|
],
|
||||||
|
[
|
||||||
|
'role' => 'user',
|
||||||
|
'content' => "Classify the following text:\n\n" . $prompt
|
||||||
|
],
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
|
||||||
|
return trim($response['choices'][0]['message']['content'] ?? 'No response');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function createMainQuery($classify, $prompt, $chatHistory)
|
||||||
|
{
|
||||||
|
// Load file JSON
|
||||||
|
$jsonPath = public_path('templates/table_config.json');
|
||||||
|
$jsonConfig = json_decode(file_get_contents($jsonPath), true);
|
||||||
|
|
||||||
|
// Pastikan kategori tersedia dalam konfigurasi
|
||||||
|
if (!isset($jsonConfig[$classify])) {
|
||||||
|
return "Error: Kategori tidak ditemukan dalam konfigurasi.";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ambil nama tabel dan kolom
|
||||||
|
$tableName = $jsonConfig[$classify]['table_name'];
|
||||||
|
$columns = implode(', ', $jsonConfig[$classify]['list_column']);
|
||||||
|
|
||||||
|
// Konversi chatHistory ke dalam format messages
|
||||||
|
$messages = [
|
||||||
|
[
|
||||||
|
'role' => 'system',
|
||||||
|
'content' => "You are an AI assistant that generates only valid MariaDB queries based on user requests.
|
||||||
|
Use the following table information to construct the SQL query:
|
||||||
|
|
||||||
|
- Table Name: $tableName
|
||||||
|
- Available Columns: $columns
|
||||||
|
|
||||||
|
Generate only the SQL query without any explanation or additional text.
|
||||||
|
The query should include `LIMIT 10` to restrict the results."
|
||||||
|
]
|
||||||
|
];
|
||||||
|
|
||||||
|
// Menambahkan chat history sebagai konteks
|
||||||
|
foreach ($chatHistory as $chat) {
|
||||||
|
if (isset($chat['user'])) {
|
||||||
|
$messages[] = ['role' => 'user', 'content' => $chat['user']];
|
||||||
|
}
|
||||||
|
if (isset($chat['rawBotResponse'])) {
|
||||||
|
$messages[] = ['role' => 'assistant', 'content' => $chat['rawBotResponse']];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tambahkan prompt utama pengguna
|
||||||
|
$messages[] = ['role' => 'user', 'content' => $prompt];
|
||||||
|
|
||||||
|
// Kirim permintaan ke model AI
|
||||||
|
$response = $this->client->chat()->create([
|
||||||
|
'model' => 'gpt-4o-mini',
|
||||||
|
'messages' => $messages
|
||||||
|
]);
|
||||||
|
|
||||||
|
return trim($response['choices'][0]['message']['content'] ?? 'No response');
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// public function createMainQuery($classify, $prompt)
|
||||||
|
// {
|
||||||
|
// // Load file JSON
|
||||||
|
// $jsonPath = public_path('templates/table_config.json');
|
||||||
|
// $jsonConfig = json_decode(file_get_contents($jsonPath), true);
|
||||||
|
|
||||||
|
// // Pastikan kategori tersedia dalam konfigurasi
|
||||||
|
// if (!isset($jsonConfig[$classify])) {
|
||||||
|
// return "Error: Kategori tidak ditemukan dalam konfigurasi.";
|
||||||
|
// }
|
||||||
|
|
||||||
|
// // Ambil nama tabel dan kolom
|
||||||
|
// $tableName = $jsonConfig[$classify]['table_name'];
|
||||||
|
// $columns = implode(', ', $jsonConfig[$classify]['list_column']);
|
||||||
|
|
||||||
|
// $response = $this->client->chat()->create([
|
||||||
|
// 'model' => 'gpt-4o-mini',
|
||||||
|
// 'messages' => [
|
||||||
|
// [
|
||||||
|
// 'role' => 'system',
|
||||||
|
// 'content' => "You are an AI assistant that generates only valid MariaDB queries based on user requests.
|
||||||
|
// Use the following table information to construct the SQL query:
|
||||||
|
|
||||||
|
// - Table Name: $tableName
|
||||||
|
// - Available Columns: $columns
|
||||||
|
|
||||||
|
// Generate only the SQL query without any explanation or additional text
|
||||||
|
// The query should include `LIMIT 10` to restrict the results."
|
||||||
|
// ],
|
||||||
|
// [
|
||||||
|
// 'role' => 'user',
|
||||||
|
// 'content' => $prompt
|
||||||
|
// ],
|
||||||
|
// ],
|
||||||
|
// ]);
|
||||||
|
|
||||||
|
// return trim($response['choices'][0]['message']['content'] ?? 'No response');
|
||||||
|
// }
|
||||||
|
}
|
||||||
@@ -51,9 +51,25 @@ class ServiceClient
|
|||||||
|
|
||||||
$resultResponse = json_decode($responseBody, true, 512, JSON_THROW_ON_ERROR);
|
$resultResponse = json_decode($responseBody, true, 512, JSON_THROW_ON_ERROR);
|
||||||
return $this->resSuccess($resultResponse);
|
return $this->resSuccess($resultResponse);
|
||||||
|
} catch (\GuzzleHttp\Exception\ClientException $e) {
|
||||||
|
// Handle 4xx errors (e.g., 401 Unauthorized)
|
||||||
|
$responseBody = (string) $e->getResponse()->getBody();
|
||||||
|
$errorResponse = json_decode($responseBody, true);
|
||||||
|
|
||||||
|
if (isset($errorResponse['code']) && $errorResponse['code'] === 'token_not_valid') {
|
||||||
|
return $this->resError('Invalid token, please refresh your token.', $errorResponse, 401);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->resError('Client error from API', $errorResponse, $e->getResponse()->getStatusCode());
|
||||||
|
} catch (\GuzzleHttp\Exception\ServerException $e) {
|
||||||
|
// Handle 5xx errors (e.g., Internal Server Error)
|
||||||
|
return $this->resError('Server error from API', (string) $e->getResponse()->getBody(), 500);
|
||||||
|
} catch (\GuzzleHttp\Exception\RequestException $e) {
|
||||||
|
// Handle network errors (e.g., timeout, connection issues)
|
||||||
|
return $this->resError('Network error: ' . $e->getMessage(), null, 503);
|
||||||
} catch (Exception $e) {
|
} catch (Exception $e) {
|
||||||
\Log::error('error from client service'. $e->getMessage());
|
// Handle unexpected errors
|
||||||
return $this->resError($e->getMessage());
|
return $this->resError('Unexpected error: ' . $e->getMessage(), null, 500);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -9,12 +9,16 @@ use App\Models\ImportDatasource;
|
|||||||
use App\Models\PbgTaskIndexIntegrations;
|
use App\Models\PbgTaskIndexIntegrations;
|
||||||
use App\Models\PbgTaskPrasarana;
|
use App\Models\PbgTaskPrasarana;
|
||||||
use App\Models\PbgTaskRetributions;
|
use App\Models\PbgTaskRetributions;
|
||||||
|
use App\Models\TaskAssignment;
|
||||||
use Exception;
|
use Exception;
|
||||||
use App\Models\PbgTask;
|
use App\Models\PbgTask;
|
||||||
use App\Traits\GlobalApiResponse;
|
use App\Traits\GlobalApiResponse;
|
||||||
use Illuminate\Support\Facades\Log;
|
use Illuminate\Support\Facades\Log;
|
||||||
use Carbon\Carbon;
|
use Carbon\Carbon;
|
||||||
use App\Services\ServiceClient;
|
use App\Services\ServiceClient;
|
||||||
|
use App\Services\GoogleSheetService;
|
||||||
|
use App\Models\DataSetting;
|
||||||
|
use App\Models\PbgTaskGoogleSheet;
|
||||||
|
|
||||||
class ServiceSIMBG
|
class ServiceSIMBG
|
||||||
{
|
{
|
||||||
@@ -24,10 +28,11 @@ class ServiceSIMBG
|
|||||||
private $simbg_host;
|
private $simbg_host;
|
||||||
private $fetch_per_page;
|
private $fetch_per_page;
|
||||||
private $service_client;
|
private $service_client;
|
||||||
|
private $googleSheetService;
|
||||||
/**
|
/**
|
||||||
* Create a new class instance.
|
* Create a new class instance.
|
||||||
*/
|
*/
|
||||||
public function __construct()
|
public function __construct(GoogleSheetService $googleSheetService)
|
||||||
{
|
{
|
||||||
$settings = GlobalSetting::whereIn('key', [
|
$settings = GlobalSetting::whereIn('key', [
|
||||||
'SIMBG_EMAIL', 'SIMBG_PASSWORD', 'SIMBG_HOST', 'FETCH_PER_PAGE'
|
'SIMBG_EMAIL', 'SIMBG_PASSWORD', 'SIMBG_HOST', 'FETCH_PER_PAGE'
|
||||||
@@ -39,6 +44,7 @@ class ServiceSIMBG
|
|||||||
$this->fetch_per_page = trim((string) ($settings['FETCH_PER_PAGE'] ?? ""));
|
$this->fetch_per_page = trim((string) ($settings['FETCH_PER_PAGE'] ?? ""));
|
||||||
|
|
||||||
$this->service_client = new ServiceClient($this->simbg_host);
|
$this->service_client = new ServiceClient($this->simbg_host);
|
||||||
|
$this->googleSheetService = $googleSheetService;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getToken(){
|
public function getToken(){
|
||||||
@@ -61,9 +67,22 @@ class ServiceSIMBG
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function syncIndexIntegration($uuid, $token)
|
public function syncIndexIntegration($uuids)
|
||||||
{
|
{
|
||||||
try{
|
try{
|
||||||
|
if(empty($uuids)){
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$initResToken = $this->getToken();
|
||||||
|
if (empty($initResToken->original['data']['token']['access'])) {
|
||||||
|
Log::error("API response indicates failure", ['token' => 'Failed to retrieve token']);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
$token = $initResToken->original['data']['token']['access'];
|
||||||
|
|
||||||
|
$integrations = [];
|
||||||
|
foreach($uuids as $uuid){
|
||||||
$url = "/api/pbg/v1/detail/" . $uuid . "/retribution/indeks-terintegrasi/";
|
$url = "/api/pbg/v1/detail/" . $uuid . "/retribution/indeks-terintegrasi/";
|
||||||
|
|
||||||
$headers = [
|
$headers = [
|
||||||
@@ -75,18 +94,17 @@ class ServiceSIMBG
|
|||||||
if (empty($res->original['success']) || !$res->original['success']) {
|
if (empty($res->original['success']) || !$res->original['success']) {
|
||||||
// Log error
|
// Log error
|
||||||
Log::error("API response indicates failure", ['url' => $url, 'uuid' => $uuid]);
|
Log::error("API response indicates failure", ['url' => $url, 'uuid' => $uuid]);
|
||||||
return false;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
$data = $res->original['data']['data'] ?? null;
|
$data = $res->original['data']['data'] ?? null;
|
||||||
if (!$data) {
|
if (!$data) {
|
||||||
Log::error("No valid data returned from API", ['url' => $url, 'uuid' => $uuid]);
|
Log::error("No valid data returned from API", ['url' => $url, 'uuid' => $uuid]);
|
||||||
return false;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
$resultData = PbgTaskIndexIntegrations::updateOrCreate(
|
$integrations[] = [
|
||||||
['pbg_task_uid' => $uuid],
|
'pbg_task_uid' => $uuid,
|
||||||
[
|
|
||||||
'indeks_fungsi_bangunan' => $data['indeks_fungsi_bangunan'] ?? null,
|
'indeks_fungsi_bangunan' => $data['indeks_fungsi_bangunan'] ?? null,
|
||||||
'indeks_parameter_kompleksitas' => $data['indeks_parameter_kompleksitas'] ?? null,
|
'indeks_parameter_kompleksitas' => $data['indeks_parameter_kompleksitas'] ?? null,
|
||||||
'indeks_parameter_permanensi' => $data['indeks_parameter_permanensi'] ?? null,
|
'indeks_parameter_permanensi' => $data['indeks_parameter_permanensi'] ?? null,
|
||||||
@@ -94,8 +112,11 @@ class ServiceSIMBG
|
|||||||
'faktor_kepemilikan' => $data['faktor_kepemilikan'] ?? null,
|
'faktor_kepemilikan' => $data['faktor_kepemilikan'] ?? null,
|
||||||
'indeks_terintegrasi' => $data['indeks_terintegrasi'] ?? null,
|
'indeks_terintegrasi' => $data['indeks_terintegrasi'] ?? null,
|
||||||
'total' => $data['total'] ?? null,
|
'total' => $data['total'] ?? null,
|
||||||
]
|
];
|
||||||
);
|
}
|
||||||
|
|
||||||
|
PbgTaskIndexIntegrations::upsert($integrations, ['pbg_task_uid'], ['indeks_fungsi_bangunan',
|
||||||
|
'indeks_parameter_kompleksitas', 'indeks_parameter_permanensi', 'indeks_parameter_ketinggian', 'faktor_kepemilikan', 'indeks_terintegrasi', 'total']);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}catch (Exception $e){
|
}catch (Exception $e){
|
||||||
@@ -104,22 +125,184 @@ class ServiceSIMBG
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function syncTaskList()
|
public function syncTaskPBG()
|
||||||
{
|
{
|
||||||
$initResToken = $this->getToken();
|
try {
|
||||||
|
Log::info("Processing google sheet sync");
|
||||||
$importDatasource = ImportDatasource::create([
|
$importDatasource = ImportDatasource::create([
|
||||||
'status' => ImportDatasourceStatus::Processing->value,
|
'status' => ImportDatasourceStatus::Processing->value,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
// sync google sheet first
|
||||||
|
$totalRowCount = $this->googleSheetService->getLastRowByColumn("C");
|
||||||
|
$sheetData = $this->googleSheetService->getSheetDataCollection($totalRowCount);
|
||||||
|
$sheet_big_data = $this->googleSheetService->get_data_by_sheet();
|
||||||
|
$data_setting_result = []; // Initialize result storage
|
||||||
|
|
||||||
|
$found_section = null; // Track which section is found
|
||||||
|
|
||||||
|
foreach ($sheet_big_data as $row) {
|
||||||
|
// Check for section headers
|
||||||
|
if (in_array("•PROSES PENERBITAN:", $row)) {
|
||||||
|
$found_section = "MENUNGGU_KLIK_DPMPTSP";
|
||||||
|
} elseif (in_array("•BERKAS AKTUAL TERVERIFIKASI DINAS TEKNIS 2024:", $row)) {
|
||||||
|
$found_section = "REALISASI_TERBIT_PBG";
|
||||||
|
} elseif (in_array("•TERPROSES DI DPUTR: belum selesai rekomtek'", $row)) {
|
||||||
|
$found_section = "PROSES_DINAS_TEKNIS";
|
||||||
|
}
|
||||||
|
|
||||||
|
// If a section is found and we reach "Grand Total", save the corresponding values
|
||||||
|
if ($found_section && isset($row[0]) && trim($row[0]) === "Grand Total") {
|
||||||
|
if ($found_section === "MENUNGGU_KLIK_DPMPTSP") {
|
||||||
|
$data_setting_result["MENUNGGU_KLIK_DPMPTSP_COUNT"] = $this->convertToInteger($row[2]) ?? null;
|
||||||
|
$data_setting_result["MENUNGGU_KLIK_DPMPTSP_SUM"] = $this->convertToDecimal($row[3]) ?? null;
|
||||||
|
} elseif ($found_section === "REALISASI_TERBIT_PBG") {
|
||||||
|
$data_setting_result["REALISASI_TERBIT_PBG_COUNT"] = $this->convertToInteger($row[2]) ?? null;
|
||||||
|
$data_setting_result["REALISASI_TERBIT_PBG_SUM"] = $this->convertToDecimal($row[4]) ?? null;
|
||||||
|
} elseif ($found_section === "PROSES_DINAS_TEKNIS") {
|
||||||
|
$data_setting_result["PROSES_DINAS_TEKNIS_COUNT"] = $this->convertToInteger($row[2]) ?? null;
|
||||||
|
$data_setting_result["PROSES_DINAS_TEKNIS_SUM"] = $this->convertToDecimal($row[3]) ?? null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reset section tracking after capturing "Grand Total"
|
||||||
|
$found_section = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Log::info("data setting result", ['result' => $data_setting_result]);
|
||||||
|
|
||||||
|
foreach ($data_setting_result as $key => $value) {
|
||||||
|
DataSetting::updateOrInsert(
|
||||||
|
["key" => $key], // Find by key
|
||||||
|
["value" => $value] // Update or insert value
|
||||||
|
);
|
||||||
|
}
|
||||||
|
$mapToUpsert = [];
|
||||||
|
|
||||||
|
foreach($sheetData as $data){
|
||||||
|
$mapToUpsert[] =
|
||||||
|
[
|
||||||
|
'no_registrasi' => $data['no__registrasi'] ?? null,
|
||||||
|
'jenis_konsultasi' => $data['jenis_konsultasi'] ?? null,
|
||||||
|
'fungsi_bg' => $data['fungsi_bg'] ?? null,
|
||||||
|
'tgl_permohonan' => $this->convertToDate($data['tgl_permohonan']),
|
||||||
|
'status_verifikasi' => $data['status_verifikasi'] ?? null,
|
||||||
|
'status_permohonan' => $this->convertToDate($data['status_permohonan']),
|
||||||
|
'alamat_pemilik' => $data['alamat_pemilik'] ?? null,
|
||||||
|
'no_hp' => $data['no__hp'] ?? null,
|
||||||
|
'email' => $data['e_mail'] ?? null,
|
||||||
|
'tanggal_catatan' => $this->convertToDate($data['tanggal_catatan']),
|
||||||
|
'catatan_kekurangan_dokumen' => $data['catatan_kekurangan_dokumen'] ?? null,
|
||||||
|
'gambar' => $data['gambar'] ?? null,
|
||||||
|
'krk_kkpr' => $data['krk_kkpr'] ?? null,
|
||||||
|
'no_krk' => $data['no__krk'] ?? null,
|
||||||
|
'lh' => $data['lh'] ?? null,
|
||||||
|
'ska' => $data['ska'] ?? null,
|
||||||
|
'keterangan' => $data['keterangan'] ?? null,
|
||||||
|
'helpdesk' => $data['helpdesk'] ?? null,
|
||||||
|
'pj' => $data['pj'] ?? null,
|
||||||
|
'kepemilikan' => $data['kepemilikan'] ?? null,
|
||||||
|
'potensi_taru' => $data['potensi_taru'] ?? null,
|
||||||
|
'validasi_dinas' => $data['validasi_dinas'] ?? null,
|
||||||
|
'kategori_retribusi' => $data['kategori_retribusi'] ?? null,
|
||||||
|
'no_urut_ba_tpt' => $data['no__urut_ba_tpt__2024_0001_'] ?? null,
|
||||||
|
'tanggal_ba_tpt' => $this->convertToDate($data['tanggal_ba_tpt']),
|
||||||
|
'no_urut_ba_tpa' => $data['no__urut_ba_tpa'] ?? null,
|
||||||
|
'tanggal_ba_tpa' => $this->convertToDate($data['tanggal_ba_tpa']),
|
||||||
|
'no_urut_skrd' => $data['no__urut_skrd__2024_0001_'] ?? null,
|
||||||
|
'tanggal_skrd' => $this->convertToDate($data['tanggal_skrd']),
|
||||||
|
'ptsp' => $data['ptsp'] ?? null,
|
||||||
|
'selesai_terbit' => $data['selesai_terbit'] ?? null,
|
||||||
|
'tanggal_pembayaran' => $this->convertToDate($data['tanggal_pembayaran__yyyy_mm_dd_']),
|
||||||
|
'format_sts' => $data['format_sts'] ?? null,
|
||||||
|
'tahun_terbit' => (int) $data['tahun_terbit'] ?? null,
|
||||||
|
'tahun_berjalan' => (int) $data['tahun_berjalan'] ?? null,
|
||||||
|
'kelurahan' => $data['kelurahan'] ?? null,
|
||||||
|
'kecamatan' => $data['kecamatan'] ?? null,
|
||||||
|
'lb' => $this->convertToDecimal($data['lb']) ?? null,
|
||||||
|
'tb' => $this->convertToDecimal($data['tb']) ?? null,
|
||||||
|
'jlb' => (int) $data['jlb'] ?? null,
|
||||||
|
'unit' => (int) $data['unit'] ?? null,
|
||||||
|
'usulan_retribusi' => (int) $data['usulan_retribusi'] ?? null,
|
||||||
|
'nilai_retribusi_keseluruhan_simbg' => $this->convertToDecimal($data['nilai_retribusi_keseluruhan__simbg_']) ?? null,
|
||||||
|
'nilai_retribusi_keseluruhan_pad' => $this->convertToDecimal($data['nilai_retribusi_keseluruhan__pad_']) ?? null,
|
||||||
|
'denda' => $this->convertToDecimal($data['denda']) ?? null,
|
||||||
|
'latitude' => $data['latitude'] ?? null,
|
||||||
|
'longitude' => $data['longitude'] ?? null,
|
||||||
|
'nik_nib' => $data['nik_nib'] ?? null,
|
||||||
|
'dok_tanah' => $data['dok__tanah'] ?? null,
|
||||||
|
'temuan' => $data['temuan'] ?? null,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
$batchSize = 1000;
|
||||||
|
$chunks = array_chunk($mapToUpsert, $batchSize);
|
||||||
|
|
||||||
|
foreach($chunks as $chunk){
|
||||||
|
PbgTaskGoogleSheet::upsert($chunk, ["no_registrasi"],[
|
||||||
|
'jenis_konsultasi',
|
||||||
|
'nama_pemilik',
|
||||||
|
'lokasi_bg',
|
||||||
|
'fungsi_bg',
|
||||||
|
'nama_bangunan',
|
||||||
|
'tgl_permohonan',
|
||||||
|
'status_verifikasi',
|
||||||
|
'status_permohonan',
|
||||||
|
'alamat_pemilik',
|
||||||
|
'no_hp',
|
||||||
|
'email',
|
||||||
|
'tanggal_catatan',
|
||||||
|
'catatan_kekurangan_dokumen',
|
||||||
|
'gambar',
|
||||||
|
'krk_kkpr',
|
||||||
|
'no_krk',
|
||||||
|
'lh',
|
||||||
|
'ska',
|
||||||
|
'keterangan',
|
||||||
|
'helpdesk',
|
||||||
|
'pj',
|
||||||
|
'kepemilikan',
|
||||||
|
'potensi_taru',
|
||||||
|
'validasi_dinas',
|
||||||
|
'kategori_retribusi',
|
||||||
|
'no_urut_ba_tpt',
|
||||||
|
'tanggal_ba_tpt',
|
||||||
|
'no_urut_ba_tpa',
|
||||||
|
'tanggal_ba_tpa',
|
||||||
|
'no_urut_skrd',
|
||||||
|
'tanggal_skrd',
|
||||||
|
'ptsp',
|
||||||
|
'selesai_terbit',
|
||||||
|
'tanggal_pembayaran',
|
||||||
|
'format_sts',
|
||||||
|
'tahun_terbit',
|
||||||
|
'tahun_berjalan',
|
||||||
|
'kelurahan',
|
||||||
|
'kecamatan',
|
||||||
|
'lb',
|
||||||
|
'tb',
|
||||||
|
'jlb',
|
||||||
|
'unit',
|
||||||
|
'usulan_retribusi',
|
||||||
|
'nilai_retribusi_keseluruhan_simbg',
|
||||||
|
'nilai_retribusi_keseluruhan_pad',
|
||||||
|
'denda',
|
||||||
|
'latitude',
|
||||||
|
'longitude',
|
||||||
|
'nik_nib',
|
||||||
|
'dok_tanah',
|
||||||
|
'temuan',
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$initResToken = $this->getToken();
|
||||||
if (empty($initResToken->original['data']['token']['access'])) {
|
if (empty($initResToken->original['data']['token']['access'])) {
|
||||||
$importDatasource->update([
|
$importDatasource->update([
|
||||||
'status' => ImportDatasourceStatus::Failed->value,
|
'status' => ImportDatasourceStatus::Failed->value,
|
||||||
'message' => 'Failed to retrieve token'
|
'response_body' => 'Failed to retrieve token'
|
||||||
]);
|
]);
|
||||||
return $this->resError("Failed to retrieve token");
|
return $this->resError("Failed to retrieve token");
|
||||||
}
|
}
|
||||||
|
|
||||||
$apiToken = $initResToken->original['data']['token']['access'];
|
$apiToken = $initResToken->original['data']['token']['access'];
|
||||||
$headers = ['Authorization' => "Bearer " . $apiToken];
|
$headers = ['Authorization' => "Bearer " . $apiToken];
|
||||||
|
|
||||||
@@ -130,40 +313,64 @@ class ServiceSIMBG
|
|||||||
if ($totalPage == 0) {
|
if ($totalPage == 0) {
|
||||||
$importDatasource->update([
|
$importDatasource->update([
|
||||||
'status' => ImportDatasourceStatus::Failed->value,
|
'status' => ImportDatasourceStatus::Failed->value,
|
||||||
'message' => 'Invalid response: no total_page'
|
'response_body' => 'Invalid response: no total_page'
|
||||||
]);
|
]);
|
||||||
return $this->resError("Invalid response from API");
|
return $this->resError("Invalid response from API");
|
||||||
}
|
}
|
||||||
|
|
||||||
$savedCount = $failedCount = 0;
|
$savedCount = $failedCount = 0;
|
||||||
|
|
||||||
|
Log::info("Fetching tasks", ['total page' => $totalPage]);
|
||||||
|
|
||||||
for ($currentPage = 1; $currentPage <= $totalPage; $currentPage++) {
|
for ($currentPage = 1; $currentPage <= $totalPage; $currentPage++) {
|
||||||
|
try {
|
||||||
$pageUrl = "/api/pbg/v1/list/?page={$currentPage}&size={$this->fetch_per_page}&sort=ASC";
|
$pageUrl = "/api/pbg/v1/list/?page={$currentPage}&size={$this->fetch_per_page}&sort=ASC";
|
||||||
$getToken = $this->getToken();
|
|
||||||
Log::info("response index integration", ['currentPage' => $currentPage]);
|
Log::info("Fetching tasks", ['currentPage' => $currentPage]);
|
||||||
if (empty($getToken->original['data']['token']['access'])) {
|
$headers = [
|
||||||
$importDatasource->update([
|
'Authorization' => "Bearer " . $apiToken, // Update headers
|
||||||
'status' => ImportDatasourceStatus::Failed->value,
|
];
|
||||||
'message' => 'Failed to retrieve token'
|
|
||||||
]);
|
for ($attempt = 0; $attempt < 2; $attempt++) { // Try twice (original + retry)
|
||||||
|
|
||||||
|
$response = $this->service_client->get($pageUrl, $headers);
|
||||||
|
|
||||||
|
if ($response instanceof \Illuminate\Http\JsonResponse) {
|
||||||
|
$decodedResponse = json_decode($response->getContent(), true);
|
||||||
|
|
||||||
|
if (isset($decodedResponse['errors']['code']) && $decodedResponse['errors']['code'] === 'token_not_valid') {
|
||||||
|
Log::warning("Token is invalid, refreshing token...");
|
||||||
|
|
||||||
|
// Regenerate token
|
||||||
|
$initResToken = $this->getToken();
|
||||||
|
|
||||||
|
// Check if new token is valid
|
||||||
|
if (!empty($initResToken->original['data']['token']['access'])) {
|
||||||
|
$new_token = $initResToken->original['data']['token']['access'];
|
||||||
|
|
||||||
|
// **Fix: Update headers before retrying**
|
||||||
|
$headers['Authorization'] = "Bearer " . $new_token;
|
||||||
|
|
||||||
|
Log::info("Token refreshed successfully, retrying API request...");
|
||||||
|
continue; // Retry with new token
|
||||||
|
} else {
|
||||||
|
Log::error("Failed to refresh token");
|
||||||
|
return $this->resError("Failed to refresh token");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Success case, break loop
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
$token = $getToken->original['data']['token']['access'];
|
|
||||||
$headers = ['Authorization' => "Bearer " . $token];
|
|
||||||
$response = $this->service_client->get($pageUrl, $headers);
|
|
||||||
$tasks = $response->original['data']['data'] ?? [];
|
$tasks = $response->original['data']['data'] ?? [];
|
||||||
|
|
||||||
if (empty($tasks)) {
|
if (empty($tasks)) {
|
||||||
$importDatasource->update([
|
|
||||||
'status' => ImportDatasourceStatus::Failed->value,
|
|
||||||
'message' => 'No data found on page'
|
|
||||||
]);
|
|
||||||
Log::warning("No data found on page", ['page' => $currentPage]);
|
Log::warning("No data found on page", ['page' => $currentPage]);
|
||||||
break;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
Log::info("executed page", ['page' => $currentPage, 'total' => $totalPage]);
|
|
||||||
|
|
||||||
$tasksCollective = [];
|
$tasksCollective = [];
|
||||||
foreach ($tasks as $item) {
|
foreach ($tasks as $item) {
|
||||||
try {
|
try {
|
||||||
@@ -190,65 +397,112 @@ class ServiceSIMBG
|
|||||||
'created_at' => now(),
|
'created_at' => now(),
|
||||||
];
|
];
|
||||||
|
|
||||||
$this->syncIndexIntegration($item['uid'], $token);
|
$this->syncTaskDetailSubmit($item['uid'], $apiToken);
|
||||||
|
$this->syncTaskAssignments($item['uid']);
|
||||||
$this->syncTaskDetailSubmit($item['uid'], $token);
|
|
||||||
|
|
||||||
$savedCount++;
|
$savedCount++;
|
||||||
} catch (Exception $e) {
|
} catch (Exception $e) {
|
||||||
$failedCount++;
|
$failedCount++;
|
||||||
$importDatasource->update([
|
|
||||||
'status' => ImportDatasourceStatus::Failed->value,
|
|
||||||
'message' => "Successfully processed: $savedCount, Failed: $failedCount"
|
|
||||||
]);
|
|
||||||
Log::error("Failed to process task", [
|
Log::error("Failed to process task", [
|
||||||
'error' => $e->getMessage(),
|
'error' => $e->getMessage(),
|
||||||
'task' => $item,
|
'task' => $item,
|
||||||
]);
|
]);
|
||||||
break;
|
continue; // Skip failed task, continue processing the rest
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!empty($tasksCollective)) {
|
||||||
PbgTask::upsert($tasksCollective, ['uuid'], [
|
PbgTask::upsert($tasksCollective, ['uuid'], [
|
||||||
'name', 'owner_name', 'application_type', 'application_type_name', 'condition',
|
'name', 'owner_name', 'application_type', 'application_type_name', 'condition',
|
||||||
'registration_number', 'document_number', 'address', 'status', 'status_name',
|
'registration_number', 'document_number', 'address', 'status', 'status_name',
|
||||||
'slf_status', 'slf_status_name', 'function_type', 'consultation_type', 'due_date',
|
'slf_status', 'slf_status_name', 'function_type', 'consultation_type', 'due_date',
|
||||||
'land_certificate_phase', 'task_created_at', 'updated_at'
|
'land_certificate_phase', 'task_created_at', 'updated_at'
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
$uuids = array_column($tasksCollective, 'uuid');
|
||||||
|
$this->syncIndexIntegration($uuids);
|
||||||
|
}
|
||||||
|
} catch (Exception $e) {
|
||||||
|
Log::error("Failed to process page", [
|
||||||
|
'error' => $e->getMessage(),
|
||||||
|
'page' => $currentPage,
|
||||||
|
]);
|
||||||
|
continue; // Skip the failed page and move to the next
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BigdataResume::generateResumeData($importDatasource->id, "all", $data_setting_result);
|
||||||
|
BigdataResume::generateResumeData($importDatasource->id, now()->year, $data_setting_result);
|
||||||
|
|
||||||
|
// Final update after processing all pages
|
||||||
$importDatasource->update([
|
$importDatasource->update([
|
||||||
'status' => ImportDatasourceStatus::Success->value,
|
'status' => ImportDatasourceStatus::Success->value,
|
||||||
'message' => "Successfully processed: $savedCount, Failed: $failedCount"
|
'message' => "Successfully processed: $savedCount, Failed: $failedCount"
|
||||||
]);
|
]);
|
||||||
|
|
||||||
BigdataResume::generateResumeData($importDatasource->id);
|
|
||||||
|
|
||||||
Log::info("syncTaskList completed", ['savedCount' => $savedCount, 'failedCount' => $failedCount]);
|
Log::info("syncTaskList completed", ['savedCount' => $savedCount, 'failedCount' => $failedCount]);
|
||||||
|
|
||||||
return $this->resSuccess(['savedCount' => $savedCount, 'failedCount' => $failedCount]);
|
return $this->resSuccess(['savedCount' => $savedCount, 'failedCount' => $failedCount]);
|
||||||
}
|
|
||||||
|
|
||||||
|
} catch (Exception $e) {
|
||||||
|
Log::error("syncTaskList failed", ['error' => $e->getMessage()]);
|
||||||
|
if (isset($importDatasource)) {
|
||||||
|
$importDatasource->update([
|
||||||
|
'status' => ImportDatasourceStatus::Failed->value,
|
||||||
|
'response_body' => 'Critical failure: ' . $e->getMessage()
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
return $this->resError("Critical failure occurred: " . $e->getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public function syncTaskDetailSubmit($uuid, $token)
|
public function syncTaskDetailSubmit($uuid, $token)
|
||||||
{
|
{
|
||||||
try{
|
try{
|
||||||
$url = "/api/pbg/v1/detail/" . $uuid . "/retribution/submit/";
|
$url = "/api/pbg/v1/detail/" . $uuid . "/retribution/submit/";
|
||||||
|
|
||||||
$headers = [
|
$headers = [
|
||||||
'Authorization' => "Bearer " . $token,
|
'Authorization' => "Bearer " . $token,
|
||||||
];
|
];
|
||||||
|
|
||||||
|
for ($attempt = 0; $attempt < 2; $attempt++) { // Try twice (original + retry)
|
||||||
$res = $this->service_client->get($url, $headers);
|
$res = $this->service_client->get($url, $headers);
|
||||||
|
|
||||||
if (empty($res->original['success']) || !$res->original['success']) {
|
// Check if response is JsonResponse and decode it
|
||||||
// Log error
|
if ($res instanceof \Illuminate\Http\JsonResponse) {
|
||||||
Log::error("API response indicates failure", ['url' => $url, 'uuid' => $uuid]);
|
$decodedResponse = json_decode($res->getContent(), true);
|
||||||
return false;
|
|
||||||
|
// Handle invalid token case
|
||||||
|
if (isset($decodedResponse['errors']['code']) && $decodedResponse['errors']['code'] === 'token_not_valid') {
|
||||||
|
Log::warning("Token is invalid, refreshing token...");
|
||||||
|
|
||||||
|
// Regenerate the token
|
||||||
|
$initResToken = $this->getToken();
|
||||||
|
|
||||||
|
// Check if the new token is valid
|
||||||
|
if (!empty($initResToken->original['data']['token']['access'])) {
|
||||||
|
$new_token = $initResToken->original['data']['token']['access'];
|
||||||
|
|
||||||
|
// **Fix: Update headers with the new token**
|
||||||
|
$headers['Authorization'] = "Bearer " . $new_token;
|
||||||
|
|
||||||
|
Log::info("Token refreshed successfully, retrying API request...");
|
||||||
|
continue; // Retry the request with the new token
|
||||||
|
} else {
|
||||||
|
Log::error("Failed to refresh token");
|
||||||
|
return $this->resError("Failed to refresh token");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$data = $res->original['data']['data'] ?? [];
|
// If request succeeds, break out of retry loop
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure response is valid before accessing properties
|
||||||
|
$responseData = $res->original ?? [];
|
||||||
|
$data = $responseData['data']['data'] ?? [];
|
||||||
if (empty($data)) {
|
if (empty($data)) {
|
||||||
Log::error("No data returned from API", ['url' => $url, 'uuid' => $uuid]);
|
Log::error("API response indicates failure", ['url' => $url, 'uuid' => $uuid, 'response' => $responseData]);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -311,5 +565,112 @@ class ServiceSIMBG
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
public function syncTaskAssignments($uuid){
|
||||||
|
try{
|
||||||
|
$init_token = $this->getToken();
|
||||||
|
$token = $init_token->original['data']['token']['access'];
|
||||||
|
$url = "/api/pbg/v1/list-tim-penilai/". $uuid . "/?page=1&size=10";
|
||||||
|
$headers = [
|
||||||
|
'Authorization' => "Bearer " . $token,
|
||||||
|
];
|
||||||
|
|
||||||
|
$response = $this->service_client->get($url, $headers);
|
||||||
|
$datas = $response->original['data']['data'] ?? [];
|
||||||
|
if(empty($datas)){
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
$task_assignments = [];
|
||||||
|
|
||||||
|
foreach ($datas as $data) {
|
||||||
|
$task_assignments[] = [
|
||||||
|
'pbg_task_uid' => $uuid, // Assuming this is a foreign key
|
||||||
|
'user_id' => $data['user_id'],
|
||||||
|
'name' => $data['name'],
|
||||||
|
'username' => $data['username'],
|
||||||
|
'email' => $data['email'],
|
||||||
|
'phone_number' => $data['phone_number'],
|
||||||
|
'role' => $data['role'],
|
||||||
|
'role_name' => $data['role_name'],
|
||||||
|
'is_active' => $data['is_active'],
|
||||||
|
'file' => json_encode($data['file']), // Store as JSON if it's an array
|
||||||
|
'expertise' => $data['expertise'],
|
||||||
|
'experience' => $data['experience'],
|
||||||
|
'is_verif' => $data['is_verif'],
|
||||||
|
'uid' => $data['uid'], // Unique identifier
|
||||||
|
'status' => $data['status'],
|
||||||
|
'status_name' => $data['status_name'],
|
||||||
|
'note' => $data['note'],
|
||||||
|
'created_at' => now(),
|
||||||
|
'updated_at' => now(),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
TaskAssignment::upsert(
|
||||||
|
$task_assignments, // Data to insert/update
|
||||||
|
['uid'], // Unique key for conflict resolution
|
||||||
|
['name', 'username', 'email', 'phone_number', 'role', 'role_name', 'is_active', 'file', 'expertise', 'experience', 'is_verif', 'status', 'status_name', 'note', 'updated_at']
|
||||||
|
);
|
||||||
|
return true;
|
||||||
|
}catch(Exception $e){
|
||||||
|
Log::error("Failed to sync task assignments", ['error' => $e->getMessage()]);
|
||||||
|
throw $e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function convertToDecimal(?string $value): ?float
|
||||||
|
{
|
||||||
|
if (empty($value)) {
|
||||||
|
return null; // Return null if the input is empty
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove all non-numeric characters except comma and dot
|
||||||
|
$value = preg_replace('/[^0-9,\.]/', '', $value);
|
||||||
|
|
||||||
|
// If the number contains both dot (.) and comma (,)
|
||||||
|
if (strpos($value, '.') !== false && strpos($value, ',') !== false) {
|
||||||
|
$value = str_replace('.', '', $value); // Remove thousands separator
|
||||||
|
$value = str_replace(',', '.', $value); // Convert decimal separator to dot
|
||||||
|
}
|
||||||
|
// If only a dot is present (assumed as thousands separator)
|
||||||
|
elseif (strpos($value, '.') !== false) {
|
||||||
|
$value = str_replace('.', '', $value); // Remove all dots (treat as thousands separators)
|
||||||
|
}
|
||||||
|
// If only a comma is present (assumed as decimal separator)
|
||||||
|
elseif (strpos($value, ',') !== false) {
|
||||||
|
$value = str_replace(',', '.', $value); // Convert comma to dot (decimal separator)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure the value is numeric before returning
|
||||||
|
return is_numeric($value) ? (float) number_format((float) $value, 2, '.', '') : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function convertToInteger($value) {
|
||||||
|
// Check if the value is an empty string, and return null if true
|
||||||
|
if (trim($value) === "") {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$cleaned = str_replace('.','', $value);
|
||||||
|
|
||||||
|
// Otherwise, cast to integer
|
||||||
|
return (int) $cleaned;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function convertToDate($dateString)
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
// Check if the string is empty
|
||||||
|
if (empty($dateString)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Try to parse the date string
|
||||||
|
$date = Carbon::parse($dateString);
|
||||||
|
|
||||||
|
// Return the Carbon instance
|
||||||
|
return $date->format('Y-m-d');
|
||||||
|
} catch (Exception $e) {
|
||||||
|
// Return null if an error occurs during parsing
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -15,7 +15,8 @@
|
|||||||
"laravel/framework": "^11.31",
|
"laravel/framework": "^11.31",
|
||||||
"laravel/sanctum": "^4.0",
|
"laravel/sanctum": "^4.0",
|
||||||
"laravel/tinker": "^2.9",
|
"laravel/tinker": "^2.9",
|
||||||
"maatwebsite/excel": "^3.1"
|
"maatwebsite/excel": "^3.1",
|
||||||
|
"openai-php/client": "^0.10.3"
|
||||||
},
|
},
|
||||||
"require-dev": {
|
"require-dev": {
|
||||||
"fakerphp/faker": "^1.23",
|
"fakerphp/faker": "^1.23",
|
||||||
@@ -56,7 +57,7 @@
|
|||||||
],
|
],
|
||||||
"dev": [
|
"dev": [
|
||||||
"Composer\\Config::disableProcessTimeout",
|
"Composer\\Config::disableProcessTimeout",
|
||||||
"npx concurrently -c \"#93c5fd,#c4b5fd,#fb7185,#fdba74\" \"php artisan serve\" \"php artisan queue:listen --tries=1\" \"php artisan pail --timeout=0\" \"npm run dev\" --names=server,queue,logs,vite"
|
"npx concurrently -c \"#93c5fd,#c4b5fd,#fb7185,#fdba74\" \"php artisan serve\" \"php artisan pail --timeout=0\" \"npm run dev\" --names=server,queue,logs,vite"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"extra": {
|
"extra": {
|
||||||
|
|||||||
300
composer.lock
generated
300
composer.lock
generated
@@ -4,7 +4,7 @@
|
|||||||
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
|
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
|
||||||
"This file is @generated automatically"
|
"This file is @generated automatically"
|
||||||
],
|
],
|
||||||
"content-hash": "52617d098d62b15c6ce8538cc8aea775",
|
"content-hash": "41bb51871a746904ab745e4095db8b46",
|
||||||
"packages": [
|
"packages": [
|
||||||
{
|
{
|
||||||
"name": "brick/math",
|
"name": "brick/math",
|
||||||
@@ -3296,6 +3296,97 @@
|
|||||||
],
|
],
|
||||||
"time": "2024-11-21T10:39:51+00:00"
|
"time": "2024-11-21T10:39:51+00:00"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"name": "openai-php/client",
|
||||||
|
"version": "v0.10.3",
|
||||||
|
"source": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/openai-php/client.git",
|
||||||
|
"reference": "4a565d145e0fb3ea1baba8fffe39d86c56b6dc2c"
|
||||||
|
},
|
||||||
|
"dist": {
|
||||||
|
"type": "zip",
|
||||||
|
"url": "https://api.github.com/repos/openai-php/client/zipball/4a565d145e0fb3ea1baba8fffe39d86c56b6dc2c",
|
||||||
|
"reference": "4a565d145e0fb3ea1baba8fffe39d86c56b6dc2c",
|
||||||
|
"shasum": ""
|
||||||
|
},
|
||||||
|
"require": {
|
||||||
|
"php": "^8.1.0",
|
||||||
|
"php-http/discovery": "^1.20.0",
|
||||||
|
"php-http/multipart-stream-builder": "^1.4.2",
|
||||||
|
"psr/http-client": "^1.0.3",
|
||||||
|
"psr/http-client-implementation": "^1.0.1",
|
||||||
|
"psr/http-factory-implementation": "*",
|
||||||
|
"psr/http-message": "^1.1.0|^2.0.0"
|
||||||
|
},
|
||||||
|
"require-dev": {
|
||||||
|
"guzzlehttp/guzzle": "^7.9.2",
|
||||||
|
"guzzlehttp/psr7": "^2.7.0",
|
||||||
|
"laravel/pint": "^1.18.1",
|
||||||
|
"mockery/mockery": "^1.6.12",
|
||||||
|
"nunomaduro/collision": "^7.11.0|^8.5.0",
|
||||||
|
"pestphp/pest": "^2.36.0|^3.5.0",
|
||||||
|
"pestphp/pest-plugin-arch": "^2.7|^3.0",
|
||||||
|
"pestphp/pest-plugin-type-coverage": "^2.8.7|^3.1.0",
|
||||||
|
"phpstan/phpstan": "^1.12.7",
|
||||||
|
"symfony/var-dumper": "^6.4.11|^7.1.5"
|
||||||
|
},
|
||||||
|
"type": "library",
|
||||||
|
"autoload": {
|
||||||
|
"files": [
|
||||||
|
"src/OpenAI.php"
|
||||||
|
],
|
||||||
|
"psr-4": {
|
||||||
|
"OpenAI\\": "src/"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"notification-url": "https://packagist.org/downloads/",
|
||||||
|
"license": [
|
||||||
|
"MIT"
|
||||||
|
],
|
||||||
|
"authors": [
|
||||||
|
{
|
||||||
|
"name": "Nuno Maduro",
|
||||||
|
"email": "enunomaduro@gmail.com"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Sandro Gehri"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"description": "OpenAI PHP is a supercharged PHP API client that allows you to interact with the Open AI API",
|
||||||
|
"keywords": [
|
||||||
|
"GPT-3",
|
||||||
|
"api",
|
||||||
|
"client",
|
||||||
|
"codex",
|
||||||
|
"dall-e",
|
||||||
|
"language",
|
||||||
|
"natural",
|
||||||
|
"openai",
|
||||||
|
"php",
|
||||||
|
"processing",
|
||||||
|
"sdk"
|
||||||
|
],
|
||||||
|
"support": {
|
||||||
|
"issues": "https://github.com/openai-php/client/issues",
|
||||||
|
"source": "https://github.com/openai-php/client/tree/v0.10.3"
|
||||||
|
},
|
||||||
|
"funding": [
|
||||||
|
{
|
||||||
|
"url": "https://www.paypal.com/paypalme/enunomaduro",
|
||||||
|
"type": "custom"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://github.com/gehrisandro",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "https://github.com/nunomaduro",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"time": "2024-11-12T20:51:16+00:00"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"name": "paragonie/constant_time_encoding",
|
"name": "paragonie/constant_time_encoding",
|
||||||
"version": "v3.0.0",
|
"version": "v3.0.0",
|
||||||
@@ -3413,6 +3504,141 @@
|
|||||||
},
|
},
|
||||||
"time": "2020-10-15T08:29:30+00:00"
|
"time": "2020-10-15T08:29:30+00:00"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"name": "php-http/discovery",
|
||||||
|
"version": "1.20.0",
|
||||||
|
"source": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/php-http/discovery.git",
|
||||||
|
"reference": "82fe4c73ef3363caed49ff8dd1539ba06044910d"
|
||||||
|
},
|
||||||
|
"dist": {
|
||||||
|
"type": "zip",
|
||||||
|
"url": "https://api.github.com/repos/php-http/discovery/zipball/82fe4c73ef3363caed49ff8dd1539ba06044910d",
|
||||||
|
"reference": "82fe4c73ef3363caed49ff8dd1539ba06044910d",
|
||||||
|
"shasum": ""
|
||||||
|
},
|
||||||
|
"require": {
|
||||||
|
"composer-plugin-api": "^1.0|^2.0",
|
||||||
|
"php": "^7.1 || ^8.0"
|
||||||
|
},
|
||||||
|
"conflict": {
|
||||||
|
"nyholm/psr7": "<1.0",
|
||||||
|
"zendframework/zend-diactoros": "*"
|
||||||
|
},
|
||||||
|
"provide": {
|
||||||
|
"php-http/async-client-implementation": "*",
|
||||||
|
"php-http/client-implementation": "*",
|
||||||
|
"psr/http-client-implementation": "*",
|
||||||
|
"psr/http-factory-implementation": "*",
|
||||||
|
"psr/http-message-implementation": "*"
|
||||||
|
},
|
||||||
|
"require-dev": {
|
||||||
|
"composer/composer": "^1.0.2|^2.0",
|
||||||
|
"graham-campbell/phpspec-skip-example-extension": "^5.0",
|
||||||
|
"php-http/httplug": "^1.0 || ^2.0",
|
||||||
|
"php-http/message-factory": "^1.0",
|
||||||
|
"phpspec/phpspec": "^5.1 || ^6.1 || ^7.3",
|
||||||
|
"sebastian/comparator": "^3.0.5 || ^4.0.8",
|
||||||
|
"symfony/phpunit-bridge": "^6.4.4 || ^7.0.1"
|
||||||
|
},
|
||||||
|
"type": "composer-plugin",
|
||||||
|
"extra": {
|
||||||
|
"class": "Http\\Discovery\\Composer\\Plugin",
|
||||||
|
"plugin-optional": true
|
||||||
|
},
|
||||||
|
"autoload": {
|
||||||
|
"psr-4": {
|
||||||
|
"Http\\Discovery\\": "src/"
|
||||||
|
},
|
||||||
|
"exclude-from-classmap": [
|
||||||
|
"src/Composer/Plugin.php"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"notification-url": "https://packagist.org/downloads/",
|
||||||
|
"license": [
|
||||||
|
"MIT"
|
||||||
|
],
|
||||||
|
"authors": [
|
||||||
|
{
|
||||||
|
"name": "Márk Sági-Kazár",
|
||||||
|
"email": "mark.sagikazar@gmail.com"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"description": "Finds and installs PSR-7, PSR-17, PSR-18 and HTTPlug implementations",
|
||||||
|
"homepage": "http://php-http.org",
|
||||||
|
"keywords": [
|
||||||
|
"adapter",
|
||||||
|
"client",
|
||||||
|
"discovery",
|
||||||
|
"factory",
|
||||||
|
"http",
|
||||||
|
"message",
|
||||||
|
"psr17",
|
||||||
|
"psr7"
|
||||||
|
],
|
||||||
|
"support": {
|
||||||
|
"issues": "https://github.com/php-http/discovery/issues",
|
||||||
|
"source": "https://github.com/php-http/discovery/tree/1.20.0"
|
||||||
|
},
|
||||||
|
"time": "2024-10-02T11:20:13+00:00"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "php-http/multipart-stream-builder",
|
||||||
|
"version": "1.4.2",
|
||||||
|
"source": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/php-http/multipart-stream-builder.git",
|
||||||
|
"reference": "10086e6de6f53489cca5ecc45b6f468604d3460e"
|
||||||
|
},
|
||||||
|
"dist": {
|
||||||
|
"type": "zip",
|
||||||
|
"url": "https://api.github.com/repos/php-http/multipart-stream-builder/zipball/10086e6de6f53489cca5ecc45b6f468604d3460e",
|
||||||
|
"reference": "10086e6de6f53489cca5ecc45b6f468604d3460e",
|
||||||
|
"shasum": ""
|
||||||
|
},
|
||||||
|
"require": {
|
||||||
|
"php": "^7.1 || ^8.0",
|
||||||
|
"php-http/discovery": "^1.15",
|
||||||
|
"psr/http-factory-implementation": "^1.0"
|
||||||
|
},
|
||||||
|
"require-dev": {
|
||||||
|
"nyholm/psr7": "^1.0",
|
||||||
|
"php-http/message": "^1.5",
|
||||||
|
"php-http/message-factory": "^1.0.2",
|
||||||
|
"phpunit/phpunit": "^7.5.15 || ^8.5 || ^9.3"
|
||||||
|
},
|
||||||
|
"type": "library",
|
||||||
|
"autoload": {
|
||||||
|
"psr-4": {
|
||||||
|
"Http\\Message\\MultipartStream\\": "src/"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"notification-url": "https://packagist.org/downloads/",
|
||||||
|
"license": [
|
||||||
|
"MIT"
|
||||||
|
],
|
||||||
|
"authors": [
|
||||||
|
{
|
||||||
|
"name": "Tobias Nyholm",
|
||||||
|
"email": "tobias.nyholm@gmail.com"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"description": "A builder class that help you create a multipart stream",
|
||||||
|
"homepage": "http://php-http.org",
|
||||||
|
"keywords": [
|
||||||
|
"factory",
|
||||||
|
"http",
|
||||||
|
"message",
|
||||||
|
"multipart stream",
|
||||||
|
"stream"
|
||||||
|
],
|
||||||
|
"support": {
|
||||||
|
"issues": "https://github.com/php-http/multipart-stream-builder/issues",
|
||||||
|
"source": "https://github.com/php-http/multipart-stream-builder/tree/1.4.2"
|
||||||
|
},
|
||||||
|
"time": "2024-09-04T13:22:54+00:00"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"name": "phpoffice/phpspreadsheet",
|
"name": "phpoffice/phpspreadsheet",
|
||||||
"version": "1.29.10",
|
"version": "1.29.10",
|
||||||
@@ -7155,74 +7381,6 @@
|
|||||||
},
|
},
|
||||||
"time": "2020-07-09T08:09:16+00:00"
|
"time": "2020-07-09T08:09:16+00:00"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"name": "ibex/crud-generator",
|
|
||||||
"version": "v2.1.2",
|
|
||||||
"source": {
|
|
||||||
"type": "git",
|
|
||||||
"url": "https://github.com/awais-vteams/laravel-crud-generator.git",
|
|
||||||
"reference": "3906f4a702c91bbe3a84d940c3021d1511834320"
|
|
||||||
},
|
|
||||||
"dist": {
|
|
||||||
"type": "zip",
|
|
||||||
"url": "https://api.github.com/repos/awais-vteams/laravel-crud-generator/zipball/3906f4a702c91bbe3a84d940c3021d1511834320",
|
|
||||||
"reference": "3906f4a702c91bbe3a84d940c3021d1511834320",
|
|
||||||
"shasum": ""
|
|
||||||
},
|
|
||||||
"require": {
|
|
||||||
"laravel/framework": "^10.30|^11.0",
|
|
||||||
"php": "^8.2"
|
|
||||||
},
|
|
||||||
"type": "library",
|
|
||||||
"extra": {
|
|
||||||
"laravel": {
|
|
||||||
"providers": [
|
|
||||||
"Ibex\\CrudGenerator\\CrudServiceProvider"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"autoload": {
|
|
||||||
"psr-4": {
|
|
||||||
"Ibex\\CrudGenerator\\": "src/"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"notification-url": "https://packagist.org/downloads/",
|
|
||||||
"license": [
|
|
||||||
"MIT"
|
|
||||||
],
|
|
||||||
"authors": [
|
|
||||||
{
|
|
||||||
"name": "M Awais",
|
|
||||||
"email": "asargodha@gmail.com"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"description": "Laravel CRUD Generator",
|
|
||||||
"keywords": [
|
|
||||||
"alpine js",
|
|
||||||
"bootstrap css",
|
|
||||||
"crud",
|
|
||||||
"crud generator",
|
|
||||||
"laravel",
|
|
||||||
"laravel crud generator",
|
|
||||||
"laravel package",
|
|
||||||
"tailwind css"
|
|
||||||
],
|
|
||||||
"support": {
|
|
||||||
"issues": "https://github.com/awais-vteams/laravel-crud-generator/issues",
|
|
||||||
"source": "https://github.com/awais-vteams/laravel-crud-generator/tree/v2.1.2"
|
|
||||||
},
|
|
||||||
"funding": [
|
|
||||||
{
|
|
||||||
"url": "https://github.com/awais-vteams",
|
|
||||||
"type": "github"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"url": "https://ko-fi.com/mawais",
|
|
||||||
"type": "ko_fi"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"time": "2024-12-09T06:01:54+00:00"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"name": "laravel/pail",
|
"name": "laravel/pail",
|
||||||
"version": "v1.2.2",
|
"version": "v1.2.2",
|
||||||
@@ -9316,12 +9474,12 @@
|
|||||||
],
|
],
|
||||||
"aliases": [],
|
"aliases": [],
|
||||||
"minimum-stability": "stable",
|
"minimum-stability": "stable",
|
||||||
"stability-flags": [],
|
"stability-flags": {},
|
||||||
"prefer-stable": true,
|
"prefer-stable": true,
|
||||||
"prefer-lowest": false,
|
"prefer-lowest": false,
|
||||||
"platform": {
|
"platform": {
|
||||||
"php": "^8.2"
|
"php": "^8.2"
|
||||||
},
|
},
|
||||||
"platform-dev": [],
|
"platform-dev": {},
|
||||||
"plugin-api-version": "2.6.0"
|
"plugin-api-version": "2.6.0"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -123,5 +123,6 @@ return [
|
|||||||
'store' => env('APP_MAINTENANCE_STORE', 'database'),
|
'store' => env('APP_MAINTENANCE_STORE', 'database'),
|
||||||
],
|
],
|
||||||
|
|
||||||
'api_url' => env('API_URL', 'http://localhost:8000')
|
'api_url' => env('API_URL', 'http://localhost:8000'),
|
||||||
|
'paginate_per_page' => 50
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -109,4 +109,10 @@ return [
|
|||||||
'table' => 'failed_jobs',
|
'table' => 'failed_jobs',
|
||||||
],
|
],
|
||||||
|
|
||||||
|
// set timeout queue
|
||||||
|
|
||||||
|
'worker' => [
|
||||||
|
'timeout' => 300
|
||||||
|
]
|
||||||
|
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -0,0 +1,32 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
use Illuminate\Database\Migrations\Migration;
|
||||||
|
use Illuminate\Database\Schema\Blueprint;
|
||||||
|
use Illuminate\Support\Facades\Schema;
|
||||||
|
|
||||||
|
return new class extends Migration
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Run the migrations.
|
||||||
|
*/
|
||||||
|
public function up(): void
|
||||||
|
{
|
||||||
|
Schema::table('bigdata_resumes', function (Blueprint $table) {
|
||||||
|
$table->integer('spatial_count')->default(0);
|
||||||
|
$table->decimal('spatial_sum', 20,2)->default(0);
|
||||||
|
$table->string('year');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverse the migrations.
|
||||||
|
*/
|
||||||
|
public function down(): void
|
||||||
|
{
|
||||||
|
Schema::table('bigdata_resumes', function (Blueprint $table) {
|
||||||
|
$table->dropColumn('spatial_count');
|
||||||
|
$table->dropColumn('spatial_sum');
|
||||||
|
$table->dropColumn('year');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
use Illuminate\Database\Migrations\Migration;
|
||||||
|
use Illuminate\Database\Schema\Blueprint;
|
||||||
|
use Illuminate\Support\Facades\Schema;
|
||||||
|
|
||||||
|
return new class extends Migration
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Run the migrations.
|
||||||
|
*/
|
||||||
|
public function up(): void
|
||||||
|
{
|
||||||
|
Schema::table('pbg_task_google_sheet', function (Blueprint $table) {
|
||||||
|
$table->string('formatted_registration_number')->nullable()->after('no_registrasi');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverse the migrations.
|
||||||
|
*/
|
||||||
|
public function down(): void
|
||||||
|
{
|
||||||
|
Schema::table('pbg_task_google_sheet', function (Blueprint $table) {
|
||||||
|
$table->dropColumn('formatted_registration_number');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -0,0 +1,38 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
use Illuminate\Database\Migrations\Migration;
|
||||||
|
use Illuminate\Database\Schema\Blueprint;
|
||||||
|
use Illuminate\Support\Facades\Schema;
|
||||||
|
|
||||||
|
return new class extends Migration
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Run the migrations.
|
||||||
|
*/
|
||||||
|
public function up(): void
|
||||||
|
{
|
||||||
|
Schema::table('bigdata_resumes', function (Blueprint $table) {
|
||||||
|
$table->integer('waiting_click_dpmptsp_count')->default(0);
|
||||||
|
$table->decimal('waiting_click_dpmptsp_sum', 20,2)->default(0);
|
||||||
|
$table->integer('issuance_realization_pbg_count')->default(0);
|
||||||
|
$table->decimal('issuance_realization_pbg_sum', 20,2)->default(0);
|
||||||
|
$table->integer('process_in_technical_office_count')->default(0);
|
||||||
|
$table->decimal('process_in_technical_office_sum', 20,2)->default(0);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverse the migrations.
|
||||||
|
*/
|
||||||
|
public function down(): void
|
||||||
|
{
|
||||||
|
Schema::table('bigdata_resumes', function (Blueprint $table) {
|
||||||
|
$table->dropColumn('waiting_click_dpmptsp_count');
|
||||||
|
$table->dropColumn('waiting_click_dpmptsp_sum');
|
||||||
|
$table->dropColumn('issuance_realization_pbg_count');
|
||||||
|
$table->dropColumn('issuance_realization_pbg_sum');
|
||||||
|
$table->dropColumn('process_in_technical_office_count');
|
||||||
|
$table->dropColumn('process_in_technical_office_sum');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -0,0 +1,48 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
use Illuminate\Database\Migrations\Migration;
|
||||||
|
use Illuminate\Database\Schema\Blueprint;
|
||||||
|
use Illuminate\Support\Facades\Schema;
|
||||||
|
|
||||||
|
return new class extends Migration
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Run the migrations.
|
||||||
|
*/
|
||||||
|
public function up(): void
|
||||||
|
{
|
||||||
|
Schema::create('task_assignments', function (Blueprint $table) {
|
||||||
|
$table->id(); // Auto-increment primary key
|
||||||
|
|
||||||
|
// Foreign key reference to pbg_tasks (uid column)
|
||||||
|
$table->string('pbg_task_uid');
|
||||||
|
$table->foreign('pbg_task_uid')->references('uuid')->on('pbg_task')->onDelete('cascade');
|
||||||
|
|
||||||
|
$table->unsignedBigInteger('user_id'); // Reference to users table
|
||||||
|
$table->string('name');
|
||||||
|
$table->string('username')->unique();
|
||||||
|
$table->string('email')->unique();
|
||||||
|
$table->string('phone_number')->nullable();
|
||||||
|
$table->unsignedInteger('role'); // Assuming role is numeric
|
||||||
|
$table->string('role_name');
|
||||||
|
$table->boolean('is_active')->default(true);
|
||||||
|
$table->json('file')->nullable(); // Store as JSON if 'file' is an array
|
||||||
|
$table->string('expertise')->nullable();
|
||||||
|
$table->string('experience')->nullable();
|
||||||
|
$table->boolean('is_verif')->default(false);
|
||||||
|
$table->string('uid')->unique();
|
||||||
|
$table->unsignedTinyInteger('status')->default(0); // Assuming status is a small integer
|
||||||
|
$table->string('status_name')->nullable();
|
||||||
|
$table->text('note')->nullable();
|
||||||
|
$table->timestamps();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverse the migrations.
|
||||||
|
*/
|
||||||
|
public function down(): void
|
||||||
|
{
|
||||||
|
Schema::dropIfExists('task_assignments');
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -71,9 +71,16 @@ class UsersRoleMenuSeeder extends Seeder
|
|||||||
[
|
[
|
||||||
"name" => "Laporan",
|
"name" => "Laporan",
|
||||||
"url" => "/laporan",
|
"url" => "/laporan",
|
||||||
"icon" => "mingcute:task-line",
|
"icon" => "mingcute:report-line",
|
||||||
"parent_id" => null,
|
"parent_id" => null,
|
||||||
"sort_order" => 6,
|
"sort_order" => 6,
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"name" => "Neng Bedas",
|
||||||
|
"url" => "/chat",
|
||||||
|
"icon" => "mingcute:wechat-line",
|
||||||
|
"parent_id" => null,
|
||||||
|
"sort_order" => 7,
|
||||||
]
|
]
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -92,6 +99,7 @@ class UsersRoleMenuSeeder extends Seeder
|
|||||||
$dataSettings = Menu::where('name', 'Data Settings')->first();
|
$dataSettings = Menu::where('name', 'Data Settings')->first();
|
||||||
$data = Menu::where('name', 'Data')->first();
|
$data = Menu::where('name', 'Data')->first();
|
||||||
$laporan = Menu::where('name', 'Laporan')->first();
|
$laporan = Menu::where('name', 'Laporan')->first();
|
||||||
|
$chat_bedas = Menu::where('name', 'Neng Bedas')->first();
|
||||||
|
|
||||||
// create children menu
|
// create children menu
|
||||||
$children_menus = [
|
$children_menus = [
|
||||||
@@ -167,7 +175,7 @@ class UsersRoleMenuSeeder extends Seeder
|
|||||||
],
|
],
|
||||||
[
|
[
|
||||||
"name" => "Reklame",
|
"name" => "Reklame",
|
||||||
"url" => "advertisements.index",
|
"url" => "web.advertisements.index",
|
||||||
"icon" => null,
|
"icon" => null,
|
||||||
"parent_id" => $data->id,
|
"parent_id" => $data->id,
|
||||||
"sort_order" => 2,
|
"sort_order" => 2,
|
||||||
@@ -181,21 +189,21 @@ class UsersRoleMenuSeeder extends Seeder
|
|||||||
],
|
],
|
||||||
[
|
[
|
||||||
"name" => "UMKM",
|
"name" => "UMKM",
|
||||||
"url" => "umkm.index",
|
"url" => "web-umkm.index",
|
||||||
"icon" => null,
|
"icon" => null,
|
||||||
"parent_id" => $data->id,
|
"parent_id" => $data->id,
|
||||||
"sort_order" => 4,
|
"sort_order" => 4,
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"name" => "Pariwisata",
|
"name" => "Pariwisata",
|
||||||
"url" => "tourisms.index",
|
"url" => "web-tourisms.index",
|
||||||
"icon" => null,
|
"icon" => null,
|
||||||
"parent_id" => $data->id,
|
"parent_id" => $data->id,
|
||||||
"sort_order" => 5,
|
"sort_order" => 5,
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
"name" => "Tata Ruang",
|
"name" => "Tata Ruang",
|
||||||
"url" => "spatial-plannings.index",
|
"url" => "web-spatial-plannings.index",
|
||||||
"icon" => null,
|
"icon" => null,
|
||||||
"parent_id" => $data->id,
|
"parent_id" => $data->id,
|
||||||
"sort_order" => 6,
|
"sort_order" => 6,
|
||||||
@@ -209,11 +217,25 @@ class UsersRoleMenuSeeder extends Seeder
|
|||||||
],
|
],
|
||||||
[
|
[
|
||||||
"name" => "Lap Pariwisata",
|
"name" => "Lap Pariwisata",
|
||||||
"url" => "tourisms.index",
|
"url" => "tourisms-report.index",
|
||||||
"icon" => null,
|
"icon" => null,
|
||||||
"parent_id" => $laporan->id,
|
"parent_id" => $laporan->id,
|
||||||
"sort_order" => 1,
|
"sort_order" => 1,
|
||||||
],
|
],
|
||||||
|
[
|
||||||
|
"name" => "Lap Pimpinan",
|
||||||
|
"url" => "bigdata-resumes",
|
||||||
|
"icon" => null,
|
||||||
|
"parent_id" => $laporan->id,
|
||||||
|
"sort_order" => 2,
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"name" => "Chat",
|
||||||
|
"url" => "main-chatbot.index",
|
||||||
|
"icon" => null,
|
||||||
|
"parent_id" => $chat_bedas->id,
|
||||||
|
"sort_order" => 1,
|
||||||
|
],
|
||||||
];
|
];
|
||||||
|
|
||||||
foreach ($children_menus as $child_menu) {
|
foreach ($children_menus as $child_menu) {
|
||||||
@@ -237,6 +259,8 @@ class UsersRoleMenuSeeder extends Seeder
|
|||||||
$spatial_plannings = Menu::where('name', 'Tata Ruang')->first();
|
$spatial_plannings = Menu::where('name', 'Tata Ruang')->first();
|
||||||
$pdam = Menu::where('name', 'PDAM')->first();
|
$pdam = Menu::where('name', 'PDAM')->first();
|
||||||
$peta = Menu::where('name', 'PETA')->first();
|
$peta = Menu::where('name', 'PETA')->first();
|
||||||
|
$bigdata_resume = Menu::where('name', 'Lap Pimpinan')->first();
|
||||||
|
$chatbot = Menu::where('name', 'Chat')->first();
|
||||||
|
|
||||||
// Superadmin gets all menus
|
// Superadmin gets all menus
|
||||||
$superadmin->menus()->sync([
|
$superadmin->menus()->sync([
|
||||||
@@ -247,6 +271,7 @@ class UsersRoleMenuSeeder extends Seeder
|
|||||||
$dataSettings->id => ["allow_show" => true, "allow_create" => false, "allow_update" => false, "allow_destroy" => false],
|
$dataSettings->id => ["allow_show" => true, "allow_create" => false, "allow_update" => false, "allow_destroy" => false],
|
||||||
$data->id => ["allow_show" => true, "allow_create" => false, "allow_update" => false, "allow_destroy" => false],
|
$data->id => ["allow_show" => true, "allow_create" => false, "allow_update" => false, "allow_destroy" => false],
|
||||||
$laporan->id => ["allow_show" => true, "allow_create" => false, "allow_update" => false, "allow_destroy" => false],
|
$laporan->id => ["allow_show" => true, "allow_create" => false, "allow_update" => false, "allow_destroy" => false],
|
||||||
|
$chat_bedas->id => ["allow_show" => true, "allow_create" => false, "allow_update" => false, "allow_destroy" => false],
|
||||||
// children
|
// children
|
||||||
$dashboard_pimpinan->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
$dashboard_pimpinan->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
||||||
$dashboard_pbg->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
$dashboard_pbg->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
||||||
@@ -265,6 +290,8 @@ class UsersRoleMenuSeeder extends Seeder
|
|||||||
$spatial_plannings->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
$spatial_plannings->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
||||||
$pdam->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
$pdam->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
||||||
$peta->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
$peta->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
||||||
|
$bigdata_resume->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
||||||
|
$chatbot->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// Admin gets limited menus
|
// Admin gets limited menus
|
||||||
@@ -279,6 +306,7 @@ class UsersRoleMenuSeeder extends Seeder
|
|||||||
$dashboard->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
$dashboard->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
||||||
$data->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
$data->id => ["allow_show" => true, "allow_create" => true, "allow_update" => true, "allow_destroy" => true],
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// Attach User to role super admin
|
// Attach User to role super admin
|
||||||
User::findOrFail(1)->roles()->sync([$superadmin->id]);
|
User::findOrFail(1)->roles()->sync([$superadmin->id]);
|
||||||
}
|
}
|
||||||
|
|||||||
21
database/view_query/v_advertisements.sql
Normal file
21
database/view_query/v_advertisements.sql
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
CREATE VIEW v_advertisements AS
|
||||||
|
SELECT
|
||||||
|
a.no,
|
||||||
|
a.business_name,
|
||||||
|
a.npwpd,
|
||||||
|
a.advertisement_type,
|
||||||
|
a.advertisement_content,
|
||||||
|
a.business_address,
|
||||||
|
a.advertisement_location,
|
||||||
|
v.village_name AS village_name,
|
||||||
|
d.district_name AS district_name,
|
||||||
|
a.length,
|
||||||
|
a.width,
|
||||||
|
a.viewing_angle,
|
||||||
|
a.face,
|
||||||
|
a.area,
|
||||||
|
a.angle,
|
||||||
|
a.contact
|
||||||
|
FROM advertisements a
|
||||||
|
JOIN villages v ON a.village_code = v.village_code
|
||||||
|
JOIN districts d ON a.district_code = d.district_code;
|
||||||
8
database/view_query/v_tourism_base_kbli.sql
Normal file
8
database/view_query/v_tourism_base_kbli.sql
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
CREATE VIEW v_tourisms_based_kbli AS
|
||||||
|
SELECT kbli_title, total_records
|
||||||
|
FROM (
|
||||||
|
SELECT kbli, kbli_title, COUNT(*) AS total_records
|
||||||
|
FROM tourisms
|
||||||
|
GROUP BY kbli, kbli_title
|
||||||
|
) AS subquery
|
||||||
|
ORDER BY total_records DESC;
|
||||||
29
database/view_query/v_tourisms.sql
Normal file
29
database/view_query/v_tourisms.sql
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
CREATE VIEW v_tourisms AS
|
||||||
|
SELECT
|
||||||
|
t.project_id,
|
||||||
|
t.project_type_id,
|
||||||
|
t.nib,
|
||||||
|
t.business_name,
|
||||||
|
t.oss_publication_date,
|
||||||
|
t.investment_status_description,
|
||||||
|
t.business_form,
|
||||||
|
t.project_risk,
|
||||||
|
t.project_name,
|
||||||
|
t.business_scale,
|
||||||
|
t.business_address,
|
||||||
|
v.village_name as village_name,
|
||||||
|
d.district_name as district_name,
|
||||||
|
t.longitude,
|
||||||
|
t.latitude,
|
||||||
|
t.project_submission_date,
|
||||||
|
t.kbli_title,
|
||||||
|
t.supervisory_sector,
|
||||||
|
t.user_name,
|
||||||
|
t.email,
|
||||||
|
t.contact,
|
||||||
|
t.land_area_in_m2,
|
||||||
|
t.investment_amount,
|
||||||
|
t.tki
|
||||||
|
FROM tourisms t
|
||||||
|
JOIN villages v on t.village_code = v.village_code
|
||||||
|
JOIN districts d on t.district_code = d.district_code;
|
||||||
28
database/view_query/v_umkms.sql
Normal file
28
database/view_query/v_umkms.sql
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
CREATE VIEW v_umkms AS
|
||||||
|
SELECT
|
||||||
|
u.business_address,
|
||||||
|
u.business_contact,
|
||||||
|
u.business_desc,
|
||||||
|
bf.business_form,
|
||||||
|
u.business_id_number,
|
||||||
|
u.business_name,
|
||||||
|
bs.business_scale,
|
||||||
|
u.business_type,
|
||||||
|
u.created_at,
|
||||||
|
d.district_name,
|
||||||
|
u.land_area,
|
||||||
|
u.number_of_employee,
|
||||||
|
u.owner_address,
|
||||||
|
u.owner_contact,
|
||||||
|
u.owner_id,
|
||||||
|
u.owner_name,
|
||||||
|
ps.permit_status,
|
||||||
|
u.revenue,
|
||||||
|
u.updated_at,
|
||||||
|
v.village_name
|
||||||
|
FROM umkms u
|
||||||
|
JOIN business_form bf on u.business_form_id = bf.id
|
||||||
|
JOIN permit_status ps on u.permit_status_id = ps.id
|
||||||
|
JOIn business_scale bs on u.business_scale_id = bs.id
|
||||||
|
JOIN villages v on u.village_code = v.village_code
|
||||||
|
JOIN districts d on u.district_code = v.district_code;
|
||||||
34
deploy.sh
Executable file
34
deploy.sh
Executable file
@@ -0,0 +1,34 @@
|
|||||||
|
GIT_BRANCH="dev"
|
||||||
|
PHP_VERSION="php8.3"
|
||||||
|
|
||||||
|
echo "🚀 Starting deployment..."
|
||||||
|
php artisan down
|
||||||
|
|
||||||
|
echo "📥 Pulling latest changes from Git..."
|
||||||
|
git fetch origin $GIT_BRANCH
|
||||||
|
git reset --hard origin/$GIT_BRANCH
|
||||||
|
git pull origin $GIT_BRANCH
|
||||||
|
|
||||||
|
echo "⚡ Installing NPM dependencies and building assets..."
|
||||||
|
npm ci --no-audit --no-fund
|
||||||
|
npm run build
|
||||||
|
|
||||||
|
echo "📦 Installing composer dependencies..."
|
||||||
|
COMPOSER_ALLOW_SUPERUSER=1 composer install --no-interaction --optimize-autoloader
|
||||||
|
|
||||||
|
echo "🗄️ Running migrations..."
|
||||||
|
php artisan migrate --force
|
||||||
|
|
||||||
|
echo "⚡ Optimizing application..."
|
||||||
|
php artisan optimize:clear
|
||||||
|
|
||||||
|
echo "🔄 Restarting PHP service..."
|
||||||
|
systemctl restart $PHP_VERSION-fpm
|
||||||
|
|
||||||
|
echo "🔁 Restarting Supervisor queue workers..."
|
||||||
|
supervisorctl stop all
|
||||||
|
supervisorctl reload
|
||||||
|
supervisorctl start all
|
||||||
|
|
||||||
|
php artisan up
|
||||||
|
echo "🚀 Deployment completed successfully!"
|
||||||
9
package-lock.json
generated
9
package-lock.json
generated
@@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"name": "sibedas-pbg",
|
"name": "sibedas-pbg-web",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
@@ -15,6 +15,7 @@
|
|||||||
"gridjs": "^5.1.0",
|
"gridjs": "^5.1.0",
|
||||||
"iconify-icon": "^2.1.0",
|
"iconify-icon": "^2.1.0",
|
||||||
"jsvectormap": "^1.5.1",
|
"jsvectormap": "^1.5.1",
|
||||||
|
"leaflet": "^1.9.4",
|
||||||
"moment": "^2.29.4",
|
"moment": "^2.29.4",
|
||||||
"node-waves": "^0.7.6",
|
"node-waves": "^0.7.6",
|
||||||
"quill": "^1.3.7",
|
"quill": "^1.3.7",
|
||||||
@@ -1467,6 +1468,12 @@
|
|||||||
"vite": "^5.0.0 || ^6.0.0"
|
"vite": "^5.0.0 || ^6.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/leaflet": {
|
||||||
|
"version": "1.9.4",
|
||||||
|
"resolved": "https://registry.npmjs.org/leaflet/-/leaflet-1.9.4.tgz",
|
||||||
|
"integrity": "sha512-nxS1ynzJOmOlHp+iL3FyWqK89GtNL8U8rvlMOsQdTTssxZwCXh8N2NB3GDQOL+YR3XnWyZAxwQixURb+FA74PA==",
|
||||||
|
"license": "BSD-2-Clause"
|
||||||
|
},
|
||||||
"node_modules/lilconfig": {
|
"node_modules/lilconfig": {
|
||||||
"version": "2.1.0",
|
"version": "2.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-2.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-2.1.0.tgz",
|
||||||
|
|||||||
@@ -26,6 +26,7 @@
|
|||||||
"gridjs": "^5.1.0",
|
"gridjs": "^5.1.0",
|
||||||
"iconify-icon": "^2.1.0",
|
"iconify-icon": "^2.1.0",
|
||||||
"jsvectormap": "^1.5.1",
|
"jsvectormap": "^1.5.1",
|
||||||
|
"leaflet": "^1.9.4",
|
||||||
"moment": "^2.29.4",
|
"moment": "^2.29.4",
|
||||||
"node-waves": "^0.7.6",
|
"node-waves": "^0.7.6",
|
||||||
"quill": "^1.3.7",
|
"quill": "^1.3.7",
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
BIN
public/images/iconchatbot.jpeg
Normal file
BIN
public/images/iconchatbot.jpeg
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 177 KiB |
@@ -5,6 +5,7 @@ use Illuminate\Http\Request;
|
|||||||
define('LARAVEL_START', microtime(true));
|
define('LARAVEL_START', microtime(true));
|
||||||
|
|
||||||
ini_set('max_execution_time',14400);
|
ini_set('max_execution_time',14400);
|
||||||
|
ini_set('memory_limit', '2G');
|
||||||
|
|
||||||
// Determine if the application is in maintenance mode...
|
// Determine if the application is in maintenance mode...
|
||||||
if (file_exists($maintenance = __DIR__.'/../storage/framework/maintenance.php')) {
|
if (file_exists($maintenance = __DIR__.'/../storage/framework/maintenance.php')) {
|
||||||
|
|||||||
11
public/templates/contentTemplatePrompt.json
Normal file
11
public/templates/contentTemplatePrompt.json
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
{
|
||||||
|
"RETRIBUTION": {
|
||||||
|
"prompt": "You are a MariaDB SQL expert. Your task is to generate an efficient and optimized SQL query based on user input.\n\n The query should retrieve data from the view table `v_pbg_task_with_retributions`, specifically selecting the following columns:\n\n - nilai_retribusi_bangunan\n - land_certificate_phase\n - due_date\n - consultation_type\n - function_type\n - slf_status_name\n - slf_status\n - status_name\n - status\n - address\n - document_number\n - registration_number\n - application_type_name\n - application_type\n - owner_name\n - name\n\n Ensure the query is well-structured, uses best indexing practices, and avoids performance bottlenecks.\n\n Consider the following context: \"$mainContent\".\n\n Always return only the SQL query without any additional explanation.\n\n The query should include `LIMIT 5` to restrict the results."
|
||||||
|
},
|
||||||
|
"DOCUMENT VALIDATION": {
|
||||||
|
"prompt": "You are a MariaDB SQL expert. Your task is to generate an efficient and optimized SQL query based on user input.\n\n The query should retrieve data from the view table `pbg_task`, specifically selecting the following columns:\n\n - name\n - owner_name\n - application_type\n - application_type_name\n - registration_number\n - document_number\n - address\n - status_name\n - slf_status\n - slf_status_name\n - function_type\n - consultation_type\n - function_type\n - consultation_type\n - due_date\n - land_certificate_phase\n\n Ensure the query is well-structured, uses best indexing practices, and avoids performance bottlenecks.\n\n Consider the following context: \"$mainContent\".\n\n Always return only the SQL query without any additional explanation.\n\n The query should include `LIMIT 5` to restrict the results."
|
||||||
|
},
|
||||||
|
"DATA SUMMARY": {
|
||||||
|
"prompt": "You are a MariaDB SQL expert. Your task is to generate an efficient and optimized SQL query based on user input.\n\n The query should retrieve data from the view table `bigdata_resumes`, specifically selecting the following columns:\n\n - potention_count\n - potention_sum\n - non_verified_count\n - non_verified_sum\n - verified_sum\n - verified_count\n - business_count\n - business_sum\n - non_business_count\n - non_business_sum\n - spatial_count\n - spatial_sum\n - updated_at\n\n Ensure the query is well-structured, uses best indexing practices, and avoids performance bottlenecks.\n\n Consider the following context: \"$mainContent\".\n\n Always return only the SQL query without any additional explanation.\n\n The query should include `LIMIT 5` to restrict the results."
|
||||||
|
}
|
||||||
|
}
|
||||||
169
public/templates/table_config.json
Normal file
169
public/templates/table_config.json
Normal file
@@ -0,0 +1,169 @@
|
|||||||
|
{
|
||||||
|
"reklame": {
|
||||||
|
"table_name": "v_advertisements",
|
||||||
|
"list_column": [
|
||||||
|
"no",
|
||||||
|
"business_name",
|
||||||
|
"npwpd",
|
||||||
|
"advertisement_type",
|
||||||
|
"advertisement_content",
|
||||||
|
"business_address",
|
||||||
|
"advertisement_location",
|
||||||
|
"village_name",
|
||||||
|
"district_name",
|
||||||
|
"length",
|
||||||
|
"width",
|
||||||
|
"viewing_angle",
|
||||||
|
"face",
|
||||||
|
"area",
|
||||||
|
"angle",
|
||||||
|
"contact"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"business_or_industries": {
|
||||||
|
"table_name": "business_or_industries",
|
||||||
|
"list_column": [
|
||||||
|
"nama_kecamatan",
|
||||||
|
"nama_kelurahan",
|
||||||
|
"nop",
|
||||||
|
"nama_wajib_pajak",
|
||||||
|
"alamat_wajib_pajak",
|
||||||
|
"alamat_objek_pajak",
|
||||||
|
"luas_bumi",
|
||||||
|
"luas_bangunan",
|
||||||
|
"njop_bumi",
|
||||||
|
"njop_bangunan",
|
||||||
|
"ketetapan",
|
||||||
|
"tahun_pajak",
|
||||||
|
"created_at",
|
||||||
|
"updated_at"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"customers": {
|
||||||
|
"table_name": "customers",
|
||||||
|
"list_column": [
|
||||||
|
"nomor_pelanggan",
|
||||||
|
"kota_pelayanan",
|
||||||
|
"nama",
|
||||||
|
"alamat",
|
||||||
|
"latitude",
|
||||||
|
"longitude",
|
||||||
|
"created_at",
|
||||||
|
"updated_at"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"pbg": {
|
||||||
|
"table_name": "pbg_task",
|
||||||
|
"list_column": [
|
||||||
|
"uuid",
|
||||||
|
"name",
|
||||||
|
"owner_name",
|
||||||
|
"application_type",
|
||||||
|
"application_type_name",
|
||||||
|
"condition",
|
||||||
|
"registration_number",
|
||||||
|
"document_number",
|
||||||
|
"address",
|
||||||
|
"status_name",
|
||||||
|
"slf_status_name",
|
||||||
|
"function_type",
|
||||||
|
"consultation_type",
|
||||||
|
"due_date",
|
||||||
|
"land_certificate_phase",
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
"task_created_at"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"retribusi": {
|
||||||
|
"table_name": "v_pbg_task_with_retributions",
|
||||||
|
"list_column": [
|
||||||
|
"uuid",
|
||||||
|
"name",
|
||||||
|
"owner_name",
|
||||||
|
"application_type",
|
||||||
|
"application_type_name",
|
||||||
|
"condition",
|
||||||
|
"registration_number",
|
||||||
|
"document_number",
|
||||||
|
"address",
|
||||||
|
"status_name",
|
||||||
|
"slf_status_name",
|
||||||
|
"consultation_type",
|
||||||
|
"due_date",
|
||||||
|
"land_certificate_phase",
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
"task_created_at",
|
||||||
|
"nilai_retribusi_bangunan"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"spatial_plannings": {
|
||||||
|
"table_name": "spatial_plannings",
|
||||||
|
"list_column": [
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
"name",
|
||||||
|
"kbli",
|
||||||
|
"activities",
|
||||||
|
"area",
|
||||||
|
"location",
|
||||||
|
"number",
|
||||||
|
"date"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"tourisms": {
|
||||||
|
"table_name": "v_tourisms",
|
||||||
|
"list_column": [
|
||||||
|
"project_id",
|
||||||
|
"project_type_id",
|
||||||
|
"nib",
|
||||||
|
"business_name",
|
||||||
|
"oss_publication_date",
|
||||||
|
"investment_status_description",
|
||||||
|
"business_form",
|
||||||
|
"project_risk",
|
||||||
|
"project_name",
|
||||||
|
"business_scale",
|
||||||
|
"business_address",
|
||||||
|
"village_name",
|
||||||
|
"district_name",
|
||||||
|
"longitude",
|
||||||
|
"latitude",
|
||||||
|
"project_submission_date",
|
||||||
|
"kbli_title",
|
||||||
|
"supervisory_sector",
|
||||||
|
"user_name",
|
||||||
|
"email",
|
||||||
|
"contact",
|
||||||
|
"land_area_in_m2",
|
||||||
|
"investment_amount",
|
||||||
|
"tki"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"umkms": {
|
||||||
|
"table_name": "v_umkms",
|
||||||
|
"list_column": [
|
||||||
|
"business_address",
|
||||||
|
"business_contact",
|
||||||
|
"business_desc",
|
||||||
|
"business_form",
|
||||||
|
"business_id_number",
|
||||||
|
"business_name",
|
||||||
|
"business_scale",
|
||||||
|
"business_type",
|
||||||
|
"created_at",
|
||||||
|
"district_name",
|
||||||
|
"land_area",
|
||||||
|
"number_of_employee",
|
||||||
|
"owner_address",
|
||||||
|
"owner_contact",
|
||||||
|
"owner_id",
|
||||||
|
"owner_name",
|
||||||
|
"permit_status",
|
||||||
|
"revenue",
|
||||||
|
"updated_at",
|
||||||
|
"village_name"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,6 +2,30 @@ import bootstrap from "bootstrap/dist/js/bootstrap";
|
|||||||
window.bootstrap = bootstrap;
|
window.bootstrap = bootstrap;
|
||||||
import "iconify-icon";
|
import "iconify-icon";
|
||||||
import "simplebar/dist/simplebar";
|
import "simplebar/dist/simplebar";
|
||||||
|
// import flatpickr from "flatpickr";
|
||||||
|
// import "flatpickr/dist/flatpickr.min.css";
|
||||||
|
|
||||||
|
// class InitDatePicker {
|
||||||
|
// constructor(selector = ".datepicker") {
|
||||||
|
// this.selector = selector;
|
||||||
|
// }
|
||||||
|
// init() {
|
||||||
|
// const elements = document.querySelectorAll(this.selector);
|
||||||
|
// if (elements.length === 0) return; // Skip if no elements found
|
||||||
|
|
||||||
|
// const today = new Date();
|
||||||
|
// const minYear = today.getFullYear() - 5;
|
||||||
|
|
||||||
|
// elements.forEach((element) => {
|
||||||
|
// flatpickr(element, {
|
||||||
|
// enableTime: false,
|
||||||
|
// dateFormat: "Y-m-d",
|
||||||
|
// minDate: `${minYear}-01-01`,
|
||||||
|
// maxDate: today,
|
||||||
|
// });
|
||||||
|
// });
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
class Components {
|
class Components {
|
||||||
initBootstrapComponents() {
|
initBootstrapComponents() {
|
||||||
@@ -107,6 +131,7 @@ class FormValidation {
|
|||||||
}
|
}
|
||||||
document.addEventListener("DOMContentLoaded", function (e) {
|
document.addEventListener("DOMContentLoaded", function (e) {
|
||||||
new Components().init(), new FormValidation().init();
|
new Components().init(), new FormValidation().init();
|
||||||
|
// new InitDatePicker().init();
|
||||||
});
|
});
|
||||||
class ThemeLayout {
|
class ThemeLayout {
|
||||||
constructor() {
|
constructor() {
|
||||||
|
|||||||
172
resources/js/bigdata-resumes/index.js
Normal file
172
resources/js/bigdata-resumes/index.js
Normal file
@@ -0,0 +1,172 @@
|
|||||||
|
import { Grid } from "gridjs/dist/gridjs.umd.js";
|
||||||
|
import gridjs from "gridjs/dist/gridjs.umd.js";
|
||||||
|
import "gridjs/dist/gridjs.umd.js";
|
||||||
|
import GlobalConfig, { addThousandSeparators } from "../global-config.js";
|
||||||
|
import Swal from "sweetalert2";
|
||||||
|
import moment from "moment";
|
||||||
|
|
||||||
|
class BigdataResume {
|
||||||
|
constructor() {
|
||||||
|
this.toastMessage = document.getElementById("toast-message");
|
||||||
|
this.toastElement = document.getElementById("toastNotification");
|
||||||
|
this.toast = new bootstrap.Toast(this.toastElement);
|
||||||
|
this.table = null;
|
||||||
|
|
||||||
|
// Initialize functions
|
||||||
|
this.initTableDataSettings();
|
||||||
|
// this.initEvents();
|
||||||
|
}
|
||||||
|
initEvents() {
|
||||||
|
document.body.addEventListener("click", async (event) => {
|
||||||
|
const deleteButton = event.target.closest(
|
||||||
|
".btn-delete-data-settings"
|
||||||
|
);
|
||||||
|
if (deleteButton) {
|
||||||
|
event.preventDefault();
|
||||||
|
await this.handleDelete(deleteButton);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
initTableDataSettings() {
|
||||||
|
let tableContainer = document.getElementById("table-bigdata-resumes");
|
||||||
|
this.table = new Grid({
|
||||||
|
columns: [
|
||||||
|
{ name: "ID" },
|
||||||
|
{ name: "Jumlah Potensi" },
|
||||||
|
{ name: "Total Potensi" },
|
||||||
|
{ name: "Jumlah Berkas Belum Terverifikasi" },
|
||||||
|
{ name: "Total Berkas Belum Terverifikasi" },
|
||||||
|
{ name: "Jumlah Berkas Terverifikasi" },
|
||||||
|
{ name: "Total Berkas Terverifikasi" },
|
||||||
|
{ name: "Jumlah Usaha" },
|
||||||
|
{ name: "Total Usaha" },
|
||||||
|
{ name: "Jumlah Non Usaha" },
|
||||||
|
{ name: "Total Non Usaha" },
|
||||||
|
{ name: "Jumlah Tata Ruang" },
|
||||||
|
{ name: "Total Tata Ruang" },
|
||||||
|
{ name: "Jumlah Menunggu Klik DPMPTSP" },
|
||||||
|
{ name: "Total Menunggu Klik DPMPTSP" },
|
||||||
|
{ name: "Jumlah Realisasi Terbit PBG" },
|
||||||
|
{ name: "Total Realisasi Terbit PBG" },
|
||||||
|
{ name: "Jumlah Proses Dinas Teknis" },
|
||||||
|
{ name: "Total Proses Dinas Teknis" },
|
||||||
|
{
|
||||||
|
name: "Created",
|
||||||
|
attributes: { style: "width: 200px; white-space: nowrap;" }, // Set width dynamically
|
||||||
|
},
|
||||||
|
],
|
||||||
|
pagination: {
|
||||||
|
limit: 50,
|
||||||
|
server: {
|
||||||
|
url: (prev, page) =>
|
||||||
|
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
||||||
|
page + 1
|
||||||
|
}`,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
sort: true,
|
||||||
|
// search: {
|
||||||
|
// server: {
|
||||||
|
// url: (prev, keyword) => `${prev}?search=${keyword}`,
|
||||||
|
// },
|
||||||
|
// },
|
||||||
|
server: {
|
||||||
|
url: `${GlobalConfig.apiHost}/api/bigdata-report`,
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${document
|
||||||
|
.querySelector('meta[name="api-token"]')
|
||||||
|
.getAttribute("content")}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
then: (data) => {
|
||||||
|
return data.data.map((item) => [
|
||||||
|
item.id,
|
||||||
|
item.potention_count,
|
||||||
|
addThousandSeparators(item.potention_sum),
|
||||||
|
item.non_verified_count,
|
||||||
|
addThousandSeparators(item.non_verified_sum),
|
||||||
|
item.verified_count,
|
||||||
|
addThousandSeparators(item.verified_sum),
|
||||||
|
item.business_count,
|
||||||
|
addThousandSeparators(item.business_sum),
|
||||||
|
item.non_business_count,
|
||||||
|
addThousandSeparators(item.non_business_sum),
|
||||||
|
item.spatial_count,
|
||||||
|
addThousandSeparators(item.spatial_sum),
|
||||||
|
item.waiting_click_dpmptsp_count,
|
||||||
|
addThousandSeparators(item.waiting_click_dpmptsp_sum),
|
||||||
|
item.issuance_realization_pbg_count,
|
||||||
|
addThousandSeparators(
|
||||||
|
item.issuance_realization_pbg_sum
|
||||||
|
),
|
||||||
|
item.process_in_technical_office_count,
|
||||||
|
addThousandSeparators(
|
||||||
|
item.process_in_technical_office_sum
|
||||||
|
),
|
||||||
|
moment(item.created_at).format("YYYY-MM-DD H:mm:ss"),
|
||||||
|
]);
|
||||||
|
},
|
||||||
|
total: (data) => data.total,
|
||||||
|
},
|
||||||
|
width: "auto",
|
||||||
|
}).render(tableContainer);
|
||||||
|
}
|
||||||
|
handleSearch() {}
|
||||||
|
async handleDelete(deleteButton) {
|
||||||
|
const id = deleteButton.getAttribute("data-id");
|
||||||
|
|
||||||
|
const result = await Swal.fire({
|
||||||
|
title: "Are you sure?",
|
||||||
|
text: "You won't be able to revert this!",
|
||||||
|
icon: "warning",
|
||||||
|
showCancelButton: true,
|
||||||
|
confirmButtonColor: "#3085d6",
|
||||||
|
cancelButtonColor: "#d33",
|
||||||
|
confirmButtonText: "Yes, delete it!",
|
||||||
|
});
|
||||||
|
|
||||||
|
if (result.isConfirmed) {
|
||||||
|
try {
|
||||||
|
let response = await fetch(
|
||||||
|
`${GlobalConfig.apiHost}/api/data-settings/${id}`,
|
||||||
|
{
|
||||||
|
method: "DELETE",
|
||||||
|
credentials: "include",
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${document
|
||||||
|
.querySelector('meta[name="api-token"]')
|
||||||
|
.getAttribute("content")}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (response.ok) {
|
||||||
|
let result = await response.json();
|
||||||
|
this.toastMessage.innerText =
|
||||||
|
result.message || "Deleted successfully!";
|
||||||
|
this.toast.show();
|
||||||
|
|
||||||
|
// Refresh Grid.js table
|
||||||
|
if (typeof this.table !== "undefined") {
|
||||||
|
this.table.updateConfig({}).forceRender();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
let error = await response.json();
|
||||||
|
console.error("Delete failed:", error);
|
||||||
|
this.toastMessage.innerText =
|
||||||
|
error.message || "Delete failed!";
|
||||||
|
this.toast.show();
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error deleting item:", error);
|
||||||
|
this.toastMessage.innerText = "An error occurred!";
|
||||||
|
this.toast.show();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
document.addEventListener("DOMContentLoaded", function (e) {
|
||||||
|
new BigdataResume();
|
||||||
|
});
|
||||||
@@ -5,9 +5,9 @@ Dropzone.autoDiscover = false;
|
|||||||
var previewTemplate,
|
var previewTemplate,
|
||||||
dropzone,
|
dropzone,
|
||||||
dropzonePreviewNode = document.querySelector("#dropzone-preview-list");
|
dropzonePreviewNode = document.querySelector("#dropzone-preview-list");
|
||||||
console.log(previewTemplate);
|
|
||||||
console.log(dropzone);
|
const uploadButton = document.getElementById("btnUploadBusinessIndustry");
|
||||||
console.log(dropzonePreviewNode);
|
const spinner = document.getElementById("spinner");
|
||||||
|
|
||||||
const toastNotification = document.getElementById("toastNotification");
|
const toastNotification = document.getElementById("toastNotification");
|
||||||
const toast = new bootstrap.Toast(toastNotification);
|
const toast = new bootstrap.Toast(toastNotification);
|
||||||
@@ -29,97 +29,50 @@ const toast = new bootstrap.Toast(toastNotification);
|
|||||||
.getAttribute("content")}`,
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
init: function () {
|
init: function () {
|
||||||
// Listen for the success event
|
|
||||||
this.on("success", function (file, response) {
|
this.on("success", function (file, response) {
|
||||||
console.log("File successfully uploaded:", file);
|
|
||||||
console.log("API Response:", response);
|
|
||||||
|
|
||||||
// Show success toast
|
|
||||||
document.getElementById("toast-message").innerText =
|
document.getElementById("toast-message").innerText =
|
||||||
response.message;
|
response.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
document.getElementById("submit-upload").innerHTML =
|
|
||||||
"Upload Files";
|
|
||||||
// Tunggu sebentar lalu reload halaman
|
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = "/data/business-industries";
|
window.location.href = "/data/business-industries";
|
||||||
}, 2000);
|
}, 2000);
|
||||||
});
|
});
|
||||||
// Listen for the error event
|
|
||||||
this.on("error", function (file, errorMessage) {
|
this.on("error", function (file, errorMessage) {
|
||||||
console.error("Error uploading file:", file);
|
console.error("Error uploading file:", file);
|
||||||
console.error("Error message:", errorMessage);
|
console.error("Error message:", errorMessage);
|
||||||
// Handle the error response
|
|
||||||
|
|
||||||
// Show error toast
|
|
||||||
document.getElementById("toast-message").innerText =
|
document.getElementById("toast-message").innerText =
|
||||||
errorMessage.message;
|
errorMessage.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
document.getElementById("submit-upload").innerHTML =
|
uploadButton.disabled = false;
|
||||||
"Upload Files";
|
spinner.classList.add("d-none");
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
})));
|
})));
|
||||||
|
|
||||||
// Add event listener to control the submission manually
|
// Add event listener to control the submission manually
|
||||||
document.querySelector("#submit-upload").addEventListener("click", function () {
|
document
|
||||||
|
.querySelector("#btnUploadBusinessIndustry")
|
||||||
|
.addEventListener("click", function () {
|
||||||
console.log("Ini adalah value dropzone", dropzone.files[0]);
|
console.log("Ini adalah value dropzone", dropzone.files[0]);
|
||||||
const formData = new FormData();
|
const formData = new FormData();
|
||||||
console.log("Dropzonefiles", dropzone.files);
|
|
||||||
|
|
||||||
this.innerHTML =
|
|
||||||
'<span class="spinner-border spinner-border-sm me-1" role="status" aria-hidden="true"></span>Loading...';
|
|
||||||
|
|
||||||
// Pastikan ada file dalam queue sebelum memprosesnya
|
|
||||||
if (dropzone.files.length > 0) {
|
if (dropzone.files.length > 0) {
|
||||||
formData.append("file", dropzone.files[0]);
|
formData.append("file", dropzone.files[0]);
|
||||||
console.log("ini adalah form data on submit", ...formData);
|
|
||||||
dropzone.processQueue(); // Ini akan manual memicu upload
|
dropzone.processQueue(); // Ini akan manual memicu upload
|
||||||
|
uploadButton.disabled = true;
|
||||||
|
spinner.classList.remove("d-none");
|
||||||
} else {
|
} else {
|
||||||
// Show error toast when no file is selected
|
|
||||||
document.getElementById("toast-message").innerText =
|
document.getElementById("toast-message").innerText =
|
||||||
"Please add a file first.";
|
"Please add a file first.";
|
||||||
toast.show();
|
toast.show();
|
||||||
|
uploadButton.disabled = false;
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
spinner.classList.add("d-none");
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Optional: Listen for the 'addedfile' event to log or control file add behavior
|
dropzone.on("addedfile", function (file) {});
|
||||||
dropzone.on("addedfile", function (file) {
|
|
||||||
console.log("File ditambahkan:", file);
|
|
||||||
console.log("Nama File:", file.name);
|
|
||||||
console.log("Tipe File:", file.type);
|
|
||||||
console.log("Ukuran File:", (file.size / 1024).toFixed(2) + " KB");
|
|
||||||
});
|
|
||||||
|
|
||||||
dropzone.on("complete", function (file) {
|
dropzone.on("complete", function (file) {
|
||||||
dropzone.removeFile(file);
|
dropzone.removeFile(file);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Function to show toast
|
|
||||||
// function showToast(iconClass, iconColor, message) {
|
|
||||||
// const toastElement = document.getElementById("toastUploadAdvertisement");
|
|
||||||
// const toastBody = toastElement.querySelector(".toast-body");
|
|
||||||
// const toastHeader = toastElement.querySelector(".toast-header");
|
|
||||||
|
|
||||||
// // Remove existing icon (if any) before adding the new one
|
|
||||||
// const existingIcon = toastHeader.querySelector(".bx");
|
|
||||||
// if (existingIcon) {
|
|
||||||
// toastHeader.querySelector(".auth-logo").removeChild(existingIcon); // Remove the existing icon
|
|
||||||
// }
|
|
||||||
|
|
||||||
// // Add the new icon to the toast header
|
|
||||||
// const icon = document.createElement("i");
|
|
||||||
// icon.classList.add("bx", iconClass);
|
|
||||||
// icon.style.fontSize = "25px";
|
|
||||||
// icon.style.color = iconColor;
|
|
||||||
// toastHeader.querySelector(".auth-logo").appendChild(icon);
|
|
||||||
|
|
||||||
// // Set the toast message
|
|
||||||
// toastBody.textContent = message;
|
|
||||||
|
|
||||||
// // Show the toast
|
|
||||||
// const toast = new bootstrap.Toast(toastElement); // Inisialisasi Bootstrap Toast
|
|
||||||
// toast.show();
|
|
||||||
// }
|
|
||||||
|
|||||||
@@ -2,58 +2,36 @@ import { Grid } from "gridjs/dist/gridjs.umd.js";
|
|||||||
import gridjs from "gridjs/dist/gridjs.umd.js";
|
import gridjs from "gridjs/dist/gridjs.umd.js";
|
||||||
import "gridjs/dist/gridjs.umd.js";
|
import "gridjs/dist/gridjs.umd.js";
|
||||||
import GlobalConfig from "../global-config.js";
|
import GlobalConfig from "../global-config.js";
|
||||||
|
import Swal from "sweetalert2";
|
||||||
|
|
||||||
class BusinessIndustries {
|
class BusinessIndustries {
|
||||||
constructor() {
|
constructor() {
|
||||||
this.table = null; // Store Grid.js instance
|
this.toastMessage = document.getElementById("toast-message");
|
||||||
|
this.toastElement = document.getElementById("toastNotification");
|
||||||
|
this.toast = new bootstrap.Toast(this.toastElement);
|
||||||
|
this.table = null;
|
||||||
|
|
||||||
|
// Initialize functions
|
||||||
|
this.initTableBusinessIndustries();
|
||||||
|
this.initEvents();
|
||||||
}
|
}
|
||||||
init() {
|
initEvents() {
|
||||||
this.getFetchApiData();
|
document.body.addEventListener("click", async (event) => {
|
||||||
|
const deleteButton = event.target.closest(
|
||||||
|
".btn-delete-business-industry"
|
||||||
|
);
|
||||||
|
if (deleteButton) {
|
||||||
|
event.preventDefault();
|
||||||
|
await this.handleDelete(deleteButton);
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
getFetchApiData() {
|
initTableBusinessIndustries() {
|
||||||
let tableContainer = document.getElementById(
|
let tableContainer = document.getElementById(
|
||||||
"table-business-industries"
|
"table-business-industries"
|
||||||
);
|
);
|
||||||
|
// Create a new Grid.js instance only if it doesn't exist
|
||||||
if (this.table) {
|
|
||||||
// If table exists, update its data instead of recreating
|
|
||||||
this.table
|
|
||||||
.updateConfig({
|
|
||||||
server: {
|
|
||||||
url: `${GlobalConfig.apiHost}/api/api-business-industries`,
|
|
||||||
credentials: "include",
|
|
||||||
headers: {
|
|
||||||
Authorization: `Bearer ${document
|
|
||||||
.querySelector('meta[name="api-token"]')
|
|
||||||
.getAttribute("content")}`,
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
then: (data) =>
|
|
||||||
data.data.map((item) => [
|
|
||||||
item.id,
|
|
||||||
item.nama_kecamatan,
|
|
||||||
item.nama_kelurahan,
|
|
||||||
item.nop,
|
|
||||||
item.nama_wajib_pajak,
|
|
||||||
item.alamat_wajib_pajak,
|
|
||||||
item.alamat_objek_pajak,
|
|
||||||
item.luas_bumi,
|
|
||||||
item.luas_bangunan,
|
|
||||||
item.njop_bumi,
|
|
||||||
item.njop_bangunan,
|
|
||||||
item.ketetapan,
|
|
||||||
item.tahun_pajak,
|
|
||||||
item.created_at,
|
|
||||||
item.id,
|
|
||||||
]),
|
|
||||||
total: (data) => data.total,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
.forceRender();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.table = new Grid({
|
this.table = new Grid({
|
||||||
columns: [
|
columns: [
|
||||||
{ name: "ID", width: "80px", hidden: false },
|
{ name: "ID", width: "80px", hidden: false },
|
||||||
@@ -71,29 +49,22 @@ class BusinessIndustries {
|
|||||||
{ name: "Tahun Pajak", width: "120px" },
|
{ name: "Tahun Pajak", width: "120px" },
|
||||||
{ name: "Created", width: "180px" },
|
{ name: "Created", width: "180px" },
|
||||||
{
|
{
|
||||||
name: "Actions",
|
name: "Action",
|
||||||
width: "120px",
|
formatter: (cell) =>
|
||||||
formatter: function (cell) {
|
gridjs.html(`
|
||||||
return gridjs.html(`
|
|
||||||
<div class="d-flex justify-content-center gap-2">
|
<div class="d-flex justify-content-center gap-2">
|
||||||
<a href="/data/business-industries/${cell}/edit" class="btn btn-yellow btn-sm d-inline-flex align-items-center justify-content-center">
|
<a href="/data/business-industries/${cell}/edit" class="btn btn-yellow btn-sm d-inline-flex align-items-center justify-content-center">
|
||||||
<i class='bx bx-edit'></i>
|
<i class='bx bx-edit'></i>
|
||||||
</a>
|
</a>
|
||||||
<button class="btn btn-sm btn-red d-inline-flex align-items-center justify-content-center btn-delete-business-industries" data-id="${cell}">
|
<button data-id="${cell}" class="btn btn-sm btn-red btn-delete-business-industry d-inline-flex align-items-center justify-content-center">
|
||||||
<i class='bx bxs-trash'></i>
|
<i class='bx bxs-trash' ></i>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
`);
|
`),
|
||||||
},
|
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
search: {
|
|
||||||
server: {
|
|
||||||
url: (prev, keyword) => `${prev}?search=${keyword}`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
pagination: {
|
pagination: {
|
||||||
limit: 15,
|
limit: 50,
|
||||||
server: {
|
server: {
|
||||||
url: (prev, page) =>
|
url: (prev, page) =>
|
||||||
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
||||||
@@ -102,6 +73,11 @@ class BusinessIndustries {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
sort: true,
|
sort: true,
|
||||||
|
search: {
|
||||||
|
server: {
|
||||||
|
url: (prev, keyword) => `${prev}?search=${keyword}`,
|
||||||
|
},
|
||||||
|
},
|
||||||
server: {
|
server: {
|
||||||
url: `${GlobalConfig.apiHost}/api/api-business-industries`,
|
url: `${GlobalConfig.apiHost}/api/api-business-industries`,
|
||||||
headers: {
|
headers: {
|
||||||
@@ -131,57 +107,28 @@ class BusinessIndustries {
|
|||||||
total: (data) => data.total,
|
total: (data) => data.total,
|
||||||
},
|
},
|
||||||
}).render(tableContainer);
|
}).render(tableContainer);
|
||||||
|
|
||||||
document.addEventListener("click", this.handleDelete.bind(this));
|
|
||||||
}
|
}
|
||||||
handleDelete(event) {
|
async handleDelete(deleteButton) {
|
||||||
if (event.target.classList.contains("btn-delete-business-industries")) {
|
const id = deleteButton.getAttribute("data-id");
|
||||||
event.preventDefault();
|
|
||||||
const id = event.target.getAttribute("data-id");
|
|
||||||
let modalElement = document.getElementById("modalConfirmation");
|
|
||||||
let toastMessage = document.getElementById("toast-message");
|
|
||||||
|
|
||||||
if (!modalElement) {
|
const result = await Swal.fire({
|
||||||
console.error("Modal element not found!");
|
title: "Are you sure?",
|
||||||
return;
|
text: "You won't be able to revert this!",
|
||||||
}
|
icon: "warning",
|
||||||
|
showCancelButton: true,
|
||||||
let modal = new bootstrap.Modal(modalElement);
|
confirmButtonColor: "#3085d6",
|
||||||
let btnSaveConfirmation = document.getElementById(
|
cancelButtonColor: "#d33",
|
||||||
"btnSaveConfirmation"
|
confirmButtonText: "Yes, delete it!",
|
||||||
);
|
});
|
||||||
let toastElement = document.getElementById("toastNotification");
|
|
||||||
let toast = new bootstrap.Toast(toastElement);
|
|
||||||
|
|
||||||
// Remove previous event listeners to avoid multiple bindings
|
|
||||||
btnSaveConfirmation.replaceWith(
|
|
||||||
btnSaveConfirmation.cloneNode(true)
|
|
||||||
);
|
|
||||||
btnSaveConfirmation = document.getElementById(
|
|
||||||
"btnSaveConfirmation"
|
|
||||||
);
|
|
||||||
|
|
||||||
// Set the role ID on the confirm button inside the modal
|
|
||||||
btnSaveConfirmation.setAttribute("data-business-industries-id", id);
|
|
||||||
|
|
||||||
// Show the modal
|
|
||||||
modal.show();
|
|
||||||
|
|
||||||
btnSaveConfirmation.addEventListener("click", async () => {
|
|
||||||
let deletedId = btnSaveConfirmation.getAttribute(
|
|
||||||
"data-business-industries-id"
|
|
||||||
);
|
|
||||||
|
|
||||||
|
if (result.isConfirmed) {
|
||||||
try {
|
try {
|
||||||
let response = await fetch(
|
let response = await fetch(
|
||||||
`${GlobalConfig.apiHost}/api/api-business-industries/${deletedId}`,
|
`${GlobalConfig.apiHost}/api/api-business-industries/${id}`,
|
||||||
{
|
{
|
||||||
method: "DELETE",
|
method: "DELETE",
|
||||||
credentials: "include",
|
credentials: "include",
|
||||||
headers: {
|
headers: {
|
||||||
"X-CSRF-TOKEN": document
|
|
||||||
.querySelector('meta[name="csrf-token"]')
|
|
||||||
.getAttribute("content"),
|
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`,
|
.getAttribute("content")}`,
|
||||||
@@ -192,69 +139,29 @@ class BusinessIndustries {
|
|||||||
|
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
let result = await response.json();
|
let result = await response.json();
|
||||||
toastMessage.innerText =
|
this.toastMessage.innerText =
|
||||||
result.message || "Deleted successfully!";
|
result.message || "Deleted successfully!";
|
||||||
toast.show();
|
this.toast.show();
|
||||||
|
|
||||||
// Hide modal
|
|
||||||
modal.hide();
|
|
||||||
|
|
||||||
// Refresh Grid.js table
|
// Refresh Grid.js table
|
||||||
this.refreshDataSettings();
|
if (typeof this.table !== "undefined") {
|
||||||
|
this.table.updateConfig({}).forceRender();
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
let error = await response.json();
|
let error = await response.json();
|
||||||
console.error("Delete failed:", error);
|
console.error("Delete failed:", error);
|
||||||
toastMessage.innerText =
|
this.toastMessage.innerText =
|
||||||
error.message || "Delete failed!";
|
error.message || "Delete failed!";
|
||||||
toast.show();
|
this.toast.show();
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error deleting item:", error);
|
console.error("Error deleting item:", error);
|
||||||
toastMessage.innerText = "An error occurred!";
|
this.toastMessage.innerText = "An error occurred!";
|
||||||
toast.show();
|
this.toast.show();
|
||||||
}
|
}
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
refreshDataSettings() {
|
|
||||||
if (this.table) {
|
|
||||||
this.table
|
|
||||||
.updateConfig({
|
|
||||||
server: {
|
|
||||||
url: `${GlobalConfig.apiHost}/api/api-business-industries`,
|
|
||||||
credentials: "include",
|
|
||||||
headers: {
|
|
||||||
Authorization: `Bearer ${document
|
|
||||||
.querySelector('meta[name="api-token"]')
|
|
||||||
.getAttribute("content")}`,
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
then: (data) =>
|
|
||||||
data.data.map((item) => [
|
|
||||||
item.id,
|
|
||||||
item.nama_kecamatan,
|
|
||||||
item.nama_kelurahan,
|
|
||||||
item.nop,
|
|
||||||
item.nama_wajib_pajak,
|
|
||||||
item.alamat_wajib_pajak,
|
|
||||||
item.alamat_objek_pajak,
|
|
||||||
item.luas_bumi,
|
|
||||||
item.luas_bangunan,
|
|
||||||
item.njop_bumi,
|
|
||||||
item.njop_bangunan,
|
|
||||||
item.ketetapan,
|
|
||||||
item.tahun_pajak,
|
|
||||||
item.created_at,
|
|
||||||
item.id, // ID for Actions column
|
|
||||||
]),
|
|
||||||
total: (data) => data.total,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
.forceRender();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
document.addEventListener("DOMContentLoaded", function (e) {
|
document.addEventListener("DOMContentLoaded", function (e) {
|
||||||
new BusinessIndustries().init();
|
new BusinessIndustries();
|
||||||
});
|
});
|
||||||
|
|||||||
155
resources/js/chatbot-pimpinan/index.js
Normal file
155
resources/js/chatbot-pimpinan/index.js
Normal file
@@ -0,0 +1,155 @@
|
|||||||
|
import GlobalConfig from "../global-config.js";
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", function () {
|
||||||
|
const timeElements = document.querySelectorAll(".sending-message-time p");
|
||||||
|
|
||||||
|
timeElements.forEach((element) => {
|
||||||
|
element.textContent = getCurrentTime();
|
||||||
|
});
|
||||||
|
|
||||||
|
const textarea = document.getElementById("user-message");
|
||||||
|
const sendButton = document.getElementById("send");
|
||||||
|
const conversationArea = document.querySelector(".row.flex-grow");
|
||||||
|
const chatHistory = [];
|
||||||
|
|
||||||
|
// Fungsi untuk mengirim pesan
|
||||||
|
async function sendMessage() {
|
||||||
|
const userText = textarea.value.trim();
|
||||||
|
if (userText !== "") {
|
||||||
|
// Kosongkan textarea setelah mengirim
|
||||||
|
textarea.value = "";
|
||||||
|
|
||||||
|
// Tambahkan pesan user ke UI
|
||||||
|
addMessage(userText, "user");
|
||||||
|
|
||||||
|
// Tambahkan pesan bot sementara dengan "Loading..."
|
||||||
|
const botMessageElement = addMessage('<div class="bot-message-text">...</div>', "bot");
|
||||||
|
|
||||||
|
const messageTextContainer = botMessageElement.querySelector(".bot-message-text");
|
||||||
|
if (messageTextContainer) {
|
||||||
|
messageTextContainer.innerHTML = '<div class="loader ms-3"></div>';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Panggil API untuk mendapatkan respons dari bot
|
||||||
|
const botResponse = await getBotResponse(userText, chatHistory);
|
||||||
|
|
||||||
|
// Perbarui pesan bot dengan respons yang sebenarnya
|
||||||
|
if (messageTextContainer) {
|
||||||
|
messageTextContainer.innerHTML = botResponse;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Event listener untuk klik tombol
|
||||||
|
sendButton.addEventListener("click", sendMessage);
|
||||||
|
|
||||||
|
// Event listener untuk menekan Enter di textarea
|
||||||
|
textarea.addEventListener("keydown", function (event) {
|
||||||
|
if (event.key === "Enter" && !event.shiftKey) {
|
||||||
|
event.preventDefault(); // Mencegah newline di textarea
|
||||||
|
sendMessage(); // Panggil fungsi kirim pesan
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
function getCurrentTime() {
|
||||||
|
const now = new Date();
|
||||||
|
return now.getHours().toString().padStart(2, "0") + ":" + now.getMinutes().toString().padStart(2, "0");
|
||||||
|
}
|
||||||
|
|
||||||
|
function addMessage(text, sender) {
|
||||||
|
const messageRow = document.createElement("div");
|
||||||
|
// Atur posisi berdasarkan sender (user -> end, bot -> start)
|
||||||
|
messageRow.classList.add("row", "flex-grow", "overflow-auto", sender === "user" ? "justify-content-end" : "justify-content-start");
|
||||||
|
|
||||||
|
const messageCol = document.createElement("div");
|
||||||
|
messageCol.classList.add("col-9", "w-auto");
|
||||||
|
|
||||||
|
// Atur lebar maksimum berdasarkan sender
|
||||||
|
messageCol.style.maxWidth = sender === "user" ? "50%" : "75%";
|
||||||
|
|
||||||
|
// Container untuk menyimpan nama dan bubble chat
|
||||||
|
const messageWrapper = document.createElement("div");
|
||||||
|
messageWrapper.classList.add("d-flex", "flex-column");
|
||||||
|
|
||||||
|
// Tambahkan Nama di luar bubble chat
|
||||||
|
const messageName = document.createElement("p");
|
||||||
|
messageName.classList.add("fw-bolder", sender === "user" ? "text-end" : "text-start", "mb-1");
|
||||||
|
messageName.textContent = sender === "user" ? "You" : "Neng Bedas";
|
||||||
|
|
||||||
|
// Bubble Chat
|
||||||
|
const messageContainer = document.createElement("div");
|
||||||
|
messageContainer.classList.add("p-2", "rounded", "mb-2", "d-inline-block");
|
||||||
|
if (sender === "user") {
|
||||||
|
messageContainer.classList.add("user-response", "bg-primary", "text-white");
|
||||||
|
} else {
|
||||||
|
messageContainer.classList.add("bot-response", "bg-light");
|
||||||
|
}
|
||||||
|
|
||||||
|
const messageContent = document.createElement("div");
|
||||||
|
messageContent.classList.add("bot-message-text", "mb-0", "text-start");
|
||||||
|
messageContent.textContent = text;
|
||||||
|
|
||||||
|
// Waktu di dalam bubble chat
|
||||||
|
const messageTime = document.createElement("div");
|
||||||
|
messageTime.classList.add("sending-message-time", "text-end", "mt-1");
|
||||||
|
messageTime.innerHTML = `<p class="small mb-0 ${sender === "user" ? "text-white text-start" : "text-muted"}">${getCurrentTime()}</p>`;
|
||||||
|
|
||||||
|
messageContainer.appendChild(messageContent);
|
||||||
|
messageContainer.appendChild(messageTime);
|
||||||
|
|
||||||
|
// Jika pengirim adalah bot, tambahkan avatar
|
||||||
|
if (sender !== "user") {
|
||||||
|
const avatarContainer = document.createElement("div");
|
||||||
|
avatarContainer.classList.add("col-auto", "pe-0");
|
||||||
|
|
||||||
|
const avatarImg = document.createElement("img");
|
||||||
|
avatarImg.classList.add("rounded-circle");
|
||||||
|
avatarImg.width = 45;
|
||||||
|
avatarImg.src = "/images/iconchatbot.jpeg";
|
||||||
|
avatarImg.alt = "bot-avatar";
|
||||||
|
|
||||||
|
avatarContainer.appendChild(avatarImg);
|
||||||
|
messageRow.appendChild(avatarContainer);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Masukkan nama dan bubble ke dalam wrapper
|
||||||
|
messageWrapper.appendChild(messageName);
|
||||||
|
messageWrapper.appendChild(messageContainer);
|
||||||
|
messageCol.appendChild(messageWrapper);
|
||||||
|
messageRow.appendChild(messageCol);
|
||||||
|
|
||||||
|
conversationArea.appendChild(messageRow);
|
||||||
|
conversationArea.scrollTop = conversationArea.scrollHeight;
|
||||||
|
|
||||||
|
return messageContainer;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fungsi untuk memanggil API
|
||||||
|
async function getBotResponse(userText, historyChat) {
|
||||||
|
try {
|
||||||
|
const url = `${GlobalConfig.apiHost}/api/main-generate-text`;
|
||||||
|
const response = await fetch(url, {
|
||||||
|
method: "POST",
|
||||||
|
body: JSON.stringify({prompt: userText, chatHistory: historyChat}),
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${document
|
||||||
|
.querySelector('meta[name="api-token"]')
|
||||||
|
.getAttribute("content")}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
const rawBotResponse = data.nlpResponse;
|
||||||
|
// Tambahkan ke chatHistory
|
||||||
|
chatHistory.push({
|
||||||
|
user: userText,
|
||||||
|
rawBotResponse: rawBotResponse,
|
||||||
|
});
|
||||||
|
return data.response || "Maaf, saya tidak mengerti.";
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error fetching bot response:", error);
|
||||||
|
return "Terjadi kesalahan, coba lagi nanti.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
218
resources/js/chatbot/index.js
Normal file
218
resources/js/chatbot/index.js
Normal file
@@ -0,0 +1,218 @@
|
|||||||
|
import GlobalConfig from "../global-config.js";
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", function () {
|
||||||
|
const tabs = document.querySelectorAll(".nav-link");
|
||||||
|
const timeElements = document.querySelectorAll(".sending-message-time p");
|
||||||
|
|
||||||
|
timeElements.forEach((element) => {
|
||||||
|
element.textContent = getCurrentTime();
|
||||||
|
});
|
||||||
|
|
||||||
|
function activateTab(tab) {
|
||||||
|
tabs.forEach(btn => {
|
||||||
|
btn.classList.remove("border-3", "bg-primary", "text-white"); // Reset semua tab
|
||||||
|
});
|
||||||
|
tab.classList.add("border-3", "bg-primary", "text-white"); // Tambahkan warna pada tab aktif
|
||||||
|
}
|
||||||
|
|
||||||
|
tabs.forEach(tab => {
|
||||||
|
tab.addEventListener("click", function () {
|
||||||
|
activateTab(this);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// Set warna awal untuk tab aktif (jika ada)
|
||||||
|
const initialActiveTab = document.querySelector(".nav-link.active");
|
||||||
|
if (initialActiveTab) {
|
||||||
|
activateTab(initialActiveTab);
|
||||||
|
}
|
||||||
|
|
||||||
|
document.querySelectorAll(".nav-link").forEach(tab => {
|
||||||
|
tab.addEventListener("click", function () {
|
||||||
|
setTimeout(() => {
|
||||||
|
const tab_active = getActiveTabId();
|
||||||
|
console.log("Active Tab ID:", tab_active);
|
||||||
|
|
||||||
|
// Hapus semua chat kecuali pesan default bot
|
||||||
|
conversationArea.innerHTML = `
|
||||||
|
<div class="row flex-grow overflow-auto align-items-start">
|
||||||
|
<!-- Avatar -->
|
||||||
|
<div class="col-auto alignpe-0">
|
||||||
|
<img class="rounded-circle" width="45" src="/images/iconchatbot.jpeg" alt="avatar-3">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Nama dan Bubble Chat -->
|
||||||
|
<div class="col-9 w-auto">
|
||||||
|
<!-- Nama Bot -->
|
||||||
|
<p class="fw-bolder mb-1">Neng Bedas</p>
|
||||||
|
|
||||||
|
<!-- Bubble Chat -->
|
||||||
|
<div class="bot-response p-2 bg-light rounded mb-2 d-inline-block">
|
||||||
|
<p class="mb-0">Halo! Ada yang bisa saya bantu?</p>
|
||||||
|
|
||||||
|
<!-- Waktu (Tetap di Dalam Bubble Chat) -->
|
||||||
|
<div class="sending-message-time text-end mt-1">
|
||||||
|
<p class="text-muted small mb-0">Now</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}, 100); // Timeout untuk memastikan class `active` sudah diperbarui
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const textarea = document.getElementById("user-message");
|
||||||
|
const sendButton = document.getElementById("send");
|
||||||
|
const conversationArea = document.querySelector(".row.flex-grow");
|
||||||
|
const chatHistory = [];
|
||||||
|
|
||||||
|
// Fungsi untuk mengirim pesan
|
||||||
|
async function sendMessage() {
|
||||||
|
const userText = textarea.value.trim();
|
||||||
|
if (userText !== "") {
|
||||||
|
// Kosongkan textarea setelah mengirim
|
||||||
|
textarea.value = "";
|
||||||
|
|
||||||
|
// Ambil tab aktif saat ini
|
||||||
|
const currentTab = getActiveTabId();
|
||||||
|
|
||||||
|
// Tambahkan pesan user ke UI
|
||||||
|
addMessage(userText, "user");
|
||||||
|
|
||||||
|
// Tambahkan pesan bot sementara dengan "Loading..."
|
||||||
|
const botMessageElement = addMessage('<div class="bot-message-text">...</div>', "bot");
|
||||||
|
|
||||||
|
const messageTextContainer = botMessageElement.querySelector(".bot-message-text");
|
||||||
|
if (messageTextContainer) {
|
||||||
|
messageTextContainer.innerHTML = '<div class="loader ms-3"></div>';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Panggil API untuk mendapatkan respons dari bot
|
||||||
|
const botResponse = await getBotResponse(currentTab, userText, chatHistory);
|
||||||
|
|
||||||
|
// Perbarui pesan bot dengan respons yang sebenarnya
|
||||||
|
if (messageTextContainer) {
|
||||||
|
messageTextContainer.innerHTML = botResponse;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Event listener untuk klik tombol
|
||||||
|
sendButton.addEventListener("click", sendMessage);
|
||||||
|
|
||||||
|
// Event listener untuk menekan Enter di textarea
|
||||||
|
textarea.addEventListener("keydown", function (event) {
|
||||||
|
if (event.key === "Enter" && !event.shiftKey) {
|
||||||
|
event.preventDefault(); // Mencegah newline di textarea
|
||||||
|
sendMessage(); // Panggil fungsi kirim pesan
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
function getCurrentTime() {
|
||||||
|
const now = new Date();
|
||||||
|
return now.getHours().toString().padStart(2, "0") + ":" + now.getMinutes().toString().padStart(2, "0");
|
||||||
|
}
|
||||||
|
|
||||||
|
function addMessage(text, sender) {
|
||||||
|
const messageRow = document.createElement("div");
|
||||||
|
// Atur posisi berdasarkan sender (user -> end, bot -> start)
|
||||||
|
messageRow.classList.add("row", "flex-grow", "overflow-auto", sender === "user" ? "justify-content-end" : "justify-content-start");
|
||||||
|
|
||||||
|
const messageCol = document.createElement("div");
|
||||||
|
messageCol.classList.add("col-9", "w-auto");
|
||||||
|
|
||||||
|
// Atur lebar maksimum berdasarkan sender
|
||||||
|
messageCol.style.maxWidth = sender === "user" ? "50%" : "75%";
|
||||||
|
|
||||||
|
// Container untuk menyimpan nama dan bubble chat
|
||||||
|
const messageWrapper = document.createElement("div");
|
||||||
|
messageWrapper.classList.add("d-flex", "flex-column");
|
||||||
|
|
||||||
|
// Tambahkan Nama di luar bubble chat
|
||||||
|
const messageName = document.createElement("p");
|
||||||
|
messageName.classList.add("fw-bolder", sender === "user" ? "text-end" : "text-start", "mb-1");
|
||||||
|
messageName.textContent = sender === "user" ? "You" : "Neng Bedas";
|
||||||
|
|
||||||
|
// Bubble Chat
|
||||||
|
const messageContainer = document.createElement("div");
|
||||||
|
messageContainer.classList.add("p-2", "rounded", "mb-2", "d-inline-block");
|
||||||
|
if (sender === "user") {
|
||||||
|
messageContainer.classList.add("user-response", "bg-primary", "text-white");
|
||||||
|
} else {
|
||||||
|
messageContainer.classList.add("bot-response", "bg-light");
|
||||||
|
}
|
||||||
|
|
||||||
|
const messageContent = document.createElement("div");
|
||||||
|
messageContent.classList.add("bot-message-text", "mb-0", "text-start");
|
||||||
|
messageContent.textContent = text;
|
||||||
|
|
||||||
|
// Waktu di dalam bubble chat
|
||||||
|
const messageTime = document.createElement("div");
|
||||||
|
messageTime.classList.add("sending-message-time", "text-end", "mt-1");
|
||||||
|
messageTime.innerHTML = `<p class="small mb-0 ${sender === "user" ? "text-white text-start" : "text-muted"}">${getCurrentTime()}</p>`;
|
||||||
|
|
||||||
|
messageContainer.appendChild(messageContent);
|
||||||
|
messageContainer.appendChild(messageTime);
|
||||||
|
|
||||||
|
// Jika pengirim adalah bot, tambahkan avatar
|
||||||
|
if (sender !== "user") {
|
||||||
|
const avatarContainer = document.createElement("div");
|
||||||
|
avatarContainer.classList.add("col-auto", "pe-0");
|
||||||
|
|
||||||
|
const avatarImg = document.createElement("img");
|
||||||
|
avatarImg.classList.add("rounded-circle");
|
||||||
|
avatarImg.width = 45;
|
||||||
|
avatarImg.src = "/images/iconchatbot.jpeg";
|
||||||
|
avatarImg.alt = "bot-avatar";
|
||||||
|
|
||||||
|
avatarContainer.appendChild(avatarImg);
|
||||||
|
messageRow.appendChild(avatarContainer);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Masukkan nama dan bubble ke dalam wrapper
|
||||||
|
messageWrapper.appendChild(messageName);
|
||||||
|
messageWrapper.appendChild(messageContainer);
|
||||||
|
messageCol.appendChild(messageWrapper);
|
||||||
|
messageRow.appendChild(messageCol);
|
||||||
|
|
||||||
|
conversationArea.appendChild(messageRow);
|
||||||
|
conversationArea.scrollTop = conversationArea.scrollHeight;
|
||||||
|
|
||||||
|
return messageContainer;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fungsi untuk memanggil API
|
||||||
|
async function getBotResponse(tab_active, userText, historyChat) {
|
||||||
|
try {
|
||||||
|
const url = `${GlobalConfig.apiHost}/api/generate-text`;
|
||||||
|
const response = await fetch(url, {
|
||||||
|
method: "POST",
|
||||||
|
body: JSON.stringify({tab_active:tab_active, prompt: userText, chatHistory: historyChat }),
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${document
|
||||||
|
.querySelector('meta[name="api-token"]')
|
||||||
|
.getAttribute("content")}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
const rawBotResponse = data.nlpResponse;
|
||||||
|
// Tambahkan ke chatHistory
|
||||||
|
chatHistory.push({
|
||||||
|
user: userText,
|
||||||
|
rawBotResponse: rawBotResponse,
|
||||||
|
});
|
||||||
|
return data.response || "Maaf, saya tidak mengerti.";
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error fetching bot response:", error);
|
||||||
|
return "Terjadi kesalahan, coba lagi nanti.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
function getActiveTabId() {
|
||||||
|
const activeTab = document.querySelector(".nav-link.active");
|
||||||
|
return activeTab ? activeTab.id : null;
|
||||||
|
}
|
||||||
@@ -1,9 +1,9 @@
|
|||||||
class UpdateCustomer {
|
class UpdateCustomer {
|
||||||
constructor() {
|
constructor() {
|
||||||
this.initUpdateSpatial();
|
this.initUpdateCustomer();
|
||||||
}
|
}
|
||||||
|
|
||||||
initUpdateSpatial() {
|
initUpdateCustomer() {
|
||||||
const toastNotification = document.getElementById("toastNotification");
|
const toastNotification = document.getElementById("toastNotification");
|
||||||
const toast = new bootstrap.Toast(toastNotification);
|
const toast = new bootstrap.Toast(toastNotification);
|
||||||
document
|
document
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ class Customers {
|
|||||||
this.table = null;
|
this.table = null;
|
||||||
|
|
||||||
// Initialize functions
|
// Initialize functions
|
||||||
this.initTableSpatialPlannings();
|
this.initTableCustomers();
|
||||||
this.initEvents();
|
this.initEvents();
|
||||||
}
|
}
|
||||||
initEvents() {
|
initEvents() {
|
||||||
@@ -25,7 +25,7 @@ class Customers {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
initTableSpatialPlannings() {
|
initTableCustomers() {
|
||||||
let tableContainer = document.getElementById("table-customers");
|
let tableContainer = document.getElementById("table-customers");
|
||||||
// Create a new Grid.js instance only if it doesn't exist
|
// Create a new Grid.js instance only if it doesn't exist
|
||||||
this.table = new Grid({
|
this.table = new Grid({
|
||||||
@@ -53,7 +53,7 @@ class Customers {
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
pagination: {
|
pagination: {
|
||||||
limit: 15,
|
limit: 50,
|
||||||
server: {
|
server: {
|
||||||
url: (prev, page) =>
|
url: (prev, page) =>
|
||||||
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
||||||
|
|||||||
@@ -1,167 +1,31 @@
|
|||||||
import Big from "big.js";
|
import Big from "big.js";
|
||||||
import GlobalConfig, { addThousandSeparators } from "../global-config.js";
|
import GlobalConfig, { addThousandSeparators } from "../global-config.js";
|
||||||
import flatpickr from "flatpickr";
|
import InitDatePicker from "../utils/InitDatePicker.js";
|
||||||
import "flatpickr/dist/flatpickr.min.css";
|
|
||||||
|
|
||||||
class BigData {
|
class BigData {
|
||||||
async init() {
|
async init() {
|
||||||
try {
|
try {
|
||||||
this.filterYear = new Date().getFullYear(); // Set initial year
|
new InitDatePicker(
|
||||||
|
"#datepicker-dashboard-bigdata",
|
||||||
let yearSelect = document.querySelector("#yearPicker");
|
this.handleChangeDate.bind(this)
|
||||||
let filterButton = document.querySelector("#btnFilterYear");
|
).init();
|
||||||
|
|
||||||
if (!yearSelect || !filterButton) {
|
|
||||||
console.error(
|
|
||||||
"Element #yearPicker or #btnFilterYear not found."
|
|
||||||
);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set default value for input
|
|
||||||
yearSelect.value = this.filterYear;
|
|
||||||
|
|
||||||
yearSelect.addEventListener("change", () => {
|
|
||||||
this.updateYear(yearSelect.value);
|
|
||||||
});
|
|
||||||
|
|
||||||
// Handle button click
|
|
||||||
filterButton.addEventListener("click", () => {
|
|
||||||
this.updateYear(yearSelect.value);
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log("init filter this year", this.filterYear);
|
|
||||||
|
|
||||||
// Load initial data
|
// Load initial data
|
||||||
await this.updateData(this.filterYear);
|
this.updateData("latest");
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error initializing data:", error);
|
console.error("Error initializing data:", error);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
updateYear(value) {
|
|
||||||
let inputYear = parseInt(value, 10);
|
handleChangeDate(filterDate) {
|
||||||
if (!isNaN(inputYear)) {
|
if (!filterDate) return;
|
||||||
this.filterYear = inputYear;
|
this.updateData(filterDate);
|
||||||
this.updateData(this.filterYear);
|
|
||||||
} else {
|
|
||||||
console.error("Invalid year input");
|
|
||||||
}
|
}
|
||||||
}
|
async updateData(filterDate) {
|
||||||
async updateData(year) {
|
|
||||||
try {
|
try {
|
||||||
this.totalTargetPAD = await this.getDataSettings("TARGET_PAD");
|
this.resumeBigData = await this.getBigDataResume(filterDate);
|
||||||
this.resultDataTotal = await this.getDataTotalPotensi(year);
|
|
||||||
this.dataVerification = await this.getDataVerfication(year);
|
|
||||||
this.dataNonVerification = await this.getDataNonVerfication(year);
|
|
||||||
this.dataBusiness = await this.getDataBusiness(year);
|
|
||||||
this.dataNonBusiness = await this.getDataNonBusiness(year);
|
|
||||||
this.dataTataRuang = await this.getDataSettings("TATA_RUANG");
|
|
||||||
this.dataSumRealisasiTerbit = await this.getDataSettings(
|
|
||||||
"REALISASI_TERBIT_PBG_SUM"
|
|
||||||
);
|
|
||||||
this.dataCountRealisasiTerbit = await this.getDataSettings(
|
|
||||||
"REALISASI_TERBIT_PBG_COUNT"
|
|
||||||
);
|
|
||||||
this.dataSumMenungguKlikDPMPTSP = await this.getDataSettings(
|
|
||||||
"MENUNGGU_KLIK_DPMPTSP_SUM"
|
|
||||||
);
|
|
||||||
this.dataCountMenungguKlikDPMPTSP = await this.getDataSettings(
|
|
||||||
"MENUNGGU_KLIK_DPMPTSP_COUNT"
|
|
||||||
);
|
|
||||||
this.dataSumProsesDinasTeknis = await this.getDataSettings(
|
|
||||||
"PROSES_DINAS_TEKNIS_SUM"
|
|
||||||
);
|
|
||||||
this.dataCountProsesDinasTeknis = await this.getDataSettings(
|
|
||||||
"PROSES_DINAS_TEKNIS_COUNT"
|
|
||||||
);
|
|
||||||
|
|
||||||
// total potensi
|
this.initChartTargetPAD(filterDate);
|
||||||
this.bigTargetPAD = new Big(this.totalTargetPAD ?? 0);
|
|
||||||
this.bigTotalPotensi = new Big(this.resultDataTotal.totalData ?? 0);
|
|
||||||
|
|
||||||
this.resultPercentage = 0;
|
|
||||||
if (this.bigTotalPotensi > 0 && this.bigTargetPAD > 0) {
|
|
||||||
this.resultPercentage = this.bigTotalPotensi
|
|
||||||
.div(this.bigTargetPAD)
|
|
||||||
.times(100)
|
|
||||||
.toFixed(2);
|
|
||||||
if (this.resultPercentage > 100) {
|
|
||||||
this.resultPercentage = 100;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// tata ruang
|
|
||||||
this.bigTotalTataRuang = new Big(this.dataTataRuang);
|
|
||||||
this.percentageResultTataRuang =
|
|
||||||
this.bigTotalTataRuang <= 0 || this.bigTotalPotensi <= 0
|
|
||||||
? 0
|
|
||||||
: this.bigTotalTataRuang
|
|
||||||
.div(this.bigTotalPotensi)
|
|
||||||
.times(100)
|
|
||||||
.toFixed(2);
|
|
||||||
|
|
||||||
// kekurangan potensi
|
|
||||||
this.totalKekuranganPotensi = new Big(
|
|
||||||
this.bigTargetPAD - this.bigTotalPotensi
|
|
||||||
);
|
|
||||||
|
|
||||||
this.percentageKekuranganPotensi =
|
|
||||||
this.totalKekuranganPotensi <= 0 || this.bigTargetPAD <= 0
|
|
||||||
? 0
|
|
||||||
: this.totalKekuranganPotensi
|
|
||||||
.div(this.bigTargetPAD)
|
|
||||||
.times(100)
|
|
||||||
.toFixed(2);
|
|
||||||
|
|
||||||
// non-verification documents
|
|
||||||
this.bigTotalNonVerification = new Big(
|
|
||||||
this.dataNonVerification.total
|
|
||||||
);
|
|
||||||
this.percentageResultNonVerification =
|
|
||||||
this.bigTotalNonVerification <= 0 || this.bigTotalPotensi <= 0
|
|
||||||
? 0
|
|
||||||
: this.bigTotalNonVerification
|
|
||||||
.div(this.bigTotalPotensi)
|
|
||||||
.times(100)
|
|
||||||
.toFixed(2);
|
|
||||||
|
|
||||||
// verification documents
|
|
||||||
this.bigTotalVerification = new Big(this.dataVerification.total);
|
|
||||||
this.percetageResultVerification =
|
|
||||||
this.bigTotalVerification <= 0 || this.bigTotalPotensi <= 0
|
|
||||||
? 0
|
|
||||||
: this.bigTotalVerification
|
|
||||||
.div(this.bigTargetPAD)
|
|
||||||
.times(100)
|
|
||||||
.toFixed(2);
|
|
||||||
|
|
||||||
// business documents
|
|
||||||
this.bigTotalBusiness = new Big(this.dataBusiness.total);
|
|
||||||
this.percentageResultBusiness =
|
|
||||||
this.bigTotalNonVerification <= 0 || this.bigTotalBusiness <= 0
|
|
||||||
? 0
|
|
||||||
: this.bigTotalBusiness
|
|
||||||
.div(this.bigTotalNonVerification)
|
|
||||||
.times(100)
|
|
||||||
.toFixed(2);
|
|
||||||
|
|
||||||
// non-business documents
|
|
||||||
this.bigTotalNonBusiness = new Big(this.dataNonBusiness.total);
|
|
||||||
this.percentageResultNonBusiness =
|
|
||||||
this.bigTotalNonBusiness <= 0 ||
|
|
||||||
this.bigTotalNonVerification <= 0
|
|
||||||
? 0
|
|
||||||
: this.bigTotalNonBusiness
|
|
||||||
.div(this.bigTotalNonVerification)
|
|
||||||
.times(100)
|
|
||||||
.toFixed(2);
|
|
||||||
|
|
||||||
if (!this.bigTargetPAD) {
|
|
||||||
console.error("Failed to load chart data");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.initChartTargetPAD();
|
|
||||||
this.initChartUsaha();
|
this.initChartUsaha();
|
||||||
this.initChartNonUsaha();
|
this.initChartNonUsaha();
|
||||||
this.initChartTotalPotensi();
|
this.initChartTotalPotensi();
|
||||||
@@ -177,10 +41,10 @@ class BigData {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async getDataTotalPotensi(year) {
|
async getBigDataResume(filterByDate) {
|
||||||
try {
|
try {
|
||||||
const response = await fetch(
|
const response = await fetch(
|
||||||
`${GlobalConfig.apiHost}/api/all-task-documents?year=${year}`,
|
`${GlobalConfig.apiHost}/api/bigdata-resume?filterByDate=${filterByDate}`,
|
||||||
{
|
{
|
||||||
credentials: "include",
|
credentials: "include",
|
||||||
headers: {
|
headers: {
|
||||||
@@ -198,169 +62,23 @@ class BigData {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const data = await response.json();
|
const data = await response.json();
|
||||||
return {
|
return data;
|
||||||
countData: data.data.count,
|
|
||||||
totalData: data.data.total,
|
|
||||||
};
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error fetching chart data:", error);
|
console.error("Error fetching chart data:", error);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async getDataVerfication(year) {
|
initChartTargetPAD(filterDate) {
|
||||||
try {
|
const year =
|
||||||
const response = await fetch(
|
filterDate === "latest"
|
||||||
`${GlobalConfig.apiHost}/api/verification-documents?year=${year}`,
|
? new Date().getFullYear()
|
||||||
{
|
: new Date(filterDate).getFullYear();
|
||||||
credentials: "include",
|
document
|
||||||
headers: {
|
.querySelectorAll(".document-title.chart-target-pad")
|
||||||
Authorization: `Bearer ${
|
.forEach((element) => {
|
||||||
document.querySelector("meta[name='api-token']")
|
element.innerText = `Target PAD ${year}`;
|
||||||
.content
|
});
|
||||||
}`,
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!response.ok) {
|
|
||||||
console.error("Network response was not ok", response);
|
|
||||||
}
|
|
||||||
|
|
||||||
const data = await response.json();
|
|
||||||
return {
|
|
||||||
count: data.data.count,
|
|
||||||
total: data.data.total,
|
|
||||||
};
|
|
||||||
} catch (error) {
|
|
||||||
console.error("Error fetching chart data:", error);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async getDataNonVerfication(year) {
|
|
||||||
try {
|
|
||||||
const response = await fetch(
|
|
||||||
`${GlobalConfig.apiHost}/api/non-verification-documents?year=${year}`,
|
|
||||||
{
|
|
||||||
credentials: "include",
|
|
||||||
headers: {
|
|
||||||
Authorization: `Bearer ${
|
|
||||||
document.querySelector("meta[name='api-token']")
|
|
||||||
.content
|
|
||||||
}`,
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!response.ok) {
|
|
||||||
console.error("Network response was not ok", response);
|
|
||||||
}
|
|
||||||
|
|
||||||
const data = await response.json();
|
|
||||||
return {
|
|
||||||
count: data.data.count,
|
|
||||||
total: data.data.total,
|
|
||||||
};
|
|
||||||
} catch (error) {
|
|
||||||
console.error("Error fetching chart data:", error);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async getDataBusiness(year) {
|
|
||||||
try {
|
|
||||||
const response = await fetch(
|
|
||||||
`${GlobalConfig.apiHost}/api/business-documents?year=${year}`,
|
|
||||||
{
|
|
||||||
credentials: "include",
|
|
||||||
headers: {
|
|
||||||
Authorization: `Bearer ${
|
|
||||||
document.querySelector("meta[name='api-token']")
|
|
||||||
.content
|
|
||||||
}`,
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!response.ok) {
|
|
||||||
console.error("Network response was not ok", response);
|
|
||||||
}
|
|
||||||
|
|
||||||
const data = await response.json();
|
|
||||||
return {
|
|
||||||
count: data.data.count,
|
|
||||||
total: data.data.total,
|
|
||||||
};
|
|
||||||
} catch (error) {
|
|
||||||
console.error("Error fetching chart data:", error);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async getDataNonBusiness(year) {
|
|
||||||
try {
|
|
||||||
const response = await fetch(
|
|
||||||
`${GlobalConfig.apiHost}/api/non-business-documents?year=${year}`,
|
|
||||||
{
|
|
||||||
credentials: "include",
|
|
||||||
headers: {
|
|
||||||
Authorization: `Bearer ${
|
|
||||||
document.querySelector("meta[name='api-token']")
|
|
||||||
.content
|
|
||||||
}`,
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!response.ok) {
|
|
||||||
console.error("Network response was not ok", response);
|
|
||||||
}
|
|
||||||
|
|
||||||
const data = await response.json();
|
|
||||||
return {
|
|
||||||
count: data.data.count,
|
|
||||||
total: data.data.total,
|
|
||||||
};
|
|
||||||
} catch (error) {
|
|
||||||
console.error("Error fetching chart data:", error);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async getDataSettings(string_key) {
|
|
||||||
try {
|
|
||||||
const response = await fetch(
|
|
||||||
`${GlobalConfig.apiHost}/api/api-data-settings?search=${string_key}`,
|
|
||||||
{
|
|
||||||
credentials: "include",
|
|
||||||
headers: {
|
|
||||||
Authorization: `Bearer ${
|
|
||||||
document.querySelector("meta[name='api-token']")
|
|
||||||
.content
|
|
||||||
}`,
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!response.ok) {
|
|
||||||
console.error("Network response was not ok", response);
|
|
||||||
}
|
|
||||||
|
|
||||||
const data = await response.json();
|
|
||||||
return data.data[0].value;
|
|
||||||
} catch (error) {
|
|
||||||
console.error("Error fetching chart data:", error);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
initChartTargetPAD() {
|
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-count.chart-target-pad")
|
.querySelectorAll(".document-count.chart-target-pad")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
@@ -370,53 +88,59 @@ class BigData {
|
|||||||
.querySelectorAll(".document-total.chart-target-pad")
|
.querySelectorAll(".document-total.chart-target-pad")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `Rp.${addThousandSeparators(
|
element.innerText = `Rp.${addThousandSeparators(
|
||||||
this.bigTargetPAD.toString()
|
this.resumeBigData.target_pad.sum.toString()
|
||||||
)}`;
|
)}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".small-percentage.chart-target-pad")
|
.querySelectorAll(".small-percentage.chart-target-pad")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${100}%`;
|
element.innerText = `${this.resumeBigData.target_pad.percentage}%`;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
initChartTotalPotensi() {
|
initChartTotalPotensi() {
|
||||||
const countAll = this.resultDataTotal.countData ?? 0;
|
// const countAll = this.resultDataTotal.countData ?? 0;
|
||||||
|
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-count.chart-total-potensi")
|
.querySelectorAll(".document-count.chart-total-potensi")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${countAll}`;
|
// element.innerText = `${countAll}`;
|
||||||
|
element.innerText = `${this.resumeBigData.total_potensi.count}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-total.chart-total-potensi")
|
.querySelectorAll(".document-total.chart-total-potensi")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `Rp.${addThousandSeparators(
|
element.innerText = `Rp.${addThousandSeparators(
|
||||||
this.bigTotalPotensi.toString()
|
// this.bigTotalPotensi.toString()
|
||||||
|
this.resumeBigData.total_potensi.sum.toString()
|
||||||
)}`;
|
)}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".small-percentage.chart-total-potensi")
|
.querySelectorAll(".small-percentage.chart-total-potensi")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.resultPercentage}%`;
|
// element.innerText = `${this.resultPercentage}%`;
|
||||||
|
element.innerText = `${this.resumeBigData.total_potensi.percentage}%`;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
initChartVerificationDocuments() {
|
initChartVerificationDocuments() {
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-count.chart-berkas-terverifikasi")
|
.querySelectorAll(".document-count.chart-berkas-terverifikasi")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.dataVerification.count}`;
|
// element.innerText = `${this.dataVerification.count}`;
|
||||||
|
element.innerText = `${this.resumeBigData.verified_document.count}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-total.chart-berkas-terverifikasi")
|
.querySelectorAll(".document-total.chart-berkas-terverifikasi")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `Rp.${addThousandSeparators(
|
element.innerText = `Rp.${addThousandSeparators(
|
||||||
this.bigTotalVerification.toString()
|
// this.bigTotalVerification.toString()
|
||||||
|
this.resumeBigData.verified_document.sum.toString()
|
||||||
)}`;
|
)}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".small-percentage.chart-berkas-terverifikasi")
|
.querySelectorAll(".small-percentage.chart-berkas-terverifikasi")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.percetageResultVerification}%`;
|
// element.innerText = `${this.percetageResultVerification}%`;
|
||||||
|
element.innerText = `${this.resumeBigData.verified_document.percentage}%`;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
initChartNonVerificationDocuments() {
|
initChartNonVerificationDocuments() {
|
||||||
@@ -425,7 +149,8 @@ class BigData {
|
|||||||
".document-count.chart-berkas-belum-terverifikasi"
|
".document-count.chart-berkas-belum-terverifikasi"
|
||||||
)
|
)
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.dataNonVerification.count}`;
|
// element.innerText = `${this.dataNonVerification.count}`;
|
||||||
|
element.innerText = `${this.resumeBigData.non_verified_document.count}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(
|
.querySelectorAll(
|
||||||
@@ -433,7 +158,8 @@ class BigData {
|
|||||||
)
|
)
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `Rp.${addThousandSeparators(
|
element.innerText = `Rp.${addThousandSeparators(
|
||||||
this.bigTotalNonVerification.toString()
|
// this.bigTotalNonVerification.toString()
|
||||||
|
this.resumeBigData.non_verified_document.sum.toString()
|
||||||
)}`;
|
)}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
@@ -441,45 +167,52 @@ class BigData {
|
|||||||
".small-percentage.chart-berkas-belum-terverifikasi"
|
".small-percentage.chart-berkas-belum-terverifikasi"
|
||||||
)
|
)
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.percentageResultNonVerification}%`;
|
// element.innerText = `${this.percentageResultNonVerification}%`;
|
||||||
|
element.innerText = `${this.resumeBigData.non_verified_document.percentage}%`;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
initChartUsaha() {
|
initChartUsaha() {
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-count.chart-business")
|
.querySelectorAll(".document-count.chart-business")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.dataBusiness.count}`;
|
// element.innerText = `${this.dataBusiness.count}`;
|
||||||
|
element.innerText = `${this.resumeBigData.business_document.count}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-total.chart-business")
|
.querySelectorAll(".document-total.chart-business")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `Rp.${addThousandSeparators(
|
element.innerText = `Rp.${addThousandSeparators(
|
||||||
this.bigTotalBusiness.toString()
|
// this.bigTotalBusiness.toString()
|
||||||
|
this.resumeBigData.business_document.sum.toString()
|
||||||
)}`;
|
)}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".small-percentage.chart-business")
|
.querySelectorAll(".small-percentage.chart-business")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.percentageResultBusiness}%`;
|
// element.innerText = `${this.percentageResultBusiness}%`;
|
||||||
|
element.innerText = `${this.resumeBigData.business_document.percentage}%`;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
initChartNonUsaha() {
|
initChartNonUsaha() {
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-count.chart-non-business")
|
.querySelectorAll(".document-count.chart-non-business")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.dataNonBusiness.count}`;
|
// element.innerText = `${this.dataNonBusiness.count}`;
|
||||||
|
element.innerText = `${this.resumeBigData.non_business_document.count}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-total.chart-non-business")
|
.querySelectorAll(".document-total.chart-non-business")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `Rp.${addThousandSeparators(
|
element.innerText = `Rp.${addThousandSeparators(
|
||||||
this.bigTotalNonBusiness.toString()
|
// this.bigTotalNonBusiness.toString()
|
||||||
|
this.resumeBigData.non_business_document.sum.toString()
|
||||||
)}`;
|
)}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".small-percentage.chart-non-business")
|
.querySelectorAll(".small-percentage.chart-non-business")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.percentageResultNonBusiness}%`;
|
// element.innerText = `${this.percentageResultNonBusiness}%`;
|
||||||
|
element.innerText = `${this.resumeBigData.non_business_document.percentage}%`;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
initChartKekuranganPotensi() {
|
initChartKekuranganPotensi() {
|
||||||
@@ -492,89 +225,97 @@ class BigData {
|
|||||||
.querySelectorAll(".document-total.chart-kekurangan-potensi")
|
.querySelectorAll(".document-total.chart-kekurangan-potensi")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `Rp.${addThousandSeparators(
|
element.innerText = `Rp.${addThousandSeparators(
|
||||||
this.totalKekuranganPotensi.toString()
|
// this.totalKekuranganPotensi.toString()
|
||||||
|
this.resumeBigData.kekurangan_potensi.sum.toString()
|
||||||
)}`;
|
)}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".small-percentage.chart-kekurangan-potensi")
|
.querySelectorAll(".small-percentage.chart-kekurangan-potensi")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.percentageKekuranganPotensi}%`;
|
// element.innerText = `${this.percentageKekuranganPotensi}%`;
|
||||||
|
element.innerText = `${this.resumeBigData.kekurangan_potensi.percentage}%`;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
initChartRealisasiTerbitPBG() {
|
initChartRealisasiTerbitPBG() {
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-count.chart-realisasi-tebit-pbg")
|
.querySelectorAll(".document-count.chart-realisasi-tebit-pbg")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.dataCountRealisasiTerbit}`;
|
// element.innerText = `${this.dataCountRealisasiTerbit}`;
|
||||||
|
element.innerText = `${this.resumeBigData.realisasi_terbit.count}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-total.chart-realisasi-tebit-pbg")
|
.querySelectorAll(".document-total.chart-realisasi-tebit-pbg")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `Rp.${addThousandSeparators(
|
element.innerText = `Rp.${addThousandSeparators(
|
||||||
this.dataSumRealisasiTerbit
|
// this.dataSumRealisasiTerbit
|
||||||
|
this.resumeBigData.realisasi_terbit.sum.toString()
|
||||||
)}`;
|
)}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".small-percentage.chart-realisasi-tebit-pbg")
|
.querySelectorAll(".small-percentage.chart-realisasi-tebit-pbg")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `0.00%`;
|
element.innerText = `${this.resumeBigData.realisasi_terbit.percentage}%`;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
initChartMenungguKlikDPMPTSP() {
|
initChartMenungguKlikDPMPTSP() {
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-count.chart-menunggu-klik-dpmptsp")
|
.querySelectorAll(".document-count.chart-menunggu-klik-dpmptsp")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.dataCountMenungguKlikDPMPTSP}`;
|
// element.innerText = `${this.dataCountMenungguKlikDPMPTSP}`;
|
||||||
|
element.innerText = `${this.resumeBigData.menunggu_klik_dpmptsp.count}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-total.chart-menunggu-klik-dpmptsp")
|
.querySelectorAll(".document-total.chart-menunggu-klik-dpmptsp")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `Rp.${addThousandSeparators(
|
element.innerText = `Rp.${addThousandSeparators(
|
||||||
this.dataSumMenungguKlikDPMPTSP
|
// this.dataSumMenungguKlikDPMPTSP
|
||||||
|
this.resumeBigData.menunggu_klik_dpmptsp.sum.toString()
|
||||||
)}`;
|
)}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".small-percentage.chart-menunggu-klik-dpmptsp")
|
.querySelectorAll(".small-percentage.chart-menunggu-klik-dpmptsp")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `0.00%`;
|
element.innerText = `${this.resumeBigData.menunggu_klik_dpmptsp.percentage}%`;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
initChartProsesDinasTeknis() {
|
initChartProsesDinasTeknis() {
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-count.chart-proses-dinas-teknis")
|
.querySelectorAll(".document-count.chart-proses-dinas-teknis")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.dataCountProsesDinasTeknis}`;
|
// element.innerText = `${this.dataCountProsesDinasTeknis}`;
|
||||||
|
element.innerText = `${this.resumeBigData.proses_dinas_teknis.count}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-total.chart-proses-dinas-teknis")
|
.querySelectorAll(".document-total.chart-proses-dinas-teknis")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `Rp.${addThousandSeparators(
|
element.innerText = `Rp.${addThousandSeparators(
|
||||||
this.dataSumProsesDinasTeknis
|
// this.dataSumProsesDinasTeknis
|
||||||
|
this.resumeBigData.proses_dinas_teknis.sum.toString()
|
||||||
)}`;
|
)}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".small-percentage.chart-proses-dinas-teknis")
|
.querySelectorAll(".small-percentage.chart-proses-dinas-teknis")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `0.00%`;
|
element.innerText = `${this.resumeBigData.proses_dinas_teknis.percentage}%`;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
initChartPotensiTataRuang() {
|
initChartPotensiTataRuang() {
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-count.chart-potensi-tata-ruang")
|
.querySelectorAll(".document-count.chart-potensi-tata-ruang")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = "";
|
element.innerText = `${this.resumeBigData.tata_ruang.count}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-total.chart-potensi-tata-ruang")
|
.querySelectorAll(".document-total.chart-potensi-tata-ruang")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `Rp.${addThousandSeparators(
|
element.innerText = `Rp.${addThousandSeparators(
|
||||||
this.bigTotalTataRuang.toString()
|
this.resumeBigData.tata_ruang.sum.toString()
|
||||||
)}`;
|
)}`;
|
||||||
});
|
});
|
||||||
document
|
document
|
||||||
.querySelectorAll(".small-percentage.chart-potensi-tata-ruang")
|
.querySelectorAll(".small-percentage.chart-potensi-tata-ruang")
|
||||||
.forEach((element) => {
|
.forEach((element) => {
|
||||||
element.innerText = `${this.percentageResultTataRuang}%`;
|
element.innerText = `${this.resumeBigData.tata_ruang.percentage}%`;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,21 +1,55 @@
|
|||||||
import Big from "big.js";
|
import Big from "big.js";
|
||||||
import GlobalConfig, { addThousandSeparators } from "../global-config.js";
|
import GlobalConfig, { addThousandSeparators } from "../global-config.js";
|
||||||
|
import InitDatePicker from "../utils/InitDatePicker.js";
|
||||||
|
|
||||||
class LackOfPotential {
|
class LackOfPotential {
|
||||||
async init() {
|
async init() {
|
||||||
|
new InitDatePicker(
|
||||||
|
"#datepicker-lack-of-potential",
|
||||||
|
this.handleChangedDate.bind(this)
|
||||||
|
).init();
|
||||||
this.bigTotalLackPotential = 0;
|
this.bigTotalLackPotential = 0;
|
||||||
this.totalPotensi = await this.getDataTotalPotensi(2025);
|
this.totalPotensi = await this.getDataTotalPotensi("latest");
|
||||||
this.totalTargetPAD = await this.getDataSettings("TARGET_PAD");
|
this.totalTargetPAD = await this.getDataSettings("TARGET_PAD");
|
||||||
|
this.allCountData = await this.getValueDashboard();
|
||||||
|
this.reklameCount = this.allCountData.total_reklame ?? 0;
|
||||||
|
this.pdamCount = this.allCountData.total_pdam ?? 0;
|
||||||
|
this.tataRuangCount = this.allCountData.total_tata_ruang ?? 0;
|
||||||
|
|
||||||
|
let dataReportTourism = this.allCountData.data_report;
|
||||||
|
|
||||||
|
this.totalVilla = dataReportTourism
|
||||||
|
.filter((item) => item.kbli_title.toLowerCase() === "vila")
|
||||||
|
.reduce((sum, item) => sum + item.total_records, 0);
|
||||||
|
this.totalRestoran = dataReportTourism
|
||||||
|
.filter((item) => item.kbli_title.toLowerCase() === "restoran")
|
||||||
|
.reduce((sum, item) => sum + item.total_records, 0);
|
||||||
|
this.totalPariwisata = dataReportTourism.reduce(
|
||||||
|
(sum, item) => sum + item.total_records,
|
||||||
|
0
|
||||||
|
);
|
||||||
|
|
||||||
this.bigTargetPAD = new Big(this.totalTargetPAD ?? 0);
|
this.bigTargetPAD = new Big(this.totalTargetPAD ?? 0);
|
||||||
this.bigTotalPotensi = new Big(this.totalPotensi.totalData ?? 0);
|
this.bigTotalPotensi = new Big(this.totalPotensi.total ?? 0);
|
||||||
this.bigTotalLackPotential = this.bigTargetPAD - this.bigTotalPotensi;
|
this.bigTotalLackPotential = this.bigTargetPAD.minus(
|
||||||
|
this.bigTotalPotensi
|
||||||
|
);
|
||||||
|
this.initChartKekuranganPotensi();
|
||||||
|
this.initDataValueDashboard();
|
||||||
|
}
|
||||||
|
async handleChangedDate(filterDate) {
|
||||||
|
const totalPotensi = await this.getDataTotalPotensi(filterDate);
|
||||||
|
this.bigTotalPotensi = new Big(totalPotensi.total ?? 0);
|
||||||
|
this.bigTotalLackPotential = this.bigTargetPAD.minus(
|
||||||
|
this.bigTotalPotensi
|
||||||
|
);
|
||||||
|
|
||||||
this.initChartKekuranganPotensi();
|
this.initChartKekuranganPotensi();
|
||||||
}
|
}
|
||||||
async getDataTotalPotensi(year) {
|
async getDataTotalPotensi(filterDate) {
|
||||||
try {
|
try {
|
||||||
const response = await fetch(
|
const response = await fetch(
|
||||||
`${GlobalConfig.apiHost}/api/all-task-documents?year=${year}`,
|
`${GlobalConfig.apiHost}/api/bigdata-resume?filterByDate=${filterDate}`,
|
||||||
{
|
{
|
||||||
credentials: "include",
|
credentials: "include",
|
||||||
headers: {
|
headers: {
|
||||||
@@ -34,8 +68,7 @@ class LackOfPotential {
|
|||||||
|
|
||||||
const data = await response.json();
|
const data = await response.json();
|
||||||
return {
|
return {
|
||||||
countData: data.data.count,
|
total: data.total_potensi.sum,
|
||||||
totalData: data.data.total,
|
|
||||||
};
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error fetching chart data:", error);
|
console.error("Error fetching chart data:", error);
|
||||||
@@ -45,7 +78,7 @@ class LackOfPotential {
|
|||||||
async getDataSettings(string_key) {
|
async getDataSettings(string_key) {
|
||||||
try {
|
try {
|
||||||
const response = await fetch(
|
const response = await fetch(
|
||||||
`${GlobalConfig.apiHost}/api/api-data-settings?search=${string_key}`,
|
`${GlobalConfig.apiHost}/api/data-settings?search=${string_key}`,
|
||||||
{
|
{
|
||||||
credentials: "include",
|
credentials: "include",
|
||||||
headers: {
|
headers: {
|
||||||
@@ -69,6 +102,33 @@ class LackOfPotential {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
async getValueDashboard() {
|
||||||
|
try {
|
||||||
|
const response = await fetch(
|
||||||
|
`${GlobalConfig.apiHost}/api/dashboard-potential-count`,
|
||||||
|
{
|
||||||
|
credentials: "include",
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${
|
||||||
|
document.querySelector("meta[name='api-token']")
|
||||||
|
.content
|
||||||
|
}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
console.error("Network response was not ok", response);
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
return data;
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error fetching chart data:", error);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
initChartKekuranganPotensi() {
|
initChartKekuranganPotensi() {
|
||||||
document
|
document
|
||||||
.querySelectorAll(".document-count.chart-lack-of-potential")
|
.querySelectorAll(".document-count.chart-lack-of-potential")
|
||||||
@@ -88,6 +148,21 @@ class LackOfPotential {
|
|||||||
element.innerText = ``;
|
element.innerText = ``;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
initDataValueDashboard() {
|
||||||
|
document.getElementById("reklame-count").innerText = this.reklameCount;
|
||||||
|
document.getElementById("pdam-count").innerText = this.pdamCount;
|
||||||
|
document.getElementById("pbb-bangunan-count").innerText =
|
||||||
|
this.tataRuangCount;
|
||||||
|
document.getElementById("tata-ruang-count").innerText =
|
||||||
|
this.tataRuangCount;
|
||||||
|
document.getElementById("tata-ruang-usaha-count").innerText =
|
||||||
|
this.tataRuangCount;
|
||||||
|
document.getElementById("restoran-count").innerText =
|
||||||
|
this.totalRestoran;
|
||||||
|
document.getElementById("villa-count").innerText = this.totalVilla;
|
||||||
|
document.getElementById("pariwisata-count").innerText =
|
||||||
|
this.totalPariwisata;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
document.addEventListener("DOMContentLoaded", async function (e) {
|
document.addEventListener("DOMContentLoaded", async function (e) {
|
||||||
await new LackOfPotential().init();
|
await new LackOfPotential().init();
|
||||||
|
|||||||
543
resources/js/dashboards/pbg.js
Normal file
543
resources/js/dashboards/pbg.js
Normal file
@@ -0,0 +1,543 @@
|
|||||||
|
import Big from "big.js";
|
||||||
|
import GlobalConfig, { addThousandSeparators } from "../global-config.js";
|
||||||
|
import ApexCharts from "apexcharts";
|
||||||
|
import "gridjs/dist/gridjs.umd.js";
|
||||||
|
import GeneralTable from "../table-generator.js";
|
||||||
|
|
||||||
|
var chart;
|
||||||
|
document.addEventListener("DOMContentLoaded", async function () {
|
||||||
|
await initChart();
|
||||||
|
const yearPicker = document.getElementById("yearPicker");
|
||||||
|
|
||||||
|
async function updateDataByYear(selectedYear) {
|
||||||
|
// Target PAD Element
|
||||||
|
const targetPadElement = document.getElementById("target-pad");
|
||||||
|
if (!targetPadElement) return;
|
||||||
|
const targetPadValue = await getDataSettings("TARGET_PAD");
|
||||||
|
targetPadElement.textContent = formatCurrency(targetPadValue);
|
||||||
|
|
||||||
|
// Total Potensi Berkas
|
||||||
|
const totalPotensiBerkas = document.getElementById(
|
||||||
|
"total-potensi-berkas"
|
||||||
|
);
|
||||||
|
if (!totalPotensiBerkas) return;
|
||||||
|
const totalPotensiBerkasValue = await getDataTotalPotensi(selectedYear);
|
||||||
|
totalPotensiBerkas.textContent = formatCurrency(
|
||||||
|
totalPotensiBerkasValue.totalData
|
||||||
|
);
|
||||||
|
|
||||||
|
// Total Berkas Terverifikasi
|
||||||
|
const totalBerkasTerverifikasi = document.getElementById(
|
||||||
|
"total-berkas-terverifikasi"
|
||||||
|
);
|
||||||
|
if (!totalBerkasTerverifikasi) return;
|
||||||
|
const totalBerkasTerverifikasiValue = await getDataVerification(
|
||||||
|
selectedYear
|
||||||
|
);
|
||||||
|
totalBerkasTerverifikasi.textContent = formatCurrency(
|
||||||
|
totalBerkasTerverifikasiValue.totalData
|
||||||
|
);
|
||||||
|
|
||||||
|
// Total Kekurangan potensi
|
||||||
|
const totalKekuranganPotensi = document.getElementById(
|
||||||
|
"total-kekurangan-potensi"
|
||||||
|
);
|
||||||
|
if (!totalKekuranganPotensi) return;
|
||||||
|
const totalKekuranganPotensiValue =
|
||||||
|
new Big(targetPadValue) -
|
||||||
|
new Big(totalPotensiBerkasValue.totalData);
|
||||||
|
totalKekuranganPotensi.textContent = formatCurrency(
|
||||||
|
totalKekuranganPotensiValue
|
||||||
|
);
|
||||||
|
|
||||||
|
// Total Potensi PBG dari tata ruang
|
||||||
|
const totalPotensiPBGTataRuang = document.getElementById(
|
||||||
|
"total-potensi-pbd-tata-ruang"
|
||||||
|
);
|
||||||
|
if (!totalPotensiPBGTataRuang) return;
|
||||||
|
const totalPotensiPBGTataRuangValue = await getDataSettings(
|
||||||
|
"TATA_RUANG"
|
||||||
|
);
|
||||||
|
totalPotensiPBGTataRuang.textContent = formatCurrency(
|
||||||
|
totalPotensiPBGTataRuangValue
|
||||||
|
);
|
||||||
|
|
||||||
|
// Total Berkas Belum terverifikasi
|
||||||
|
const totalBerkasBelumTerverifikasi = document.getElementById(
|
||||||
|
"total-berkas-belum-terverifikasi"
|
||||||
|
);
|
||||||
|
if (!totalBerkasBelumTerverifikasi) return;
|
||||||
|
const totalBerkasBelumTerverifikasiValue = await getDataNonVerification(
|
||||||
|
selectedYear
|
||||||
|
);
|
||||||
|
const totalBerkasBelumTerverifikasiCount =
|
||||||
|
totalBerkasBelumTerverifikasiValue.countData;
|
||||||
|
totalBerkasBelumTerverifikasi.textContent = formatCurrency(
|
||||||
|
totalBerkasBelumTerverifikasiValue.totalData
|
||||||
|
);
|
||||||
|
|
||||||
|
// Total Berkas Usaha
|
||||||
|
const totalBerkasUsahaValue = await getDataBusiness(selectedYear);
|
||||||
|
const totalBerkasUsahaCount = totalBerkasUsahaValue.countData;
|
||||||
|
const totalBerkasUsahaTotalData = totalBerkasUsahaValue.totalData;
|
||||||
|
|
||||||
|
// Total Berkas Non Usaha
|
||||||
|
const totalBerkasNonUsahaValue = await getDataNonBusiness(selectedYear);
|
||||||
|
const totalBerkasNonUsahaCount = totalBerkasNonUsahaValue.countData;
|
||||||
|
const totalBerkasNonUsahaTotalData = totalBerkasNonUsahaValue.totalData;
|
||||||
|
|
||||||
|
// Pie Chart Section
|
||||||
|
let persenUsaha =
|
||||||
|
totalBerkasBelumTerverifikasiCount > 0
|
||||||
|
? (
|
||||||
|
(totalBerkasUsahaCount /
|
||||||
|
totalBerkasBelumTerverifikasiCount) *
|
||||||
|
100
|
||||||
|
).toFixed(2)
|
||||||
|
: "0";
|
||||||
|
|
||||||
|
let persenNonUsaha =
|
||||||
|
totalBerkasBelumTerverifikasiCount > 0
|
||||||
|
? (
|
||||||
|
(totalBerkasNonUsahaCount /
|
||||||
|
totalBerkasBelumTerverifikasiCount) *
|
||||||
|
100
|
||||||
|
).toFixed(2)
|
||||||
|
: "0";
|
||||||
|
|
||||||
|
const dataSeriesPieChart = [
|
||||||
|
Number(persenUsaha),
|
||||||
|
Number(persenNonUsaha),
|
||||||
|
];
|
||||||
|
const labelsPieChart = ["Berkas Usaha", "Berkas Non Usaha"];
|
||||||
|
document.querySelector("td[data-category='non-usaha']").textContent =
|
||||||
|
formatCurrency(totalBerkasNonUsahaTotalData).toLocaleString();
|
||||||
|
document.querySelector(
|
||||||
|
"td[data-category='non-usaha-percentage']"
|
||||||
|
).textContent = persenNonUsaha + "%";
|
||||||
|
|
||||||
|
document.querySelector("td[data-category='usaha']").textContent =
|
||||||
|
formatCurrency(totalBerkasUsahaTotalData).toLocaleString();
|
||||||
|
document.querySelector(
|
||||||
|
"td[data-category='usaha-percentage']"
|
||||||
|
).textContent = persenUsaha + "%";
|
||||||
|
|
||||||
|
updatePieChart(dataSeriesPieChart, labelsPieChart);
|
||||||
|
|
||||||
|
// Load all Tourism location
|
||||||
|
const allLocation = await getAllLocation();
|
||||||
|
console.log(allLocation);
|
||||||
|
|
||||||
|
// Filter hanya data yang memiliki angka valid
|
||||||
|
let validLocations = allLocation.dataLocation.filter((loc) => {
|
||||||
|
return (
|
||||||
|
!isNaN(parseFloat(loc.longitude)) &&
|
||||||
|
!isNaN(parseFloat(loc.latitude))
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Ubah string ke float
|
||||||
|
validLocations = validLocations.map((loc) => ({
|
||||||
|
name: loc.project_name,
|
||||||
|
longitude: parseFloat(loc.longitude),
|
||||||
|
latitude: parseFloat(loc.latitude),
|
||||||
|
}));
|
||||||
|
|
||||||
|
console.log(validLocations.name);
|
||||||
|
|
||||||
|
// Inisialisasi peta
|
||||||
|
var map = L.map("map").setView([-7.023, 107.5275], 10);
|
||||||
|
L.tileLayer("https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png", {
|
||||||
|
attribution: "© OpenStreetMap contributors",
|
||||||
|
}).addTo(map);
|
||||||
|
|
||||||
|
// Tambahkan marker ke peta
|
||||||
|
validLocations.forEach(function (loc) {
|
||||||
|
L.marker([loc.latitude, loc.longitude])
|
||||||
|
.addTo(map)
|
||||||
|
.bindPopup(`<b>${loc.name}</b>`) // Popup saat diklik
|
||||||
|
.bindTooltip(loc.name, { permanent: false, direction: "top" }); // Tooltip saat di-hover
|
||||||
|
});
|
||||||
|
|
||||||
|
// Realisasi terbit PBG
|
||||||
|
const totalRealisasiTerbitPBG = document.getElementById(
|
||||||
|
"realisasi-terbit-pbg"
|
||||||
|
);
|
||||||
|
if (!totalRealisasiTerbitPBG) return;
|
||||||
|
const totalRealisasiTerbitPBGValue = await getDataSettings(
|
||||||
|
"REALISASI_TERBIT_PBG_SUM"
|
||||||
|
);
|
||||||
|
totalRealisasiTerbitPBG.textContent = formatCurrency(
|
||||||
|
totalRealisasiTerbitPBGValue
|
||||||
|
);
|
||||||
|
|
||||||
|
// Menunggu Klik DPMPTSP
|
||||||
|
const totalMenungguKlikDpmptsp = document.getElementById(
|
||||||
|
"waiting-click-dpmptsp"
|
||||||
|
);
|
||||||
|
if (!totalMenungguKlikDpmptsp) return;
|
||||||
|
const totalMenungguKlikDpmptspValue = await getDataSettings(
|
||||||
|
"MENUNGGU_KLIK_DPMPTSP_SUM"
|
||||||
|
);
|
||||||
|
totalMenungguKlikDpmptsp.textContent = formatCurrency(
|
||||||
|
totalMenungguKlikDpmptspValue
|
||||||
|
);
|
||||||
|
|
||||||
|
// Proses Dinas Teknis
|
||||||
|
const totalProsesDinasTeknis = document.getElementById(
|
||||||
|
"processing-technical-services"
|
||||||
|
);
|
||||||
|
if (!totalProsesDinasTeknis) return;
|
||||||
|
const totalProsesDinasTeknisValue = await getDataSettings(
|
||||||
|
"PROSES_DINAS_TEKNIS_SUM"
|
||||||
|
);
|
||||||
|
totalProsesDinasTeknis.textContent = formatCurrency(
|
||||||
|
totalProsesDinasTeknisValue
|
||||||
|
);
|
||||||
|
|
||||||
|
// Load Tabel Baru di Update
|
||||||
|
const tableLastUpdated = new GeneralTable(
|
||||||
|
"pbg-filter-by-updated-at",
|
||||||
|
`${GlobalConfig.apiHost}/api/api-pbg-task?isLastUpdated=true`,
|
||||||
|
`${GlobalConfig.apiHost}`,
|
||||||
|
pbgTaskColumns
|
||||||
|
);
|
||||||
|
|
||||||
|
tableLastUpdated.processData = function (data) {
|
||||||
|
console.log("Response Data:", data);
|
||||||
|
return data.data.map((item) => {
|
||||||
|
return [
|
||||||
|
item.no,
|
||||||
|
item.name,
|
||||||
|
item.registration_number,
|
||||||
|
item.document_number,
|
||||||
|
item.address,
|
||||||
|
];
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
tableLastUpdated.init();
|
||||||
|
|
||||||
|
// Load Tabel Status SK Terbit
|
||||||
|
const tableSKPBGTerbit = new GeneralTable(
|
||||||
|
"pbg-filter-by-status",
|
||||||
|
`${GlobalConfig.apiHost}/api/api-pbg-task?isLastUpdated=false`,
|
||||||
|
`${GlobalConfig.apiHost}`,
|
||||||
|
pbgTaskColumns
|
||||||
|
);
|
||||||
|
|
||||||
|
tableSKPBGTerbit.processData = function (data) {
|
||||||
|
console.log("Response Data:", data);
|
||||||
|
return data.data.map((item) => {
|
||||||
|
return [
|
||||||
|
item.no,
|
||||||
|
item.name,
|
||||||
|
item.registration_number,
|
||||||
|
item.document_number,
|
||||||
|
item.address,
|
||||||
|
];
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
tableSKPBGTerbit.init();
|
||||||
|
|
||||||
|
document.querySelector(
|
||||||
|
"#pbg-filter-by-updated-at .gridjs-search"
|
||||||
|
).hidden = true;
|
||||||
|
document.querySelector(
|
||||||
|
"#pbg-filter-by-updated-at .gridjs-footer"
|
||||||
|
).hidden = true;
|
||||||
|
document.querySelector(
|
||||||
|
"#pbg-filter-by-status .gridjs-search"
|
||||||
|
).hidden = true;
|
||||||
|
document.querySelector(
|
||||||
|
"#pbg-filter-by-status .gridjs-footer"
|
||||||
|
).hidden = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
await updateDataByYear(yearPicker.value);
|
||||||
|
|
||||||
|
yearPicker.addEventListener("change", async function () {
|
||||||
|
console.log("event change dropdown");
|
||||||
|
await updateDataByYear(yearPicker.value);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
async function getDataSettings(string_key) {
|
||||||
|
try {
|
||||||
|
const response = await fetch(
|
||||||
|
`${GlobalConfig.apiHost}/api/data-settings?search=${string_key}`,
|
||||||
|
{
|
||||||
|
credentials: "include",
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${
|
||||||
|
document.querySelector("meta[name='api-token']").content
|
||||||
|
}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
console.error("Network response was not ok", response);
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
return data.data?.[0]?.value ?? 0; // Pastikan tidak error jika data kosong
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error fetching data:", error);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getDataTotalPotensi(year) {
|
||||||
|
try {
|
||||||
|
const response = await fetch(
|
||||||
|
`${GlobalConfig.apiHost}/api/all-task-documents?year=${year}`,
|
||||||
|
{
|
||||||
|
credentials: "include",
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${
|
||||||
|
document.querySelector("meta[name='api-token']").content
|
||||||
|
}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
console.error("Network response was not ok", response);
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
return {
|
||||||
|
totalData: data.data.total,
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error fetching chart data:", error);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getDataVerification(year) {
|
||||||
|
try {
|
||||||
|
const response = await fetch(
|
||||||
|
`${GlobalConfig.apiHost}/api/verification-documents?year=${year}`,
|
||||||
|
{
|
||||||
|
credentials: "include",
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${
|
||||||
|
document.querySelector("meta[name='api-token']").content
|
||||||
|
}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
console.error("Network response was not ok", response);
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
return {
|
||||||
|
totalData: data.data.total,
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error fetching chart data:", error);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getDataNonVerification(year) {
|
||||||
|
try {
|
||||||
|
const response = await fetch(
|
||||||
|
`${GlobalConfig.apiHost}/api/non-verification-documents?year=${year}`,
|
||||||
|
{
|
||||||
|
credentials: "include",
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${
|
||||||
|
document.querySelector("meta[name='api-token']").content
|
||||||
|
}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
console.error("Network response was not ok", response);
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
return {
|
||||||
|
countData: data.data.count,
|
||||||
|
totalData: data.data.total,
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error fetching chart data:", error);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getDataBusiness(year) {
|
||||||
|
try {
|
||||||
|
const response = await fetch(
|
||||||
|
`${GlobalConfig.apiHost}/api/business-documents?year=${year}`,
|
||||||
|
{
|
||||||
|
credentials: "include",
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${
|
||||||
|
document.querySelector("meta[name='api-token']").content
|
||||||
|
}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
console.error("Network response was not ok", response);
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
return {
|
||||||
|
countData: data.data.count,
|
||||||
|
totalData: data.data.total,
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error fetching chart data:", error);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getDataNonBusiness(year) {
|
||||||
|
try {
|
||||||
|
const response = await fetch(
|
||||||
|
`${GlobalConfig.apiHost}/api/non-business-documents?year=${year}`,
|
||||||
|
{
|
||||||
|
credentials: "include",
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${
|
||||||
|
document.querySelector("meta[name='api-token']").content
|
||||||
|
}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
console.error("Network response was not ok", response);
|
||||||
|
}
|
||||||
|
const data = await response.json();
|
||||||
|
return {
|
||||||
|
countData: data.data.count,
|
||||||
|
totalData: data.data.total,
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error fetching chart data:", error);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getAllLocation() {
|
||||||
|
try {
|
||||||
|
const response = await fetch(
|
||||||
|
`${GlobalConfig.apiHost}/api/get-all-location`,
|
||||||
|
{
|
||||||
|
credentials: "include",
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${
|
||||||
|
document.querySelector("meta[name='api-token']").content
|
||||||
|
}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
if (!response.ok) {
|
||||||
|
console.error("Network response was not ok", response);
|
||||||
|
}
|
||||||
|
const data = await response.json();
|
||||||
|
return {
|
||||||
|
dataLocation: data.data,
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error fetching chart data:", error);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function initChart() {
|
||||||
|
var options = {
|
||||||
|
chart: {
|
||||||
|
height: 180,
|
||||||
|
type: "donut",
|
||||||
|
},
|
||||||
|
series: [0, 0], // Inisialisasi dengan nilai awal
|
||||||
|
labels: ["Berkas Usaha", "Berkas Non Usaha"],
|
||||||
|
legend: {
|
||||||
|
show: false,
|
||||||
|
},
|
||||||
|
stroke: {
|
||||||
|
width: 0,
|
||||||
|
},
|
||||||
|
plotOptions: {
|
||||||
|
pie: {
|
||||||
|
donut: {
|
||||||
|
size: "60%",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
colors: ["#7e67fe", "#17c553"],
|
||||||
|
dataLabels: {
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
responsive: [
|
||||||
|
{
|
||||||
|
breakpoint: 480,
|
||||||
|
options: {
|
||||||
|
chart: {
|
||||||
|
width: 200,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
fill: {
|
||||||
|
type: "gradient",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
chart = new ApexCharts(document.querySelector("#conversions"), options);
|
||||||
|
chart.render();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function updatePieChart(dataSeries, labels) {
|
||||||
|
if (!Array.isArray(dataSeries) || dataSeries.length === 0) {
|
||||||
|
console.error("Data series tidak valid:", dataSeries);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Perbarui data series chart
|
||||||
|
chart.updateSeries(dataSeries);
|
||||||
|
|
||||||
|
// Perbarui label jika diperlukan
|
||||||
|
if (Array.isArray(labels) && labels.length === dataSeries.length) {
|
||||||
|
chart.updateOptions({
|
||||||
|
labels: labels,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fungsi untuk format angka ke Rupiah
|
||||||
|
function formatCurrency(number) {
|
||||||
|
return new Intl.NumberFormat("id-ID", {
|
||||||
|
style: "currency",
|
||||||
|
currency: "IDR",
|
||||||
|
minimumFractionDigits: 0,
|
||||||
|
}).format(number);
|
||||||
|
}
|
||||||
|
|
||||||
|
const pbgTaskColumns = [
|
||||||
|
"No",
|
||||||
|
"Name",
|
||||||
|
"Nomor Registrasi",
|
||||||
|
"Nomor Dokumen",
|
||||||
|
"Alamat",
|
||||||
|
];
|
||||||
@@ -24,10 +24,9 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
method: "POST",
|
method: "POST",
|
||||||
credentials: "include",
|
credentials: "include",
|
||||||
headers: {
|
headers: {
|
||||||
"X-CSRF-TOKEN": document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="csrf-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content"),
|
.getAttribute("content")}`,
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
},
|
||||||
body: formData,
|
body: formData,
|
||||||
});
|
});
|
||||||
@@ -35,7 +34,7 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
let result = await response.json();
|
let result = await response.json();
|
||||||
document.getElementById("toast-message").innerText =
|
document.getElementById("toast-message").innerText =
|
||||||
result.message;
|
result.data.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = "/data-settings";
|
window.location.href = "/data-settings";
|
||||||
@@ -46,6 +45,8 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
error.message;
|
error.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
console.error("Error:", error);
|
console.error("Error:", error);
|
||||||
|
submitButton.disabled = false;
|
||||||
|
spinner.classList.add("d-none");
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Request failed:", error);
|
console.error("Request failed:", error);
|
||||||
|
|||||||
@@ -2,46 +2,34 @@ import { Grid } from "gridjs/dist/gridjs.umd.js";
|
|||||||
import gridjs from "gridjs/dist/gridjs.umd.js";
|
import gridjs from "gridjs/dist/gridjs.umd.js";
|
||||||
import "gridjs/dist/gridjs.umd.js";
|
import "gridjs/dist/gridjs.umd.js";
|
||||||
import GlobalConfig from "../global-config.js";
|
import GlobalConfig from "../global-config.js";
|
||||||
|
import Swal from "sweetalert2";
|
||||||
|
|
||||||
class DataSettings {
|
class DataSettings {
|
||||||
constructor() {
|
constructor() {
|
||||||
this.table = null; // Store Grid.js instance
|
this.toastMessage = document.getElementById("toast-message");
|
||||||
|
this.toastElement = document.getElementById("toastNotification");
|
||||||
|
this.toast = new bootstrap.Toast(this.toastElement);
|
||||||
|
this.table = null;
|
||||||
|
|
||||||
|
// Initialize functions
|
||||||
|
this.initTableDataSettings();
|
||||||
|
this.initEvents();
|
||||||
}
|
}
|
||||||
init() {
|
initEvents() {
|
||||||
this.getFetchApiData();
|
document.body.addEventListener("click", async (event) => {
|
||||||
|
const deleteButton = event.target.closest(
|
||||||
|
".btn-delete-data-settings"
|
||||||
|
);
|
||||||
|
if (deleteButton) {
|
||||||
|
event.preventDefault();
|
||||||
|
await this.handleDelete(deleteButton);
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
getFetchApiData() {
|
initTableDataSettings() {
|
||||||
let tableContainer = document.getElementById("table-data-settings");
|
let tableContainer = document.getElementById("table-data-settings");
|
||||||
|
// Create a new Grid.js instance only if it doesn't exist
|
||||||
if (this.table) {
|
|
||||||
// If table exists, update its data instead of recreating
|
|
||||||
this.table
|
|
||||||
.updateConfig({
|
|
||||||
server: {
|
|
||||||
url: `${GlobalConfig.apiHost}/api/api-data-settings`,
|
|
||||||
credentials: "include",
|
|
||||||
headers: {
|
|
||||||
Authorization: `Bearer ${document
|
|
||||||
.querySelector('meta[name="api-token"]')
|
|
||||||
.getAttribute("content")}`,
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
then: (data) =>
|
|
||||||
data.data.map((item) => [
|
|
||||||
item.id,
|
|
||||||
item.key,
|
|
||||||
item.value,
|
|
||||||
item.created_at,
|
|
||||||
item.id,
|
|
||||||
]),
|
|
||||||
total: (data) => data.meta.total,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
.forceRender();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.table = new Grid({
|
this.table = new Grid({
|
||||||
columns: [
|
columns: [
|
||||||
"ID",
|
"ID",
|
||||||
@@ -65,11 +53,6 @@ class DataSettings {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
search: {
|
|
||||||
server: {
|
|
||||||
url: (prev, keyword) => `${prev}?search=${keyword}`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
pagination: {
|
pagination: {
|
||||||
limit: 15,
|
limit: 15,
|
||||||
server: {
|
server: {
|
||||||
@@ -80,8 +63,13 @@ class DataSettings {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
sort: true,
|
sort: true,
|
||||||
|
search: {
|
||||||
server: {
|
server: {
|
||||||
url: `${GlobalConfig.apiHost}/api/api-data-settings`,
|
url: (prev, keyword) => `${prev}?search=${keyword}`,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
server: {
|
||||||
|
url: `${GlobalConfig.apiHost}/api/data-settings`,
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
@@ -99,116 +87,61 @@ class DataSettings {
|
|||||||
total: (data) => data.meta.total,
|
total: (data) => data.meta.total,
|
||||||
},
|
},
|
||||||
}).render(tableContainer);
|
}).render(tableContainer);
|
||||||
|
|
||||||
document.addEventListener("click", this.handleDelete.bind(this));
|
|
||||||
}
|
}
|
||||||
handleDelete(event) {
|
async handleDelete(deleteButton) {
|
||||||
if (event.target.classList.contains("btn-delete-data-settings")) {
|
const id = deleteButton.getAttribute("data-id");
|
||||||
event.preventDefault();
|
|
||||||
const id = event.target.getAttribute("data-id");
|
|
||||||
let modalElement = document.getElementById("modalConfirmation");
|
|
||||||
let toastMessage = document.getElementById("toast-message");
|
|
||||||
|
|
||||||
if (!modalElement) {
|
const result = await Swal.fire({
|
||||||
console.error("Modal element not found!");
|
title: "Are you sure?",
|
||||||
return;
|
text: "You won't be able to revert this!",
|
||||||
}
|
icon: "warning",
|
||||||
|
showCancelButton: true,
|
||||||
let modal = new bootstrap.Modal(modalElement);
|
confirmButtonColor: "#3085d6",
|
||||||
let btnSaveConfirmation = document.getElementById(
|
cancelButtonColor: "#d33",
|
||||||
"btnSaveConfirmation"
|
confirmButtonText: "Yes, delete it!",
|
||||||
);
|
});
|
||||||
let toastElement = document.getElementById("toastNotification");
|
|
||||||
let toast = new bootstrap.Toast(toastElement);
|
|
||||||
|
|
||||||
// Remove previous event listeners to avoid multiple bindings
|
|
||||||
btnSaveConfirmation.replaceWith(
|
|
||||||
btnSaveConfirmation.cloneNode(true)
|
|
||||||
);
|
|
||||||
btnSaveConfirmation = document.getElementById(
|
|
||||||
"btnSaveConfirmation"
|
|
||||||
);
|
|
||||||
|
|
||||||
// Set the role ID on the confirm button inside the modal
|
|
||||||
btnSaveConfirmation.setAttribute("data-settings-id", id);
|
|
||||||
|
|
||||||
// Show the modal
|
|
||||||
modal.show();
|
|
||||||
|
|
||||||
btnSaveConfirmation.addEventListener("click", async () => {
|
|
||||||
let dataSettingId =
|
|
||||||
btnSaveConfirmation.getAttribute("data-settings-id");
|
|
||||||
|
|
||||||
|
if (result.isConfirmed) {
|
||||||
try {
|
try {
|
||||||
let response = await fetch(
|
let response = await fetch(
|
||||||
`/data-settings/${dataSettingId}`,
|
`${GlobalConfig.apiHost}/api/data-settings/${id}`,
|
||||||
{
|
{
|
||||||
method: "DELETE",
|
method: "DELETE",
|
||||||
credentials: "include",
|
credentials: "include",
|
||||||
headers: {
|
|
||||||
"X-CSRF-TOKEN": document
|
|
||||||
.querySelector('meta[name="csrf-token"]')
|
|
||||||
.getAttribute("content"),
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
if (response.ok) {
|
|
||||||
let result = await response.json();
|
|
||||||
toastMessage.innerText =
|
|
||||||
result.message || "Deleted successfully!";
|
|
||||||
toast.show();
|
|
||||||
|
|
||||||
// Hide modal
|
|
||||||
modal.hide();
|
|
||||||
|
|
||||||
// Refresh Grid.js table
|
|
||||||
this.refreshDataSettings();
|
|
||||||
} else {
|
|
||||||
let error = await response.json();
|
|
||||||
console.error("Delete failed:", error);
|
|
||||||
toastMessage.innerText =
|
|
||||||
error.message || "Delete failed!";
|
|
||||||
toast.show();
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error("Error deleting item:", error);
|
|
||||||
toastMessage.innerText = "An error occurred!";
|
|
||||||
toast.show();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
refreshDataSettings() {
|
|
||||||
if (this.table) {
|
|
||||||
this.table
|
|
||||||
.updateConfig({
|
|
||||||
server: {
|
|
||||||
url: `${GlobalConfig.apiHost}/api/api-data-settings`,
|
|
||||||
credentials: "include",
|
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`,
|
.getAttribute("content")}`,
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
},
|
},
|
||||||
then: (data) =>
|
}
|
||||||
data.data.map((item) => [
|
);
|
||||||
item.id,
|
|
||||||
item.key,
|
if (response.ok) {
|
||||||
item.value,
|
let result = await response.json();
|
||||||
item.created_at,
|
this.toastMessage.innerText =
|
||||||
item.id,
|
result.message || "Deleted successfully!";
|
||||||
]),
|
this.toast.show();
|
||||||
total: (data) => data.meta.total,
|
|
||||||
},
|
// Refresh Grid.js table
|
||||||
})
|
if (typeof this.table !== "undefined") {
|
||||||
.forceRender();
|
this.table.updateConfig({}).forceRender();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
let error = await response.json();
|
||||||
|
console.error("Delete failed:", error);
|
||||||
|
this.toastMessage.innerText =
|
||||||
|
error.message || "Delete failed!";
|
||||||
|
this.toast.show();
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error deleting item:", error);
|
||||||
|
this.toastMessage.innerText = "An error occurred!";
|
||||||
|
this.toast.show();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
document.addEventListener("DOMContentLoaded", function (e) {
|
document.addEventListener("DOMContentLoaded", function (e) {
|
||||||
new DataSettings().init();
|
new DataSettings();
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -24,16 +24,16 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
let response = await fetch(form.action, {
|
let response = await fetch(form.action, {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
"X-CSRF-TOKEN": document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="csrf-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content"),
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
body: formData,
|
body: formData,
|
||||||
});
|
});
|
||||||
|
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
let result = await response.json();
|
let result = await response.json();
|
||||||
toastMessage.innerText = result.message;
|
toastMessage.innerText = result.data.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = "/data-settings";
|
window.location.href = "/data-settings";
|
||||||
@@ -43,6 +43,8 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
toastMessage.innerText = error.message;
|
toastMessage.innerText = error.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
console.error("Error:", error);
|
console.error("Error:", error);
|
||||||
|
submitButton.disabled = false;
|
||||||
|
spinner.classList.add("d-none");
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Request failed:", error);
|
console.error("Request failed:", error);
|
||||||
|
|||||||
@@ -16,10 +16,10 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
Loading...
|
Loading...
|
||||||
`;
|
`;
|
||||||
const isEdit = saveButton.classList.contains("btn-edit");
|
const isEdit = saveButton.classList.contains("btn-edit");
|
||||||
const formData = new FormData(form)
|
const formData = new FormData(form);
|
||||||
const toast = document.getElementById('toastEditUpdate');
|
const toast = document.getElementById("toastEditUpdate");
|
||||||
const toastBody = toast.querySelector('.toast-body');
|
const toastBody = toast.querySelector(".toast-body");
|
||||||
const toastHeader = toast.querySelector('.toast-header small');
|
const toastHeader = toast.querySelector(".toast-header small");
|
||||||
|
|
||||||
const data = {};
|
const data = {};
|
||||||
|
|
||||||
@@ -40,52 +40,54 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`,
|
.getAttribute("content")}`,
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
}
|
},
|
||||||
})
|
})
|
||||||
.then(response => response.json())
|
.then((response) => response.json())
|
||||||
.then(data => {
|
.then((data) => {
|
||||||
if (!data.errors) {
|
if (!data.errors) {
|
||||||
// Remove existing icon (if any) before adding the new one
|
// Remove existing icon (if any) before adding the new one
|
||||||
if (authLogo) {
|
if (authLogo) {
|
||||||
// Hapus ikon yang sudah ada jika ada
|
// Hapus ikon yang sudah ada jika ada
|
||||||
const existingIcon = authLogo.querySelector('.bx');
|
const existingIcon = authLogo.querySelector(".bx");
|
||||||
if (existingIcon) {
|
if (existingIcon) {
|
||||||
authLogo.removeChild(existingIcon);
|
authLogo.removeChild(existingIcon);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Buat ikon baru
|
// Buat ikon baru
|
||||||
const icon = document.createElement('i');
|
const icon = document.createElement("i");
|
||||||
icon.classList.add('bx', 'bxs-check-square');
|
icon.classList.add("bx", "bxs-check-square");
|
||||||
icon.style.fontSize = '25px';
|
icon.style.fontSize = "25px";
|
||||||
icon.style.color = 'green'; // Pastikan 'green' dalam bentuk string
|
icon.style.color = "green"; // Pastikan 'green' dalam bentuk string
|
||||||
|
|
||||||
// Tambahkan ikon ke dalam auth-logo
|
// Tambahkan ikon ke dalam auth-logo
|
||||||
authLogo.appendChild(icon);
|
authLogo.appendChild(icon);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set success message for the toast
|
// Set success message for the toast
|
||||||
toastBody.textContent = isEdit ? "Data updated successfully!" : "Data created successfully!";
|
toastBody.textContent = isEdit
|
||||||
toast.classList.add('show'); // Show the toast
|
? "Data updated successfully!"
|
||||||
|
: "Data created successfully!";
|
||||||
|
toast.classList.add("show"); // Show the toast
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
toast.classList.remove('show'); // Hide the toast after 3 seconds
|
toast.classList.remove("show"); // Hide the toast after 3 seconds
|
||||||
}, 2000);
|
}, 2000);
|
||||||
|
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = '/data/advertisements';
|
window.location.href = "/data/web-advertisements";
|
||||||
}, 1000);
|
}, 1000);
|
||||||
} else {
|
} else {
|
||||||
if (authLogo) {
|
if (authLogo) {
|
||||||
// Hapus ikon yang sudah ada jika ada
|
// Hapus ikon yang sudah ada jika ada
|
||||||
const existingIcon = authLogo.querySelector('.bx');
|
const existingIcon = authLogo.querySelector(".bx");
|
||||||
if (existingIcon) {
|
if (existingIcon) {
|
||||||
authLogo.removeChild(existingIcon);
|
authLogo.removeChild(existingIcon);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Buat ikon baru
|
// Buat ikon baru
|
||||||
const icon = document.createElement('i');
|
const icon = document.createElement("i");
|
||||||
icon.classList.add('bx', 'bxs-error-alt');
|
icon.classList.add("bx", "bxs-error-alt");
|
||||||
icon.style.fontSize = '25px';
|
icon.style.fontSize = "25px";
|
||||||
icon.style.color = 'red'; // Pastikan 'green' dalam bentuk string
|
icon.style.color = "red"; // Pastikan 'green' dalam bentuk string
|
||||||
|
|
||||||
// Tambahkan ikon ke dalam auth-logo
|
// Tambahkan ikon ke dalam auth-logo
|
||||||
authLogo.appendChild(icon);
|
authLogo.appendChild(icon);
|
||||||
@@ -96,27 +98,28 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
modalButton.innerHTML = isEdit ? "Update" : "Create";
|
modalButton.innerHTML = isEdit ? "Update" : "Create";
|
||||||
|
|
||||||
// Set error message for the toast
|
// Set error message for the toast
|
||||||
toastBody.textContent = "Failed: " + (data.message || "Something went wrong");
|
toastBody.textContent =
|
||||||
toast.classList.add('show'); // Show the toast
|
"Failed: " + (data.message || "Something went wrong");
|
||||||
|
toast.classList.add("show"); // Show the toast
|
||||||
|
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
toast.classList.remove('show'); // Hide the toast after 3 seconds
|
toast.classList.remove("show"); // Hide the toast after 3 seconds
|
||||||
}, 3000);
|
}, 3000);
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.catch(errors => {
|
.catch((errors) => {
|
||||||
if (authLogo) {
|
if (authLogo) {
|
||||||
// Hapus ikon yang sudah ada jika ada
|
// Hapus ikon yang sudah ada jika ada
|
||||||
const existingIcon = authLogo.querySelector('.bx');
|
const existingIcon = authLogo.querySelector(".bx");
|
||||||
if (existingIcon) {
|
if (existingIcon) {
|
||||||
authLogo.removeChild(existingIcon);
|
authLogo.removeChild(existingIcon);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Buat ikon baru
|
// Buat ikon baru
|
||||||
const icon = document.createElement('i');
|
const icon = document.createElement("i");
|
||||||
icon.classList.add('bx', 'bxs-error-alt');
|
icon.classList.add("bx", "bxs-error-alt");
|
||||||
icon.style.fontSize = '25px';
|
icon.style.fontSize = "25px";
|
||||||
icon.style.color = 'red'; // Pastikan 'green' dalam bentuk string
|
icon.style.color = "red"; // Pastikan 'green' dalam bentuk string
|
||||||
|
|
||||||
// Tambahkan ikon ke dalam auth-logo
|
// Tambahkan ikon ke dalam auth-logo
|
||||||
authLogo.appendChild(icon);
|
authLogo.appendChild(icon);
|
||||||
@@ -126,11 +129,12 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
modalButton.innerHTML = isEdit ? "Update" : "Create";
|
modalButton.innerHTML = isEdit ? "Update" : "Create";
|
||||||
|
|
||||||
// Set error message for the toast
|
// Set error message for the toast
|
||||||
toastBody.textContent = "An error occurred while processing your request.";
|
toastBody.textContent =
|
||||||
toast.classList.add('show'); // Show the toast
|
"An error occurred while processing your request.";
|
||||||
|
toast.classList.add("show"); // Show the toast
|
||||||
|
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
toast.classList.remove('show'); // Hide the toast after 3 seconds
|
toast.classList.remove("show"); // Hide the toast after 3 seconds
|
||||||
}, 3000);
|
}, 3000);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -141,7 +145,11 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
if (inputValue.length < 2) return;
|
if (inputValue.length < 2) return;
|
||||||
let districtValue = document.getElementById("district_name").value; // Ambil kecamatan terpilih
|
let districtValue = document.getElementById("district_name").value; // Ambil kecamatan terpilih
|
||||||
|
|
||||||
let url = `${GlobalConfig.apiHost}/api/combobox/search-options?query=${encodeURIComponent(inputValue)}&field=${field}`;
|
let url = `${
|
||||||
|
GlobalConfig.apiHost
|
||||||
|
}/api/combobox/search-options?query=${encodeURIComponent(
|
||||||
|
inputValue
|
||||||
|
)}&field=${field}`;
|
||||||
|
|
||||||
// Jika field desa, tambahkan kecamatan sebagai filter
|
// Jika field desa, tambahkan kecamatan sebagai filter
|
||||||
if (field === "village_name") {
|
if (field === "village_name") {
|
||||||
@@ -149,30 +157,30 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fetch(url, {
|
fetch(url, {
|
||||||
method: 'GET',
|
method: "GET",
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`,
|
.getAttribute("content")}`,
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
}
|
},
|
||||||
})
|
})
|
||||||
.then(response => response.json())
|
.then((response) => response.json())
|
||||||
.then(data => {
|
.then((data) => {
|
||||||
let dataList = document.getElementById(field + "Options");
|
let dataList = document.getElementById(field + "Options");
|
||||||
dataList.innerHTML = "";
|
dataList.innerHTML = "";
|
||||||
|
|
||||||
data.forEach(item => {
|
data.forEach((item) => {
|
||||||
let option = document.createElement("option");
|
let option = document.createElement("option");
|
||||||
option.value = item.name;
|
option.value = item.name;
|
||||||
option.dataset.code = item.code;
|
option.dataset.code = item.code;
|
||||||
dataList.appendChild(option);
|
dataList.appendChild(option);
|
||||||
});
|
});
|
||||||
})
|
})
|
||||||
.catch(error => console.error("Error fetching options:", error));
|
.catch((error) => console.error("Error fetching options:", error));
|
||||||
};
|
};
|
||||||
|
|
||||||
document.querySelector('.btn-back').addEventListener('click', function() {
|
document.querySelector(".btn-back").addEventListener("click", function () {
|
||||||
window.history.back();
|
window.history.back();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -25,51 +25,54 @@ console.log(dropzonePreviewNode);
|
|||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
init: function() {
|
init: function () {
|
||||||
// Listen for the success event
|
// Listen for the success event
|
||||||
this.on("success", function(file, response) {
|
this.on("success", function (file, response) {
|
||||||
console.log("File successfully uploaded:", file);
|
console.log("File successfully uploaded:", file);
|
||||||
console.log("API Response:", response);
|
console.log("API Response:", response);
|
||||||
|
|
||||||
// Show success toast
|
// Show success toast
|
||||||
showToast('bxs-check-square', 'green', response.message);
|
showToast("bxs-check-square", "green", response.message);
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
document.getElementById("submit-upload").innerHTML =
|
||||||
|
"Upload Files";
|
||||||
// Tunggu sebentar lalu reload halaman
|
// Tunggu sebentar lalu reload halaman
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = "/data/advertisements";
|
window.location.href = "/data/web-advertisements";
|
||||||
}, 2000);
|
}, 2000);
|
||||||
});
|
});
|
||||||
// Listen for the error event
|
// Listen for the error event
|
||||||
this.on("error", function(file, errorMessage) {
|
this.on("error", function (file, errorMessage) {
|
||||||
console.error("Error uploading file:", file);
|
console.error("Error uploading file:", file);
|
||||||
console.error("Error message:", errorMessage);
|
console.error("Error message:", errorMessage);
|
||||||
// Handle the error response
|
// Handle the error response
|
||||||
|
|
||||||
// Show error toast
|
// Show error toast
|
||||||
showToast('bxs-error-alt', 'red', errorMessage.message);
|
showToast("bxs-error-alt", "red", errorMessage.message);
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
document.getElementById("submit-upload").innerHTML =
|
||||||
|
"Upload Files";
|
||||||
});
|
});
|
||||||
}
|
},
|
||||||
})));
|
})));
|
||||||
|
|
||||||
// Add event listener to control the submission manually
|
// Add event listener to control the submission manually
|
||||||
document.querySelector("#submit-upload").addEventListener("click", function() {
|
document.querySelector("#submit-upload").addEventListener("click", function () {
|
||||||
console.log("Ini adalah value dropzone", dropzone.files[0]);
|
console.log("Ini adalah value dropzone", dropzone.files[0]);
|
||||||
const formData = new FormData()
|
const formData = new FormData();
|
||||||
console.log("Dropzonefiles",dropzone.files);
|
console.log("Dropzonefiles", dropzone.files);
|
||||||
|
|
||||||
this.innerHTML = '<span class="spinner-border spinner-border-sm me-1" role="status" aria-hidden="true"></span>Loading...';
|
this.innerHTML =
|
||||||
|
'<span class="spinner-border spinner-border-sm me-1" role="status" aria-hidden="true"></span>Loading...';
|
||||||
|
|
||||||
// Pastikan ada file dalam queue sebelum memprosesnya
|
// Pastikan ada file dalam queue sebelum memprosesnya
|
||||||
if (dropzone.files.length > 0) {
|
if (dropzone.files.length > 0) {
|
||||||
formData.append('file', dropzone.files[0])
|
formData.append("file", dropzone.files[0]);
|
||||||
console.log("ini adalah form data on submit", ...formData);
|
console.log("ini adalah form data on submit", ...formData);
|
||||||
dropzone.processQueue(); // Ini akan manual memicu upload
|
dropzone.processQueue(); // Ini akan manual memicu upload
|
||||||
} else {
|
} else {
|
||||||
// Show error toast when no file is selected
|
// Show error toast when no file is selected
|
||||||
showToast('bxs-error-alt', 'red', "Please add a file first.");
|
showToast("bxs-error-alt", "red", "Please add a file first.");
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -82,22 +85,24 @@ dropzone.on("addedfile", function (file) {
|
|||||||
console.log("Ukuran File:", (file.size / 1024).toFixed(2) + " KB");
|
console.log("Ukuran File:", (file.size / 1024).toFixed(2) + " KB");
|
||||||
});
|
});
|
||||||
|
|
||||||
dropzone.on("complete", function(file) {
|
dropzone.on("complete", function (file) {
|
||||||
dropzone.removeFile(file);
|
dropzone.removeFile(file);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Add event listener to donwload file template
|
// Add event listener to donwload file template
|
||||||
document.getElementById('downloadtempadvertisement').addEventListener('click', function() {
|
document
|
||||||
|
.getElementById("downloadtempadvertisement")
|
||||||
|
.addEventListener("click", function () {
|
||||||
var url = `${GlobalConfig.apiHost}/api/download-template-advertisement`;
|
var url = `${GlobalConfig.apiHost}/api/download-template-advertisement`;
|
||||||
fetch(url, {
|
fetch(url, {
|
||||||
method: 'GET',
|
method: "GET",
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
.then(response => {
|
.then((response) => {
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
return response.blob();
|
return response.blob();
|
||||||
} else {
|
} else {
|
||||||
@@ -106,38 +111,42 @@ document.getElementById('downloadtempadvertisement').addEventListener('click', f
|
|||||||
})
|
})
|
||||||
.then((blob) => {
|
.then((blob) => {
|
||||||
const url = window.URL.createObjectURL(blob);
|
const url = window.URL.createObjectURL(blob);
|
||||||
const a = document.createElement('a');
|
const a = document.createElement("a");
|
||||||
a.style.display = 'none';
|
a.style.display = "none";
|
||||||
a.href = url;
|
a.href = url;
|
||||||
a.download = 'template_reklame.xlsx';
|
a.download = "template_reklame.xlsx";
|
||||||
document.body.appendChild(a);
|
document.body.appendChild(a);
|
||||||
a.click();
|
a.click();
|
||||||
window.URL.revokeObjectURL(url);
|
window.URL.revokeObjectURL(url);
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
console.error("Gagal mendownload file:", error);
|
console.error("Gagal mendownload file:", error);
|
||||||
showToast('bxs-error-alt', 'red', "Template file is not already exist.");
|
showToast(
|
||||||
})
|
"bxs-error-alt",
|
||||||
})
|
"red",
|
||||||
|
"Template file is not already exist."
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
// Function to show toast
|
// Function to show toast
|
||||||
function showToast(iconClass, iconColor, message) {
|
function showToast(iconClass, iconColor, message) {
|
||||||
const toastElement = document.getElementById('toastUploadAdvertisement');
|
const toastElement = document.getElementById("toastUploadAdvertisement");
|
||||||
const toastBody = toastElement.querySelector('.toast-body');
|
const toastBody = toastElement.querySelector(".toast-body");
|
||||||
const toastHeader = toastElement.querySelector('.toast-header');
|
const toastHeader = toastElement.querySelector(".toast-header");
|
||||||
|
|
||||||
// Remove existing icon (if any) before adding the new one
|
// Remove existing icon (if any) before adding the new one
|
||||||
const existingIcon = toastHeader.querySelector('.bx');
|
const existingIcon = toastHeader.querySelector(".bx");
|
||||||
if (existingIcon) {
|
if (existingIcon) {
|
||||||
toastHeader.querySelector('.auth-logo').removeChild(existingIcon); // Remove the existing icon
|
toastHeader.querySelector(".auth-logo").removeChild(existingIcon); // Remove the existing icon
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add the new icon to the toast header
|
// Add the new icon to the toast header
|
||||||
const icon = document.createElement('i');
|
const icon = document.createElement("i");
|
||||||
icon.classList.add('bx', iconClass);
|
icon.classList.add("bx", iconClass);
|
||||||
icon.style.fontSize = '25px';
|
icon.style.fontSize = "25px";
|
||||||
icon.style.color = iconColor;
|
icon.style.color = iconColor;
|
||||||
toastHeader.querySelector('.auth-logo').appendChild(icon);
|
toastHeader.querySelector(".auth-logo").appendChild(icon);
|
||||||
|
|
||||||
// Set the toast message
|
// Set the toast message
|
||||||
toastBody.textContent = message;
|
toastBody.textContent = message;
|
||||||
@@ -146,4 +155,3 @@ function showToast(iconClass, iconColor, message) {
|
|||||||
const toast = new bootstrap.Toast(toastElement); // Inisialisasi Bootstrap Toast
|
const toast = new bootstrap.Toast(toastElement); // Inisialisasi Bootstrap Toast
|
||||||
toast.show();
|
toast.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -73,7 +73,7 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
}, 3000);
|
}, 3000);
|
||||||
|
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = "/data/spatial-plannings";
|
window.location.href = "/data/web-spatial-plannings";
|
||||||
}, 3000);
|
}, 3000);
|
||||||
} else {
|
} else {
|
||||||
if (authLogo) {
|
if (authLogo) {
|
||||||
|
|||||||
@@ -25,51 +25,54 @@ console.log(dropzonePreviewNode);
|
|||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
init: function() {
|
init: function () {
|
||||||
// Listen for the success event
|
// Listen for the success event
|
||||||
this.on("success", function(file, response) {
|
this.on("success", function (file, response) {
|
||||||
console.log("File successfully uploaded:", file);
|
console.log("File successfully uploaded:", file);
|
||||||
console.log("API Response:", response);
|
console.log("API Response:", response);
|
||||||
|
|
||||||
// Show success toast
|
// Show success toast
|
||||||
showToast('bxs-check-square', 'green', response.message);
|
showToast("bxs-check-square", "green", response.message);
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
document.getElementById("submit-upload").innerHTML =
|
||||||
|
"Upload Files";
|
||||||
// Tunggu sebentar lalu reload halaman
|
// Tunggu sebentar lalu reload halaman
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = "/data/spatial-plannings";
|
window.location.href = "/data/web-spatial-plannings";
|
||||||
}, 2000);
|
}, 2000);
|
||||||
});
|
});
|
||||||
// Listen for the error event
|
// Listen for the error event
|
||||||
this.on("error", function(file, errorMessage) {
|
this.on("error", function (file, errorMessage) {
|
||||||
console.error("Error uploading file:", file);
|
console.error("Error uploading file:", file);
|
||||||
console.error("Error message:", errorMessage);
|
console.error("Error message:", errorMessage);
|
||||||
// Handle the error response
|
// Handle the error response
|
||||||
|
|
||||||
// Show error toast
|
// Show error toast
|
||||||
showToast('bxs-error-alt', 'red', errorMessage.message);
|
showToast("bxs-error-alt", "red", errorMessage.message);
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
document.getElementById("submit-upload").innerHTML =
|
||||||
|
"Upload Files";
|
||||||
});
|
});
|
||||||
}
|
},
|
||||||
})));
|
})));
|
||||||
|
|
||||||
// Add event listener to control the submission manually
|
// Add event listener to control the submission manually
|
||||||
document.querySelector("#submit-upload").addEventListener("click", function() {
|
document.querySelector("#submit-upload").addEventListener("click", function () {
|
||||||
console.log("Ini adalah value dropzone", dropzone.files[0]);
|
console.log("Ini adalah value dropzone", dropzone.files[0]);
|
||||||
const formData = new FormData()
|
const formData = new FormData();
|
||||||
console.log("Dropzonefiles",dropzone.files);
|
console.log("Dropzonefiles", dropzone.files);
|
||||||
|
|
||||||
this.innerHTML = '<span class="spinner-border spinner-border-sm me-1" role="status" aria-hidden="true"></span>Loading...';
|
this.innerHTML =
|
||||||
|
'<span class="spinner-border spinner-border-sm me-1" role="status" aria-hidden="true"></span>Loading...';
|
||||||
|
|
||||||
// Pastikan ada file dalam queue sebelum memprosesnya
|
// Pastikan ada file dalam queue sebelum memprosesnya
|
||||||
if (dropzone.files.length > 0) {
|
if (dropzone.files.length > 0) {
|
||||||
formData.append('file', dropzone.files[0])
|
formData.append("file", dropzone.files[0]);
|
||||||
console.log("ini adalah form data on submit", ...formData);
|
console.log("ini adalah form data on submit", ...formData);
|
||||||
dropzone.processQueue(); // Ini akan manual memicu upload
|
dropzone.processQueue(); // Ini akan manual memicu upload
|
||||||
} else {
|
} else {
|
||||||
// Show error toast when no file is selected
|
// Show error toast when no file is selected
|
||||||
showToast('bxs-error-alt', 'red', "Please add a file first.");
|
showToast("bxs-error-alt", "red", "Please add a file first.");
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -82,22 +85,24 @@ dropzone.on("addedfile", function (file) {
|
|||||||
console.log("Ukuran File:", (file.size / 1024).toFixed(2) + " KB");
|
console.log("Ukuran File:", (file.size / 1024).toFixed(2) + " KB");
|
||||||
});
|
});
|
||||||
|
|
||||||
dropzone.on("complete", function(file) {
|
dropzone.on("complete", function (file) {
|
||||||
dropzone.removeFile(file);
|
dropzone.removeFile(file);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Add event listener to download file template
|
// Add event listener to download file template
|
||||||
document.getElementById('downloadtempspatialPlannings').addEventListener('click', function() {
|
document
|
||||||
|
.getElementById("downloadtempspatialPlannings")
|
||||||
|
.addEventListener("click", function () {
|
||||||
var url = `${GlobalConfig.apiHost}/api/download-template-spatialPlannings`;
|
var url = `${GlobalConfig.apiHost}/api/download-template-spatialPlannings`;
|
||||||
fetch(url, {
|
fetch(url, {
|
||||||
method: 'GET',
|
method: "GET",
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
.then(response => {
|
.then((response) => {
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
return response.blob(); // Jika respons OK, konversi menjadi blob
|
return response.blob(); // Jika respons OK, konversi menjadi blob
|
||||||
} else {
|
} else {
|
||||||
@@ -107,38 +112,42 @@ document.getElementById('downloadtempspatialPlannings').addEventListener('click'
|
|||||||
.then((blob) => {
|
.then((blob) => {
|
||||||
console.log(blob);
|
console.log(blob);
|
||||||
const url = window.URL.createObjectURL(blob);
|
const url = window.URL.createObjectURL(blob);
|
||||||
const a = document.createElement('a');
|
const a = document.createElement("a");
|
||||||
a.style.display = 'none';
|
a.style.display = "none";
|
||||||
a.href = url;
|
a.href = url;
|
||||||
a.download = 'template_rencana_tata_ruang.xlsx';
|
a.download = "template_rencana_tata_ruang.xlsx";
|
||||||
document.body.appendChild(a);
|
document.body.appendChild(a);
|
||||||
a.click();
|
a.click();
|
||||||
window.URL.revokeObjectURL(url);
|
window.URL.revokeObjectURL(url);
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
console.error("Gagal mendownload file:", error);
|
console.error("Gagal mendownload file:", error);
|
||||||
showToast('bxs-error-alt', 'red', "Template file is not already exist.");
|
showToast(
|
||||||
})
|
"bxs-error-alt",
|
||||||
})
|
"red",
|
||||||
|
"Template file is not already exist."
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
// Function to show toast
|
// Function to show toast
|
||||||
function showToast(iconClass, iconColor, message) {
|
function showToast(iconClass, iconColor, message) {
|
||||||
const toastElement = document.getElementById('toastUploadSpatialPlannings');
|
const toastElement = document.getElementById("toastUploadSpatialPlannings");
|
||||||
const toastBody = toastElement.querySelector('.toast-body');
|
const toastBody = toastElement.querySelector(".toast-body");
|
||||||
const toastHeader = toastElement.querySelector('.toast-header');
|
const toastHeader = toastElement.querySelector(".toast-header");
|
||||||
|
|
||||||
// Remove existing icon (if any) before adding the new one
|
// Remove existing icon (if any) before adding the new one
|
||||||
const existingIcon = toastHeader.querySelector('.bx');
|
const existingIcon = toastHeader.querySelector(".bx");
|
||||||
if (existingIcon) {
|
if (existingIcon) {
|
||||||
toastHeader.querySelector('.auth-logo').removeChild(existingIcon); // Remove the existing icon
|
toastHeader.querySelector(".auth-logo").removeChild(existingIcon); // Remove the existing icon
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add the new icon to the toast header
|
// Add the new icon to the toast header
|
||||||
const icon = document.createElement('i');
|
const icon = document.createElement("i");
|
||||||
icon.classList.add('bx', iconClass);
|
icon.classList.add("bx", iconClass);
|
||||||
icon.style.fontSize = '25px';
|
icon.style.fontSize = "25px";
|
||||||
icon.style.color = iconColor;
|
icon.style.color = iconColor;
|
||||||
toastHeader.querySelector('.auth-logo').appendChild(icon);
|
toastHeader.querySelector(".auth-logo").appendChild(icon);
|
||||||
|
|
||||||
// Set the toast message
|
// Set the toast message
|
||||||
toastBody.textContent = message;
|
toastBody.textContent = message;
|
||||||
@@ -147,4 +156,3 @@ function showToast(iconClass, iconColor, message) {
|
|||||||
const toast = new bootstrap.Toast(toastElement); // Inisialisasi Bootstrap Toast
|
const toast = new bootstrap.Toast(toastElement); // Inisialisasi Bootstrap Toast
|
||||||
toast.show();
|
toast.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -20,9 +20,9 @@ const dataTourismsColumns = [
|
|||||||
name: "Actions",
|
name: "Actions",
|
||||||
widht: "120px",
|
widht: "120px",
|
||||||
formatter: function (cell, row) {
|
formatter: function (cell, row) {
|
||||||
const id = row.cells[10].data;
|
const id = row.cells[11].data;
|
||||||
const long = row.cells[8].data;
|
const long = row.cells[9].data;
|
||||||
const lat = row.cells[9].data;
|
const lat = row.cells[10].data;
|
||||||
const model = "data/tourisms";
|
const model = "data/tourisms";
|
||||||
return gridjs.html(`
|
return gridjs.html(`
|
||||||
<div class="d-flex justify-items-end gap-10">
|
<div class="d-flex justify-items-end gap-10">
|
||||||
|
|||||||
@@ -73,7 +73,7 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
}, 3000);
|
}, 3000);
|
||||||
|
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = "/data/tourisms";
|
window.location.href = "/data/web-tourisms";
|
||||||
}, 3000);
|
}, 3000);
|
||||||
} else {
|
} else {
|
||||||
if (authLogo) {
|
if (authLogo) {
|
||||||
|
|||||||
@@ -25,51 +25,54 @@ console.log(dropzonePreviewNode);
|
|||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
init: function() {
|
init: function () {
|
||||||
// Listen for the success event
|
// Listen for the success event
|
||||||
this.on("success", function(file, response) {
|
this.on("success", function (file, response) {
|
||||||
console.log("File successfully uploaded:", file);
|
console.log("File successfully uploaded:", file);
|
||||||
console.log("API Response:", response);
|
console.log("API Response:", response);
|
||||||
|
|
||||||
// Show success toast
|
// Show success toast
|
||||||
showToast('bxs-check-square', 'green', response.message);
|
showToast("bxs-check-square", "green", response.message);
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
document.getElementById("submit-upload").innerHTML =
|
||||||
|
"Upload Files";
|
||||||
// Tunggu sebentar lalu reload halaman
|
// Tunggu sebentar lalu reload halaman
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = "/data/tourisms";
|
window.location.href = "/data/web-tourisms";
|
||||||
}, 2000);
|
}, 2000);
|
||||||
});
|
});
|
||||||
// Listen for the error event
|
// Listen for the error event
|
||||||
this.on("error", function(file, errorMessage) {
|
this.on("error", function (file, errorMessage) {
|
||||||
console.error("Error uploading file:", file);
|
console.error("Error uploading file:", file);
|
||||||
console.error("Error message:", errorMessage);
|
console.error("Error message:", errorMessage);
|
||||||
// Handle the error response
|
// Handle the error response
|
||||||
|
|
||||||
// Show error toast
|
// Show error toast
|
||||||
showToast('bxs-error-alt', 'red', errorMessage.message);
|
showToast("bxs-error-alt", "red", errorMessage.message);
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
document.getElementById("submit-upload").innerHTML =
|
||||||
|
"Upload Files";
|
||||||
});
|
});
|
||||||
}
|
},
|
||||||
})));
|
})));
|
||||||
|
|
||||||
// Add event listener to control the submission manually
|
// Add event listener to control the submission manually
|
||||||
document.querySelector("#submit-upload").addEventListener("click", function() {
|
document.querySelector("#submit-upload").addEventListener("click", function () {
|
||||||
console.log("Ini adalah value dropzone", dropzone.files[0]);
|
console.log("Ini adalah value dropzone", dropzone.files[0]);
|
||||||
const formData = new FormData()
|
const formData = new FormData();
|
||||||
console.log("Dropzonefiles",dropzone.files);
|
console.log("Dropzonefiles", dropzone.files);
|
||||||
|
|
||||||
this.innerHTML = '<span class="spinner-border spinner-border-sm me-1" role="status" aria-hidden="true"></span>Loading...';
|
this.innerHTML =
|
||||||
|
'<span class="spinner-border spinner-border-sm me-1" role="status" aria-hidden="true"></span>Loading...';
|
||||||
|
|
||||||
// Pastikan ada file dalam queue sebelum memprosesnya
|
// Pastikan ada file dalam queue sebelum memprosesnya
|
||||||
if (dropzone.files.length > 0) {
|
if (dropzone.files.length > 0) {
|
||||||
formData.append('file', dropzone.files[0])
|
formData.append("file", dropzone.files[0]);
|
||||||
console.log("ini adalah form data on submit", ...formData);
|
console.log("ini adalah form data on submit", ...formData);
|
||||||
dropzone.processQueue(); // Ini akan manual memicu upload
|
dropzone.processQueue(); // Ini akan manual memicu upload
|
||||||
} else {
|
} else {
|
||||||
// Show error toast when no file is selected
|
// Show error toast when no file is selected
|
||||||
showToast('bxs-error-alt', 'red', "Please add a file first.");
|
showToast("bxs-error-alt", "red", "Please add a file first.");
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -82,22 +85,24 @@ dropzone.on("addedfile", function (file) {
|
|||||||
console.log("Ukuran File:", (file.size / 1024).toFixed(2) + " KB");
|
console.log("Ukuran File:", (file.size / 1024).toFixed(2) + " KB");
|
||||||
});
|
});
|
||||||
|
|
||||||
dropzone.on("complete", function(file) {
|
dropzone.on("complete", function (file) {
|
||||||
dropzone.removeFile(file);
|
dropzone.removeFile(file);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Add event listener to download file template
|
// Add event listener to download file template
|
||||||
document.getElementById('downloadtemptourisms').addEventListener('click', function() {
|
document
|
||||||
|
.getElementById("downloadtemptourisms")
|
||||||
|
.addEventListener("click", function () {
|
||||||
var url = `${GlobalConfig.apiHost}/api/download-template-tourism`;
|
var url = `${GlobalConfig.apiHost}/api/download-template-tourism`;
|
||||||
fetch(url, {
|
fetch(url, {
|
||||||
method: 'GET',
|
method: "GET",
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
.then(response => {
|
.then((response) => {
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
return response.blob(); // Jika respons OK, konversi menjadi blob
|
return response.blob(); // Jika respons OK, konversi menjadi blob
|
||||||
} else {
|
} else {
|
||||||
@@ -107,38 +112,42 @@ document.getElementById('downloadtemptourisms').addEventListener('click', functi
|
|||||||
.then((blob) => {
|
.then((blob) => {
|
||||||
console.log(blob);
|
console.log(blob);
|
||||||
const url = window.URL.createObjectURL(blob);
|
const url = window.URL.createObjectURL(blob);
|
||||||
const a = document.createElement('a');
|
const a = document.createElement("a");
|
||||||
a.style.display = 'none';
|
a.style.display = "none";
|
||||||
a.href = url;
|
a.href = url;
|
||||||
a.download = 'template_pariwisata.xlsx';
|
a.download = "template_pariwisata.xlsx";
|
||||||
document.body.appendChild(a);
|
document.body.appendChild(a);
|
||||||
a.click();
|
a.click();
|
||||||
window.URL.revokeObjectURL(url);
|
window.URL.revokeObjectURL(url);
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
console.error("Gagal mendownload file:", error);
|
console.error("Gagal mendownload file:", error);
|
||||||
showToast('bxs-error-alt', 'red', "Template file is not already exist.");
|
showToast(
|
||||||
})
|
"bxs-error-alt",
|
||||||
})
|
"red",
|
||||||
|
"Template file is not already exist."
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
// Function to show toast
|
// Function to show toast
|
||||||
function showToast(iconClass, iconColor, message) {
|
function showToast(iconClass, iconColor, message) {
|
||||||
const toastElement = document.getElementById('toastUploadTourisms');
|
const toastElement = document.getElementById("toastUploadTourisms");
|
||||||
const toastBody = toastElement.querySelector('.toast-body');
|
const toastBody = toastElement.querySelector(".toast-body");
|
||||||
const toastHeader = toastElement.querySelector('.toast-header');
|
const toastHeader = toastElement.querySelector(".toast-header");
|
||||||
|
|
||||||
// Remove existing icon (if any) before adding the new one
|
// Remove existing icon (if any) before adding the new one
|
||||||
const existingIcon = toastHeader.querySelector('.bx');
|
const existingIcon = toastHeader.querySelector(".bx");
|
||||||
if (existingIcon) {
|
if (existingIcon) {
|
||||||
toastHeader.querySelector('.auth-logo').removeChild(existingIcon); // Remove the existing icon
|
toastHeader.querySelector(".auth-logo").removeChild(existingIcon); // Remove the existing icon
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add the new icon to the toast header
|
// Add the new icon to the toast header
|
||||||
const icon = document.createElement('i');
|
const icon = document.createElement("i");
|
||||||
icon.classList.add('bx', iconClass);
|
icon.classList.add("bx", iconClass);
|
||||||
icon.style.fontSize = '25px';
|
icon.style.fontSize = "25px";
|
||||||
icon.style.color = iconColor;
|
icon.style.color = iconColor;
|
||||||
toastHeader.querySelector('.auth-logo').appendChild(icon);
|
toastHeader.querySelector(".auth-logo").appendChild(icon);
|
||||||
|
|
||||||
// Set the toast message
|
// Set the toast message
|
||||||
toastBody.textContent = message;
|
toastBody.textContent = message;
|
||||||
@@ -147,4 +156,3 @@ function showToast(iconClass, iconColor, message) {
|
|||||||
const toast = new bootstrap.Toast(toastElement); // Inisialisasi Bootstrap Toast
|
const toast = new bootstrap.Toast(toastElement); // Inisialisasi Bootstrap Toast
|
||||||
toast.show();
|
toast.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ const dataUMKMColumns = [
|
|||||||
name: "Actions",
|
name: "Actions",
|
||||||
widht: "120px",
|
widht: "120px",
|
||||||
formatter: function(cell, row) {
|
formatter: function(cell, row) {
|
||||||
const id = row.cells[18].data;
|
const id = row.cells[19].data;
|
||||||
const model = "data/umkm";
|
const model = "data/umkm";
|
||||||
return gridjs.html(`
|
return gridjs.html(`
|
||||||
<div class="d-flex justify-items-end gap-10">
|
<div class="d-flex justify-items-end gap-10">
|
||||||
|
|||||||
@@ -16,10 +16,10 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
Loading...
|
Loading...
|
||||||
`;
|
`;
|
||||||
const isEdit = saveButton.classList.contains("btn-edit");
|
const isEdit = saveButton.classList.contains("btn-edit");
|
||||||
const formData = new FormData(form)
|
const formData = new FormData(form);
|
||||||
const toast = document.getElementById('toastEditUpdate');
|
const toast = document.getElementById("toastEditUpdate");
|
||||||
const toastBody = toast.querySelector('.toast-body');
|
const toastBody = toast.querySelector(".toast-body");
|
||||||
const toastHeader = toast.querySelector('.toast-header small');
|
const toastHeader = toast.querySelector(".toast-header small");
|
||||||
|
|
||||||
const data = {};
|
const data = {};
|
||||||
|
|
||||||
@@ -40,53 +40,55 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`,
|
.getAttribute("content")}`,
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
}
|
},
|
||||||
})
|
})
|
||||||
.then(response => response.json())
|
.then((response) => response.json())
|
||||||
.then(data => {
|
.then((data) => {
|
||||||
console.log("Response data:", data);
|
console.log("Response data:", data);
|
||||||
if (!data.errors) {
|
if (!data.errors) {
|
||||||
// Remove existing icon (if any) before adding the new one
|
// Remove existing icon (if any) before adding the new one
|
||||||
if (authLogo) {
|
if (authLogo) {
|
||||||
// Hapus ikon yang sudah ada jika ada
|
// Hapus ikon yang sudah ada jika ada
|
||||||
const existingIcon = authLogo.querySelector('.bx');
|
const existingIcon = authLogo.querySelector(".bx");
|
||||||
if (existingIcon) {
|
if (existingIcon) {
|
||||||
authLogo.removeChild(existingIcon);
|
authLogo.removeChild(existingIcon);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Buat ikon baru
|
// Buat ikon baru
|
||||||
const icon = document.createElement('i');
|
const icon = document.createElement("i");
|
||||||
icon.classList.add('bx', 'bxs-check-square');
|
icon.classList.add("bx", "bxs-check-square");
|
||||||
icon.style.fontSize = '25px';
|
icon.style.fontSize = "25px";
|
||||||
icon.style.color = 'green'; // Pastikan 'green' dalam bentuk string
|
icon.style.color = "green"; // Pastikan 'green' dalam bentuk string
|
||||||
|
|
||||||
// Tambahkan ikon ke dalam auth-logo
|
// Tambahkan ikon ke dalam auth-logo
|
||||||
authLogo.appendChild(icon);
|
authLogo.appendChild(icon);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set success message for the toast
|
// Set success message for the toast
|
||||||
toastBody.textContent = isEdit ? "Data updated successfully!" : "Data created successfully!";
|
toastBody.textContent = isEdit
|
||||||
toast.classList.add('show'); // Show the toast
|
? "Data updated successfully!"
|
||||||
|
: "Data created successfully!";
|
||||||
|
toast.classList.add("show"); // Show the toast
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
toast.classList.remove('show'); // Hide the toast after 3 seconds
|
toast.classList.remove("show"); // Hide the toast after 3 seconds
|
||||||
}, 2000);
|
}, 2000);
|
||||||
|
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = '/data/umkm';
|
window.location.href = "/data/web-umkm";
|
||||||
}, 1000);
|
}, 1000);
|
||||||
} else {
|
} else {
|
||||||
if (authLogo) {
|
if (authLogo) {
|
||||||
// Hapus ikon yang sudah ada jika ada
|
// Hapus ikon yang sudah ada jika ada
|
||||||
const existingIcon = authLogo.querySelector('.bx');
|
const existingIcon = authLogo.querySelector(".bx");
|
||||||
if (existingIcon) {
|
if (existingIcon) {
|
||||||
authLogo.removeChild(existingIcon);
|
authLogo.removeChild(existingIcon);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Buat ikon baru
|
// Buat ikon baru
|
||||||
const icon = document.createElement('i');
|
const icon = document.createElement("i");
|
||||||
icon.classList.add('bx', 'bxs-error-alt');
|
icon.classList.add("bx", "bxs-error-alt");
|
||||||
icon.style.fontSize = '25px';
|
icon.style.fontSize = "25px";
|
||||||
icon.style.color = 'red'; // Pastikan 'green' dalam bentuk string
|
icon.style.color = "red"; // Pastikan 'green' dalam bentuk string
|
||||||
|
|
||||||
// Tambahkan ikon ke dalam auth-logo
|
// Tambahkan ikon ke dalam auth-logo
|
||||||
authLogo.appendChild(icon);
|
authLogo.appendChild(icon);
|
||||||
@@ -96,30 +98,30 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
modalButton.disabled = false;
|
modalButton.disabled = false;
|
||||||
modalButton.innerHTML = isEdit ? "Update" : "Create";
|
modalButton.innerHTML = isEdit ? "Update" : "Create";
|
||||||
|
|
||||||
|
|
||||||
// Set error message for the toast
|
// Set error message for the toast
|
||||||
toastBody.textContent = "Error: " + (data.message || "Something went wrong");
|
toastBody.textContent =
|
||||||
toast.classList.add('show'); // Show the toast
|
"Error: " + (data.message || "Something went wrong");
|
||||||
|
toast.classList.add("show"); // Show the toast
|
||||||
|
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
toast.classList.remove('show'); // Hide the toast after 3 seconds
|
toast.classList.remove("show"); // Hide the toast after 3 seconds
|
||||||
}, 3000);
|
}, 3000);
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.catch(error => {
|
.catch((error) => {
|
||||||
console.error("Error:", error);
|
console.error("Error:", error);
|
||||||
if (authLogo) {
|
if (authLogo) {
|
||||||
// Hapus ikon yang sudah ada jika ada
|
// Hapus ikon yang sudah ada jika ada
|
||||||
const existingIcon = authLogo.querySelector('.bx');
|
const existingIcon = authLogo.querySelector(".bx");
|
||||||
if (existingIcon) {
|
if (existingIcon) {
|
||||||
authLogo.removeChild(existingIcon);
|
authLogo.removeChild(existingIcon);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Buat ikon baru
|
// Buat ikon baru
|
||||||
const icon = document.createElement('i');
|
const icon = document.createElement("i");
|
||||||
icon.classList.add('bx', 'bxs-error-alt');
|
icon.classList.add("bx", "bxs-error-alt");
|
||||||
icon.style.fontSize = '25px';
|
icon.style.fontSize = "25px";
|
||||||
icon.style.color = 'red'; // Pastikan 'green' dalam bentuk string
|
icon.style.color = "red"; // Pastikan 'green' dalam bentuk string
|
||||||
|
|
||||||
// Tambahkan ikon ke dalam auth-logo
|
// Tambahkan ikon ke dalam auth-logo
|
||||||
authLogo.appendChild(icon);
|
authLogo.appendChild(icon);
|
||||||
@@ -130,11 +132,12 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
modalButton.innerHTML = isEdit ? "Update" : "Create";
|
modalButton.innerHTML = isEdit ? "Update" : "Create";
|
||||||
|
|
||||||
// Set error message for the toast
|
// Set error message for the toast
|
||||||
toastBody.textContent = "An error occurred while processing your request.";
|
toastBody.textContent =
|
||||||
toast.classList.add('show'); // Show the toast
|
"An error occurred while processing your request.";
|
||||||
|
toast.classList.add("show"); // Show the toast
|
||||||
|
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
toast.classList.remove('show'); // Hide the toast after 3 seconds
|
toast.classList.remove("show"); // Hide the toast after 3 seconds
|
||||||
}, 3000);
|
}, 3000);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -145,7 +148,11 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
if (inputValue.length < 2) return;
|
if (inputValue.length < 2) return;
|
||||||
let districtValue = document.getElementById("district_name").value; // Ambil kecamatan terpilih
|
let districtValue = document.getElementById("district_name").value; // Ambil kecamatan terpilih
|
||||||
|
|
||||||
let url = `${GlobalConfig.apiHost}/api/combobox/search-options?query=${encodeURIComponent(inputValue)}&field=${field}`;
|
let url = `${
|
||||||
|
GlobalConfig.apiHost
|
||||||
|
}/api/combobox/search-options?query=${encodeURIComponent(
|
||||||
|
inputValue
|
||||||
|
)}&field=${field}`;
|
||||||
|
|
||||||
// Jika field desa, tambahkan kecamatan sebagai filter
|
// Jika field desa, tambahkan kecamatan sebagai filter
|
||||||
if (field === "village_name") {
|
if (field === "village_name") {
|
||||||
@@ -153,30 +160,30 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fetch(url, {
|
fetch(url, {
|
||||||
method: 'GET',
|
method: "GET",
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`,
|
.getAttribute("content")}`,
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
}
|
},
|
||||||
})
|
})
|
||||||
.then(response => response.json())
|
.then((response) => response.json())
|
||||||
.then(data => {
|
.then((data) => {
|
||||||
let dataList = document.getElementById(field + "Options");
|
let dataList = document.getElementById(field + "Options");
|
||||||
dataList.innerHTML = "";
|
dataList.innerHTML = "";
|
||||||
|
|
||||||
data.forEach(item => {
|
data.forEach((item) => {
|
||||||
let option = document.createElement("option");
|
let option = document.createElement("option");
|
||||||
option.value = item.name;
|
option.value = item.name;
|
||||||
option.dataset.code = item.code;
|
option.dataset.code = item.code;
|
||||||
dataList.appendChild(option);
|
dataList.appendChild(option);
|
||||||
});
|
});
|
||||||
})
|
})
|
||||||
.catch(error => console.error("Error fetching options:", error));
|
.catch((error) => console.error("Error fetching options:", error));
|
||||||
};
|
};
|
||||||
|
|
||||||
document.querySelector('.btn-back').addEventListener('click', function() {
|
document.querySelector(".btn-back").addEventListener("click", function () {
|
||||||
window.history.back();
|
window.history.back();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -25,51 +25,54 @@ console.log(dropzonePreviewNode);
|
|||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
init: function() {
|
init: function () {
|
||||||
// Listen for the success event
|
// Listen for the success event
|
||||||
this.on("success", function(file, response) {
|
this.on("success", function (file, response) {
|
||||||
console.log("File successfully uploaded:", file);
|
console.log("File successfully uploaded:", file);
|
||||||
console.log("API Response:", response);
|
console.log("API Response:", response);
|
||||||
|
|
||||||
// Show success toast
|
// Show success toast
|
||||||
showToast('bxs-check-square', 'green', response.message);
|
showToast("bxs-check-square", "green", response.message);
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
document.getElementById("submit-upload").innerHTML =
|
||||||
|
"Upload Files";
|
||||||
// Tunggu sebentar lalu reload halaman
|
// Tunggu sebentar lalu reload halaman
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = "/data/umkm";
|
window.location.href = "/data/web-umkm";
|
||||||
}, 2000);
|
}, 2000);
|
||||||
});
|
});
|
||||||
// Listen for the error event
|
// Listen for the error event
|
||||||
this.on("error", function(file, errorMessage) {
|
this.on("error", function (file, errorMessage) {
|
||||||
console.error("Error uploading file:", file);
|
console.error("Error uploading file:", file);
|
||||||
console.error("Error message:", errorMessage);
|
console.error("Error message:", errorMessage);
|
||||||
// Handle the error response
|
// Handle the error response
|
||||||
|
|
||||||
// Show error toast
|
// Show error toast
|
||||||
showToast('bxs-error-alt', 'red', errorMessage.message);
|
showToast("bxs-error-alt", "red", errorMessage.message);
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
document.getElementById("submit-upload").innerHTML =
|
||||||
|
"Upload Files";
|
||||||
});
|
});
|
||||||
}
|
},
|
||||||
})));
|
})));
|
||||||
|
|
||||||
// Add event listener to control the submission manually
|
// Add event listener to control the submission manually
|
||||||
document.querySelector("#submit-upload").addEventListener("click", function() {
|
document.querySelector("#submit-upload").addEventListener("click", function () {
|
||||||
console.log("Ini adalah value dropzone", dropzone.files[0]);
|
console.log("Ini adalah value dropzone", dropzone.files[0]);
|
||||||
const formData = new FormData()
|
const formData = new FormData();
|
||||||
console.log("Dropzonefiles",dropzone.files);
|
console.log("Dropzonefiles", dropzone.files);
|
||||||
|
|
||||||
this.innerHTML = '<span class="spinner-border spinner-border-sm me-1" role="status" aria-hidden="true"></span>Loading...';
|
this.innerHTML =
|
||||||
|
'<span class="spinner-border spinner-border-sm me-1" role="status" aria-hidden="true"></span>Loading...';
|
||||||
|
|
||||||
// Pastikan ada file dalam queue sebelum memprosesnya
|
// Pastikan ada file dalam queue sebelum memprosesnya
|
||||||
if (dropzone.files.length > 0) {
|
if (dropzone.files.length > 0) {
|
||||||
formData.append('file', dropzone.files[0])
|
formData.append("file", dropzone.files[0]);
|
||||||
console.log("ini adalah form data on submit", ...formData);
|
console.log("ini adalah form data on submit", ...formData);
|
||||||
dropzone.processQueue(); // Ini akan manual memicu upload
|
dropzone.processQueue(); // Ini akan manual memicu upload
|
||||||
} else {
|
} else {
|
||||||
// Show error toast when no file is selected
|
// Show error toast when no file is selected
|
||||||
showToast('bxs-error-alt', 'red', "Please add a file first.");
|
showToast("bxs-error-alt", "red", "Please add a file first.");
|
||||||
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
document.getElementById("submit-upload").innerHTML = "Upload Files";
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -82,22 +85,24 @@ dropzone.on("addedfile", function (file) {
|
|||||||
console.log("Ukuran File:", (file.size / 1024).toFixed(2) + " KB");
|
console.log("Ukuran File:", (file.size / 1024).toFixed(2) + " KB");
|
||||||
});
|
});
|
||||||
|
|
||||||
dropzone.on("complete", function(file) {
|
dropzone.on("complete", function (file) {
|
||||||
dropzone.removeFile(file);
|
dropzone.removeFile(file);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Add event listener to download file template
|
// Add event listener to download file template
|
||||||
document.getElementById('downloadtempumkm').addEventListener('click', function() {
|
document
|
||||||
|
.getElementById("downloadtempumkm")
|
||||||
|
.addEventListener("click", function () {
|
||||||
var url = `${GlobalConfig.apiHost}/api/download-template-umkm`;
|
var url = `${GlobalConfig.apiHost}/api/download-template-umkm`;
|
||||||
fetch(url, {
|
fetch(url, {
|
||||||
method: 'GET',
|
method: "GET",
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
.then(response => {
|
.then((response) => {
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
return response.blob(); // Jika respons OK, konversi menjadi blob
|
return response.blob(); // Jika respons OK, konversi menjadi blob
|
||||||
} else {
|
} else {
|
||||||
@@ -106,38 +111,42 @@ document.getElementById('downloadtempumkm').addEventListener('click', function()
|
|||||||
})
|
})
|
||||||
.then((blob) => {
|
.then((blob) => {
|
||||||
const url = window.URL.createObjectURL(blob);
|
const url = window.URL.createObjectURL(blob);
|
||||||
const a = document.createElement('a');
|
const a = document.createElement("a");
|
||||||
a.style.display = 'none';
|
a.style.display = "none";
|
||||||
a.href = url;
|
a.href = url;
|
||||||
a.download = 'template_umkm.xlsx';
|
a.download = "template_umkm.xlsx";
|
||||||
document.body.appendChild(a);
|
document.body.appendChild(a);
|
||||||
a.click();
|
a.click();
|
||||||
window.URL.revokeObjectURL(url);
|
window.URL.revokeObjectURL(url);
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
console.error("Gagal mendownload file:", error);
|
console.error("Gagal mendownload file:", error);
|
||||||
showToast('bxs-error-alt', 'red', "Template file is not already exist.");
|
showToast(
|
||||||
})
|
"bxs-error-alt",
|
||||||
})
|
"red",
|
||||||
|
"Template file is not already exist."
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
// Function to show toast
|
// Function to show toast
|
||||||
function showToast(iconClass, iconColor, message) {
|
function showToast(iconClass, iconColor, message) {
|
||||||
const toastElement = document.getElementById('toastUploadUmkm');
|
const toastElement = document.getElementById("toastUploadUmkm");
|
||||||
const toastBody = toastElement.querySelector('.toast-body');
|
const toastBody = toastElement.querySelector(".toast-body");
|
||||||
const toastHeader = toastElement.querySelector('.toast-header');
|
const toastHeader = toastElement.querySelector(".toast-header");
|
||||||
|
|
||||||
// Remove existing icon (if any) before adding the new one
|
// Remove existing icon (if any) before adding the new one
|
||||||
const existingIcon = toastHeader.querySelector('.bx');
|
const existingIcon = toastHeader.querySelector(".bx");
|
||||||
if (existingIcon) {
|
if (existingIcon) {
|
||||||
toastHeader.querySelector('.auth-logo').removeChild(existingIcon); // Remove the existing icon
|
toastHeader.querySelector(".auth-logo").removeChild(existingIcon); // Remove the existing icon
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add the new icon to the toast header
|
// Add the new icon to the toast header
|
||||||
const icon = document.createElement('i');
|
const icon = document.createElement("i");
|
||||||
icon.classList.add('bx', iconClass);
|
icon.classList.add("bx", iconClass);
|
||||||
icon.style.fontSize = '25px';
|
icon.style.fontSize = "25px";
|
||||||
icon.style.color = iconColor;
|
icon.style.color = iconColor;
|
||||||
toastHeader.querySelector('.auth-logo').appendChild(icon);
|
toastHeader.querySelector(".auth-logo").appendChild(icon);
|
||||||
|
|
||||||
// Set the toast message
|
// Set the toast message
|
||||||
toastBody.textContent = message;
|
toastBody.textContent = message;
|
||||||
@@ -146,4 +155,3 @@ function showToast(iconClass, iconColor, message) {
|
|||||||
const toast = new bootstrap.Toast(toastElement); // Inisialisasi Bootstrap Toast
|
const toast = new bootstrap.Toast(toastElement); // Inisialisasi Bootstrap Toast
|
||||||
toast.show();
|
toast.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
79
resources/js/maps/maps-kml.js
Normal file
79
resources/js/maps/maps-kml.js
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
import L from "leaflet";
|
||||||
|
import "leaflet/dist/leaflet.css";
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", function () {
|
||||||
|
var map = L.map("map").setView([-6.9175, 107.6191], 10); // Bandung
|
||||||
|
|
||||||
|
L.tileLayer("https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png", {
|
||||||
|
attribution: "© OpenStreetMap contributors",
|
||||||
|
}).addTo(map);
|
||||||
|
|
||||||
|
// Dapatkan elemen loading
|
||||||
|
const loadingDiv = document.getElementById("loading");
|
||||||
|
loadingDiv.style.display = "flex"; // Tampilkan loading
|
||||||
|
|
||||||
|
fetch("/storage/maps/rencana-polaruang.geojson")
|
||||||
|
.then((res) => res.json())
|
||||||
|
.then((geojson) => {
|
||||||
|
let colorMapping = {
|
||||||
|
BJ: "rgb(235, 30, 30)",
|
||||||
|
BA: "rgb(151, 219, 242)",
|
||||||
|
CA: "rgb(70, 70, 165)",
|
||||||
|
"P-2": "rgb(230, 255, 75)",
|
||||||
|
HL: "rgb(50, 95, 40)",
|
||||||
|
HPT: "rgb(75, 155, 55)",
|
||||||
|
HP: "rgb(125, 180, 55)",
|
||||||
|
W: "rgb(255, 165, 255)",
|
||||||
|
PTL: "rgb(0, 255, 205)",
|
||||||
|
"IK-2": "rgb(130, 185, 210)",
|
||||||
|
"P-3": "rgb(175, 175, 55)",
|
||||||
|
PS: "rgb(5, 215, 215)",
|
||||||
|
PD: "rgb(235, 155, 60)",
|
||||||
|
PK: "rgb(245, 155, 30)",
|
||||||
|
HK: "rgb(155, 0, 255)",
|
||||||
|
KPI: "rgb(105, 0, 0)",
|
||||||
|
MBT: "rgb(95, 115, 145)",
|
||||||
|
"P-4": "rgb(185, 235, 185)",
|
||||||
|
TB: "rgb(70, 150, 255)",
|
||||||
|
"P-1": "rgb(200, 245, 70)",
|
||||||
|
TR: "rgb(215, 55, 0)",
|
||||||
|
THR: "rgb(185, 165, 255)",
|
||||||
|
TWA: "rgb(210, 190, 255)",
|
||||||
|
};
|
||||||
|
var geoLayer = L.geoJSON(geojson, {
|
||||||
|
style: function (feature) {
|
||||||
|
let htmlString = feature.properties.description.toString();
|
||||||
|
|
||||||
|
let match = htmlString.match(
|
||||||
|
/<td>Kode Kawasan<\/td>\s*<td>(.*?)<\/td>/
|
||||||
|
);
|
||||||
|
|
||||||
|
console.log("Kode Kawasan ", match[1]);
|
||||||
|
|
||||||
|
let color_code = match[1];
|
||||||
|
|
||||||
|
return {
|
||||||
|
color: colorMapping[color_code],
|
||||||
|
fillColor: colorMapping[color_code] || "#cccccc",
|
||||||
|
fillOpacity: 0.6,
|
||||||
|
weight: 1.5,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
onEachFeature: function (feature, layer) {
|
||||||
|
if (feature.properties && feature.properties.name) {
|
||||||
|
layer.bindPopup(feature.properties.name);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}).addTo(map);
|
||||||
|
|
||||||
|
map.fitBounds(geoLayer.getBounds());
|
||||||
|
|
||||||
|
// Sembunyikan loading setelah selesai render
|
||||||
|
loadingDiv.style.display = "none";
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.error("Error loading GeoJSON:", error);
|
||||||
|
loadingDiv.innerHTML =
|
||||||
|
"<div class='loading-text' style='background: red;'>Failed to load data!</div>";
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -31,7 +31,7 @@ class UsersTable {
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
pagination: {
|
pagination: {
|
||||||
limit: 15,
|
limit: 50,
|
||||||
server: {
|
server: {
|
||||||
url: (prev, page) =>
|
url: (prev, page) =>
|
||||||
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
||||||
|
|||||||
@@ -1,4 +1,9 @@
|
|||||||
document.addEventListener("DOMContentLoaded", function (e) {
|
class CreateMenu {
|
||||||
|
constructor() {
|
||||||
|
this.initCreateMenu();
|
||||||
|
}
|
||||||
|
|
||||||
|
initCreateMenu() {
|
||||||
const toastNotification = document.getElementById("toastNotification");
|
const toastNotification = document.getElementById("toastNotification");
|
||||||
const toast = new bootstrap.Toast(toastNotification);
|
const toast = new bootstrap.Toast(toastNotification);
|
||||||
document
|
document
|
||||||
@@ -23,9 +28,9 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
let response = await fetch(form.action, {
|
let response = await fetch(form.action, {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
"X-CSRF-TOKEN": document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="csrf-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content"),
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
body: formData,
|
body: formData,
|
||||||
});
|
});
|
||||||
@@ -44,6 +49,8 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
error.message;
|
error.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
console.error("Error:", error);
|
console.error("Error:", error);
|
||||||
|
submitButton.disabled = false;
|
||||||
|
spinner.classList.add("d-none");
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Request failed:", error);
|
console.error("Request failed:", error);
|
||||||
@@ -52,4 +59,9 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
toast.show();
|
toast.show();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", function (e) {
|
||||||
|
new CreateMenu();
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -2,13 +2,27 @@ import { Grid } from "gridjs/dist/gridjs.umd.js";
|
|||||||
import gridjs from "gridjs/dist/gridjs.umd.js";
|
import gridjs from "gridjs/dist/gridjs.umd.js";
|
||||||
import "gridjs/dist/gridjs.umd.js";
|
import "gridjs/dist/gridjs.umd.js";
|
||||||
import GlobalConfig from "../global-config";
|
import GlobalConfig from "../global-config";
|
||||||
|
import Swal from "sweetalert2";
|
||||||
|
|
||||||
class Menus {
|
class Menus {
|
||||||
constructor() {
|
constructor() {
|
||||||
|
this.toastMessage = document.getElementById("toast-message");
|
||||||
|
this.toastElement = document.getElementById("toastNotification");
|
||||||
|
this.toast = new bootstrap.Toast(this.toastElement);
|
||||||
this.table = null;
|
this.table = null;
|
||||||
}
|
|
||||||
init() {
|
// Initialize functions
|
||||||
this.initTableMenus();
|
this.initTableMenus();
|
||||||
|
this.initEvents();
|
||||||
|
}
|
||||||
|
initEvents() {
|
||||||
|
document.body.addEventListener("click", async (event) => {
|
||||||
|
const deleteButton = event.target.closest(".btn-delete-menu");
|
||||||
|
if (deleteButton) {
|
||||||
|
event.preventDefault();
|
||||||
|
await this.handleDelete(deleteButton);
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
initTableMenus() {
|
initTableMenus() {
|
||||||
@@ -19,7 +33,7 @@ class Menus {
|
|||||||
this.table
|
this.table
|
||||||
.updateConfig({
|
.updateConfig({
|
||||||
server: {
|
server: {
|
||||||
url: `${GlobalConfig.apiHost}/api/api-menus`,
|
url: `${GlobalConfig.apiHost}/api/menus`,
|
||||||
credentials: "include",
|
credentials: "include",
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
@@ -60,7 +74,8 @@ class Menus {
|
|||||||
<a href="/menus/${cell}/edit" class="btn btn-yellow btn-sm d-inline-flex align-items-center justify-content-center">
|
<a href="/menus/${cell}/edit" class="btn btn-yellow btn-sm d-inline-flex align-items-center justify-content-center">
|
||||||
<i class='bx bx-edit'></i>
|
<i class='bx bx-edit'></i>
|
||||||
</a>
|
</a>
|
||||||
<button data-id="${cell}" class="btn btn-red btn-sm btn-delete-menu d-inline-flex align-items-center justify-content-center">
|
<button data-id="${cell}"
|
||||||
|
class="btn btn-red btn-sm btn-delete-menu d-inline-flex align-items-center justify-content-center">
|
||||||
<i class='bx bxs-trash' ></i>
|
<i class='bx bxs-trash' ></i>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
@@ -68,7 +83,7 @@ class Menus {
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
pagination: {
|
pagination: {
|
||||||
limit: 15,
|
limit: 50,
|
||||||
server: {
|
server: {
|
||||||
url: (prev, page) =>
|
url: (prev, page) =>
|
||||||
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
||||||
@@ -83,7 +98,7 @@ class Menus {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
server: {
|
server: {
|
||||||
url: `${GlobalConfig.apiHost}/api/api-menus`,
|
url: `${GlobalConfig.apiHost}/api/menus`,
|
||||||
credentials: "include",
|
credentials: "include",
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
@@ -104,118 +119,63 @@ class Menus {
|
|||||||
total: (data) => data.total,
|
total: (data) => data.total,
|
||||||
},
|
},
|
||||||
}).render(tableContainer);
|
}).render(tableContainer);
|
||||||
|
|
||||||
document.addEventListener("click", this.handleDelete.bind(this));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
handleDelete(event) {
|
async handleDelete(button) {
|
||||||
if (event.target.classList.contains("btn-delete-menu")) {
|
const id = button.getAttribute("data-id");
|
||||||
event.preventDefault();
|
|
||||||
const id = event.target.getAttribute("data-id");
|
|
||||||
let modalElement = document.getElementById("modalConfirmation");
|
|
||||||
let toastMessage = document.getElementById("toast-message");
|
|
||||||
|
|
||||||
if (!modalElement) {
|
const result = await Swal.fire({
|
||||||
console.error("Modal element not found!");
|
title: "Are you sure?",
|
||||||
return;
|
text: "You won't be able to revert this!",
|
||||||
}
|
icon: "warning",
|
||||||
|
showCancelButton: true,
|
||||||
let modal = new bootstrap.Modal(modalElement);
|
confirmButtonColor: "#3085d6",
|
||||||
let btnSaveConfirmation = document.getElementById(
|
cancelButtonColor: "#d33",
|
||||||
"btnSaveConfirmation"
|
confirmButtonText: "Yes, delete it!",
|
||||||
);
|
});
|
||||||
let toastElement = document.getElementById("toastNotification");
|
|
||||||
let toast = new bootstrap.Toast(toastElement);
|
|
||||||
|
|
||||||
// Remove previous event listeners to avoid multiple bindings
|
|
||||||
btnSaveConfirmation.replaceWith(
|
|
||||||
btnSaveConfirmation.cloneNode(true)
|
|
||||||
);
|
|
||||||
btnSaveConfirmation = document.getElementById(
|
|
||||||
"btnSaveConfirmation"
|
|
||||||
);
|
|
||||||
|
|
||||||
// Set the role ID on the confirm button inside the modal
|
|
||||||
btnSaveConfirmation.setAttribute("data-menu-id", id);
|
|
||||||
|
|
||||||
// Show the modal
|
|
||||||
modal.show();
|
|
||||||
|
|
||||||
btnSaveConfirmation.addEventListener("click", async () => {
|
|
||||||
let menuId = btnSaveConfirmation.getAttribute("data-menu-id");
|
|
||||||
|
|
||||||
|
if (result.isConfirmed) {
|
||||||
try {
|
try {
|
||||||
let response = await fetch(`/menus/${menuId}`, {
|
let response = await fetch(
|
||||||
|
`${GlobalConfig.apiHost}/api/menus/${id}`,
|
||||||
|
{
|
||||||
method: "DELETE",
|
method: "DELETE",
|
||||||
credentials: "include",
|
credentials: "include",
|
||||||
headers: {
|
|
||||||
"X-CSRF-TOKEN": document
|
|
||||||
.querySelector('meta[name="csrf-token"]')
|
|
||||||
.getAttribute("content"),
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (response.ok) {
|
|
||||||
let result = await response.json();
|
|
||||||
toastMessage.innerText =
|
|
||||||
result.message || "Deleted successfully!";
|
|
||||||
toast.show();
|
|
||||||
|
|
||||||
// Hide modal
|
|
||||||
modal.hide();
|
|
||||||
|
|
||||||
// Refresh Grid.js table
|
|
||||||
this.refreshTableMenus();
|
|
||||||
} else {
|
|
||||||
let error = await response.json();
|
|
||||||
console.error("Delete failed:", error);
|
|
||||||
toastMessage.innerText =
|
|
||||||
error.message || "Delete failed!";
|
|
||||||
toast.show();
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error("Error deleting item:", error);
|
|
||||||
toastMessage.innerText = "An error occurred!";
|
|
||||||
toast.show();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
refreshTableMenus() {
|
|
||||||
if (this.table) {
|
|
||||||
this.table
|
|
||||||
.updateConfig({
|
|
||||||
server: {
|
|
||||||
url: `${GlobalConfig.apiHost}/api/api-menus`,
|
|
||||||
credentials: "include",
|
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`,
|
.getAttribute("content")}`,
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
},
|
},
|
||||||
then: (data) =>
|
}
|
||||||
data.data.map((item) => [
|
);
|
||||||
item.id,
|
|
||||||
item.name,
|
if (response.ok) {
|
||||||
item.url,
|
let result = await response.json();
|
||||||
item.icon,
|
this.toastMessage.innerText =
|
||||||
item.parent_id,
|
result.message || "Deleted successfully!";
|
||||||
item.sort_order,
|
this.toast.show();
|
||||||
item.id,
|
|
||||||
]),
|
// Refresh Grid.js table
|
||||||
total: (data) => data.total,
|
if (typeof this.table !== "undefined") {
|
||||||
},
|
this.table.updateConfig({}).forceRender();
|
||||||
})
|
}
|
||||||
.forceRender();
|
|
||||||
} else {
|
} else {
|
||||||
this.initTableMenus(); // If no table exists, reinitialize it
|
let error = await response.json();
|
||||||
|
console.error("Delete failed:", error);
|
||||||
|
this.toastMessage.innerText =
|
||||||
|
error.message || "Delete failed!";
|
||||||
|
this.toast.show();
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error deleting item:", error);
|
||||||
|
this.toastMessage.innerText = "An error occurred!";
|
||||||
|
this.toast.show();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
document.addEventListener("DOMContentLoaded", function (e) {
|
document.addEventListener("DOMContentLoaded", function (e) {
|
||||||
new Menus().init();
|
new Menus();
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,13 +1,17 @@
|
|||||||
document.addEventListener("DOMContentLoaded", function (e) {
|
class UpdateMenu {
|
||||||
let form = document.getElementById("formUpdateMenus");
|
constructor() {
|
||||||
let submitButton = document.getElementById("btnUpdateMenus");
|
this.initUpdateMenu();
|
||||||
let spinner = document.getElementById("spinner");
|
}
|
||||||
let toastMessage = document.getElementById("toast-message");
|
|
||||||
let toast = new bootstrap.Toast(
|
initUpdateMenu() {
|
||||||
document.getElementById("toastNotification")
|
const toastNotification = document.getElementById("toastNotification");
|
||||||
);
|
const toast = new bootstrap.Toast(toastNotification);
|
||||||
submitButton.addEventListener("click", async function () {
|
document
|
||||||
|
.getElementById("btnUpdateMenus")
|
||||||
|
.addEventListener("click", async function () {
|
||||||
let submitButton = this;
|
let submitButton = this;
|
||||||
|
let spinner = document.getElementById("spinner");
|
||||||
|
let form = document.getElementById("formUpdateMenus");
|
||||||
|
|
||||||
if (!form) {
|
if (!form) {
|
||||||
console.error("Form element not found!");
|
console.error("Form element not found!");
|
||||||
@@ -24,30 +28,40 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
let response = await fetch(form.action, {
|
let response = await fetch(form.action, {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
"X-CSRF-TOKEN": document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="csrf-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content"),
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
body: formData,
|
body: formData,
|
||||||
});
|
});
|
||||||
|
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
let result = await response.json();
|
let result = await response.json();
|
||||||
toastMessage.innerText = result.message;
|
document.getElementById("toast-message").innerText =
|
||||||
|
result.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = "/menus";
|
window.location.href = "/menus";
|
||||||
}, 2000);
|
}, 2000);
|
||||||
} else {
|
} else {
|
||||||
let error = await response.json();
|
let error = await response.json();
|
||||||
toastMessage.innerText = error.message;
|
document.getElementById("toast-message").innerText =
|
||||||
|
error.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
console.error("Error:", error);
|
console.error("Error:", error);
|
||||||
|
submitButton.disabled = false;
|
||||||
|
spinner.classList.add("d-none");
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Request failed:", error);
|
console.error("Request failed:", error);
|
||||||
toastMessage.innerText = error.message;
|
document.getElementById("toast-message").innerText =
|
||||||
|
error.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", function (e) {
|
||||||
|
new UpdateMenu();
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,4 +1,9 @@
|
|||||||
document.addEventListener("DOMContentLoaded", function (e) {
|
class CreateRoles {
|
||||||
|
constructor() {
|
||||||
|
this.initCreateRole();
|
||||||
|
}
|
||||||
|
|
||||||
|
initCreateRole() {
|
||||||
const toastNotification = document.getElementById("toastNotification");
|
const toastNotification = document.getElementById("toastNotification");
|
||||||
const toast = new bootstrap.Toast(toastNotification);
|
const toast = new bootstrap.Toast(toastNotification);
|
||||||
document
|
document
|
||||||
@@ -23,9 +28,9 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
let response = await fetch(form.action, {
|
let response = await fetch(form.action, {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
"X-CSRF-TOKEN": document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="csrf-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content"),
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
body: formData,
|
body: formData,
|
||||||
});
|
});
|
||||||
@@ -44,6 +49,8 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
error.message;
|
error.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
console.error("Error:", error);
|
console.error("Error:", error);
|
||||||
|
submitButton.disabled = false;
|
||||||
|
spinner.classList.add("d-none");
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Request failed:", error);
|
console.error("Request failed:", error);
|
||||||
@@ -52,4 +59,9 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
toast.show();
|
toast.show();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", function (e) {
|
||||||
|
new CreateRoles();
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -2,46 +2,31 @@ import { Grid } from "gridjs/dist/gridjs.umd.js";
|
|||||||
import gridjs from "gridjs/dist/gridjs.umd.js";
|
import gridjs from "gridjs/dist/gridjs.umd.js";
|
||||||
import "gridjs/dist/gridjs.umd.js";
|
import "gridjs/dist/gridjs.umd.js";
|
||||||
import GlobalConfig from "../global-config";
|
import GlobalConfig from "../global-config";
|
||||||
|
import Swal from "sweetalert2";
|
||||||
|
|
||||||
class Roles {
|
class Roles {
|
||||||
constructor() {
|
constructor() {
|
||||||
this.table = null; // Store Grid.js instance
|
this.toastMessage = document.getElementById("toast-message");
|
||||||
}
|
this.toastElement = document.getElementById("toastNotification");
|
||||||
|
this.toast = new bootstrap.Toast(this.toastElement);
|
||||||
|
this.table = null;
|
||||||
|
|
||||||
init() {
|
// Initialize functions
|
||||||
this.initTableRoles();
|
this.initTableRoles();
|
||||||
|
this.initEvents();
|
||||||
|
}
|
||||||
|
initEvents() {
|
||||||
|
document.body.addEventListener("click", async (event) => {
|
||||||
|
const deleteButton = event.target.closest(".btn-delete-role");
|
||||||
|
if (deleteButton) {
|
||||||
|
event.preventDefault();
|
||||||
|
await this.handleDelete(deleteButton);
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
initTableRoles() {
|
initTableRoles() {
|
||||||
let tableContainer = document.getElementById("table-roles");
|
let tableContainer = document.getElementById("table-roles");
|
||||||
|
|
||||||
// If table instance already exists, update it instead of re-creating
|
|
||||||
if (this.table) {
|
|
||||||
this.table
|
|
||||||
.updateConfig({
|
|
||||||
server: {
|
|
||||||
url: `${GlobalConfig.apiHost}/api/api-roles`,
|
|
||||||
credentials: "include",
|
|
||||||
headers: {
|
|
||||||
Authorization: `Bearer ${document
|
|
||||||
.querySelector('meta[name="api-token"]')
|
|
||||||
.getAttribute("content")}`,
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
then: (data) =>
|
|
||||||
data.data.map((item) => [
|
|
||||||
item.id,
|
|
||||||
item.name,
|
|
||||||
item.description,
|
|
||||||
item.id,
|
|
||||||
]),
|
|
||||||
total: (data) => data.total,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
.forceRender();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create a new Grid.js instance only if it doesn't exist
|
// Create a new Grid.js instance only if it doesn't exist
|
||||||
this.table = new gridjs.Grid({
|
this.table = new gridjs.Grid({
|
||||||
columns: [
|
columns: [
|
||||||
@@ -67,7 +52,7 @@ class Roles {
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
pagination: {
|
pagination: {
|
||||||
limit: 15,
|
limit: 50,
|
||||||
server: {
|
server: {
|
||||||
url: (prev, page) =>
|
url: (prev, page) =>
|
||||||
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
||||||
@@ -82,7 +67,7 @@ class Roles {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
server: {
|
server: {
|
||||||
url: `${GlobalConfig.apiHost}/api/api-roles`,
|
url: `${GlobalConfig.apiHost}/api/roles`,
|
||||||
credentials: "include",
|
credentials: "include",
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
@@ -100,116 +85,63 @@ class Roles {
|
|||||||
total: (data) => data.total,
|
total: (data) => data.total,
|
||||||
},
|
},
|
||||||
}).render(tableContainer);
|
}).render(tableContainer);
|
||||||
|
|
||||||
document.addEventListener("click", this.handleDelete.bind(this));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
handleDelete(event) {
|
async handleDelete(deleteButton) {
|
||||||
if (event.target.classList.contains("btn-delete-role")) {
|
const id = deleteButton.getAttribute("data-id");
|
||||||
event.preventDefault();
|
|
||||||
|
|
||||||
const id = event.target.getAttribute("data-id");
|
const result = await Swal.fire({
|
||||||
let modalElement = document.getElementById("modalConfirmation");
|
title: "Are you sure?",
|
||||||
let toastMessage = document.getElementById("toast-message");
|
text: "You won't be able to revert this!",
|
||||||
|
icon: "warning",
|
||||||
if (!modalElement) {
|
showCancelButton: true,
|
||||||
console.error("Modal element not found!");
|
confirmButtonColor: "#3085d6",
|
||||||
return;
|
cancelButtonColor: "#d33",
|
||||||
}
|
confirmButtonText: "Yes, delete it!",
|
||||||
|
});
|
||||||
let modal = new bootstrap.Modal(modalElement);
|
|
||||||
let btnSaveConfirmation = document.getElementById(
|
|
||||||
"btnSaveConfirmation"
|
|
||||||
);
|
|
||||||
let toastElement = document.getElementById("toastNotification");
|
|
||||||
let toast = new bootstrap.Toast(toastElement);
|
|
||||||
|
|
||||||
// Remove previous event listeners to avoid multiple bindings
|
|
||||||
btnSaveConfirmation.replaceWith(
|
|
||||||
btnSaveConfirmation.cloneNode(true)
|
|
||||||
);
|
|
||||||
btnSaveConfirmation = document.getElementById(
|
|
||||||
"btnSaveConfirmation"
|
|
||||||
);
|
|
||||||
|
|
||||||
// Set the role ID on the confirm button inside the modal
|
|
||||||
btnSaveConfirmation.setAttribute("data-role-id", id);
|
|
||||||
|
|
||||||
// Show the modal
|
|
||||||
modal.show();
|
|
||||||
|
|
||||||
btnSaveConfirmation.addEventListener("click", async () => {
|
|
||||||
let roleId = btnSaveConfirmation.getAttribute("data-role-id");
|
|
||||||
|
|
||||||
|
if (result.isConfirmed) {
|
||||||
try {
|
try {
|
||||||
let response = await fetch(`/roles/${roleId}`, {
|
let response = await fetch(
|
||||||
|
`${GlobalConfig.apiHost}/api/roles/${id}`,
|
||||||
|
{
|
||||||
method: "DELETE",
|
method: "DELETE",
|
||||||
credentials: "include",
|
credentials: "include",
|
||||||
headers: {
|
|
||||||
"X-CSRF-TOKEN": document
|
|
||||||
.querySelector('meta[name="csrf-token"]')
|
|
||||||
.getAttribute("content"),
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (response.ok) {
|
|
||||||
let result = await response.json();
|
|
||||||
toastMessage.innerText =
|
|
||||||
result.message || "Deleted successfully!";
|
|
||||||
toast.show();
|
|
||||||
|
|
||||||
// Hide modal
|
|
||||||
modal.hide();
|
|
||||||
|
|
||||||
// Refresh Grid.js table
|
|
||||||
this.refreshRolesTable();
|
|
||||||
} else {
|
|
||||||
let error = await response.json();
|
|
||||||
console.error("Delete failed:", error);
|
|
||||||
toastMessage.innerText =
|
|
||||||
error.message || "Delete failed!";
|
|
||||||
toast.show();
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error("Error deleting item:", error);
|
|
||||||
toastMessage.innerText = "An error occurred!";
|
|
||||||
toast.show();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
refreshRolesTable() {
|
|
||||||
if (this.table) {
|
|
||||||
this.table
|
|
||||||
.updateConfig({
|
|
||||||
server: {
|
|
||||||
url: `${GlobalConfig.apiHost}/api/api-roles`,
|
|
||||||
credentials: "include",
|
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="api-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content")}`,
|
.getAttribute("content")}`,
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
},
|
},
|
||||||
then: (data) =>
|
}
|
||||||
data.data.map((item) => [
|
);
|
||||||
item.id,
|
|
||||||
item.name,
|
if (response.ok) {
|
||||||
item.description,
|
let result = await response.json();
|
||||||
item.id,
|
this.toastMessage.innerText =
|
||||||
]),
|
result.message || "Deleted successfully!";
|
||||||
total: (data) => data.total,
|
this.toast.show();
|
||||||
},
|
|
||||||
})
|
// Refresh Grid.js table
|
||||||
.forceRender();
|
if (typeof this.table !== "undefined") {
|
||||||
|
this.table.updateConfig({}).forceRender();
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
this.initTableRoles(); // If the table is null, reinitialize
|
let error = await response.json();
|
||||||
|
console.error("Delete failed:", error);
|
||||||
|
this.toastMessage.innerText =
|
||||||
|
error.message || "Delete failed!";
|
||||||
|
this.toast.show();
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error deleting item:", error);
|
||||||
|
this.toastMessage.innerText = "An error occurred!";
|
||||||
|
this.toast.show();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
document.addEventListener("DOMContentLoaded", function (e) {
|
document.addEventListener("DOMContentLoaded", function (e) {
|
||||||
new Roles().init();
|
new Roles();
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,13 +1,17 @@
|
|||||||
document.addEventListener("DOMContentLoaded", function (e) {
|
class UpdateRoles {
|
||||||
let form = document.getElementById("formUpdateRole");
|
constructor() {
|
||||||
let submitButton = document.getElementById("btnUpdateRole");
|
this.initUpdateRole();
|
||||||
let spinner = document.getElementById("spinner");
|
}
|
||||||
let toastMessage = document.getElementById("toast-message");
|
|
||||||
let toast = new bootstrap.Toast(
|
initUpdateRole() {
|
||||||
document.getElementById("toastNotification")
|
const toastNotification = document.getElementById("toastNotification");
|
||||||
);
|
const toast = new bootstrap.Toast(toastNotification);
|
||||||
submitButton.addEventListener("click", async function () {
|
document
|
||||||
|
.getElementById("btnUpdateRole")
|
||||||
|
.addEventListener("click", async function () {
|
||||||
let submitButton = this;
|
let submitButton = this;
|
||||||
|
let spinner = document.getElementById("spinner");
|
||||||
|
let form = document.getElementById("formUpdateRole");
|
||||||
|
|
||||||
if (!form) {
|
if (!form) {
|
||||||
console.error("Form element not found!");
|
console.error("Form element not found!");
|
||||||
@@ -24,30 +28,40 @@ document.addEventListener("DOMContentLoaded", function (e) {
|
|||||||
let response = await fetch(form.action, {
|
let response = await fetch(form.action, {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
"X-CSRF-TOKEN": document
|
Authorization: `Bearer ${document
|
||||||
.querySelector('meta[name="csrf-token"]')
|
.querySelector('meta[name="api-token"]')
|
||||||
.getAttribute("content"),
|
.getAttribute("content")}`,
|
||||||
},
|
},
|
||||||
body: formData,
|
body: formData,
|
||||||
});
|
});
|
||||||
|
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
let result = await response.json();
|
let result = await response.json();
|
||||||
toastMessage.innerText = result.message;
|
document.getElementById("toast-message").innerText =
|
||||||
|
result.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = "/roles";
|
window.location.href = "/roles";
|
||||||
}, 2000);
|
}, 2000);
|
||||||
} else {
|
} else {
|
||||||
let error = await response.json();
|
let error = await response.json();
|
||||||
toastMessage.innerText = error.message;
|
document.getElementById("toast-message").innerText =
|
||||||
|
error.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
console.error("Error:", error);
|
console.error("Error:", error);
|
||||||
|
submitButton.disabled = false;
|
||||||
|
spinner.classList.add("d-none");
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Request failed:", error);
|
console.error("Request failed:", error);
|
||||||
toastMessage.innerText = error.message;
|
document.getElementById("toast-message").innerText =
|
||||||
|
error.message;
|
||||||
toast.show();
|
toast.show();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", function (e) {
|
||||||
|
new UpdateRoles();
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -4,13 +4,21 @@ import "gridjs/dist/gridjs.umd.js";
|
|||||||
import GlobalConfig from "../../global-config.js";
|
import GlobalConfig from "../../global-config.js";
|
||||||
|
|
||||||
class SyncronizeTask {
|
class SyncronizeTask {
|
||||||
|
constructor() {
|
||||||
|
this.toastElement = document.getElementById("toastNotification");
|
||||||
|
this.toastMessage = document.getElementById("toast-message");
|
||||||
|
this.toast = new bootstrap.Toast(this.toastElement);
|
||||||
|
this.table = null;
|
||||||
|
}
|
||||||
init() {
|
init() {
|
||||||
this.initTableImportDatasources();
|
this.initTableImportDatasources();
|
||||||
this.handleSubmitSync();
|
this.handleSubmitSync();
|
||||||
this.handleSubmitSnycGoogleSheet();
|
|
||||||
}
|
}
|
||||||
initTableImportDatasources() {
|
initTableImportDatasources() {
|
||||||
new Grid({
|
let tableContainer = document.getElementById(
|
||||||
|
"table-import-datasources"
|
||||||
|
);
|
||||||
|
this.table = new gridjs.Grid({
|
||||||
columns: ["ID", "Message", "Response", "Status", "Created"],
|
columns: ["ID", "Message", "Response", "Status", "Created"],
|
||||||
search: {
|
search: {
|
||||||
server: {
|
server: {
|
||||||
@@ -18,7 +26,7 @@ class SyncronizeTask {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
pagination: {
|
pagination: {
|
||||||
limit: 15,
|
limit: 50,
|
||||||
server: {
|
server: {
|
||||||
url: (prev, page) =>
|
url: (prev, page) =>
|
||||||
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
`${prev}${prev.includes("?") ? "&" : "?"}page=${
|
||||||
@@ -45,20 +53,24 @@ class SyncronizeTask {
|
|||||||
]),
|
]),
|
||||||
total: (data) => data.meta.total,
|
total: (data) => data.meta.total,
|
||||||
},
|
},
|
||||||
}).render(document.getElementById("table-import-datasources"));
|
}).render(tableContainer);
|
||||||
}
|
}
|
||||||
handleSubmitSync() {
|
handleSubmitSync() {
|
||||||
const button = document.getElementById("btn-sync-submit");
|
const button = document.getElementById("btn-sync-submit");
|
||||||
|
const spinner = document.getElementById("spinner");
|
||||||
|
const apiToken = document
|
||||||
|
.querySelector('meta[name="api-token"]')
|
||||||
|
.getAttribute("content");
|
||||||
|
|
||||||
|
// Show the spinner while checking
|
||||||
|
spinner.classList.remove("d-none");
|
||||||
|
|
||||||
// Check if the button should be enabled or disabled based on the status
|
|
||||||
fetch(
|
fetch(
|
||||||
`${GlobalConfig.apiHost}/api/import-datasource/check-datasource`,
|
`${GlobalConfig.apiHost}/api/import-datasource/check-datasource`,
|
||||||
{
|
{
|
||||||
method: "GET",
|
method: "GET",
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${apiToken}`,
|
||||||
.querySelector('meta[name="api-token"]')
|
|
||||||
.getAttribute("content")}`,
|
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@@ -70,91 +82,74 @@ class SyncronizeTask {
|
|||||||
return response.json();
|
return response.json();
|
||||||
})
|
})
|
||||||
.then((data) => {
|
.then((data) => {
|
||||||
console.log("data check button sync", data.can_execute);
|
|
||||||
button.disabled = !data.can_execute;
|
button.disabled = !data.can_execute;
|
||||||
|
|
||||||
// If the button is enabled, add click event to trigger sync
|
if (!data.can_execute) {
|
||||||
if (!button.disabled) {
|
// Keep spinner visible if cannot execute
|
||||||
button.addEventListener("click", function (e) {
|
spinner.classList.remove("d-none");
|
||||||
button.disabled = true; // Disable button to prevent multiple clicks
|
} else {
|
||||||
button.textContent = "Syncing..."; // Change button text to show syncing
|
// Hide spinner when execution is allowed
|
||||||
|
spinner.classList.add("d-none");
|
||||||
|
|
||||||
// Trigger the scraping API call
|
// Remove previous event listener before adding a new one
|
||||||
fetch(`${GlobalConfig.apiHost}/api/scraping`, {
|
button.removeEventListener("click", this.handleSyncClick);
|
||||||
method: "GET",
|
button.addEventListener(
|
||||||
headers: {
|
"click",
|
||||||
Authorization: `Bearer ${document
|
this.handleSyncClick.bind(this)
|
||||||
.querySelector('meta[name="api-token"]')
|
|
||||||
.getAttribute("content")}`,
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
},
|
|
||||||
})
|
|
||||||
.then((response) => {
|
|
||||||
if (!response.ok) {
|
|
||||||
throw new Error(
|
|
||||||
"Network response was not ok"
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return response.json();
|
|
||||||
})
|
|
||||||
.then((data) => {
|
|
||||||
console.log("data sync button", data);
|
|
||||||
alert("Synchronization executed successfully");
|
|
||||||
window.location.reload();
|
|
||||||
})
|
|
||||||
.catch((err) => {
|
|
||||||
console.error("Fetch error:", err);
|
|
||||||
alert(
|
|
||||||
"An error occurred during synchronization"
|
|
||||||
);
|
|
||||||
})
|
|
||||||
.finally(() => {
|
|
||||||
button.disabled = false; // Re-enable the button after the request is complete
|
|
||||||
button.textContent = "Sync Data"; // Reset button text
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
.catch((err) => {
|
.catch((err) => {
|
||||||
console.error("Fetch error:", err);
|
console.error("Fetch error:", err);
|
||||||
alert("An error occurred while checking the datasource");
|
alert("An error occurred while checking the datasource");
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
handleSubmitSnycGoogleSheet() {
|
|
||||||
const button = document.getElementById("btn-sync-submit-google-sheet");
|
|
||||||
button.addEventListener("click", function (e) {
|
|
||||||
button.disabled = true; // Disable button to prevent multiple clicks
|
|
||||||
button.textContent = "Syncing..."; // Change button text to show syncing
|
|
||||||
|
|
||||||
// Trigger the scraping API call
|
handleSyncClick() {
|
||||||
fetch(`${GlobalConfig.apiHost}/api/sync-pbg-task-google-sheet`, {
|
const button = document.getElementById("btn-sync-submit");
|
||||||
|
const spinner = document.getElementById("spinner");
|
||||||
|
const apiToken = document
|
||||||
|
.querySelector('meta[name="api-token"]')
|
||||||
|
.getAttribute("content");
|
||||||
|
|
||||||
|
button.disabled = true; // Prevent multiple clicks
|
||||||
|
spinner.classList.remove("d-none"); // Show spinner during sync
|
||||||
|
|
||||||
|
fetch(`${GlobalConfig.apiHost}/api/scraping`, {
|
||||||
method: "GET",
|
method: "GET",
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${document
|
Authorization: `Bearer ${apiToken}`,
|
||||||
.querySelector('meta[name="api-token"]')
|
|
||||||
.getAttribute("content")}`,
|
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
.then((response) => {
|
.then(async (response) => {
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error("Network response was not ok");
|
throw new Error(`HTTP error! Status: ${response.status}`);
|
||||||
}
|
}
|
||||||
return response.json();
|
|
||||||
|
let data;
|
||||||
|
try {
|
||||||
|
data = await response.json();
|
||||||
|
} catch (jsonError) {
|
||||||
|
throw new Error("Failed to parse JSON response");
|
||||||
|
}
|
||||||
|
|
||||||
|
return data;
|
||||||
})
|
})
|
||||||
.then((data) => {
|
.then((data) => {
|
||||||
console.log("data sync button", data);
|
this.toastMessage.innerText =
|
||||||
alert("Synchronization executed successfully");
|
data.data.message || "Synchronize successfully!";
|
||||||
window.location.reload();
|
this.toast.show();
|
||||||
|
|
||||||
|
// Update the table if it exists
|
||||||
|
if (this.table) {
|
||||||
|
this.table.updateConfig({}).forceRender();
|
||||||
|
}
|
||||||
})
|
})
|
||||||
.catch((err) => {
|
.catch((err) => {
|
||||||
console.error("Fetch error:", err);
|
console.error("Fetch error:", err);
|
||||||
alert("An error occurred during synchronization");
|
alert("An error occurred during synchronization" + err.message);
|
||||||
})
|
button.disabled = false;
|
||||||
.finally(() => {
|
|
||||||
button.disabled = false; // Re-enable the button after the request is complete
|
|
||||||
button.textContent = "Sync Google Sheet"; // Reset button text
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
34
resources/js/utils/InitDatePicker.js
Normal file
34
resources/js/utils/InitDatePicker.js
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
import flatpickr from "flatpickr";
|
||||||
|
import "flatpickr/dist/flatpickr.min.css";
|
||||||
|
|
||||||
|
class InitDatePicker {
|
||||||
|
constructor(selector = ".datepicker", onChangeCallback = null) {
|
||||||
|
this.selector = selector;
|
||||||
|
this.onChangeCallback = onChangeCallback;
|
||||||
|
}
|
||||||
|
|
||||||
|
init() {
|
||||||
|
const today = new Date();
|
||||||
|
|
||||||
|
document.querySelectorAll(this.selector).forEach((element) => {
|
||||||
|
flatpickr(element, {
|
||||||
|
enableTime: false,
|
||||||
|
dateFormat: "Y-m-d",
|
||||||
|
maxDate: today,
|
||||||
|
onChange: (selectedDates, dateStr) => {
|
||||||
|
if (this.onChangeCallback) {
|
||||||
|
this.onChangeCallback(dateStr); // Call callback with selected date
|
||||||
|
}
|
||||||
|
},
|
||||||
|
onReady: (selectedDates, dateStr, instance) => {
|
||||||
|
// Call the callback with the default date when initialized
|
||||||
|
if (this.onChangeCallback && dateStr) {
|
||||||
|
this.onChangeCallback(dateStr);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default InitDatePicker;
|
||||||
@@ -21,8 +21,8 @@
|
|||||||
// overflow: hidden;
|
// overflow: hidden;
|
||||||
|
|
||||||
.circle-content {
|
.circle-content {
|
||||||
width: 180px; /* Ukuran lingkaran dalam */
|
min-width: 180px; /* Ukuran lingkaran dalam */
|
||||||
height: 180px;
|
min-height: 180px;
|
||||||
background-color: var(--circle-color); /* Warna lingkaran dalam */
|
background-color: var(--circle-color); /* Warna lingkaran dalam */
|
||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -57,17 +57,20 @@
|
|||||||
padding: 0 7px;
|
padding: 0 7px;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
|
max-width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.circle-content .document-count {
|
.circle-content .document-count {
|
||||||
font-size: 24px;
|
font-size: 24px;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
|
max-width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.circle-content .document-type {
|
.circle-content .document-type {
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
|
max-width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.small-circle-container {
|
.small-circle-container {
|
||||||
|
|||||||
104
resources/views/bigdata-resumes/index.blade.php
Normal file
104
resources/views/bigdata-resumes/index.blade.php
Normal file
@@ -0,0 +1,104 @@
|
|||||||
|
@extends('layouts.vertical', ['subtitle' => 'Laporan Pimpinan'])
|
||||||
|
|
||||||
|
@section('css')
|
||||||
|
@vite(['node_modules/gridjs/dist/theme/mermaid.min.css'])
|
||||||
|
<style>
|
||||||
|
/* Ensure the Grid.js container allows full scrolling */
|
||||||
|
#table-bigdata-resumes {
|
||||||
|
overflow-x: auto;
|
||||||
|
overflow-y: hidden; /* Prevent vertical scrolling */
|
||||||
|
white-space: nowrap;
|
||||||
|
max-width: 100%;
|
||||||
|
height: 100%; /* Adjust height if necessary */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ensure Grid.js wrapper is scrollable */
|
||||||
|
.gridjs-wrapper {
|
||||||
|
max-width: 100%;
|
||||||
|
overflow-x: auto;
|
||||||
|
overflow-y: hidden;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Make the entire scrollbar much bigger */
|
||||||
|
.gridjs-wrapper::-webkit-scrollbar {
|
||||||
|
height: 40px; /* Increase scrollbar height */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Scrollbar track (background) */
|
||||||
|
.gridjs-wrapper::-webkit-scrollbar-track {
|
||||||
|
background: #ddd;
|
||||||
|
border-radius: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Scrollbar thumb (draggable part) */
|
||||||
|
.gridjs-wrapper::-webkit-scrollbar-thumb {
|
||||||
|
background: #007bff;
|
||||||
|
border-radius: 20px;
|
||||||
|
width: 40px; /* Wider scrollbar thumb */
|
||||||
|
min-width: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Scrollbar thumb hover effect */
|
||||||
|
.gridjs-wrapper::-webkit-scrollbar-thumb:hover {
|
||||||
|
background: #0056b3;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Bigger Scrollbar Buttons */
|
||||||
|
.gridjs-wrapper::-webkit-scrollbar-button {
|
||||||
|
background: #007bff;
|
||||||
|
height: 40px; /* Force bigger button height */
|
||||||
|
width: 40px; /* Force bigger button width */
|
||||||
|
border-radius: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Left Scroll Button */
|
||||||
|
.gridjs-wrapper::-webkit-scrollbar-button:horizontal:decrement {
|
||||||
|
display: block;
|
||||||
|
background: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="white"><path d="M15 18l-6-6 6-6"/></svg>') no-repeat center;
|
||||||
|
background-size: 30px;
|
||||||
|
width: 40px; /* Ensure button size */
|
||||||
|
height: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Right Scroll Button */
|
||||||
|
.gridjs-wrapper::-webkit-scrollbar-button:horizontal:increment {
|
||||||
|
display: block;
|
||||||
|
background: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="white"><path d="M9 18l6-6-6-6"/></svg>') no-repeat center;
|
||||||
|
background-size: 30px;
|
||||||
|
width: 40px; /* Ensure button size */
|
||||||
|
height: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Scrollbar button hover effect */
|
||||||
|
.gridjs-wrapper::-webkit-scrollbar-button:hover {
|
||||||
|
background: #0056b3;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
@endsection
|
||||||
|
|
||||||
|
@section('content')
|
||||||
|
|
||||||
|
@include('layouts.partials/page-title', ['title' => 'Laporan', 'subtitle' => 'Laporan Pimpinan'])
|
||||||
|
|
||||||
|
<x-toast-notification />
|
||||||
|
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-12">
|
||||||
|
<div class="card w-100 h-100">
|
||||||
|
<div class="card-header d-flex align-items-center">
|
||||||
|
<input type="text" class="form-control me-2 w-auto" />
|
||||||
|
<button id="search-btn" class="btn btn-md btn-info text-white">Cari</button>
|
||||||
|
</div>
|
||||||
|
<div class="card-body">
|
||||||
|
<div id="table-bigdata-resumes"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@endsection
|
||||||
|
|
||||||
|
@section('scripts')
|
||||||
|
@vite(['resources/js/bigdata-resumes/index.js'])
|
||||||
|
@endsection
|
||||||
@@ -64,7 +64,10 @@
|
|||||||
<!-- end dropzon-preview -->
|
<!-- end dropzon-preview -->
|
||||||
</div>
|
</div>
|
||||||
<div class="d-flex justify-content-end">
|
<div class="d-flex justify-content-end">
|
||||||
<button id="submit-upload" class="btn btn-primary">Upload Files</button>
|
<button type="button" class="btn btn-primary" id="btnUploadBusinessIndustry">
|
||||||
|
<span id="spinner" class="spinner-border spinner-border-sm me-1 d-none" role="status" aria-hidden="true"></span>
|
||||||
|
Upload
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div> <!-- end card body -->
|
</div> <!-- end card body -->
|
||||||
</div> <!-- end card -->
|
</div> <!-- end card -->
|
||||||
|
|||||||
@@ -9,14 +9,13 @@
|
|||||||
@include('layouts.partials/page-title', ['title' => 'Data', 'subtitle' => 'Business Industries'])
|
@include('layouts.partials/page-title', ['title' => 'Data', 'subtitle' => 'Business Industries'])
|
||||||
|
|
||||||
<x-toast-notification />
|
<x-toast-notification />
|
||||||
<x-modal-confirmation buttonText="Delete" confirmationMessage="Are you sure you want to delete this?" />
|
|
||||||
|
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-12">
|
<div class="col-12">
|
||||||
<div class="card w-100">
|
<div class="card w-100">
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<div class="d-flex flex-wrap justify-content-end align-items-center mb-2">
|
<div class="d-flex flex-wrap justify-content-end align-items-center mb-2">
|
||||||
<a href="{{ route('business-industries.create')}}" class="btn btn-success btn-sm d-block d-sm-inline w-auto">Create</a>
|
<a href="{{ route('business-industries.create')}}" class="btn btn-success btn-sm d-block d-sm-inline w-auto">Upload</a>
|
||||||
</div>
|
</div>
|
||||||
<div id="table-business-industries"></div>
|
<div id="table-business-industries"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
77
resources/views/chatbot-pimpinan/index.blade.php
Normal file
77
resources/views/chatbot-pimpinan/index.blade.php
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
@extends('layouts.vertical', ['subtitle' => 'Main Chatbot'])
|
||||||
|
|
||||||
|
@section('css')
|
||||||
|
@vite(['node_modules/gridjs/dist/theme/mermaid.min.css'])
|
||||||
|
<style>
|
||||||
|
#user-message {
|
||||||
|
height: 60px; /* Menambah tinggi textarea */
|
||||||
|
font-size: 1.1rem; /* Memperbesar font */
|
||||||
|
padding: 10px; /* Menambah ruang di dalam textarea */
|
||||||
|
resize: none; /* Mencegah resize manual */
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 10px;
|
||||||
|
aspect-ratio: 1;
|
||||||
|
border-radius: 50%;
|
||||||
|
animation: l5 1s infinite linear alternate;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes l5 {
|
||||||
|
0% {box-shadow: 10px 0 #000, -10px 0 #0002; background: #000 }
|
||||||
|
33% {box-shadow: 10px 0 #000, -10px 0 #0002; background: #0002}
|
||||||
|
66% {box-shadow: 10px 0 #0002, -10px 0 #000; background: #0002}
|
||||||
|
100%{box-shadow: 10px 0 #0002, -10px 0 #000; background: #000 }
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
@endsection
|
||||||
|
|
||||||
|
@section('content')
|
||||||
|
@include('layouts.partials/page-title', ['title' => 'Main Chatbot', 'subtitle' => 'Main Chatbot'])
|
||||||
|
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-body d-flex flex-column" style="height: 700px;">
|
||||||
|
<!-- Conversation Area -->
|
||||||
|
|
||||||
|
<!-- Bot Response -->
|
||||||
|
<div class="row flex-grow overflow-auto align-items-start">
|
||||||
|
<!-- Avatar -->
|
||||||
|
<div class="col-auto alignpe-0">
|
||||||
|
<img class="rounded-circle" width="45" src="/images/iconchatbot.jpeg" alt="avatar-3">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Nama dan Bubble Chat -->
|
||||||
|
<div class="col-9 w-auto">
|
||||||
|
<!-- Nama Bot -->
|
||||||
|
<p class="fw-bolder mb-1">Neng Bedas</p>
|
||||||
|
|
||||||
|
<!-- Bubble Chat -->
|
||||||
|
<div class="bot-response p-2 bg-light rounded mb-2 d-inline-block">
|
||||||
|
<p class="mb-0">Halo! Ada yang bisa saya bantu?</p>
|
||||||
|
|
||||||
|
<!-- Waktu (Tetap di Dalam Bubble Chat) -->
|
||||||
|
<div class="sending-message-time text-end mt-1">
|
||||||
|
<p class="text-muted small mb-0">Now</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<!-- Input & Button (Selalu di Bawah) -->
|
||||||
|
<div class="row mt-auto">
|
||||||
|
<div class="col-xl-12 d-flex align-items-end gap-1">
|
||||||
|
<textarea class="form-control" id="user-message"></textarea>
|
||||||
|
<button id="send" class="btn btn-primary btn-lg h-100 d-flex align-items-center">
|
||||||
|
<i class='bx bx-send'></i>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@endsection
|
||||||
|
|
||||||
|
@section('scripts')
|
||||||
|
@vite(['resources/js/chatbot-pimpinan/index.js'])
|
||||||
|
@endsection
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user