Update Manga

This commit is contained in:
ahmadafriadi 2025-10-13 14:35:26 +07:00
parent 54981bebc8
commit 4f266bcdd8
3 changed files with 757 additions and 0 deletions

View File

@ -0,0 +1,309 @@
<template>
<div>
<ul class="flex space-x-2 rtl:space-x-reverse">
<li>
<a href="javascript:;" class="text-primary hover:underline">Entertainment</a>
</li>
<li class="before:mr-2 before:content-['/'] rtl:before:ml-2">
<NuxtLink to="/entertainment/manga/list" class="text-primary hover:underline">Manga</NuxtLink>
</li>
<li class="before:mr-2 before:content-['/'] rtl:before:ml-2">
<span>Edit Manga</span>
</li>
</ul>
<div class="grid grid-cols-1 gap-6 pt-5">
<div class="panel">
<div class="mb-5 flex items-center justify-between">
<h5 class="text-lg font-semibold dark:text-white-light">Edit Karakter</h5>
<NuxtLink to="/entertainment/manga/list" class="dark:text-white-light btn btn-dark !py-1">
<icon-arrow-backward class="me-1" />
Daftar
</NuxtLink>
</div>
<div class="mb-5">
<form v-if="form" class="space-y-5" @submit.prevent="submitForm()">
<div class="grid grid-cols-1 gap-4 md:grid-cols-2 gap-2">
<div :class="{ 'has-error': $validate.form.title.$error, 'has-success': isSubmitted && !$validate.form.title.$error }">
<label for="title">Judul Manga</label>
<input id="title" type="text" class="form-input" v-model="form.title" />
<template v-if="isSubmitted && $validate.form.title.$error">
<p class="text-danger mt-1">Judul Manga harus diisi</p>
</template>
</div>
<div :class="{ 'has-error': $validate.form.slug.$error, 'has-success': isSubmitted && !$validate.form.slug.$error }">
<label for="slug">Slug Manga</label>
<input id="slug" type="text" class="form-input" v-model="form.slug" />
<template v-if="isSubmitted && $validate.form.slug.$error">
<p class="text-danger mt-1">Slug Manga harus diisi</p>
</template>
</div>
</div>
<div class="grid grid-cols-1 gap-4 md:grid-cols-4 gap-2">
<div :class="{ 'has-error': $validate.form.genre.$error, 'has-success': isSubmitted && !$validate.form.genre.$error }">
<label for="genre">Genre</label>
<multiselect id="genre"
v-model="form.genre"
:options="genreOptions"
class="custom-multiselect"
:searchable="true"
placeholder="Pilih genre manga"
selected-label=""
select-label=""
deselect-label=""
label="label"
track-by="value"
></multiselect>
<template v-if="isSubmitted && $validate.form.genre.$error">
<p class="text-danger mt-1">Genre Manga harus diisi</p>
</template>
</div>
<div :class="{ 'has-error': $validate.form.coin.$error, 'has-success': isSubmitted && !$validate.form.coin.$error }">
<label for="coin">Coin</label>
<input id="coin" type="number" class="form-input" v-model="form.coin" />
<template v-if="isSubmitted && $validate.form.coin.$error">
<p class="text-danger mt-1">Coin Manga harus diisi</p>
</template>
</div>
<div :class="{ 'has-error': $validate.form.color.$error, 'has-success': isSubmitted && !$validate.form.color.$error }">
<label for="color">Color</label>
<input id="color" type="text" class="form-input" v-model="form.color" />
<template v-if="isSubmitted && $validate.form.color.$error">
<p class="text-danger mt-1">Color Manga harus diisi</p>
</template>
</div>
<div :class="{ 'has-error': $validate.form.status.$error, 'has-success': isSubmitted && !$validate.form.status.$error }">
<label for="status">Status</label>
<multiselect id="status"
v-model="form.status"
:options="statusOptions"
class="custom-multiselect"
:searchable="true"
placeholder="Pilih status manga"
selected-label=""
select-label=""
deselect-label=""
label="label"
track-by="value"
></multiselect>
<template v-if="isSubmitted && $validate.form.status.$error">
<p class="text-danger mt-1">Status Manga harus diisi</p>
</template>
</div>
</div>
<div class="grid grid-cols-1 gap-4 md:grid-cols-1 gap-2">
<div :class="{ 'has-error': $validate.form.synopsis.$error, 'has-success': isSubmitted && !$validate.form.synopsis.$error }">
<label for="synopsis">Synopsis</label>
<textarea id="synopsis" rows="3" class="form-textarea" v-model="form.synopsis"></textarea>
<template v-if="isSubmitted && $validate.form.synopsis.$error">
<p class="text-danger mt-1">Deskripsi karakter harus diisi</p>
</template>
</div>
</div>
<div class="grid grid-cols-1 gap-4 md:grid-cols-2 gap-2">
<div :class="{ 'has-error': $validate.form.posted_at.$error, 'has-success': isSubmitted && !$validate.form.posted_at.$error }">
<label for="posted_at">Tanggal Unggah</label>
<input id="posted_at" type="datetime-local" class="form-input" v-model="form.posted_at" />
<template v-if="isSubmitted && $validate.form.posted_at.$error">
<p class="text-danger mt-1">Tanggal unggah Manga harus diisi</p>
</template>
</div>
<div :class="{ 'has-error': $validate.form.archived_at.$error, 'has-success': isSubmitted && !$validate.form.archived_at.$error }">
<label for="archived_at">Tanggal Arsip</label>
<input id="archived_at" type="datetime-local" class="form-input" v-model="form.archived_at" />
<template v-if="isSubmitted && $validate.form.archived_at.$error">
<p class="text-danger mt-1">Tanggal arsip Manga harus diisi</p>
</template>
</div>
</div>
<div class="grid grid-cols-2 gap-4">
<div class="custom-file-container" data-upload-id="featuredImage"
:class="{ 'has-error': $validate.form.featured_image.$error, 'has-success': isSubmitted && !$validate.form.featured_image.$error }">
<div class="label-container">
<label for="featured_image">Gambar </label> <a href="javascript:;" class="custom-file-container__image-clear" title="Clear Image">×</a>
</div>
<label class="custom-file-container__custom-file" >
<input id="featured_image" type="file" class="custom-file-container__custom-file__custom-file-input"
accept="image/png" @change="handleImageChange" />
<input type="hidden" name="MAX_FILE_SIZE" value="10485760" />
<span class="custom-file-container__custom-file__custom-file-control ltr:pr-20 rtl:pl-20"></span>
</label>
<template v-if="isSubmitted && $validate.form.featured_image.$error">
<p class="text-danger mt-1">Gambar karakter harus diupload dengan file berjenis PNG atau JPEG dengan ukuran maksimal 1 MB</p>
</template>
<div class="custom-file-container__image-preview"></div>
</div>
</div>
<div class="flex items-center ltr:ml-auto mt-8">
<button type="submit" class="btn btn-success !py-1" :disabled="isLoading"><icon-save class="me-1" /> Simpan</button>
<button type="reset" class="btn btn-dark !py-1 ml-1" @click="resetForm"><icon-restore class="me-1" />Reset</button>
</div>
</form>
</div>
</div>
</div>
</div>
</template>
<script lang="ts" setup>
import { useVuelidate } from '@vuelidate/core';
import { required, helpers } from '@vuelidate/validators';
import Multiselect from '@suadelabs/vue3-multiselect';
import '@suadelabs/vue3-multiselect/dist/vue3-multiselect.css';
import '@/assets/css/file-upload-preview.css';
useHead({ title: 'Edit Karakter' });
const isSubmitted = ref(false);
const imageType = helpers.withMessage('Format gambar harus PNG atau JPG', (value) => {
if (!value) return true;
if (typeof value === 'string') return true;
return ['image/png', 'image/jpeg', 'image/jpg'].includes(value.type);
});
const maxFileSize = helpers.withMessage('Ukuran gambar tidak boleh lebih dari 1MB', (value) => {
if (!value) return true;
if (typeof value === 'string') return true;
return value.size <= 1048576; // 1MB dalam byte
});
const rules = {
form: {
title: { required },
slug: { required },
genre: { required },
coin: { required },
color: { required },
status: { required },
synopsis: { required },
posted_at: { required },
archived_at: { required },
featured_image: { imageType, maxFileSize },
}
};
const router = useRouter();
const route = useRoute();
const config = useRuntimeConfig();
const { $api } = useNuxtApp();
const genreOptions = [{ "value": "komedi", "label": "Komedi" }, { "value": "petualangan", "label": "Petualangan" }, { "value": "fiksi ilmiah", "label": "Fiksi Ilmiah" }, { "value": "fantasi", "label": "Fantasi" }, { "value": "kehidupan sehari-hari", "label": "Kehidupan Sehari-hari" } ]
const statusOptions = [{ "value": "draft", "label": "Draft" }, { "value": "published", "label": "Dipublikasikan" }, { "value": "archived", "label": "Diarsipkan" } ]
const isLoading = ref(false);
const featuredImageUploader = ref(null);
const { data: form, refresh } = await useAsyncData('manga-get',
() => $api(`/entertainment/manga/${route.params.id}/`, {}),
);
const $validate = useVuelidate(rules, { form });
onMounted(async () => {
if (!form.value) return;
const fileupload = await import('file-upload-with-preview');
let FileUploadWithPreview = fileupload.default;
featuredImageUploader.value =new FileUploadWithPreview('featuredImage', {
images: {
baseImage: form.value.featured_image || '/assets/images/file-preview.svg',
backgroundImage: '',
},
});
if (form.value.genre) {
const foundGenre = genreOptions.find(opt => opt.value === form.value.genre);
if (foundGenre) form.value.genre = foundGenre;
}
if (form.value.status) {
const foundStatus = statusOptions.find(opt => opt.value === form.value.status);
if (foundStatus) form.value.status = foundStatus;
}
if (form.value.posted_at) {
form.value.posted_at = form.value.posted_at.replace('Z', '').slice(0, 16);
}
if (form.value.archived_at) {
form.value.archived_at = form.value.archived_at.replace('Z', '').slice(0, 16);
}
});
const submitForm = async () => {
isSubmitted.value = true;
$validate.value.form.$touch();
if ($validate.value.form.$invalid) {
return false;
}
const formData = new FormData();
formData.append('title', form.value.title);
formData.append('slug', form.value.slug);
formData.append('genre', form.value.genre.value);
formData.append('coin', form.value.coin);
formData.append('color', form.value.color);
formData.append('status', form.value.status.value);
formData.append('synopsis', form.value.synopsis);
formData.append('posted_at', new Date(form.value.posted_at).toISOString());
formData.append('archived_at', new Date(form.value.archived_at).toISOString());
if (form.value.featured_image && typeof form.value.featured_image !== 'string') {
formData.append('featured_image', form.value.featured_image);
}
isLoading.value = true;
await $api(`/entertainment/manga/${route.params.id}/`, {
method: 'PUT',
body: formData
}).then(() => {
router.push({ path: "/entertainment/manga/list" });
})
.catch((error) => {
console.log(error);
})
.finally(() => {
isLoading.value = false;
});
}
function handleImageChange(event: { target: { files: any[]; }; }) {
const file = event.target.files[0]
if (!file) return
const allowed = ['image/png', 'image/jpeg', 'image/jpg']
if (!allowed.includes(file.type)) {
alert('Hanya PNG atau JPG yang diizinkan')
return
}
form.value.featured_image = file
}
function handleIconChange(event: { target: { files: any[]; }; }) {
const file = event.target.files[0]
if (!file) return
const allowed = ['image/png', 'image/jpeg', 'image/jpg']
if (!allowed.includes(file.type)) {
alert('Hanya PNG atau JPG yang diizinkan')
return
}
form.value.featured_icon = file
}
const resetForm = () => {
refresh()
isSubmitted.value = false;
$validate.value.form.$reset();
featuredImageUploader?.value?.clearPreviewPanel();
featuredIconUploader?.value?.clearPreviewPanel();
};
</script>

View File

@ -0,0 +1,292 @@
<template>
<div>
<ul class="flex space-x-2 rtl:space-x-reverse">
<li>
<a href="javascript:;" class="text-primary hover:underline">Intertainment</a>
</li>
<li class="before:mr-2 before:content-['/'] rtl:before:ml-2">
<NuxtLink to="/entertainment/manga/list" class="text-primary hover:underline">Daftar Manga</NuxtLink>
</li>
<li class="before:mr-2 before:content-['/'] rtl:before:ml-2">
<span>Tambah Manga</span>
</li>
</ul>
<div class="grid grid-cols-1 gap-6 pt-5">
<div class="panel">
<div class="mb-5 flex items-center justify-between">
<h5 class="text-lg font-semibold dark:text-white-light">Tambah Manga</h5>
<NuxtLink to="/entertainment/manga/list" class="dark:text-white-light btn btn-dark !py-1">
<icon-arrow-backward class="me-1" />
Daftar
</NuxtLink>
</div>
<div class="mb-5">
<form class="space-y-5" @submit.prevent="submitForm()">
<div class="grid grid-cols-1 gap-4 md:grid-cols-2 gap-2">
<div :class="{ 'has-error': $validate.form.title.$error, 'has-success': isSubmitted && !$validate.form.title.$error }">
<label for="title">Title</label>
<input id="title" type="text" class="form-input" v-model="form.title" />
<template v-if="isSubmitted && $validate.form.title.$error">
<p class="text-danger mt-1">Title harus diisi</p>
</template>
</div>
<div :class="{ 'has-error': $validate.form.slug.$error, 'has-success': isSubmitted && !$validate.form.slug.$error }">
<label for="slug">Slug</label>
<input id="slug" type="text" class="form-input" v-model="form.slug" />
<template v-if="isSubmitted && $validate.form.slug.$error">
<p class="text-danger mt-1">slug harus diisi</p>
</template>
</div>
</div>
<div class="grid grid-cols-1 gap-4 md:grid-cols-4 gap-2">
<div :class="{ 'has-error': $validate.form.genre.$error, 'has-success': isSubmitted && !$validate.form.genre.$error }">
<label for="genre">Genre</label>
<multiselect id="format"
v-model="form.genre"
:options="genreOptions"
class="custom-multiselect"
:searchable="true"
placeholder="Pilih genre manga"
selected-label=""
select-label=""
deselect-label=""
label="label"
track-by="value"
></multiselect>
<template v-if="isSubmitted && $validate.form.genre.$error">
<p class="text-danger mt-1">Jenis karakter harus diisi</p>
</template>
</div>
<div :class="{ 'has-error': $validate.form.coin.$error, 'has-success': isSubmitted && !$validate.coin.slug.$error }">
<label for="coin">Coin</label>
<input id="coin" type="number" class="form-input" v-model="form.coin" />
<template v-if="isSubmitted && $validate.form.coin.$error">
<p class="text-danger mt-1">Coin harus diisi</p>
</template>
</div>
<div :class="{ 'has-error': $validate.form.color.$error, 'has-success': isSubmitted && !$validate.form.color.$error }">
<label for="color">Color</label>
<input id="color" type="text" class="form-input" v-model="form.color" />
<template v-if="isSubmitted && $validate.form.color.$error">
<p class="text-danger mt-1">color harus diisi</p>
</template>
</div>
<div :class="{ 'has-error': $validate.form.status.$error, 'has-success': isSubmitted && !$validate.form.status.$error }">
<label for="status">Status</label>
<multiselect id="format"
v-model="form.status"
:options="statusOptions"
class="custom-multiselect"
:searchable="true"
placeholder="Pilih status manga"
selected-label=""
select-label=""
deselect-label=""
label="label"
track-by="value"
></multiselect>
<template v-if="isSubmitted && $validate.form.status.$error">
<p class="text-danger mt-1">Jenis karakter harus diisi</p>
</template>
</div>
</div>
<div class="grid grid-cols-1 gap-4 md:grid-cols-1 gap-2">
<div :class="{ 'has-error': $validate.form.synopsis.$error, 'has-success': isSubmitted && !$validate.form.synopsis.$error }">
<label for="synopsis">Sinopsis</label>
<textarea id="synopsis" rows="3" class="form-textarea" v-model="form.synopsis"></textarea>
<template v-if="isSubmitted && $validate.form.synopsis.$error">
<p class="text-danger mt-1">Sinopsis manga harus diisi</p>
</template>
</div>
</div>
<div class="grid grid-cols-1 gap-4 md:grid-cols-4 gap-2">
<div :class="{ 'has-error': $validate.form.posted_at.$error, 'has-success': isSubmitted && !$validate.form.posted_at.$error }">
<label for="posted_at">Tanggal Unggah</label>
<input id="posted_at" type="date" class="form-input" v-model="form.posted_at" />
<template v-if="isSubmitted && $validate.form.posted_at.$error">
<p class="text-danger mt-1">Tanggal Harus Diisi</p>
</template>
</div>
<div :class="{ 'has-error': $validate.form.archived_at.$error, 'has-success': isSubmitted && !$validate.form.archived_at.$error }">
<label for="archived_at">Tanggal Arsip</label>
<input id="archived_at" type="date" class="form-input" v-model="form.archived_at" />
<template v-if="isSubmitted && $validate.form.archived_at.$error">
<p class="text-danger mt-1">Tanggal Harus Diisi</p>
</template>
</div>
</div>
<div class="grid grid-cols-2 gap-4">
<div class="custom-file-container" data-upload-id="featuredImage"
:class="{ 'has-error': $validate.form.featured_image.$error, 'has-success': isSubmitted && !$validate.form.featured_image.$error }">
<div class="label-container">
<label for="featured_image">Gambar </label> <a href="javascript:;" class="custom-file-container__image-clear" title="Clear Image">×</a>
</div>
<label class="custom-file-container__custom-file" >
<input id="featured_image" type="file" class="custom-file-container__custom-file__custom-file-input"
accept="image/png" @change="handleImageChange" />
<input type="hidden" name="MAX_FILE_SIZE" value="10485760" />
<span class="custom-file-container__custom-file__custom-file-control ltr:pr-20 rtl:pl-20"></span>
</label>
<template v-if="isSubmitted && $validate.form.featured_image.$error">
<p class="text-danger mt-1">Gambar karakter harus diupload dengan file berjenis PNG atau JPEG dengan ukuran maksimal 1 MB</p>
</template>
<div class="custom-file-container__image-preview"></div>
</div>
</div>
<div class="flex items-center ltr:ml-auto mt-8">
<button type="submit" class="btn btn-success !py-1" :disabled="isLoading"><icon-save class="me-1" /> {{ isLoading ? 'Menyimpan...' : 'Simpan' }}</button>
<button type="button" class="btn btn-dark !py-1 ml-1" @click="resetForm"><icon-restore class="me-1" />Reset</button>
</div>
</form>
</div>
</div>
</div>
</div>
</template>
<script lang="ts" setup>
import { reactive } from 'vue'
import { useVuelidate } from '@vuelidate/core';
import { required, helpers } from '@vuelidate/validators';
import Multiselect from '@suadelabs/vue3-multiselect';
import '@suadelabs/vue3-multiselect/dist/vue3-multiselect.css';
import '@/assets/css/file-upload-preview.css';
useHead({ title: 'Tambah Manga' });
const form = ref({
title: '',
slug: '',
genre: '',
coin: '',
synopsis: '',
color: '',
status: '',
posted_at: '',
archived_at: '',
featured_image: '',
});
const isSubmitted = ref(false);
const imageType = helpers.withMessage('Format gambar harus PNG atau JPG', (value) => {
if (!value) return false;
return ['image/png', 'image/jpeg', 'image/jpg'].includes(value.type);
});
const maxFileSize = helpers.withMessage('Ukuran gambar tidak boleh lebih dari 1MB', (value) => {
if (!value) return false;
return value.size <= 1048576; // 1MB dalam byte
});
const rules = {
form: {
title: { required },
slug: { required },
genre: { required },
coin: { required },
synopsis: { required },
color: { required },
status: { required },
posted_at: { required },
archived_at: { required },
featured_image: { required, imageType, maxFileSize },
}
};
const $validate = useVuelidate(rules, { form });
const router = useRouter();
const config = useRuntimeConfig();
const { $api } = useNuxtApp();
const genreOptions = [{ "value": "komedi", "label": "Komedi" }, { "value": "petualangan", "label": "Petualangan" }, { "value": "fiksi ilmiah", "label": "Fiksi Ilmiah" }, { "value": "fantasi", "label": "Fantasi" }, { "value": "kehidupan sehari-hari", "label": "Kehidupan Sehari-hari" } ]
const statusOptions = [{ "value": "draft", "label": "Draft" }, { "value": "published", "label": "Dipublikasikan" }, { "value": "archived", "label": "Diarsipkan" } ]
const isLoading = ref(false);
const featuredImageUploader = ref(null);
onMounted(async () => {
const fileupload = await import('file-upload-with-preview');
let FileUploadWithPreview = fileupload.default;
featuredImageUploader.value = new FileUploadWithPreview('featuredImage', {
images: {
baseImage: '/assets/images/file-preview.svg',
backgroundImage: '',
},
});
});
const submitForm = async () => {
isSubmitted.value = true;
$validate.value.form.$touch();
if ($validate.value.form.$invalid) {
return false;
}
const formData = new FormData();
formData.append('title', form.value.title);
formData.append('slug', form.value.slug);
formData.append('genre', form.value.genre);
formData.append('coin', form.value.coin);
formData.append('synopsis', form.value.synopsis);
formData.append('color', form.value.color);
formData.append('status', form.value.status);
formData.append('posted_at', form.value.posted_at);
formData.append('archived_at', form.value.archived_at);
if (form.value.featured_image) {
formData.append('featured_image', form.value.featured_image);
}
isLoading.value = true;
await $api(`/entertainment/manga/`, {
method: 'POST',
body: formData
}).then(() => {
router.push({ path: "/entertainment/manga/list" });
})
.catch((error) => {
console.log(error);
})
.finally(() => {
isLoading.value = false;
});
}
function handleImageChange(event: { target: { files: any[]; }; }) {
const file = event.target.files[0]
if (!file) return
const allowed = ['image/png', 'image/jpeg', 'image/jpg']
if (!allowed.includes(file.type)) {
alert('Hanya PNG atau JPG yang diizinkan')
return
}
form.value.featured_image = file
}
const resetForm = () => {
form.value = {
title: '',
slug: '',
genre: null,
coin: null,
synopsis: '',
color: '',
status: null,
posted_at: '',
archived_at: '',
featured_image: '',
};
isSubmitted.value = false;
$validate.value.form.$reset();
featuredImageUploader?.value?.clearPreviewPanel();
};
</script>

View File

@ -0,0 +1,156 @@
<template>
<div>
<ul class="flex space-x-2 rtl:space-x-reverse">
<li>
<a href="javascript:;" class="text-primary hover:underline">Intertainment</a>
</li>
<li class="before:mr-2 before:content-['/'] rtl:before:ml-2">
<span>Daftar Manga</span>
</li>
</ul>
<div class="grid grid-cols-1 gap-6 pt-5">
<div class="panel">
<div class="mb-5 flex items-center justify-between">
<h5 class="text-lg font-semibold dark:text-white-light">Daftar Manga</h5>
<NuxtLink to="/entertainment/manga/add" class="dark:text-white-light btn btn-primary !py-1">
<icon-plus class="me-1" />
Tambah
</NuxtLink>
</div>
<div class="mb-5">
<input v-model.lazy="params.search" type="text" class="form-input max-w-xs" placeholder="Cari... (tombol Enter untuk mencari)" @change="changeSearch"/>
</div>
<div class="mb-5">
<div class="datatable">
<vue3-datatable
:rows="rows"
:columns="cols"
:totalRows="totalRows"
:isServerMode=true
:page="params.current_page"
:pageSize="params.pagesize"
:sortable="true"
:sortColumn="params.sort_column"
:sortDirection="params.sort_direction"
@change="changeServer"
skin="whitespace-nowrap bh-table-hover"
firstArrow='<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" class="w-4.5 h-4.5 rtl:rotate-180"> <path d="M13 19L7 12L13 5" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/> <path opacity="0.5" d="M16.9998 19L10.9998 12L16.9998 5" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/> </svg>'
lastArrow='<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" class="w-4.5 h-4.5 rtl:rotate-180"> <path d="M11 19L17 12L11 5" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/> <path opacity="0.5" d="M6.99976 19L12.9998 12L6.99976 5" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/> </svg> '
previousArrow='<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" class="w-4.5 h-4.5 rtl:rotate-180"> <path d="M15 5L9 12L15 19" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/> </svg>'
nextArrow='<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" class="w-4.5 h-4.5 rtl:rotate-180"> <path d="M9 5L15 12L9 19" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/> </svg>'
>
<template #actions="data">
<div class="flex justify-end gap-1">
<button type="button" class="btn btn-success !py-1" @click="viewData(data.value)">
<icon-edit class="me-1" />
Edit
</button>
<button type="button" class="btn btn-danger !py-1" @click="deleteData(data.value)">
<icon-trash class="me-1" />
Delete
</button>
</div>
</template>
</vue3-datatable>
</div>
</div>
</div>
</div>
</div>
</template>
<script lang="ts" setup>
import { ref, computed, onMounted } from 'vue';
import Vue3Datatable from '@bhplugin/vue3-datatable';
import { useDebounceFn } from '@vueuse/core';
useHead({ title: 'Daftar Karakter' });
const router = useRouter();
const config = useRuntimeConfig();
const { $api } = useNuxtApp();
const cols =
ref([
{ field: 'title', title: 'Judul' },
{ field: 'genre', title: 'Genre' },
{
field: 'actions',
title: 'Aksi',
headerClass: 'text-right',
cellClass: 'text-right',
sort: false,
width: '150px'
}
]) || [];
const params = reactive({
search: null,
current_page: 1,
pagesize: 10,
sort_column: 'name',
sort_direction: 'asc',
});
const rows = computed(() => manga.value?.results ?? []);
const totalRows = computed(() => manga.value?.count ?? 0);
const { data: manga } = await useAsyncData('manga',
() => {
return $api(`/entertainment/manga/`, {
params: {
page: params.current_page,
page_size: params.pagesize,
ordering: (params.sort_direction == 'desc' ? '-' : '') + params.sort_column,
search: params.search
},
headers: {
'Content-Type': 'application/json',
'Authorization': `Bearer ${useAuthStore().accessToken}`
},
});
}, {
watch: [params],
default: () => ({
results: [],
count: 0,
}),
}
);
const debouncedSearch = useDebounceFn(() => {
params.current_page = 1;
refreshNuxtData('manga');
}, 500);
watch(() => params.search, debouncedSearch);
const changeServer = (data: any) => {
params.pagesize = data.pagesize;
params.sort_column = data.sort_column;
params.sort_direction = data.sort_direction;
params.current_page = data.current_page;
};
const changeSearch = (data: any) => {
params.current_page = 1;
};
const viewData = (data: any) => {
router.push({ path: "/entertainment/manga/" + data.id });
};
const deleteData = async (data: any) => {
try {
await $api(`/entertainment/manga/${data.id}/`, {
method: 'DELETE',
headers: {
'Content-Type': 'application/json',
'Authorization': `Bearer ${useAuthStore().accessToken}`
},
});
refreshNuxtData('manga');
} catch (err) {
console.error('Gagal menghapus data', err);
}
}
</script>