questionnaire.js 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276
  1. 'use strict';
  2. const assert = require('assert');
  3. const _ = require('lodash');
  4. const { ObjectId } = require('mongoose').Types;
  5. const { CrudService } = require('naf-framework-mongoose/lib/service');
  6. const { BusinessError, ErrorCode } = require('naf-core').Error;
  7. class QuestionnaireService extends CrudService {
  8. constructor(ctx) {
  9. super(ctx, 'qusetionnaire');
  10. this.questionnairemodel = this.ctx.model.Questionnaire;
  11. this.questionmodel = this.ctx.model.Question;
  12. }
  13. // 插入问卷
  14. async create(data) {
  15. const { name, num, type, question } = data;
  16. assert(name, '问卷名称不能为空');
  17. assert(num, '问卷序号不能为空');
  18. assert(type, '问卷类型不能为空');
  19. const quedata = [];
  20. for (const elm of question) {
  21. const ques = await this.questionmodel.findById(elm);
  22. if (ques) {
  23. quedata.push(ques);
  24. }
  25. }
  26. data.question = quedata;
  27. return await this.questionnairemodel.create(data);
  28. }
  29. // 根据id删除问卷
  30. async delete({ id }) {
  31. await this.questionnairemodel.findByIdAndDelete(id);
  32. return 'deleted';
  33. }
  34. // 根据id更新问卷信息
  35. async update({ id }, data) {
  36. const { name, num, type, question } = data;
  37. const questionnaire = await this.questionnairemodel.findById(id);
  38. if (name) {
  39. questionnaire.name = name;
  40. }
  41. if (num) {
  42. questionnaire.num = num;
  43. }
  44. if (type) {
  45. questionnaire.type = type;
  46. }
  47. if (question) {
  48. questionnaire.question = question;
  49. }
  50. return await questionnaire.save();
  51. }
  52. // 查询
  53. async query({ skip, limit, ...num }) {
  54. const total = await this.questionnairemodel.count(num);
  55. const data = await this.questionnairemodel
  56. .find(num)
  57. .skip(Number(skip))
  58. .limit(Number(limit));
  59. const result = { total, data };
  60. return result;
  61. }
  62. // 查询详情
  63. async show({ id }) {
  64. const questionnaire = await this.questionnairemodel.findById(id);
  65. return questionnaire;
  66. }
  67. /**
  68. * 导出问卷
  69. * @param {Object} Object 数据集合
  70. * @property Object range 学生的查询范围
  71. * @property String direction horizontal横向/vertical纵向
  72. * @property String questionnaireid 问卷id
  73. * @property Array modelList 要导出的字段,学生和问卷都在这里, 学生的table:student, 问卷的table:questionnaire
  74. */
  75. async export({ range, direction, questionnaireid, modelList }) {
  76. // 获取问卷
  77. let questionnaire = await this.questionnairemodel.findById(questionnaireid);
  78. if (!questionnaire) { throw new BusinessError(ErrorCode.DATA_NOT_EXIST, '未找到问卷信息'); }
  79. questionnaire = JSON.parse(JSON.stringify(questionnaire));
  80. // 获取学生
  81. let { data: studentList } = await this.ctx.service.student.query(range);
  82. if (studentList.length <= 0) { throw new BusinessError(ErrorCode.DATA_NOT_EXIST, '未找到任何学生信息'); }
  83. studentList = JSON.parse(JSON.stringify(studentList));
  84. // 再获取问卷
  85. let questAnswerList = await this.ctx.model.Uploadquestion.find({
  86. ...range,
  87. questionnaireid,
  88. });
  89. if (!questAnswerList) { throw new BusinessError(ErrorCode.DATA_NOT_EXIST, '未找到任何完成的问卷'); }
  90. questAnswerList = JSON.parse(JSON.stringify(questAnswerList));
  91. // fn,根据范围+问卷 得出文件名
  92. const fn = await this.toSetFileName(questionnaire.name, range);
  93. // 将期批班转换
  94. modelList = modelList.map(i => {
  95. const { model } = i;
  96. if (model === 'termid') i.model = 'termname';
  97. else if (model === 'batchid') i.model = 'batchname';
  98. else if (model === 'classid') i.model = 'classname';
  99. return i;
  100. });
  101. let excelData;
  102. if (direction === 'horizontal') {
  103. excelData = this.horizontalSetData(studentList, questAnswerList, modelList);
  104. } else if (direction === 'vertical') {
  105. excelData = this.verticaSetData(studentList, questAnswerList, modelList);
  106. } else {
  107. throw new BusinessError(ErrorCode.DATA_NOT_EXIST, '未找到excel的表头排列方式');
  108. }
  109. if (excelData) {
  110. const { head, data } = excelData;
  111. return await this.ctx.service.util.toExcel(data, head, fn);
  112. }
  113. }
  114. /**
  115. * 获得导出文件的文件名
  116. * @param {String} questName 问卷名
  117. * @param {Object} range 范围(计划-期-批-班)
  118. * @return {String} fn 文件名
  119. */
  120. async toSetFileName(questName, range) {
  121. let fn = `-${questName}`;
  122. const { planid, termid, batchid, classid } = range;
  123. const getData = (termid, batchid, termnum) => {
  124. let res = '';
  125. if (termid) {
  126. const termInfo = termnum.id(termid);
  127. if (!termInfo) return res;
  128. const { term, batchnum } = termInfo;
  129. if (term) res = `第${term}期${res}`;
  130. if (batchid && batchnum) {
  131. const batchInfo = batchnum.id(batchid);
  132. if (!batchInfo) return res;
  133. const { batch } = batchInfo;
  134. res = `${res}第${batch}批`;
  135. }
  136. }
  137. return res;
  138. };
  139. if (classid) {
  140. const cla = await this.ctx.service.class.fetch({ id: classid });
  141. if (cla) {
  142. const { name, term } = cla;
  143. if (name) fn = `${name.includes('班') ? name : `${name}班`}${fn}`;
  144. if (term) fn = `第${term}期${fn}`;
  145. }
  146. } else {
  147. const condition = {};
  148. if (planid) condition._id = ObjectId(planid);
  149. if (termid) condition['termnum._id'] = ObjectId(termid);
  150. if (batchid) condition['termnum.batchnum._id'] = ObjectId(batchid);
  151. const trainPlan = await this.ctx.model.Trainplan.findOne(condition);
  152. const { termnum, title } = trainPlan;
  153. if (!termnum) return;
  154. if (termid || batchid) {
  155. const r = getData(termid, batchid, termnum);
  156. fn = `${r}${fn}`;
  157. } else {
  158. fn = `${title}${fn}`;
  159. }
  160. }
  161. return fn;
  162. }
  163. /**
  164. * 横向组织数据
  165. * TODO 里面整理方法可以提取出来和纵向一起用
  166. * @param {Array} studentList 学生列表
  167. * @param {Array} questAnswerList 学生回答问卷列表
  168. * @param {Array} modelList 需要导出的字段
  169. */
  170. horizontalSetData(studentList, questAnswerList, modelList) {
  171. // 第一步,组织excel表头
  172. const head = modelList.map(i => {
  173. const { zh, model, _id } = i;
  174. const headObj = { header: zh };
  175. if (model) headObj.key = model;
  176. else if (_id) headObj.key = _id;
  177. headObj.width = 20;
  178. return headObj;
  179. });
  180. // 第二步,组织数据需要按照modelList的顺序进行整理
  181. const data = [];
  182. for (const stu of studentList) {
  183. const obj = {};
  184. for (const m of modelList) {
  185. const { model, table, _id } = m;
  186. if (table === 'student') {
  187. const prop = _.get(stu, model, '');
  188. obj[model] = prop;
  189. } else if (table === 'questionnaire') {
  190. const qar = questAnswerList.find(f => f.studentid === stu._id);
  191. // 没找到该学生的问卷,下一个
  192. if (!qar) continue;
  193. const { answers } = qar;
  194. // 回答的数据不存在/不是数组,数据格式不对,下一个
  195. if (!(answers && _.isArray(answers))) continue;
  196. const ar = answers.find(f => f.questionid === _id);
  197. // 没找到答案,下个
  198. if (!ar) continue;
  199. const { answer } = ar;
  200. // 没有答案,下个
  201. if (!answer) continue;
  202. obj[_id] = _.trim(answer, '"');
  203. }
  204. }
  205. data.push(obj);
  206. }
  207. const obj = {};
  208. if (head)obj.head = head;
  209. if (data) obj.data = data;
  210. return obj;
  211. }
  212. /**
  213. * 纵向组织数据
  214. * TODO 里面整理方法可以提取出来和横向一起用
  215. * @param {Array} studentList 学生列表
  216. * @param {Array} questAnswerList 学生回答问卷列表
  217. * @param {Array} modelList 需要导出的字段
  218. */
  219. verticaSetData(studentList, questAnswerList, modelList) {
  220. // 第一步,组织excel表头,纵向不需要第一行表头,开始就是数据,表头是第一列
  221. const head = [{ key: 'c', width: 20 }];
  222. for (let i = 1; i <= studentList.length; i++) {
  223. head.push({ key: `c${i}`, width: 20 });
  224. }
  225. // 第二部,整理数据
  226. const data = [];
  227. for (const m of modelList) {
  228. const { table, model, _id, zh } = m;
  229. const obj = { c: zh };
  230. if (table === 'student') {
  231. for (let i = 0; i < studentList.length; i++) {
  232. const stu = studentList[i];
  233. const value = _.get(stu, model, '');
  234. if (value) obj[`c${i + 1}`] = value;
  235. }
  236. } else if (table === 'questionnaire') {
  237. for (let i = 0; i < studentList.length; i++) {
  238. const stu = studentList[i];
  239. const qar = questAnswerList.find(f => f.studentid === stu._id);
  240. // 没找到该学生的问卷,下一个
  241. if (!qar) continue;
  242. const { answers } = qar;
  243. // 回答的数据不存在/不是数组,数据格式不对,下一个
  244. if (!(answers && _.isArray(answers))) continue;
  245. const ar = answers.find(f => f.questionid === _id);
  246. // 没找到答案,下个
  247. if (!ar) continue;
  248. const { answer } = ar;
  249. // 没有答案,下个
  250. if (!answer) continue;
  251. obj[`c${i + 1}`] = _.trim(answer, '"');
  252. }
  253. }
  254. data.push(obj);
  255. }
  256. const obj = {};
  257. if (head) obj.head = head;
  258. if (data) obj.data = data;
  259. return obj;
  260. }
  261. }
  262. module.exports = QuestionnaireService;