开发背景
随着全民健身的遍及,为进一步贯彻落实《“衰弱中国2030”布局大纲》和《全民健身条例》,更好地满足宽广青少年学生和人民大众就近、便当加入体育健身活动的需要,继续扩充公共服务配套资源供应, 中小学校园的操场,静止设施周也开始对市民凋谢,那么为了正当工夫,不便各项设施的无效利用, 缩小不必要的人员汇集, 预订小程序提供了线上预约的便捷性
性能概要设计
技术选型
- 本我的项目应用微信小程序平台进行开发。
- 应用腾讯专门的小程序云开发技术,云资源蕴含云函数,数据库,带宽,存储空间,定时器等,资源配额价格低廉,无需域名和服务器即可搭建。
- 小程序自身的即用即走,适宜小工具的应用场景,也适宜疾速开发迭代。
- 云开发技术采纳腾讯外部链路,没有被黑客攻击的危险,不会 DDOS攻打,节俭防火墙费用,安全性高且免保护。
- 资源承载力可依据业务倒退须要随时弹性扩大。
数据库设计
EnrollJoinModel.DB_STRUCTURE = { _pid: 'string|true', ENROLL_JOIN_ID: 'string|true', ENROLL_JOIN_PRICE: 'int|true|default=0', ENROLL_JOIN_IS_ADMIN: 'int|true|default=0|comment=是否管理员增加 0/1', ENROLL_JOIN_ENROLL_ID: 'string|true|comment=报名PK', ENROLL_JOIN_ENROLL_TITLE: 'string|false', ENROLL_JOIN_CATE_ID: 'string|false|default=0|comment=分类', ENROLL_JOIN_CATE_NAME: 'string|false|comment=分类冗余', ENROLL_JOIN_CODE: 'string|true|comment=核验码15位', ENROLL_JOIN_IS_CHECKIN: 'int|true|default=0|comment=是否核销 0/1 ', ENROLL_JOIN_CHECKIN_TIME: 'int|true|default=0', ENROLL_JOIN_DAY: 'string|false|comment=日期', ENROLL_JOIN_START: 'string|false|comment=开始工夫', ENROLL_JOIN_END: 'string|false|comment=完结工夫', ENROLL_JOIN_END_POINT: 'string|false|comment=完结工夫开端', ENROLL_JOIN_END_FULL: 'string|false|comment=残缺的完结工夫 YYYY-MM-DD hh:mm', ENROLL_JOIN_START_FULL: 'string|false|comment=残缺的开始工夫 YYYY-MM-DD hh:mm', ENROLL_JOIN_USER_ID: 'string|true|comment=用户ID', ENROLL_JOIN_FORMS: 'array|true|default=[]|comment=表单', ENROLL_JOIN_OBJ: 'object|true|default={}', ENROLL_JOIN_STATUS: 'int|true|default=1|comment=状态 0=待审核 1=胜利, 9=用户勾销, 99=零碎勾销', ENROLL_JOIN_LAST_TIME: 'int|true|default=0', ENROLL_JOIN_ADD_TIME: 'int|true', ENROLL_JOIN_EDIT_TIME: 'int|true', ENROLL_JOIN_ADD_IP: 'string|false', ENROLL_JOIN_EDIT_IP: 'string|false',};// 字段前缀EnrollJoinModel.FIELD_PREFIX = "ENROLL_JOIN_";/** * 状态 0=待审核 1=胜利,9=用户勾销, 99=审核未过 */EnrollJoinModel.STATUS = { WAIT: 0, SUCC: 1, CANCEL: 9, ADMIN_CANCEL: 99};EnrollJoinModel.STATUS_DESC = { WAIT: '待审核', SUCC: '胜利', CANCEL: '用户勾销', ADMIN_CANCEL: '零碎勾销'};EnrollModel.DB_STRUCTURE = { _pid: 'string|true', ENROLL_ID: 'string|true', ENROLL_TITLE: 'string|true|comment=题目', ENROLL_STATUS: 'int|true|default=1|comment=状态 0=未启用,1=应用中', ENROLL_CATE_ID: 'string|true|default=0|comment=分类', ENROLL_CATE_NAME: 'string|false|comment=分类冗余', ENROLL_CANCEL_SET: 'int|true|default=1|comment=勾销设置 0=不允,1=容许,2=开始前可勾销,3=完结前可勾销', ENROLL_EDIT_SET: 'int|true|default=1|comment=批改 0=不允,1=容许,2=开始前可批改,3=完结前可批改', ENROLL_ORDER: 'int|true|default=9999', ENROLL_VOUCH: 'int|true|default=0', ENROLL_FORMS: 'array|true|default=[]', ENROLL_OBJ: 'object|true|default={}', ENROLL_JOIN_FORMS: 'array|true|default=[]', ENROLL_DAYS: 'array|true|default=[]|comment=最近一次批改保留的可用日期', ENROLL_DAY_CNT: 'int|true|default=0', ENROLL_QR: 'string|false', ENROLL_VIEW_CNT: 'int|true|default=0', ENROLL_JOIN_CNT: 'int|true|default=0', ENROLL_ADD_TIME: 'int|true', ENROLL_EDIT_TIME: 'int|true', ENROLL_ADD_IP: 'string|false', ENROLL_EDIT_IP: 'string|false',};
要害难点
// 获取某天某个场合下的可约工夫点 async getOneDayTimePoint(day, enrollId) { let where = { DAY_ENROLL_ID: enrollId, day }; let fields = 'times'; let data = await DayModel.getOne(where, fields); if (!data) data = []; else data = data.times; return data; } // 获得某天内所有场地信息 async getAllEnroll(cateId, day) { let where = { ENROLL_CATE_ID: String(cateId), ENROLL_STATUS: EnrollModel.STATUS.COMM } let orderBy = { ENROLL_ORDER: 'asc', ENROLL_ADD_TIME: 'asc' } let list = await EnrollModel.getAll(where, '*', orderBy); let arr = []; let startTime = 23; let endTime = 0; for (let k = 0; k < list.length; k++) { let times = await this.getOneDayTimePoint(day, list[k]._id); // 合成小时 let t = []; for (let j = 0; j < times.length; j++) { if (times[j].start < startTime) startTime = times[j].start; if (times[j].end > endTime) endTime = times[j].end; for (let i = times[j].start; i <= times[j].end; i++) { let node = { t: i, //工夫点 price: times[j].price, //价格 }; t.push(node); } } if (t.length > 0) arr.push({ enrollId: list[k]._id, label: list[k].ENROLL_TITLE, timePrice: t }) } // 获得可预订的最大日期 let maxDay = await DayModel.max({ day: ['>=', day], DAY_CATE_ID: cateId }, 'day'); if (maxDay == 0) maxDay = ''; return { maxDay, startTime, endTime, list: arr }; } // 获取某天预订状况 async getUsedByDay(cateId, day) { let where = { ENROLL_JOIN_CATE_ID: String(cateId), ENROLL_JOIN_DAY: day, ENROLL_JOIN_STATUS: ['in', [EnrollJoinModel.STATUS.WAIT, EnrollJoinModel.STATUS.SUCC]], }; return EnrollJoinModel.getAll(where); } /** 获得我的注销分页列表 */ async getMyEnrollJoinList(userId, { search, // 搜寻条件 sortType, // 搜寻菜单 sortVal, // 搜寻菜单 orderBy, // 排序 page, size, isTotal = true, oldTotal }) { orderBy = orderBy || { 'ENROLL_JOIN_ADD_TIME': 'desc' }; let fields = 'ENROLL_JOIN_IS_CHECKIN,ENROLL_JOIN_CATE_NAME,ENROLL_JOIN_ENROL_TITLE,ENROLL_JOIN_PRICE,ENROLL_JOIN_END_FULL,ENROLL_JOIN_OBJ,ENROLL_JOIN_DAY,ENROLL_JOIN_START,ENROLL_JOIN_END,ENROLL_JOIN_END_POINT,ENROLL_JOIN_LAST_TIME,ENROLL_JOIN_ENROLL_ID,ENROLL_JOIN_STATUS,ENROLL_JOIN_ADD_TIME,enroll.ENROLL_TITLE,enroll.ENROLL_EDIT_SET,enroll.ENROLL_CANCEL_SET'; let where = { ENROLL_JOIN_USER_ID: userId }; if (util.isDefined(search) && search) { where['ENROLL_JOIN_OBJ.name'] = { $regex: '.*' + search, $options: 'i' }; } else if (sortType) { // 搜寻菜单 switch (sortType) { case 'timedesc': { //按工夫倒序 orderBy = { 'ENROLL_JOIN_START_FULL': 'desc' }; break; } case 'timeasc': { //按工夫正序 orderBy = { 'ENROLL_JOIN_START_FULL': 'asc' }; break; } case 'status': { break; } case 'today': { where.ENROLL_JOIN_DAY = timeUtil.time('Y-M-D'); where.ENROLL_JOIN_STATUS = EnrollJoinModel.STATUS.SUCC; break; } case 'run': { where.ENROLL_JOIN_END_FULL = ['>', timeUtil.time('Y-M-D h:m')]; where.ENROLL_JOIN_STATUS = EnrollJoinModel.STATUS.SUCC; where.ENROLL_JOIN_IS_CHECKIN = 0; break; } case 'check': { where.ENROLL_JOIN_END_FULL = ['>', timeUtil.time('Y-M-D h:m')]; where.ENROLL_JOIN_STATUS = EnrollJoinModel.STATUS.SUCC; where.ENROLL_JOIN_IS_CHECKIN = 1; break; } case 'out': { where.ENROLL_JOIN_END_FULL = ['<=', timeUtil.time('Y-M-D h:m')]; where.ENROLL_JOIN_STATUS = EnrollJoinModel.STATUS.SUCC; break; } case 'cancel': { where.ENROLL_JOIN_STATUS = EnrollJoinModel.STATUS.CANCEL; break; } case 'syscancel': { where.ENROLL_JOIN_STATUS = EnrollJoinModel.STATUS.ADMIN_CANCEL; break; } } } let joinParams = { from: EnrollModel.CL, localField: 'ENROLL_JOIN_ENROLL_ID', foreignField: '_id', as: 'enroll', }; let result = await EnrollJoinModel.getListJoin(joinParams, where, fields, orderBy, page, size, isTotal, oldTotal); return result; } /** 获得我的注销详情 */ async getMyEnrollJoinDetail(enrollJoinId) { let fields = '*'; let where = { _id: enrollJoinId }; let enrollJoin = await EnrollJoinModel.getOne(where, fields); if (enrollJoin) { enrollJoin.enroll = await EnrollModel.getOne(enrollJoin.ENROLL_JOIN_ENROLL_ID, 'ENROLL_TITLE'); } return enrollJoin; } //################## 注销 // 把工夫格局'hh:mm'转为数组[1,2,3,4] getTimeArr(start, end) { start = start.replace(':00', '').trim(); start = start.replace(':30', '').trim(); start = Number(start); end = end.replace(':00', ''); end = end.replace(':30', '').trim(); end = Number(end); let ret = []; for (let k = start; k <= end; k++) { ret.push(k); } return ret; } // 注销 async enrollJoin(userId, { enrollId, price, start, end, endPoint, day, forms }) { // 注销是否完结 let whereEnroll = { _id: enrollId, ENROLL_STATUS: EnrollModel.STATUS.COMM } let enroll = await EnrollModel.getOne(whereEnroll); if (!enroll) this.AppError('该' + ENROLL_NAME + '不存在或者曾经进行'); // 判断是否曾经被约(数组交加) let nowTimeArr = this.getTimeArr(start, end); let joinWhere = { ENROLL_JOIN_ENROLL_ID: enrollId, ENROLL_JOIN_DAY: day, ENROLL_JOIN_STATUS: ['in', [EnrollJoinModel.STATUS.WAIT, EnrollJoinModel.STATUS.SUCC]], } let joinList = await EnrollJoinModel.getAll(joinWhere, 'ENROLL_JOIN_START,ENROLL_JOIN_END', { 'ENROLL_JOIN_START': 'asc' }); for (let k = 0; k < joinList.length; k++) { let listTimeArr = this.getTimeArr(joinList[k].ENROLL_JOIN_START, joinList[k].ENROLL_JOIN_END); for (let j = 0; j < nowTimeArr.length; j++) { if (listTimeArr.includes(nowTimeArr[j])) { this.AppError(nowTimeArr[j] + '点曾经被预订,请从新抉择'); } } } // 入库 let data = { ENROLL_JOIN_USER_ID: userId, ENROLL_JOIN_ENROLL_ID: enrollId, ENROLL_JOIN_CATE_ID: enroll.ENROLL_CATE_ID, ENROLL_JOIN_CATE_NAME: enroll.ENROLL_CATE_NAME, ENROLL_JOIN_CODE: dataUtil.genRandomIntString(15), ENROLL_JOIN_PRICE: price, ENROLL_JOIN_START: start, ENROLL_JOIN_END: end, ENROLL_JOIN_END_POINT: endPoint, ENROLL_JOIN_DAY: day, ENROLL_JOIN_ENROLL_TITLE: enroll.ENROLL_TITLE, ENROLL_JOIN_END_FULL: day + ' ' + endPoint, ENROLL_JOIN_START_FULL: day + ' ' + start, ENROLL_JOIN_FORMS: forms, ENROLL_JOIN_OBJ: dataUtil.dbForms2Obj(forms), } let enrollJoinId = await EnrollJoinModel.insert(data); // 统计数量 this.statEnrollJoin(enrollId); return { enrollJoinId } } // 批改注销 async enrollJoinEdit(userId, enrollId, enrollJoinId, forms) { let whereJoin = { _id: enrollJoinId, ENROLL_JOIN_USER_ID: userId, ENROLL_JOIN_ENROLL_ID: enrollId, ENROLL_JOIN_STATUS: ['in', [EnrollJoinModel.STATUS.WAIT, EnrollJoinModel.STATUS.SUCC]], } let enrollJoin = await EnrollJoinModel.getOne(whereJoin); if (!enrollJoin) this.AppError('该' + ENROLL_NAME + '记录不存在或者曾经被零碎勾销'); // 注销是否完结 let whereEnroll = { _id: enrollId, ENROLL_STATUS: EnrollModel.STATUS.COMM } let enroll = await EnrollModel.getOne(whereEnroll); if (!enroll) this.AppError('该' + ENROLL_NAME + '不存在或者曾经进行'); if (enrollJoin.ENROLL_JOIN_IS_CHECKIN == 1) this.AppError('该预订已核销,不能批改'); if (enroll.ENROLL_EDIT_SET == 0) this.AppError('该' + ENROLL_NAME + '不容许批改材料'); if (enroll.ENROLL_EDIT_SET == 2 && enrollJoin.ENROLL_JOIN_START_FULL <= timeUtil.time('Y-M-D h:m')) this.AppError('该' + ENROLL_NAME + '曾经开始,不能批改材料'); if (enroll.ENROLL_EDIT_SET == 3 && enrollJoin.ENROLL_JOIN_END_FULL <= timeUtil.time('Y-M-D h:m')) this.AppError('该' + ENROLL_NAME + '曾经完结,不能批改材料'); let data = { ENROLL_JOIN_FORMS: forms, ENROLL_JOIN_OBJ: dataUtil.dbForms2Obj(forms), ENROLL_JOIN_LAST_TIME: this._timestamp, } await EnrollJoinModel.edit(whereJoin, data); } async statEnrollJoin(id) { let where = { ENROLL_JOIN_ENROLL_ID: id, ENROLL_JOIN_STATUS: ['in', [EnrollJoinModel.STATUS.WAIT, EnrollJoinModel.STATUS.SUCC]] } let cnt = await EnrollJoinModel.count(where); await EnrollModel.edit(id, { ENROLL_JOIN_CNT: cnt }); } /** 注销前获取要害信息 */ async detailForEnrollJoin(userId, enrollId, enrollJoinId = '') { let fields = 'ENROLL_JOIN_FORMS, ENROLL_TITLE, ENROLL_CATE_NAME'; let where = { _id: enrollId, ENROLL_STATUS: EnrollModel.STATUS.COMM } let enroll = await EnrollModel.getOne(where, fields); if (!enroll) this.AppError('该' + ENROLL_NAME + '不存在'); let joinMy = null; if (enrollJoinId) { // 编辑 let whereMy = { ENROLL_JOIN_USER_ID: userId, _id: enrollJoinId } joinMy = await EnrollJoinModel.getOne(whereMy); enroll.join = { start: joinMy.ENROLL_JOIN_START, end: joinMy.ENROLL_JOIN_END, endPoint: joinMy.ENROLL_JOIN_END_POINT, day: joinMy.ENROLL_JOIN_DAY, } } else { // 取出自己最近一次的填写表单 let whereMy = { ENROLL_JOIN_USER_ID: userId, } let orderByMy = { ENROLL_JOIN_ADD_TIME: 'desc' } joinMy = await EnrollJoinModel.getOne(whereMy, 'ENROLL_JOIN_FORMS', orderByMy); } let myForms = joinMy ? joinMy.ENROLL_JOIN_FORMS : []; enroll.myForms = myForms; return enroll; } /** 勾销我的注销 */ async cancelMyEnrollJoin(userId, enrollJoinId) { let where = { ENROLL_JOIN_USER_ID: userId, _id: enrollJoinId, ENROLL_JOIN_STATUS: ['in', [EnrollJoinModel.STATUS.WAIT, EnrollJoinModel.STATUS.SUCC]] }; let enrollJoin = await EnrollJoinModel.getOne(where); if (!enrollJoin) { this.AppError('未找到可勾销的记录'); } if (enrollJoin.ENROLL_JOIN_IS_CHECKIN == 1) this.AppError('该预订已核销,不能取消'); let enroll = await EnrollModel.getOne(enrollJoin.ENROLL_JOIN_ENROLL_ID); if (!enroll) this.AppError('该' + ENROLL_NAME + '不存在'); if (enroll.ENROLL_CANCEL_SET == 0) this.AppError('该' + ENROLL_NAME + '不能取消'); if (enroll.ENROLL_CANCEL_SET == 2 && enrollJoin.ENROLL_JOIN_START_FULL <= timeUtil.time('Y-M-D h:m')) this.AppError('该' + ENROLL_NAME + '曾经开始,不能取消'); if (enroll.ENROLL_CANCEL_SET == 3 && enrollJoin.ENROLL_JOIN_END_FULL <= timeUtil.time('Y-M-D h:m')) this.AppError('该' + ENROLL_NAME + '曾经完结,不能取消'); if (enroll.ENROLL_CANCEL_SET > 20) { let step = enroll.ENROLL_CANCEL_SET - 20; let day = timeUtil.time2Timestamp(enrollJoin.ENROLL_JOIN_END_FULL + ':00') - step * 86400 * 1000; day = timeUtil.timestamp2Time(day, 'Y-M-D'); let now = timeUtil.time('Y-M-D'); if (now > day) this.AppError('仅开始前' + step + '天可勾销'); } await EnrollJoinModel.edit(where, { ENROLL_JOIN_STATUS: EnrollJoinModel.STATUS.CANCEL, ENROLL_JOIN_IS_CHECKIN: 0 }); await this.statEnrollJoin(enrollJoin.ENROLL_JOIN_ENROLL_ID); }
前端UI设计
后盾管理系统UI设计
git源码
git源码下载