feat(parameter): create master indicator page with update func

This commit is contained in:
arieeefajar 2025-01-28 21:24:16 +07:00
parent 40c73aeea2
commit bdb51e4f71
9 changed files with 781 additions and 1 deletions

View File

@ -0,0 +1,69 @@
<?php
namespace App\Http\Controllers\MasterData;
use App\Http\Controllers\Controller;
use App\Models\Indicator;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Validator;
class IndicatorController extends Controller
{
public function index()
{
$indicators = Indicator::all();
return view('master-data.indikator.index', compact('indicators'));
}
public function update(Request $request, $id)
{
$customMessage = [
"name.required" => "Nama wajib diisi",
"name.max" => "Nama maksimal 255 karakter",
"name.string" => "Nama harus berupa string",
"description.required" => "Deskripsi wajib diisi",
"description.max" => "Deskripsi maksimal 255 karakter",
"description.string" => "Deskripsi harus berupa string",
"ideal_min.required" => "Nilai ideal minimum wajib diisi",
"ideal_min.numeric" => "Nilai ideal minimum harus berupa angka",
"ideal_max.required" => "Nilai ideal maksimum wajib diisi",
"ideal_max.numeric" => "Nilai ideal maksimum harus berupa angka",
"unit.required" => "Unit wajib diisi",
"unit.max" => "Unit maksimal 255 karakter",
"unit.string" => "Unit harus berupa string",
];
$validator = Validator::make($request->all(), [
'name' => 'required|string|max:255',
'description' => 'required|string|max:255',
'ideal_min' => 'required|numeric',
'ideal_max' => 'required|numeric',
'unit' => 'required|string|max:255',
], $customMessage);
if ($validator->fails()) {
toast($validator->messages()->all()[0], 'error')->position('top')->autoclose(3000);
return redirect()->back()->withInput();
}
$indicator = Indicator::find($id);
$indicator->name = $request->name;
$indicator->description = $request->description;
$indicator->ideal_min = $request->ideal_min;
$indicator->ideal_max = $request->ideal_max;
$indicator->unit = $request->unit;
try {
$indicator->save();
toast('Data berhasil diubah', 'success')->position('top-right')->autoclose(3000);
return redirect()->back();
} catch (\Throwable $th) {
toast('Terjadi kesalahan', 'error')->position('top')->autoclose(3000);
return redirect()->back();
}
}
}

21
app/Models/Indicator.php Normal file
View File

@ -0,0 +1,21 @@
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;
class Indicator extends Model
{
use HasFactory;
protected $table = 'indicator';
protected $fillable = [
'name',
'description',
'ideal_min',
'ideal_max',
'unit',
];
}

View File

@ -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::create('indicator', function (Blueprint $table) {
$table->id();
$table->string('name');
$table->string('description');
$table->float('ideal_min');
$table->float('ideal_max');
$table->string('unit');
$table->timestamps();
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('indicator');
}
};

View File

@ -21,7 +21,8 @@ public function run(): void
$this->call([ $this->call([
UsersSeeder::class, UsersSeeder::class,
LandSeeder::class LandSeeder::class,
IndicatorSeeder::class
]); ]);
} }
} }

View File

@ -0,0 +1,47 @@
<?php
namespace Database\Seeders;
use App\Models\Indicator;
use Illuminate\Database\Console\Seeds\WithoutModelEvents;
use Illuminate\Database\Seeder;
class IndicatorSeeder extends Seeder
{
/**
* Run the database seeds.
*/
public function run(): void
{
[
Indicator::create([
"name" => "pH Tanah",
"description" => "Tingkat keasaman tanah.",
"ideal_min" => 5.5,
"ideal_max" => 7.0,
"unit" => "pH"
]),
Indicator::create([
"name" => "Ketinggian Tempat",
"description" => "Ketinggian dari permukaan laut..",
"ideal_min" => 0,
"ideal_max" => 1000,
"unit" => "meter"
]),
Indicator::create([
"name" => "Ketersediaan Air",
"description" => "Ketersediaan air di lahan.",
"ideal_min" => 70,
"ideal_max" => 100,
"unit" => "persen"
]),
Indicator::create([
"name" => "Curah Hujan",
"description" => "Intensitas curah hujan.",
"ideal_min" => 1000,
"ideal_max" => 2000,
"unit" => "mm/tahun"
]),
];
}
}

View File

@ -0,0 +1,324 @@
function updateData(indicator) {
console.log(indicator);
var form = document.getElementById("edit-form");
var indicatorName = form.querySelector('input[name="name"]');
var description = form.querySelector('textarea[name="description"]');
var idealMin = form.querySelector('input[name="ideal_min"]');
var idealMax = form.querySelector('input[name="ideal_max"]');
var unit = form.querySelector('input[name="unit"]');
indicatorName.value = indicator.name;
description.value = indicator.description;
idealMin.value = indicator.ideal_min;
idealMax.value = indicator.ideal_max;
unit.value = indicator.unit;
form.action = "/data-indikator/" + indicator.id;
}
var checkAll = document.getElementById("checkAll");
checkAll &&
(checkAll.onclick = function () {
for (
var e = document.querySelectorAll(
'.form-check-all input[type="checkbox"]'
),
t = 0;
t < e.length;
t++
)
(e[t].checked = this.checked),
e[t].checked
? e[t].closest("tr").classList.add("table-active")
: e[t].closest("tr").classList.remove("table-active");
});
var perPage = 8,
options = {
valueNames: [
"id",
"indicator_name",
"description",
"ideal_min",
"ideal_max",
"status",
],
page: perPage,
pagination: !0,
plugins: [ListPagination({ left: 2, right: 2 })],
},
customerList = new List("customerList", options).on(
"updated",
function (e) {
0 == e.matchingItems.length
? (document.getElementsByClassName(
"noresult"
)[0].style.display = "block")
: (document.getElementsByClassName(
"noresult"
)[0].style.display = "none");
var t = 1 == e.i,
a = e.i > e.matchingItems.length - e.page;
document.querySelector(".pagination-prev.disabled") &&
document
.querySelector(".pagination-prev.disabled")
.classList.remove("disabled"),
document.querySelector(".pagination-next.disabled") &&
document
.querySelector(".pagination-next.disabled")
.classList.remove("disabled"),
t &&
document
.querySelector(".pagination-prev")
.classList.add("disabled"),
a &&
document
.querySelector(".pagination-next")
.classList.add("disabled"),
e.matchingItems.length <= perPage
? (document.querySelector(
".pagination-wrap"
).style.display = "none")
: (document.querySelector(
".pagination-wrap"
).style.display = "flex"),
e.matchingItems.length == perPage &&
document
.querySelector(".pagination.listjs-pagination")
.firstElementChild.children[0].click(),
0 < e.matchingItems.length
? (document.getElementsByClassName(
"noresult"
)[0].style.display = "none")
: (document.getElementsByClassName(
"noresult"
)[0].style.display = "block");
}
);
isCount = new DOMParser().parseFromString(
customerList.items.slice(-1)[0]._values.id,
"text/html"
);
var isValue = isCount.body.firstElementChild.innerHTML,
idField = document.getElementById("id-field"),
customerNameField = document.getElementById("customername-field"),
emailField = document.getElementById("email-field"),
dateField = document.getElementById("date-field"),
phoneField = document.getElementById("phone-field"),
statusField = document.getElementById("status-field"),
addBtn = document.getElementById("add-btn"),
editBtn = document.getElementById("edit-btn"),
removeBtns = document.getElementsByClassName("remove-item-btn"),
editBtns = document.getElementsByClassName("edit-item-btn");
function filterContact(e) {
var t = e;
customerList.filter(function (e) {
matchData = new DOMParser().parseFromString(
e.values().status,
"text/html"
);
e = matchData.body.firstElementChild.innerHTML;
return "All" == e || "All" == t || e == t;
}),
customerList.update();
}
function updateList() {
var a = document.querySelector("input[name=status]:checked").value;
(data = userList.filter(function (e) {
var t = !1;
return (
"All" == a
? (t = !0)
: ((t = e.values().sts == a), console.log(t, "statusFilter")),
t
);
})),
userList.update();
}
refreshCallbacks(),
filterContact("All"),
document
.getElementById("showModal")
.addEventListener("show.bs.modal", function (e) {
e.relatedTarget.classList.contains("edit-item-btn")
? ((document.getElementById("exampleModalLabel").innerHTML =
"Edit Data Indikator"),
(document
.getElementById("showModal")
.querySelector(".modal-footer").style.display = "block"),
// (document.getElementById("add-btn").style.display = "none"),
(document.getElementById("edit-btn").style.display = "block"))
: e.relatedTarget.classList.contains("add-btn")
? ((document.getElementById("exampleModalLabel").innerHTML =
"Add Customer"),
(document
.getElementById("showModal")
.querySelector(".modal-footer").style.display = "block"),
(document.getElementById("edit-btn").style.display = "none"),
(document.getElementById("add-btn").style.display = "block"))
: ((document.getElementById("exampleModalLabel").innerHTML =
"List Customer"),
(document
.getElementById("showModal")
.querySelector(".modal-footer").style.display = "none"));
}),
ischeckboxcheck(),
document
.getElementById("showModal")
.addEventListener("hidden.bs.modal", function () {
clearFields();
}),
document
.querySelector("#customerList")
.addEventListener("click", function () {
refreshCallbacks(), ischeckboxcheck();
});
var table = document.getElementById("customerTable"),
tr = table.getElementsByTagName("tr"),
trlist = table.querySelectorAll(".list tr"),
count = Number(isValue.replace(/[^0-9]/g, "")) + 1;
// addBtn.addEventListener("click", function (e) {
// "" !== customerNameField.value &&
// "" !== emailField.value &&
// "" !== dateField.value &&
// "" !== phoneField.value &&
// (customerList.add({
// id:
// '<a href="javascript:void(0);" class="fw-medium link-primary">#VZ' +
// count +
// "</a>",
// customer_name: customerNameField.value,
// email: emailField.value,
// date: dateField.value,
// phone: phoneField.value,
// status: isStatus(statusField.value),
// }),
// document.getElementById("close-modal").click(),
// clearFields(),
// refreshCallbacks(),
// filterContact("All"),
// count++);
// }),
editBtn.addEventListener("click", function (e) {
(document.getElementById("exampleModalLabel").innerHTML = "Edit Customer"),
customerList.get({ id: idField.value }).forEach(function (e) {
(isid = new DOMParser().parseFromString(e._values.id, "text/html")),
isid.body.firstElementChild.innerHTML == itemId &&
e.values({
id:
'<a href="javascript:void(0);" class="fw-medium link-primary">' +
idField.value +
"</a>",
customer_name: customerNameField.value,
email: emailField.value,
date: dateField.value,
phone: phoneField.value,
status: isStatus(statusField.value),
});
}),
document.getElementById("close-modal").click(),
clearFields();
});
// var statusVal = new Choices(statusField);
function isStatus(e) {
switch (e) {
case "Active":
return (
'<span class="badge badge-soft-success text-uppercase">' +
e +
"</span>"
);
case "Block":
return (
'<span class="badge badge-soft-danger text-uppercase">' +
e +
"</span>"
);
}
}
function ischeckboxcheck() {
document.getElementsByName("checkAll").forEach(function (e) {
e.addEventListener("click", function (e) {
e.target.checked
? e.target.closest("tr").classList.add("table-active")
: e.target.closest("tr").classList.remove("table-active");
});
});
}
function refreshCallbacks() {
removeBtns.forEach(function (e) {
e.addEventListener("click", function (e) {
e.target.closest("tr").children[1].innerText,
(itemId = e.target.closest("tr").children[1].innerText),
customerList.get({ id: itemId }).forEach(function (e) {
deleteid = new DOMParser().parseFromString(
e._values.id,
"text/html"
);
var t = deleteid.body.firstElementChild;
deleteid.body.firstElementChild.innerHTML == itemId &&
document
.getElementById("delete-record")
.addEventListener("click", function () {
customerList.remove("id", t.outerHTML),
document
.getElementById("deleteRecordModal")
.click();
});
});
});
}),
editBtns.forEach(function (e) {
e.addEventListener("click", function (e) {
e.target.closest("tr").children[1].innerText,
(itemId = e.target.closest("tr").children[1].innerText),
customerList.get({ id: itemId }).forEach(function (e) {
isid = new DOMParser().parseFromString(
e._values.id,
"text/html"
);
var t = isid.body.firstElementChild.innerHTML;
t == itemId &&
((idField.value = t),
(customerNameField.value = e._values.customer_name),
(emailField.value = e._values.email),
(dateField.value = e._values.date),
(phoneField.value = e._values.phone),
statusVal && statusVal.destroy(),
(statusVal = new Choices(statusField)),
(val = new DOMParser().parseFromString(
e._values.status,
"text/html"
)),
(t = val.body.firstElementChild.innerHTML),
statusVal.setChoiceByValue(t),
flatpickr("#date-field", {
dateFormat: "d M, Y",
defaultDate: e._values.date,
}));
});
});
});
}
document
.querySelector(".pagination-next")
.addEventListener("click", function () {
!document.querySelector(".pagination.listjs-pagination") ||
(document
.querySelector(".pagination.listjs-pagination")
.querySelector(".active") &&
document
.querySelector(".pagination.listjs-pagination")
.querySelector(".active")
.nextElementSibling.children[0].click());
}),
document
.querySelector(".pagination-prev")
.addEventListener("click", function () {
!document.querySelector(".pagination.listjs-pagination") ||
(document
.querySelector(".pagination.listjs-pagination")
.querySelector(".active") &&
document
.querySelector(".pagination.listjs-pagination")
.querySelector(".active")
.previousSibling.children[0].click());
});

View File

@ -0,0 +1,272 @@
@extends('layouts.app')
@push('title', 'Data Parameter')
@section('content')
<div class="page-content">
<div class="container-fluid">
<!-- start page title -->
<div class="row">
<div class="col-12">
<div class="page-title-box d-sm-flex align-items-center justify-content-between">
<h4 class="mb-sm-0">Data Indikator</h4>
<div class="page-title-right">
<ol class="breadcrumb m-0">
<li class="breadcrumb-item">
<a href="javascript: void(0);">Master Data</a>
</li>
<li class="breadcrumb-item active">Data Indikator</li>
</ol>
</div>
</div>
</div>
</div>
<!-- end page title -->
<div class="row">
<div class="col-lg-12">
<div class="card">
<div class="card-header">
<h4 class="card-title mb-0">Data Indikator</h4>
</div>
<!-- end card header -->
<div class="card-body">
<div id="customerList">
<div class="row g-4 mb-3">
{{-- <div class="col-sm-auto">
<div>
<button type="button" class="btn btn-primary add-btn" data-bs-toggle="modal"
id="create-btn" data-bs-target="#showModal">
<i class="ri-add-line align-bottom me-1"></i> Tambah
</button>
</div>
</div> --}}
<div class="col-sm">
<div class="d-flex justify-content-sm-end">
<div class="search-box ms-2">
<input type="text" class="form-control search" placeholder="Search..." />
<i class="ri-search-line search-icon"></i>
</div>
</div>
</div>
</div>
<div class="table-responsive table-card mt-3 mb-1">
<table class="table align-middle table-nowrap" id="customerTable">
<thead class="table-light">
<tr class="text-center">
<th class="sort" data-sort="no">
No
</th>
<th class="sort" data-sort="indicator_name">
Name
</th>
<th class="sort" data-sort="description">Description</th>
<th class="sort" data-sort="ideal_min">Ideal Min</th>
<th class="sort" data-sort="ideal_max">
Ideal Max
</th>
<th class="sort" data-sort="status">
Unit
</th>
<th class="sort" data-sort="action">Action</th>
</tr>
</thead>
<tbody class="list form-check-all">
@foreach ($indicators as $indicator)
<tr class="text-center">
<td class="no">{{ $loop->iteration }}</td>
<td class="id" style="display: none">
<a href="javascript:void(0);"
class="fw-medium link-primary">#VZ2101</a>
</td>
<td class="indicator_name">{{ $indicator->name }}</td>
<td class="description">{{ $indicator->description }}</td>
<td class="ideal_min">{{ $indicator->ideal_min }}</td>
<td class="ideal_max">{{ $indicator->ideal_max }}</td>
<td class="status">
<span>{{ $indicator->unit }}</span>
</td>
<td>
<div class="d-flex gap-2 justify-content-center">
<div class="edit">
<button class="btn btn-sm btn-warning edit-item-btn"
data-bs-toggle="modal" data-bs-target="#showModal"
onclick="updateData({{ $indicator }})">
Edit
</button>
</div>
{{-- <div class="remove">
<button class="btn btn-sm btn-success remove-item-btn"
data-bs-toggle="modal" data-bs-target="#deleteRecordModal">
Remove
</button>
</div> --}}
</div>
</td>
</tr>
@endforeach
</tbody>
</table>
<div class="noresult" style="display: none">
<div class="text-center">
<lord-icon src="https://cdn.lordicon.com/msoeawqm.json" trigger="loop"
colors="primary:#25a0e2,secondary:#00bd9d"
style="width: 75px; height: 75px">
</lord-icon>
<h5 class="mt-2">Maaf! Data Tidak Ditemukan</h5>
<p class="text-muted mb-0">Silahkan gunakan kata kunci lain</p>
</div>
</div>
</div>
<div class="d-flex justify-content-end">
<div class="pagination-wrap hstack gap-2">
<a class="page-item pagination-prev disabled" href="#">
Sebelumnya
</a>
<ul class="pagination listjs-pagination mb-0"></ul>
<a class="page-item pagination-next" href="#">
Selanjutnya
</a>
</div>
</div>
</div>
</div>
<!-- end card -->
</div>
<!-- end col -->
</div>
<!-- end col -->
</div>
<!-- end row -->
{{-- edit modal --}}
<div class="modal fade" id="showModal" tabindex="-1" aria-labelledby="exampleModalLabel" aria-hidden="true">
<div class="modal-dialog modal-dialog-centered">
<div class="modal-content">
<div class="modal-header bg-light p-3">
<h5 class="modal-title" id="exampleModalLabel"></h5>
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"
id="close-modal"></button>
</div>
<form action="" class="needs-validation" method="POST" novalidate id="edit-form">
@csrf
@method('PUT')
<div class="modal-body">
<div class="mb-3" id="modal-id" style="display: none">
<label for="id-field" class="form-label">ID</label>
<input type="text" id="id-field" class="form-control" placeholder="ID"
readonly />
</div>
<div class="mb-3">
<label for="indicatorname-field" class="form-label">Indikator</label>
<input type="text" id="indicatorname-field" class="form-control" name="name"
value="{{ old('name') }}" placeholder="Masukan Nama Indikator" required />
<div class="invalid-feedback">
Masukan Nama Indikator
</div>
</div>
<div class="mb-3">
<label for="description-field" class="form-label">Description</label>
<textarea name="description" id="description-field" id="description-field" rows="5" class="form-control"
placeholder="Masukan Deskripsi" required>{{ old('description') }}</textarea>
<div class="invalid-feedback">
Masukan Deskripsi
</div>
</div>
<div class="mb-3">
<label for="idealmin-field" class="form-label">Ideal Min</label>
<input type="text" id="idealmin-field" class="form-control" name="ideal_min"
value="{{ old('ideal_min') }}" placeholder="Masukan Ideal Min." required />
<div class="invalid-feedback">
Masukan angka ideal min
</div>
</div>
<div class="mb-3">
<label for="idealmax-field" class="form-label">Ideal Max</label>
<input type="text" id="idealmax-field" class="form-control" name="ideal_max"
value="{{ old('ideal_max') }}" placeholder="Masukan Ideal Max." required />
<div class="invalid-feedback">
Masukan angka ideal max
</div>
</div>
<div>
<label for="status-field" class="form-label">Unit</label>
<input type="text" id="status-field" class="form-control" name="unit"
value="{{ old('unit') }}" placeholder="Masukan Unit" required>
<div class="invalid-feedback">
Masukan Unit
</div>
</div>
</div>
<div class="modal-footer">
<div class="hstack gap-2 justify-content-end">
<button type="button" class="btn btn-light" data-bs-dismiss="modal">
Tutup
</button>
<button type="submit" class="btn btn-success" id="edit-btn">
Ubah
</button>
</div>
</div>
</form>
</div>
</div>
</div>
<!-- Modal -->
<div class="modal fade zoomIn" id="deleteRecordModal" tabindex="-1" aria-hidden="true">
<div class="modal-dialog modal-dialog-centered">
<div class="modal-content">
<div class="modal-header">
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"
id="btn-close"></button>
</div>
<div class="modal-body">
<div class="mt-2 text-center">
<lord-icon src="https://cdn.lordicon.com/gsqxdxog.json" trigger="loop"
colors="primary:#25a0e2,secondary:#00bd9d"
style="width: 100px; height: 100px"></lord-icon>
<div class="mt-4 pt-2 fs-15 mx-4 mx-sm-5">
<h4>Are you sure ?</h4>
<p class="text-muted mx-4 mb-0">
Are you sure you want to remove this record ?
</p>
</div>
</div>
<div class="d-flex gap-2 justify-content-center mt-4 mb-2">
<button type="button" class="btn w-sm btn-light" data-bs-dismiss="modal">
Close
</button>
<button type="button" class="btn w-sm btn-primary" id="delete-record">
Yes, Delete It!
</button>
</div>
</div>
</div>
</div>
</div>
<!--end modal -->
</div>
<!-- container-fluid -->
</div>
<!-- End Page-content -->
@push('other-js')
<!-- prismjs plugin -->
<script src="assets/libs/prismjs/prism.js"></script>
<script src="assets/libs/list.js/list.min.js"></script>
<script src="assets/libs/list.pagination.js/list.pagination.min.js"></script>
<!-- listjs init -->
<script src="assets/js/pages/customJs/master-data/indikator/index.js"></script>
<script src="assets/js/pages/form-validation.init.js"></script>
@endpush
@endsection

View File

@ -63,6 +63,12 @@ class="nav-link {{ request()->routeIs('master_data.lahan.*') ? 'active' : '' }}"
data-key="t-calendar"> Lahan data-key="t-calendar"> Lahan
</a> </a>
</li> </li>
<li class="nav-item">
<a href="{{ route('master_data.indikator.index') }}"
class="nav-link {{ request()->routeIs('master_data.indikator.*') ? 'active' : '' }}"
data-key="t-calendar"> Indikator
</a>
</li>
</ul> </ul>
</div> </div>
</li> </li>

View File

@ -5,6 +5,7 @@
use App\Http\Controllers\Auth\RegisteredUserController; use App\Http\Controllers\Auth\RegisteredUserController;
use App\Http\Controllers\Auth\TwoStepVerifyController; use App\Http\Controllers\Auth\TwoStepVerifyController;
use App\Http\Controllers\DashboardController; use App\Http\Controllers\DashboardController;
use App\Http\Controllers\MasterData\IndicatorController;
use App\Http\Controllers\MasterData\LandController; use App\Http\Controllers\MasterData\LandController;
use App\Http\Controllers\MasterData\UserController; use App\Http\Controllers\MasterData\UserController;
use Illuminate\Routing\RouteGroup; use Illuminate\Routing\RouteGroup;
@ -74,5 +75,12 @@
Route::put('/{id}', 'update')->name('update'); Route::put('/{id}', 'update')->name('update');
Route::delete('/{id}', 'destroy')->name('destroy'); Route::delete('/{id}', 'destroy')->name('destroy');
}); });
Route::prefix('data-indikator')->controller(IndicatorController::class)->name('indikator.')->group(function () {
Route::get('/', 'index')->name('index');
// Route::post('/', 'store')->name('store');
Route::put('/{id}', 'update')->name('update');
// Route::delete('/{id}', 'destroy')->name('destroy');
});
}); });
}); });