feat(admin): 公共管理列表添加分页
后端 listForAdmin 支持 page/pageSize 分页查询,控制器接收分页参数并校验边界,前端 Contents 页面添加 el-pagination 组件并同步分页状态。 🤖 Generated with [Qoder][https://qoder.com]
This commit is contained in:
@@ -130,6 +130,9 @@ const filterStatus = ref<ContentStatus | ''>('');
|
|||||||
const loading = ref(false);
|
const loading = ref(false);
|
||||||
const saving = ref(false);
|
const saving = ref(false);
|
||||||
const items = ref<ContentItem[]>([]);
|
const items = ref<ContentItem[]>([]);
|
||||||
|
const total = ref(0);
|
||||||
|
const page = ref(1);
|
||||||
|
const pageSize = ref(10);
|
||||||
const tableRef = ref<TableInstance>();
|
const tableRef = ref<TableInstance>();
|
||||||
const selectedRows = ref<ContentItem[]>([]);
|
const selectedRows = ref<ContentItem[]>([]);
|
||||||
|
|
||||||
@@ -206,9 +209,12 @@ async function load() {
|
|||||||
params: {
|
params: {
|
||||||
type: activeType.value,
|
type: activeType.value,
|
||||||
status: filterStatus.value || undefined,
|
status: filterStatus.value || undefined,
|
||||||
|
page: page.value,
|
||||||
|
pageSize: pageSize.value,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
items.value = data.data ?? [];
|
items.value = data.data.items ?? [];
|
||||||
|
total.value = data.data.total ?? 0;
|
||||||
selectedRows.value = [];
|
selectedRows.value = [];
|
||||||
tableRef.value?.clearSelection();
|
tableRef.value?.clearSelection();
|
||||||
} catch (e: unknown) {
|
} catch (e: unknown) {
|
||||||
@@ -219,7 +225,19 @@ async function load() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function onPageChange(p: number) {
|
||||||
|
page.value = p;
|
||||||
|
load();
|
||||||
|
}
|
||||||
|
|
||||||
|
function onSizeChange(size: number) {
|
||||||
|
pageSize.value = size;
|
||||||
|
page.value = 1;
|
||||||
|
load();
|
||||||
|
}
|
||||||
|
|
||||||
watch([activeType, filterStatus], () => {
|
watch([activeType, filterStatus], () => {
|
||||||
|
page.value = 1;
|
||||||
selectedRows.value = [];
|
selectedRows.value = [];
|
||||||
tableRef.value?.clearSelection();
|
tableRef.value?.clearSelection();
|
||||||
void load();
|
void load();
|
||||||
@@ -290,7 +308,7 @@ function batchDelete() {
|
|||||||
|
|
||||||
function resetForm() {
|
function resetForm() {
|
||||||
form.value = {
|
form.value = {
|
||||||
sortOrder: items.value.length + 1,
|
sortOrder: total.value + 1,
|
||||||
status: 'DRAFT',
|
status: 'DRAFT',
|
||||||
linkType: '',
|
linkType: '',
|
||||||
linkTarget: '',
|
linkTarget: '',
|
||||||
@@ -562,6 +580,18 @@ void load();
|
|||||||
</el-table-column>
|
</el-table-column>
|
||||||
</el-table>
|
</el-table>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="pager">
|
||||||
|
<el-pagination
|
||||||
|
v-model:current-page="page"
|
||||||
|
v-model:page-size="pageSize"
|
||||||
|
:total="total"
|
||||||
|
:page-sizes="[10, 20, 50, 100]"
|
||||||
|
layout="total, sizes, prev, pager, next"
|
||||||
|
background
|
||||||
|
@current-change="onPageChange"
|
||||||
|
@size-change="onSizeChange"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</el-card>
|
</el-card>
|
||||||
|
|
||||||
<el-dialog v-model="dialogVisible" :title="dialogTitle" width="640px" destroy-on-close>
|
<el-dialog v-model="dialogVisible" :title="dialogTitle" width="640px" destroy-on-close>
|
||||||
|
|||||||
@@ -2249,9 +2249,13 @@ export class AdminController {
|
|||||||
async listContents(
|
async listContents(
|
||||||
@Query('type') type?: string,
|
@Query('type') type?: string,
|
||||||
@Query('status') status?: string,
|
@Query('status') status?: string,
|
||||||
|
@Query('page') page?: string,
|
||||||
|
@Query('pageSize') pageSize?: string,
|
||||||
) {
|
) {
|
||||||
const items = await this.content.listForAdmin(type, status);
|
const p = Math.max(1, page ? parseInt(page, 10) || 1 : 1);
|
||||||
return jsonResponse(items);
|
const size = Math.min(Math.max(1, pageSize ? parseInt(pageSize, 10) : 20), 100);
|
||||||
|
const result = await this.content.listForAdmin(type, status, p, size);
|
||||||
|
return jsonResponse(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Get('contents/:id')
|
@Get('contents/:id')
|
||||||
|
|||||||
@@ -255,7 +255,12 @@ export class ContentService {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async listForAdmin(contentType?: string, status?: string) {
|
async listForAdmin(
|
||||||
|
contentType?: string,
|
||||||
|
status?: string,
|
||||||
|
page = 1,
|
||||||
|
pageSize = 20,
|
||||||
|
) {
|
||||||
const typeWhere =
|
const typeWhere =
|
||||||
contentType === ANNOUNCEMENT_ADMIN_TYPE
|
contentType === ANNOUNCEMENT_ADMIN_TYPE
|
||||||
? { contentType: { in: [...ANNOUNCEMENT_TYPES] } }
|
? { contentType: { in: [...ANNOUNCEMENT_TYPES] } }
|
||||||
@@ -263,15 +268,28 @@ export class ContentService {
|
|||||||
? { contentType }
|
? { contentType }
|
||||||
: {};
|
: {};
|
||||||
|
|
||||||
const items = await this.prisma.content.findMany({
|
const where = {
|
||||||
where: {
|
...typeWhere,
|
||||||
...typeWhere,
|
...(status ? { status } : {}),
|
||||||
...(status ? { status } : {}),
|
};
|
||||||
},
|
|
||||||
include: { translations: true },
|
const [items, total] = await Promise.all([
|
||||||
orderBy: [{ sortOrder: 'asc' }, { id: 'asc' }],
|
this.prisma.content.findMany({
|
||||||
});
|
where,
|
||||||
return items.map((item) => this.mapAdminItem(item));
|
include: { translations: true },
|
||||||
|
orderBy: [{ sortOrder: 'asc' }, { id: 'asc' }],
|
||||||
|
skip: (page - 1) * pageSize,
|
||||||
|
take: pageSize,
|
||||||
|
}),
|
||||||
|
this.prisma.content.count({ where }),
|
||||||
|
]);
|
||||||
|
|
||||||
|
return {
|
||||||
|
items: items.map((item) => this.mapAdminItem(item)),
|
||||||
|
total,
|
||||||
|
page,
|
||||||
|
pageSize,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
async getForAdmin(id: bigint) {
|
async getForAdmin(id: bigint) {
|
||||||
@@ -442,6 +460,7 @@ export class ContentService {
|
|||||||
|
|
||||||
/** @deprecated use listForAdmin */
|
/** @deprecated use listForAdmin */
|
||||||
async listAll(contentType?: string) {
|
async listAll(contentType?: string) {
|
||||||
return this.listForAdmin(contentType);
|
const result = await this.listForAdmin(contentType);
|
||||||
|
return result.items;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user