questionnaire.js 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  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. console.log(_.head(studentList));
  85. // 再获取问卷
  86. let questAnswerList = await this.ctx.model.Uploadquestion.find({
  87. ...range,
  88. questionnaireid,
  89. });
  90. if (!questAnswerList) { throw new BusinessError(ErrorCode.DATA_NOT_EXIST, '未找到任何完成的问卷'); }
  91. questAnswerList = JSON.parse(JSON.stringify(questAnswerList));
  92. // fn,根据范围+问卷 得出文件名
  93. const fn = await this.toSetFileName(questionnaire.name, range);
  94. // 将期批班转换
  95. modelList = modelList.map(i => {
  96. const { model } = i;
  97. if (model === 'termid') i.model = 'termname';
  98. else if (model === 'batchid') i.model = 'batchname';
  99. else if (model === 'classid') i.model = 'classname';
  100. return i;
  101. });
  102. // TODO 整理导出数据
  103. let excelData;
  104. if (direction === 'horizontal') {
  105. excelData = this.horizontalSetData(studentList, questAnswerList, modelList);
  106. } else if (direction === 'vertical') {
  107. excelData = this.verticaSetData(studentList, questAnswerList, modelList);
  108. } else {
  109. throw new BusinessError(ErrorCode.DATA_NOT_EXIST, '未找到excel的表头排列方式');
  110. }
  111. if (excelData) {
  112. const { head, data } = excelData;
  113. return await this.ctx.service.util.toExcel(data, head, fn);
  114. }
  115. }
  116. /**
  117. * 获得导出文件的文件名
  118. * @param {String} questName 问卷名
  119. * @param {Object} range 范围(计划-期-批-班)
  120. * @return {String} fn 文件名
  121. */
  122. async toSetFileName(questName, range) {
  123. let fn = `-${questName}`;
  124. const { planid, termid, batchid, classid } = range;
  125. const getData = (termid, batchid, termnum) => {
  126. let res = '';
  127. if (termid) {
  128. const termInfo = termnum.id(termid);
  129. if (!termInfo) return res;
  130. const { term, batchnum } = termInfo;
  131. if (term) res = `第${term}期${res}`;
  132. if (batchid && batchnum) {
  133. const batchInfo = batchnum.id(batchid);
  134. if (!batchInfo) return res;
  135. const { batch } = batchInfo;
  136. res = `${res}第${batch}批`;
  137. }
  138. }
  139. return res;
  140. };
  141. if (classid) {
  142. const cla = await this.ctx.service.class.fetch({ id: classid });
  143. if (cla) {
  144. const { name, term } = cla;
  145. if (name) fn = `${name.includes('班') ? name : `${name}班`}${fn}`;
  146. if (term) fn = `第${term}期${fn}`;
  147. }
  148. } else {
  149. const condition = {};
  150. if (planid) condition._id = ObjectId(planid);
  151. if (termid) condition['termnum._id'] = ObjectId(termid);
  152. if (batchid) condition['termnum.batchnum._id'] = ObjectId(batchid);
  153. const trainPlan = await this.ctx.model.Trainplan.findOne(condition);
  154. const { termnum, title } = trainPlan;
  155. if (!termnum) return;
  156. if (termid || batchid) {
  157. const r = getData(termid, batchid, termnum);
  158. fn = `${r}${fn}`;
  159. } else {
  160. fn = `${title}${fn}`;
  161. }
  162. }
  163. return fn;
  164. }
  165. /**
  166. * 横向组织数据
  167. * TODO 里面整理方法可以提取出来和纵向一起用
  168. * @param {Array} studentList 学生列表
  169. * @param {Array} questAnswerList 学生回答问卷列表
  170. * @param {Array} modelList 需要导出的字段
  171. */
  172. horizontalSetData(studentList, questAnswerList, modelList) {
  173. // 第一步,组织excel表头
  174. const head = modelList.map(i => {
  175. const { zh, model, _id } = i;
  176. const headObj = { header: zh };
  177. if (model) headObj.key = model;
  178. else if (_id) headObj.key = _id;
  179. headObj.width = 20;
  180. return headObj;
  181. });
  182. // 第二步,组织数据需要按照modelList的顺序进行整理
  183. const data = [];
  184. for (const stu of studentList) {
  185. const obj = {};
  186. for (const m of modelList) {
  187. const { model, table, _id } = m;
  188. if (table === 'student') {
  189. const prop = _.get(stu, model, '');
  190. obj[model] = prop;
  191. } else if (table === 'questionnaire') {
  192. const qar = questAnswerList.find(f => f.studentid === stu._id);
  193. // 没找到该学生的问卷,下一个
  194. if (!qar) continue;
  195. const { answers } = qar;
  196. // 回答的数据不存在/不是数组,数据格式不对,下一个
  197. if (!(answers && _.isArray(answers))) continue;
  198. const ar = answers.find(f => f.questionid === _id);
  199. // 没找到答案,下个
  200. if (!ar) continue;
  201. const { answer } = ar;
  202. // 没有答案,下个
  203. if (!answer) continue;
  204. obj[_id] = _.trim(answer, '"');
  205. }
  206. }
  207. data.push(obj);
  208. }
  209. const obj = {};
  210. if (head)obj.head = head;
  211. if (data) obj.data = data;
  212. return obj;
  213. }
  214. /**
  215. * 纵向组织数据
  216. * TODO 里面整理方法可以提取出来和横向一起用
  217. * @param {Array} studentList 学生列表
  218. * @param {Array} questAnswerList 学生回答问卷列表
  219. * @param {Array} modelList 需要导出的字段
  220. */
  221. verticaSetData(studentList, questAnswerList, modelList) {
  222. // 第一步,组织excel表头,纵向不需要第一行表头,开始就是数据,表头是第一列
  223. const head = [{ key: 'c', width: 20 }];
  224. for (let i = 1; i <= studentList.length; i++) {
  225. head.push({ key: `c${i}`, width: 20 });
  226. }
  227. // 第二部,整理数据
  228. const data = [];
  229. for (const m of modelList) {
  230. const { table, model, _id, zh } = m;
  231. const obj = { c: zh };
  232. if (table === 'student') {
  233. for (let i = 0; i < studentList.length; i++) {
  234. const stu = studentList[i];
  235. const value = _.get(stu, model, '');
  236. if (value) obj[`c${i + 1}`] = value;
  237. }
  238. } else if (table === 'questionnaire') {
  239. for (let i = 0; i < studentList.length; i++) {
  240. const stu = studentList[i];
  241. const qar = questAnswerList.find(f => f.studentid === stu._id);
  242. // 没找到该学生的问卷,下一个
  243. if (!qar) continue;
  244. const { answers } = qar;
  245. // 回答的数据不存在/不是数组,数据格式不对,下一个
  246. if (!(answers && _.isArray(answers))) continue;
  247. const ar = answers.find(f => f.questionid === _id);
  248. // 没找到答案,下个
  249. if (!ar) continue;
  250. const { answer } = ar;
  251. // 没有答案,下个
  252. if (!answer) continue;
  253. obj[`c${i + 1}`] = _.trim(answer, '"');
  254. }
  255. }
  256. data.push(obj);
  257. }
  258. const obj = {};
  259. if (head) obj.head = head;
  260. if (data) obj.data = data;
  261. return obj;
  262. }
  263. }
  264. module.exports = QuestionnaireService;