feat: update post edit page
This commit is contained in:
@@ -75,7 +75,46 @@ func (ctl *posts) Create(ctx fiber.Ctx, form *PostForm) error {
|
|||||||
// @Router /v1/admin/posts/:id [put]
|
// @Router /v1/admin/posts/:id [put]
|
||||||
// @Bind id path
|
// @Bind id path
|
||||||
// @Bind form body
|
// @Bind form body
|
||||||
func (ctl *posts) Update(ctx fiber.Ctx, id int64, form *model.Posts) error {
|
func (ctl *posts) Update(ctx fiber.Ctx, id int64, form *PostForm) error {
|
||||||
|
oldPost, err := models.Posts.GetByID(ctx.Context(), id)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
post := &model.Posts{
|
||||||
|
Title: form.Title,
|
||||||
|
Price: form.Price,
|
||||||
|
Discount: form.Discount,
|
||||||
|
Description: form.Introduction,
|
||||||
|
Status: form.Status,
|
||||||
|
Content: "",
|
||||||
|
Tags: fields.Json[[]string]{},
|
||||||
|
Assets: fields.Json[[]fields.MediaAsset]{},
|
||||||
|
CreatedAt: oldPost.CreatedAt,
|
||||||
|
UpdatedAt: oldPost.UpdatedAt,
|
||||||
|
DeletedAt: oldPost.DeletedAt,
|
||||||
|
Views: oldPost.Views,
|
||||||
|
Likes: oldPost.Likes,
|
||||||
|
}
|
||||||
|
|
||||||
|
if form.Medias != nil {
|
||||||
|
medias, err := models.Medias.GetByIds(ctx.Context(), form.Medias)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
assets := lo.Map(medias, func(media *model.Medias, _ int) fields.MediaAsset {
|
||||||
|
return fields.MediaAsset{
|
||||||
|
Type: models.Medias.ConvertFileTypeByMimeType(media.MimeType),
|
||||||
|
Media: media.ID,
|
||||||
|
Mark: nil,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
post.Assets = fields.ToJson(assets)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := models.Posts.Update(ctx.Context(), id, post); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ import (
|
|||||||
. "go.ipao.vip/atom/fen"
|
. "go.ipao.vip/atom/fen"
|
||||||
"mime/multipart"
|
"mime/multipart"
|
||||||
"quyun/app/requests"
|
"quyun/app/requests"
|
||||||
"quyun/database/schemas/public/model"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// @provider contracts.HttpRoute atom.GroupRoutes
|
// @provider contracts.HttpRoute atom.GroupRoutes
|
||||||
@@ -53,7 +52,7 @@ func (r *Routes) Register(router fiber.Router) {
|
|||||||
router.Put("/v1/admin/posts/:id", Func2(
|
router.Put("/v1/admin/posts/:id", Func2(
|
||||||
r.posts.Update,
|
r.posts.Update,
|
||||||
PathParam[int64]("id"),
|
PathParam[int64]("id"),
|
||||||
Body[model.Posts]("form"),
|
Body[PostForm]("form"),
|
||||||
))
|
))
|
||||||
|
|
||||||
router.Delete("/v1/admin/posts/:id", Func1(
|
router.Delete("/v1/admin/posts/:id", Func1(
|
||||||
|
|||||||
@@ -90,7 +90,7 @@ func (m *postsModel) Update(ctx context.Context, id int64, model *model.Posts) e
|
|||||||
model.UpdatedAt = time.Now()
|
model.UpdatedAt = time.Now()
|
||||||
|
|
||||||
tbl := table.Posts
|
tbl := table.Posts
|
||||||
stmt := tbl.UPDATE(tbl.MutableColumns).SET(model).WHERE(tbl.ID.EQ(Int64(id)))
|
stmt := tbl.UPDATE(tbl.MutableColumns).MODEL(model).WHERE(tbl.ID.EQ(Int64(id)))
|
||||||
m.log.Infof("sql: %s", stmt.DebugSql())
|
m.log.Infof("sql: %s", stmt.DebugSql())
|
||||||
|
|
||||||
_, err := stmt.ExecContext(ctx, db)
|
_, err := stmt.ExecContext(ctx, db)
|
||||||
|
|||||||
@@ -102,6 +102,7 @@ require (
|
|||||||
github.com/jcmturner/gofork v1.7.6 // indirect
|
github.com/jcmturner/gofork v1.7.6 // indirect
|
||||||
github.com/jcmturner/gokrb5/v8 v8.4.4 // indirect
|
github.com/jcmturner/gokrb5/v8 v8.4.4 // indirect
|
||||||
github.com/jcmturner/rpc/v2 v2.0.3 // indirect
|
github.com/jcmturner/rpc/v2 v2.0.3 // indirect
|
||||||
|
github.com/jinzhu/copier v0.4.0 // indirect
|
||||||
github.com/josharian/intern v1.0.0 // indirect
|
github.com/josharian/intern v1.0.0 // indirect
|
||||||
github.com/json-iterator/go v1.1.12 // indirect
|
github.com/json-iterator/go v1.1.12 // indirect
|
||||||
github.com/jtolds/gls v4.20.0+incompatible // indirect
|
github.com/jtolds/gls v4.20.0+incompatible // indirect
|
||||||
|
|||||||
@@ -168,6 +168,8 @@ github.com/jcmturner/gokrb5/v8 v8.4.4 h1:x1Sv4HaTpepFkXbt2IkL29DXRf8sOfZXo8eRKh6
|
|||||||
github.com/jcmturner/gokrb5/v8 v8.4.4/go.mod h1:1btQEpgT6k+unzCwX1KdWMEwPPkkgBtP+F6aCACiMrs=
|
github.com/jcmturner/gokrb5/v8 v8.4.4/go.mod h1:1btQEpgT6k+unzCwX1KdWMEwPPkkgBtP+F6aCACiMrs=
|
||||||
github.com/jcmturner/rpc/v2 v2.0.3 h1:7FXXj8Ti1IaVFpSAziCZWNzbNuZmnvw/i6CqLNdWfZY=
|
github.com/jcmturner/rpc/v2 v2.0.3 h1:7FXXj8Ti1IaVFpSAziCZWNzbNuZmnvw/i6CqLNdWfZY=
|
||||||
github.com/jcmturner/rpc/v2 v2.0.3/go.mod h1:VUJYCIDm3PVOEHw8sgt091/20OJjskO/YJki3ELg/Hc=
|
github.com/jcmturner/rpc/v2 v2.0.3/go.mod h1:VUJYCIDm3PVOEHw8sgt091/20OJjskO/YJki3ELg/Hc=
|
||||||
|
github.com/jinzhu/copier v0.4.0 h1:w3ciUoD19shMCRargcpm0cm91ytaBhDvuRpz1ODO/U8=
|
||||||
|
github.com/jinzhu/copier v0.4.0/go.mod h1:DfbEm0FYsaqBcKcFuvmOZb218JkPGtvSHsKg8S8hyyg=
|
||||||
github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY=
|
github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY=
|
||||||
github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y=
|
github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y=
|
||||||
github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM=
|
github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM=
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
<script setup>
|
<script setup>
|
||||||
|
import { mediaService } from '@/api/mediaService';
|
||||||
import { postService } from '@/api/postService';
|
import { postService } from '@/api/postService';
|
||||||
import { useToast } from 'primevue/usetoast';
|
import { useToast } from 'primevue/usetoast';
|
||||||
import { onMounted, reactive, ref } from 'vue';
|
import { computed, onMounted, reactive, ref } from 'vue';
|
||||||
import { useRoute, useRouter } from 'vue-router';
|
import { useRoute, useRouter } from 'vue-router';
|
||||||
|
|
||||||
// PrimeVue components
|
// PrimeVue components
|
||||||
@@ -13,6 +14,7 @@ import Dialog from 'primevue/dialog';
|
|||||||
import InputNumber from 'primevue/inputnumber';
|
import InputNumber from 'primevue/inputnumber';
|
||||||
import InputText from 'primevue/inputtext';
|
import InputText from 'primevue/inputtext';
|
||||||
import ProgressSpinner from 'primevue/progressspinner';
|
import ProgressSpinner from 'primevue/progressspinner';
|
||||||
|
import RadioButton from 'primevue/radiobutton';
|
||||||
import Textarea from 'primevue/textarea';
|
import Textarea from 'primevue/textarea';
|
||||||
import Toast from 'primevue/toast';
|
import Toast from 'primevue/toast';
|
||||||
|
|
||||||
@@ -53,6 +55,16 @@ const selectedMediaItems = ref([]);
|
|||||||
const mediaLoading = ref(false);
|
const mediaLoading = ref(false);
|
||||||
const mediaGlobalFilter = ref('');
|
const mediaGlobalFilter = ref('');
|
||||||
|
|
||||||
|
// Add pagination state for media dialog
|
||||||
|
const mediaFirst = ref(0);
|
||||||
|
const mediaRows = ref(10);
|
||||||
|
const mediaTotalRecords = ref(0);
|
||||||
|
const mediaCurrentPage = ref(1);
|
||||||
|
|
||||||
|
const mediaTotalPages = computed(() => {
|
||||||
|
return Math.ceil(mediaTotalRecords.value / mediaRows.value);
|
||||||
|
});
|
||||||
|
|
||||||
// Sample media data - in a real app, this would come from an API
|
// Sample media data - in a real app, this would come from an API
|
||||||
const mediaItems = ref([
|
const mediaItems = ref([
|
||||||
{
|
{
|
||||||
@@ -113,7 +125,7 @@ const fetchPost = async (id) => {
|
|||||||
post.title = postData.title;
|
post.title = postData.title;
|
||||||
post.price = postData.price;
|
post.price = postData.price;
|
||||||
post.discount = postData.discount || 100;
|
post.discount = postData.discount || 100;
|
||||||
post.introduction = postData.introduction;
|
post.introduction = postData.description;
|
||||||
post.status = postData.status;
|
post.status = postData.status;
|
||||||
post.selectedMedia = postData.medias || [];
|
post.selectedMedia = postData.medias || [];
|
||||||
post.medias = postData.medias?.map(media => media.id) || [];
|
post.medias = postData.medias?.map(media => media.id) || [];
|
||||||
@@ -128,6 +140,8 @@ const fetchPost = async (id) => {
|
|||||||
// Open media selection dialog
|
// Open media selection dialog
|
||||||
const openMediaDialog = () => {
|
const openMediaDialog = () => {
|
||||||
mediaDialogVisible.value = true;
|
mediaDialogVisible.value = true;
|
||||||
|
mediaCurrentPage.value = 1;
|
||||||
|
mediaFirst.value = 0;
|
||||||
loadMediaItems();
|
loadMediaItems();
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -135,13 +149,12 @@ const openMediaDialog = () => {
|
|||||||
const loadMediaItems = async () => {
|
const loadMediaItems = async () => {
|
||||||
mediaLoading.value = true;
|
mediaLoading.value = true;
|
||||||
try {
|
try {
|
||||||
// In a real app, this would be an API call
|
const response = await mediaService.getMedias({
|
||||||
// const response = await mediaApi.getMediaFiles();
|
page: mediaCurrentPage.value,
|
||||||
// mediaItems.value = response.data;
|
limit: mediaRows.value
|
||||||
|
});
|
||||||
// Simulate API delay
|
mediaItems.value = response.data.items;
|
||||||
await new Promise(resolve => setTimeout(resolve, 500));
|
mediaTotalRecords.value = response.data.total;
|
||||||
// Using sample data already defined above
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
toast.add({ severity: 'error', summary: '错误', detail: '加载媒体文件失败', life: 3000 });
|
toast.add({ severity: 'error', summary: '错误', detail: '加载媒体文件失败', life: 3000 });
|
||||||
} finally {
|
} finally {
|
||||||
@@ -249,6 +262,27 @@ const getFileIcon = (file) => {
|
|||||||
return `pi ${map[file.fileType] || 'pi-file'}`;
|
return `pi ${map[file.fileType] || 'pi-file'}`;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Add pagination handler
|
||||||
|
const onMediaPage = (event) => {
|
||||||
|
mediaFirst.value = event.first;
|
||||||
|
mediaRows.value = event.rows;
|
||||||
|
mediaCurrentPage.value = Math.floor(event.first / event.rows) + 1;
|
||||||
|
loadMediaItems();
|
||||||
|
};
|
||||||
|
|
||||||
|
// Add format file size helper
|
||||||
|
const formatFileSize = (bytes) => {
|
||||||
|
if (!bytes && bytes !== 0) return '0 B';
|
||||||
|
const sizes = ['B', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'];
|
||||||
|
const base = 1024;
|
||||||
|
let size = Number(bytes);
|
||||||
|
if (isNaN(size)) return '0 B';
|
||||||
|
|
||||||
|
const i = Math.floor(Math.log(size) / Math.log(base));
|
||||||
|
size = size / Math.pow(base, i);
|
||||||
|
return `${size.toFixed(2)} ${sizes[i]}`;
|
||||||
|
};
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
// Get post ID from route params
|
// Get post ID from route params
|
||||||
const postId = route.params.id;
|
const postId = route.params.id;
|
||||||
@@ -373,9 +407,22 @@ onMounted(() => {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<DataTable v-model:selection="selectedMediaItems" :value="mediaItems" :loading="mediaLoading" dataKey="id"
|
<DataTable v-model:selection="selectedMediaItems" :value="mediaItems" :loading="mediaLoading" dataKey="id"
|
||||||
selectionMode="multiple" :globalFilterFields="['fileName', 'fileType']"
|
:paginator="true" v-model:first="mediaFirst" v-model:rows="mediaRows" :totalRecords="mediaTotalRecords"
|
||||||
:filters="{ global: { value: mediaGlobalFilter, matchMode: 'contains' } }" stripedRows
|
@page="onMediaPage" selectionMode="multiple"
|
||||||
responsiveLayout="scroll">
|
paginatorTemplate="FirstPageLink PrevPageLink PageLinks NextPageLink LastPageLink CurrentPageReport"
|
||||||
|
:rows-per-page-options="[10, 25, 50]" currentPageReportTemplate="第 {first} 到 {last} 条,共 {totalRecords} 条"
|
||||||
|
:lazy="true" :showCurrentPageReport="true">
|
||||||
|
|
||||||
|
<template #paginatorLeft>
|
||||||
|
<div class="flex items-center">
|
||||||
|
每页: {{ mediaRows }}
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<template #paginatorRight>
|
||||||
|
<div class="flex items-center">
|
||||||
|
第 {{ mediaCurrentPage }} 页,共 {{ mediaTotalPages }} 页
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
<template #empty>
|
<template #empty>
|
||||||
<div class="text-center p-4">没有可用的媒体文件</div>
|
<div class="text-center p-4">没有可用的媒体文件</div>
|
||||||
@@ -392,27 +439,27 @@ onMounted(() => {
|
|||||||
|
|
||||||
<Column field="fileName" header="文件名">
|
<Column field="fileName" header="文件名">
|
||||||
<template #body="{ data }">
|
<template #body="{ data }">
|
||||||
<div class="flex items-center">
|
<div class="text-sm font-medium text-gray-900">{{ data.name }}</div>
|
||||||
<div v-if="data.thumbnailUrl" class="flex-shrink-0 h-10 w-10 mr-3">
|
|
||||||
<img class="h-10 w-10 object-cover rounded" :src="data.thumbnailUrl" :alt="data.fileName">
|
|
||||||
</div>
|
|
||||||
<div v-else
|
|
||||||
class="flex-shrink-0 h-10 w-10 mr-3 bg-gray-100 rounded flex items-center justify-center">
|
|
||||||
<i :class="getFileIcon(data)" class="text-2xl"></i>
|
|
||||||
</div>
|
|
||||||
<div class="text-sm font-medium text-gray-900">{{ data.fileName }}</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
</template>
|
||||||
</Column>
|
</Column>
|
||||||
|
|
||||||
<Column field="fileType" header="文件类型">
|
<Column field="fileType" header="文件类型">
|
||||||
<template #body="{ data }">
|
<template #body="{ data }">
|
||||||
<Badge :value="data.fileType" :severity="getBadgeSeverity(data.fileType)" />
|
<Badge :value="data.file_type" :severity="getBadgeSeverity(data.file_type)" />
|
||||||
</template>
|
</template>
|
||||||
</Column>
|
</Column>
|
||||||
|
|
||||||
<Column field="fileSize" header="文件大小"></Column>
|
<Column field="fileSize" header="文件大小">
|
||||||
<Column field="uploadTime" header="上传时间"></Column>
|
<template #body="{ data }">
|
||||||
|
{{ formatFileSize(data.file_size) }}
|
||||||
|
</template>
|
||||||
|
</Column>
|
||||||
|
|
||||||
|
<Column field="createdAt" header="上传时间">
|
||||||
|
<template #body="{ data }">
|
||||||
|
{{ new Date(data.upload_time).toLocaleString('zh-CN') }}
|
||||||
|
</template>
|
||||||
|
</Column>
|
||||||
</DataTable>
|
</DataTable>
|
||||||
|
|
||||||
<template #footer>
|
<template #footer>
|
||||||
|
|||||||
Reference in New Issue
Block a user