feat(course-package): detail page, publish/unpublish UI, edit lock

- Add detail/show page with conditional action buttons
- Route /course/course-package/detail with props query params
- List page click navigates to detail instead of edit
- Edit page redirects if package is published (published_at set)
- Store: add publish/unpublish API functions

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
2026-04-30 21:06:19 +08:00
parent 8a8ce597b3
commit d117eb06f6
8 changed files with 272 additions and 6 deletions
+14 -2
View File
@@ -75,7 +75,6 @@ export default defineComponent({
};
const pickCourseGroups = () => {
store.hanging = true;
router.push("/course/group/pick?target=" + PickTargets.COURSE_PACKAGE);
};
@@ -109,7 +108,20 @@ export default defineComponent({
};
onMounted(() => {
store.selectedGroups = [];
const saved = sessionStorage.getItem('cp-selected-groups');
if (saved) {
try { store.selectedGroups = JSON.parse(saved); } catch(e) {}
sessionStorage.removeItem('cp-selected-groups');
} else {
store.selectedGroups = [];
}
(window as any).__cp_debug = {
onMounted: true,
saved: saved?.substring(0, 100),
selectedGroupsLen: store.selectedGroups.length,
selectedGroupsNames: store.selectedGroups.map((g: any) => g.group_name),
ts: Date.now()
};
});
return {
+197
View File
@@ -0,0 +1,197 @@
<template>
<div class="main">
<div class="body">
<div class="header">
<h1 class="title">{{ pkg?.package_name }}</h1>
<van-tag :type="pkg?.package_status === 'ACTIVE' ? 'success' : 'warning'" size="medium">
{{ pkg?.package_status === 'ACTIVE' ? '上架中' : '已下架' }}
</van-tag>
<van-tag v-if="isPublished" type="info" size="medium" style="margin-left: 0.08rem;">已发布快照</van-tag>
</div>
<div class="section">
<h2>基本信息</h2>
<van-cell-group inset>
<van-cell title="描述" :value="pkg?.description || '无'" />
<van-cell title="课次数量" :value="pkg?.total_lessons?.toString() || '无'" />
<van-cell title="原价" :value="pkg?.original_price ? '¥' + (pkg.original_price / 100).toFixed(2) : '无'" />
<van-cell title="售价" :value="pkg?.selling_price ? '¥' + (pkg.selling_price / 100).toFixed(2) : '无'" />
<van-cell title="有效期" :value="pkg?.validity_days ? pkg.validity_days + ' 天' : '无'" />
<van-cell title="排序" :value="pkg?.sort_order?.toString() || '无'" />
</van-cell-group>
</div>
<div class="section">
<h2>包含课节</h2>
<div v-if="courseGroups.length === 0" class="empty">暂无课节</div>
<van-cell v-for="g in courseGroups" :key="g.id" :title="g.group_name">
<template #label>
<div v-if="g.course_sections?.length">
{{ g.course_sections.map((s: any) => (s.course_name || '') + '(' + s.section_name + ')').join('、') }}
</div>
<div v-else style="color: #999;">无课程</div>
</template>
</van-cell>
</div>
</div>
<div class="footer">
<van-button v-if="!isPublished" type="primary" @click="onEdit">编辑</van-button>
<van-button v-if="!isPublished" type="danger" @click="onDelete">删除</van-button>
<van-button v-if="!isPublished" type="success" @click="onPublish">发布上架</van-button>
<van-button v-if="isPublished" type="warning" @click="onUnpublish">下架</van-button>
</div>
</div>
</template>
<script lang="ts">
import {computed, defineComponent, onMounted, ref} from 'vue';
import {Button, Cell, CellGroup, Tag, showConfirmDialog, showFailToast} from 'vant';
import {useRoute, useRouter} from 'vue-router';
import useCoursePackage from '~/store/course-package';
import {CourseGroup} from '~/types';
export default defineComponent({
name: 'course-package-detail',
components: {
[Button.name]: Button,
[Cell.name]: Cell,
[CellGroup.name]: CellGroup,
[Tag.name]: Tag,
},
setup() {
const route = useRoute();
const router = useRouter();
const {store, findById, remove, listPackageItems, publish, unpublish} = useCoursePackage();
const pkg = ref<any>(null);
const courseGroups = ref<CourseGroup[]>([]);
const loading = ref(false);
const isPublished = computed(() => !!pkg.value?.published_at);
const loadData = async () => {
const id = route.query.id as string;
if (!id) {
showFailToast('缺少课包 ID');
router.go(-1);
return;
}
loading.value = true;
const data = await findById(id);
if (!data) {
router.go(-1);
return;
}
pkg.value = data;
if (data.published_snapshot?.course_groups) {
// 从快照读取课节数据
courseGroups.value = data.published_snapshot.course_groups;
} else {
// 实时获取课节数据
const groups = await listPackageItems(id);
courseGroups.value = groups || [];
}
loading.value = false;
};
const onEdit = () => {
router.push(`/course/course-package/edit?id=${pkg.value?.id}`);
};
const onDelete = async () => {
try {
await showConfirmDialog({message: `确认删除课包【${pkg.value?.package_name}】?`});
const ok = await remove(pkg.value?.id!);
if (ok) router.go(-1);
} catch { /* cancelled */ }
};
const onPublish = async () => {
try {
await showConfirmDialog({
title: '发布课包',
message: '发布后将锁定课包内容,生成快照,不可再编辑。确认发布?',
});
const id = pkg.value?.id!;
// 获取完整课节数据作为快照
let groups = courseGroups.value;
if (groups.length === 0) {
groups = (await listPackageItems(id)) || [];
}
const snapshot = { course_groups: groups };
const ok = await publish(id, snapshot);
if (ok) await loadData();
} catch { /* cancelled */ }
};
const onUnpublish = async () => {
try {
await showConfirmDialog({
title: '下架课包',
message: '下架后课包内容可编辑,确认下架?',
});
const ok = await unpublish(pkg.value?.id!);
if (ok) await loadData();
} catch { /* cancelled */ }
};
onMounted(loadData);
return {
pkg, courseGroups, loading, isPublished,
onEdit, onDelete, onPublish, onUnpublish,
};
},
});
</script>
<style scoped lang="less">
.main {
.body {
height: calc(100% - 1.06rem);
padding: 0.32rem 0;
overflow: auto;
}
.header {
text-align: center;
margin-bottom: 0.32rem;
}
.title {
font-size: 0.48rem;
margin-block-start: 0;
margin-block-end: 0.16rem;
}
h2 {
font-size: 0.32rem;
padding: 0 0.32rem;
}
.section {
margin-bottom: 0.32rem;
}
.empty {
text-align: center;
color: #999;
font-size: 0.26rem;
padding: 0.32rem;
}
.footer {
position: absolute;
bottom: 0.16rem;
width: 100%;
padding: 0.01rem 0.32rem;
display: flex;
gap: 0.16rem;
button {
flex: 1;
}
}
}
</style>
+16 -2
View File
@@ -113,6 +113,11 @@ export default defineComponent({
form.validity_days = pkg.validity_days;
form.sort_order = pkg.sort_order;
form.package_status = pkg.package_status || 'ACTIVE';
if (pkg.published_at) {
showFailToast('已发布的课包不可编辑,请先下架再修改');
router.go(-1);
return;
}
if (pkg.original_price != null) {
originalPriceStr.value = (pkg.original_price / 100).toFixed(2);
}
@@ -122,7 +127,6 @@ export default defineComponent({
};
const pickCourseGroups = () => {
store.hanging = true;
router.push("/course/group/pick?target=" + PickTargets.COURSE_PACKAGE);
};
@@ -164,7 +168,17 @@ export default defineComponent({
}
};
onMounted(() => loadPackageAndItems());
onMounted(() => {
const saved = sessionStorage.getItem('cp-selected-groups');
if (saved) {
// 从 picker 返回,恢复选中课节,只加载表单数据
try { store.selectedGroups = JSON.parse(saved); } catch(e) {}
sessionStorage.removeItem('cp-selected-groups');
loadPackage();
} else {
loadPackageAndItems();
}
});
return {
form, store, originalPriceStr, sellingPriceStr, formatPrice,
+1 -2
View File
@@ -142,8 +142,7 @@ export default defineComponent({
};
const viewDetail = (item: CoursePackage) => {
// For now, edit is the detail view since course packages are simple
edit(item);
router.push(`/course/course-package/detail?id=${item.id}`);
};
const del = async (item: CoursePackage) => {
+2
View File
@@ -237,7 +237,9 @@ export default defineComponent({
usingClazz.store.current.course_sections = {vals: course_sections};
break;
case PickTargets.COURSE_PACKAGE:
sessionStorage.setItem('cp-selected-groups', JSON.stringify(state.checked));
usingCoursePackage.store.selectedGroups = state.checked;
(window as any).__cp_debug = { action: 'onPick', checked: state.checked.length, target: params.target, ts: Date.now() };
break;
}
router.back();
+5
View File
@@ -36,6 +36,7 @@ import CourseGroupDetail from '~/pages/qumu/group/detail.vue'
import CoursePackage from '~/pages/qumu/course-package/index.vue'
import CoursePackageAdd from '~/pages/qumu/course-package/add.vue'
import CoursePackageEdit from '~/pages/qumu/course-package/edit.vue'
import CoursePackageDetail from '~/pages/qumu/course-package/detail.vue'
import Tongzhi from '~/pages/tongzhi/index.vue';
import TongzhiDetail from '~/pages/tongzhi/detail.vue';
import Tester from '~/pages/tester/index.vue'
@@ -181,6 +182,10 @@ export default [
path: '/course/course-package/edit', component: CoursePackageEdit,
meta: {title: "编辑课包"}, props: ({query}: RouteQueryAndHash) => ({params: query})
},
{
path: '/course/course-package/detail', component: CoursePackageDetail,
meta: {title: "课包详情"}, props: ({query}: RouteQueryAndHash) => ({params: query})
},
{path: '/tongzhi', component: Tongzhi, meta: {title: "消息通知"}},
{
path: '/tongzhi/detail', component: TongzhiDetail, meta: {title: "通知详情"},
+35
View File
@@ -8,12 +8,14 @@ interface CoursePackageStore {
list: CoursePackage[];
total: number;
selectedGroups: CourseGroup[];
hanging: boolean;
}
const state = reactive<CoursePackageStore>({
list: [],
total: 0,
selectedGroups: [],
hanging: false,
});
export default function useCoursePackage() {
@@ -141,6 +143,37 @@ export default function useCoursePackage() {
return undefined;
}
async function publish(id: string, snapshot: any): Promise<boolean> {
load_start();
const {r, e} = await request({
url: `/api/v1/clazz/course-package/publish/${id}`,
method: 'POST',
data: snapshot,
});
load_done();
if (r) {
showSuccessToast({message: '课包发布成功!'});
} else {
showFailToast(e);
}
return r;
}
async function unpublish(id: string): Promise<boolean> {
load_start();
const {r, e} = await request({
url: `/api/v1/clazz/course-package/unpublish/${id}`,
method: 'POST',
});
load_done();
if (r) {
showSuccessToast({message: '课包已下架!'});
} else {
showFailToast(e);
}
return r;
}
return {
store: state,
queryMyPackages,
@@ -151,5 +184,7 @@ export default function useCoursePackage() {
remove,
syncPackageItems,
listPackageItems,
publish,
unpublish,
};
}
+2
View File
@@ -770,4 +770,6 @@ export interface CoursePackage {
updated_at?: string | Date;
is_delete?: boolean;
org_id?: string;
published_snapshot?: any;
published_at?: string | Date;
}