微搭低代码MBA 培训管理系统实战 19——教务管理:从订单到课时卡的自动转化

微搭低代码MBA 培训管理系统实战 19——教务管理:从订单到课时卡的自动转化

目录

前情回顾

上一篇中我们讲解了销售在订单成交后,录入订单。此时订单的状态还是待支付的状态,需要财务确认收款情况。财务人员点击了"确认收款",订单状态变更为 已清账。此时,资金流已经闭环,但学员在系统里还只是一个"商机客户",没有上课的权限。本节我们要解决如何让系统自动识别已付钱的客户,并为其发放"数字资产"——课时卡

本节目标:

  1. 学籍自动转化:将 CRM 中的"客户"自动转录为"正式学员"。
  2. 课时自动充值:根据订单中的产品,自动生成对应的课时卡余额。

一、 数据源设计

在原有订单表的基础上,我们再创建两张表,用来记录学员的学籍和课时卡信息

1.1 学员档案表 (MBA_StudentProfiles)

定位:学员的"基本法",记录自然人属性。

字段名称字段标识字段类型说明
学员ID_id文本主键,系统自动生成
学员姓名student_name文本姓名
手机号mobile文本用于登录或通知
关联客户rel_customer_id关联关系关联 MBA_Customer 表,追溯来源
OpenIDopenid文本学员微信OpenID,用于小程序端登录和权限控制
学籍状态status枚举1-在读、2-休学、3-毕业、4-退费
入学时间enroll_date日期订单支付成功的日期
创建时间created_at日期时间自动生成
更新时间updated_at日期时间自动更新
在这里插入图片描述

1.2 课时卡表 (MBA_LearningCards)

定位:系统的"资产层"。它是财务与教务的交接单,也是待分班列表的数据源。

字段名称字段标识字段类型说明
卡片ID_id文本主键
关联学员rel_student_id关联关系属于哪个学员
关联订单rel_order_id关联关系关联 MBA_Orders 表,追溯交易来源
关联产品rel_prod_id关联关系购买的课程产品(如:MBA面试突击班)
总课时total_hours数字初始发放数
剩余课时remain_hours数字实际可用数(消课扣减此处)
分班状态assign_status枚举核心控制位:1-待分班(pending)、2-已进班(assigned)
关联班级rel_class_id关联关系当前这张卡绑定的班级(方便直接显示)
过期日期expiry_date日期课时卡有效期,默认一年
创建时间created_at日期时间自动生成
更新时间updated_at日期时间自动更新
在这里插入图片描述

二 创建管理页面

2.1 搭建财务布局

目前财务需要专属的工作台来进行资金到账的确认动作,我们来给财务创建一个布局。

点击布局设计,点击新建布局

在这里插入图片描述


选择左侧导航布局

在这里插入图片描述


重命名为财务布局

在这里插入图片描述

2.2 搭建待支付列表页面

点击创建页面的图标,创建订单确认页面,选择财务布局

切换到布局设计,添加菜单

在这里插入图片描述


切回到页面设计,在财务布局的内容插槽里添加布局组件

在这里插入图片描述


修改标题为订单确认

在这里插入图片描述


添加顶部选项卡,标签改为待办、已办

在这里插入图片描述


添加数据表格,数据模型选择订单表

在这里插入图片描述


设置数据筛选,配置为订单状态等于待支付

在这里插入图片描述


将操作列的按钮改为确认支付

在这里插入图片描述

2.3 搭建确认支付弹窗

在页面组件下添加弹窗组件

在这里插入图片描述


添加表单容器,数据模型选择订单表,场景选择更新

在这里插入图片描述


数据标识绑定为弹窗的入参

在这里插入图片描述

将订单的字段的状态改为只读

在这里插入图片描述


财务确认人的选中值绑定为当前登录对象的数据标识

在这里插入图片描述


给确认支付按钮绑定点击事件,打开弹窗,传入所在行的数据

在这里插入图片描述

2.4 自动化开课

当财务选择了已清账,需要自动给学员创建学籍,开通课时卡。我们需要创建一个自定义方法来完成操作。

export default async function afterUpdate({ event, data }){ // data.target 直接包含当前订单的完整信息(包括关联字段) const order = data.target; const orderId = order._id; const customer = order.rel_customer_id; // 关联的客户对象 const product = order.rel_prod_id; // 关联的产品对象 const order_status =$w.select10.value // 判断订单状态是否为"已清账"(3),是则执行开课逻辑 if(order_status ==='3'){ try { await $w.utils.showLoading({ title: '正在开课...'}); // 1. 查询是否已存在学员档案(避免重复创建) const studentRes = await $w.cloud.callDataSource({ dataSourceName: 'MBA_StudentProfiles', methodName: 'wedaGetRecordsV2', params: { filter: { where: { rel_customer_id: {$eq: customer._id }}}, select: {$master:true}}});let studentId;if(studentRes.records && studentRes.records.length >0){ // 更新已有学员档案 const existingStudent = studentRes.records[0]; await $w.cloud.callDataSource({ dataSourceName: 'MBA_StudentProfiles', methodName: 'wedaUpdateV2', params: { data: { status: '1', // 在读 enroll_date: Date.now(), }, filter: { where: { _id: {$eq: existingStudent._id }}}}}); studentId = existingStudent._id;}else{ // 创建新学员档案 const newStudentRes = await $w.cloud.callDataSource({ dataSourceName: 'MBA_StudentProfiles', methodName: 'wedaCreateV2', params: { data: { student_name: customer.name, mobile: customer.phone, rel_customer_id: { _id: customer._id }, openid: null, // 待学员首次登录小程序时绑定 status: '1', // 在读 enroll_date: Date.now(), }}}); studentId = newStudentRes.id;} // 2. 创建课时卡(待分班状态) const expiryDate = new Date(); expiryDate.setFullYear(expiryDate.getFullYear() + 1); await $w.cloud.callDataSource({ dataSourceName: 'MBA_LearningCards', methodName: 'wedaCreateV2', params: { data: { rel_student_id: { _id: studentId }, rel_order_id: { _id: orderId }, rel_prod_id: { _id: product._id }, total_hours: order.product_hours || product.total_hours ||0, remain_hours: order.product_hours || product.total_hours ||0, assign_status: '1', // 待分班 expiry_date: expiryDate.getTime(), }}}); console.log(`学员档案和课时卡创建成功:订单 ${orderId}`);$w.utils.showToast({ title: '自动化开课成功', icon: 'success'});} catch (error){ console.error('自动化开课失败:', error);$w.utils.showToast({ title: '开课失败:' + error.message, icon: 'error'});} finally {$w.utils.hideLoading();}}}

在表单提交的时候,调用此方法

在这里插入图片描述

三 配置门户数据

先创建财务角色

在这里插入图片描述


然后添加人员

在这里插入图片描述


配置财务门户数据

在这里插入图片描述

需要完善一下全局登录方法,添加财务岗位映射关系

 /* * 函数里面访问:通过 app.common.[name] 访问这里定义的方法或值 * 函数外面访问:通过 import(如在页面的 handler 引用的例子:import sayHi from '../../common/[name]') */ export default async function login({ event, data }){ try {$w.utils.showLoading({ title: '加载工作台中...'}); // 1. 从云开发原生 auth 对象中获取当前已登录的手机号 const phone =$w.auth.currentUser?.phone ||$w.input1.value;if(!phone){return$w.utils.showToast({ title: '获取授权信息失败,请重新登录', icon: 'error'});} // 1. 先查询是否是渠道用户 const channelRes = await $w.cloud.callDataSource({ dataSourceName: 'MBA_ChannelPartners', methodName: 'wedaGetRecordsV2', params: { filter: { where: { phone: {$eq: phone }}}, select: {$master: true, }}}); console.log("channelRes", channelRes)if(channelRes.records && channelRes.records.length >0){ // 渠道用户登录成功 const channelInfo = channelRes.records[0];$w.app.dataset.state.currentChannel = channelInfo; // 渠道用户固定角色,用于门户跳转判断 $w.app.dataset.state.userRoles =['ROLE_CHANNEL'];$w.utils.showToast({ title: '渠道登录成功', icon: 'success'});return;} // 2. 根据手机号查询业务库中的用户信息(关联查询部门和岗位) const userRes = await $w.cloud.callDataSource({ dataSourceName: 'MBA_Users', methodName: 'wedaGetRecordsV2', params: { filter: { where: { phone: {$eq: phone }}}, select: {$master: true, department_id: true, position_id: true}}});if(!userRes.records.length){return$w.utils.showToast({ title: '该手机号未在系统中注册,请联系管理员', icon: 'error'});} const user = userRes.records[0]; console.log("user", user) // 3. 异步更新用户的最后登录时间 (不阻塞后续逻辑)$w.cloud.callDataSource({ dataSourceName: 'MBA_Users', methodName: 'wedaUpdateV2', params: { data: { last_login: Date.now()}, filter: { where: { _id: {$eq: user._id }}}}}).catch(e => console.error('更新登录时间失败', e)); // 4. 获取用户关联的角色信息 const roleRes = await $w.cloud.callDataSource({ dataSourceName: 'MBA_RoleUsers', methodName: 'wedaGetRecordsV2', params: { filter: { where: { user_id: {$eq: user._id }}}, select: {$master: true, role_id: true}}}); console.log("roleRes", roleRes) // 提取角色编码 (例如: ['ROLE_ADMIN', 'ROLE_TEACHER']) const roleCodes = roleRes.records.map(item => item.role_id?.code).filter(Boolean); console.log("roleCodes", roleCodes) // 5. 构造全局 User 对象并写入页面状态 const userInfo ={...user, deptInfo: user.department_id, postInfo: user.position_id, }; // 5. 根据岗位信息确定用户类型 const userType = getUserTypeByPosition(user.position_id);$w.app.dataset.state.currentUser = userInfo;$w.app.dataset.state.userRoles = roleCodes;$w.app.dataset.state.userType = userType;} catch (e){ console.error('初始化门户失败', e);$w.utils.showToast({ title: '系统加载失败,请刷新重试', icon: 'error'});} finally {$w.utils.hideLoading();}} // 根据岗位信息确定用户类型 exportfunction getUserTypeByPosition(position){if(!position){return'employee';} const positionName = position.name ||''; const positionCode = position.code ||''; // 岗位到用户类型的映射 const positionToTypeMap ={ // 销售相关岗位 '销售':'sales', '销售经理':'sales-manager', '销售总监':'sales-manager', // 教师相关岗位 '教师':'teacher', '讲师':'teacher', '教授':'teacher', // 其他岗位 '客服':'employee', '行政':'employee', '财务':'finance'}; // 优先根据岗位编码匹配 if(positionCode){if(positionCode.includes('SALES'))return'sales';if(positionCode.includes('TEACHER'))return'teacher';if(positionCode.includes('MANAGER'))return'sales-manager';if(positionCode.includes('POST-FM'))return'finance';} // 其次根据岗位名称匹配 return positionToTypeMap[positionName]||'employee';}

最终效果

财务人员访问首页,输入手机号

在这里插入图片描述


跳转到对应的工作台进行相关的操作

在这里插入图片描述

总结

本篇我们介绍了财务确认相关业务流程的搭建,创建了相关的表结构。在财务人员点击确认后,会自动开通学籍及课时卡信息,下一篇我们介绍班级管理版块,敬请期待。

Read more

【大模型科普】AIGC技术发展与应用实践(一文读懂AIGC)

【大模型科普】AIGC技术发展与应用实践(一文读懂AIGC)

【作者主页】Francek Chen 【专栏介绍】 ⌈ ⌈ ⌈人工智能与大模型应用 ⌋ ⌋ ⌋ 人工智能(AI)通过算法模拟人类智能,利用机器学习、深度学习等技术驱动医疗、金融等领域的智能化。大模型是千亿参数的深度神经网络(如ChatGPT),经海量数据训练后能完成文本生成、图像创作等复杂任务,显著提升效率,但面临算力消耗、数据偏见等挑战。当前正加速与教育、科研融合,未来需平衡技术创新与伦理风险,推动可持续发展。 文章目录 * 一、AIGC概述 * (一)什么是AIGC * (二)AIGC与大模型的关系 * (三)常见的AIGC应用场景 * (四)AIGC技术对行业发展的影响 * (五)AIGC技术对职业发展的影响 * (六)常见的AIGC大模型工具 * (七)AIGC大模型的提示词 * 二、文本类AIGC应用实践 * (一)案例1:与DeepSeek进行对话 * (二)案例2:与百度文心一言进行对话 * (三)案例3:使用讯飞智文生成PPT

夸克网盘免费资源电子书籍安卓软件经典游戏音乐歌曲精品教程AI绘画学习资料合集

夸克网盘免费资源电子书籍安卓软件经典游戏音乐歌曲精品教程AI绘画学习资料合集

一、夸克网盘免费资源说明 夸克网盘免费资源,来自全网整理二次精选,涵盖了几乎所有资源类型,网盘资源目录的分享链接,仅限一级目录和二级目录,一级目录是网盘资源的根目录,包括电子书籍、软件资源、游戏资源、视频资源、音乐音频、美食技术和学习资料等,二级目录是一级目录的子目录,均为资源专题形式,比如,Kindle原版书籍合集、U盘车载音乐歌曲、DeepSeek全套资源、全网专业摄影书籍、TikTok全球解锁版本、IOS巨魔专用资源、TED演讲视频合集、剪映教学全套资源、全网热门漫画精选,等等,相信其中会有你所需要的。 特别说明: 1、夸克网盘与百度网盘不同,不仅支持查看分享链接的资源大小,而且支持在分享链接页面里搜索资源,可以查询其中是否有你所需要的。 2、夸克官方一直都有福利活动,新用户可以免费领取1TB空间,具体操作方法请查看文本文件(在分享链接里)。 3、一级目录《全网精选2000T优质资料》,提供了很有价值的海量夸克资源,分享链接存放在电子表格里,整个目录大小只有9.7M,建议转存收藏。 二、夸克网盘一级目录资源 电子书籍+

2026 届毕业生必看:各大学位论文 AIGC 检测率要求汇总,超过这个数真的危险了!

2026 届毕业生必看:各大学位论文 AIGC 检测率要求汇总,超过这个数真的危险了!

一、 前言 随着 2026 届毕业季的临近,很多小伙伴在写论文时都离不开 AI 的辅助。但今年最让大家头疼的不再仅仅是查重率,而是新出的AIGC 疑似度。 很多学校已经明确:如果 AIGC 检测超过阈值,直接取消答辩资格! 今天我就帮大家梳理一下目前主流的检测要求,以及如何正确应对。 二、 各大高校 AIGC 检测率“红线”汇总 虽然各校标准不一,但根据目前各大高校反馈的最新政策,基本可以划分为三个梯度: 风险等级AIGC 疑似度范围学校处理建议安全区< 20%基本无风险,属于合理参考范围。预警区20% - 40%导师需进行人工核查,可能要求提供写作痕迹证据。高危区> 40%极大可能被判定为“代写”或“学术不端”,面临延毕风险。 注意: 部分顶尖院校(如 C9

GitHub Copilot:Python开发者的AI助手

GitHub Copilot:Python开发者的AI助手 前言 大家好,我是第一程序员(名字大,人很菜)。作为一个非科班转码、正在学习Rust和Python的萌新,最近我开始使用GitHub Copilot。今天我想分享一下GitHub Copilot如何成为Python开发者的AI助手。 一、GitHub Copilot简介 1.1 什么是GitHub Copilot * AI编程助手:由GitHub和OpenAI合作开发的AI编程助手 * 代码生成:根据上下文自动生成代码 * 智能建议:提供智能的代码建议 * 多语言支持:支持多种编程语言,包括Python 1.2 GitHub Copilot的工作原理 * 基于GPT模型:使用OpenAI的GPT模型 * 代码训练:在大量开源代码上训练 * 上下文理解:理解代码的上下文和意图 * 实时建议:在编写代码时实时提供建议 二、GitHub Copilot在Python开发中的应用 2.1 代码生成 示例1:生成函数