func-tbook-export.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. import { replaceDocx, docsToPdf, createZip, uploadFileToOSS } from "../../lib/docs";
  2. // const Parse = global.Parse;
  3. const path = require("path")
  4. const fs = require("fs")
  5. var TemplateDocxPath = path.join(__dirname,"template/模板-推荐申报表.docx")
  6. if(!fs.existsSync(TemplateDocxPath)){
  7. TemplateDocxPath = path.join(__dirname,"../../template/模板-推荐申报表.docx")
  8. }
  9. /**
  10. * 定义导出申报合集文档云函数
  11. * @example
  12. * Cloud Code test
  13. 导出流程
  14. curl -X POST -H "Content-Type: application/json" -H 'X-Parse-Application-Id: edu-textbook' -d '{ "processId": "Wz34loxdbO" }' http://127.0.0.1:61337/parse/functions/tbookExportReport
  15. curl -X POST -H "Content-Type: application/json" -H 'X-Parse-Application-Id: edu-textbook' -d '{ "processId": "Wz34loxdbO" }' http://8.140.98.43/parse/functions/tbookExportReport
  16. 导出教材列表
  17. curl -X POST -H "Content-Type: application/json" -H 'X-Parse-Application-Id: edu-textbook' -d '{ "bookList": ["2YBKitpCJL","xLdiEaHGrX"] }' http://127.0.0.1:61337/parse/functions/tbookExportReport
  18. */
  19. export function defineTbookExportReport(){
  20. Parse.Cloud.define("tbookExportReport", async (request) => {
  21. let processId = request.params.processId;
  22. let bookList = request.params.bookList;
  23. try{
  24. let result
  25. if(processId){
  26. await exportProcessReportDocs(processId)
  27. }
  28. if(bookList?.length){
  29. await exportProcessReportDocs(null,bookList)
  30. }
  31. if(result?.docsList?.length==0){
  32. throw new Parse.Error(404,"合集内无申报教材")
  33. }
  34. return result
  35. }catch(err){
  36. console.error(err)
  37. throw new Parse.Error(404,"导出申报合集失败")
  38. }
  39. throw new Parse.Error(404,"未找到该流程合集")
  40. },{
  41. fields : {
  42. processId:{
  43. required:false
  44. },
  45. }
  46. });
  47. }
  48. /**
  49. * 导出流程教材申报文件
  50. * @returns
  51. * docsList
  52. * zipUrl
  53. */
  54. export async function exportProcessReportDocs(processId,bookList) {
  55. if(!processId && !bookList?.length) return {}
  56. let textbookList
  57. if(processId){ // 流程读取教材列表
  58. let query = new Parse.Query("EduTextbook")
  59. query.equalTo("eduProcess",processId);
  60. textbookList = await query.find();
  61. }
  62. if(bookList?.length){ // 直接导出教材列表
  63. let query = new Parse.Query("EduTextbook")
  64. query.containedIn("objectId",bookList);
  65. textbookList = await query.find();
  66. }
  67. let docsList = []
  68. for (let index = 0; index < textbookList.length; index++) {
  69. let textbook = textbookList[index];
  70. let result = await renderReportDocsByTextbook(textbook);
  71. docsList.push(result)
  72. }
  73. let zipPath,zipUrl
  74. if(docsList?.length){
  75. let now = new Date();
  76. let zipName = `申报书导出-${now.getFullYear()}${now.getMonth()+1}${now.getDate()}-${now.getHours()}${now.getMinutes()}${now.getSeconds()}.zip`
  77. zipPath = await createZip(docsList?.map(item=>item?.filePath),zipName)
  78. if(zipPath){
  79. zipUrl = (await uploadFileToOSS(zipPath))?.url || null
  80. }
  81. docsList = docsList.map(item=>{return {code:item.code,title:item.title,url:item?.url}})
  82. }
  83. let result = {
  84. docsList,
  85. zipUrl
  86. }
  87. return result
  88. }
  89. module.exports.exportProcessReportDocs = exportProcessReportDocs
  90. function renderReportDocsByTextbook(textbook){
  91. let json = textbook.toJSON();
  92. console.log(json)
  93. // 圆圈选中未选 ○ 未选 ● 选中
  94. let circleCheck = ["○","●"];
  95. // 方块选中未选 ○ 未选 ● 选中
  96. let squareCheck = [``,`☑`];
  97. // 联系电话:默认为作者首个存在的电话;
  98. let mobile = json?.authorList?.find(item => item.mobile)?.mobile || ""
  99. // 填报时间:默认为创建时间
  100. let createdAt = new Date(textbook?.createdAt);
  101. let createdDate = `${createdAt?.getFullYear()}年${createdAt?.getMonth()+1}月${createdAt?.getDate()}日`;
  102. // 专业代码:前四位
  103. let majorCode = json?.majorId || json?.major?.code
  104. if(majorCode?.length>4){
  105. majorCode = majorCode.slice(0,4)
  106. }
  107. let majorName = json?.majorName || json?.major?.name
  108. // 是否重点立项
  109. let isJC = circleCheck[(json?.approval?.indexOf("基础")>-1)?1:0];
  110. let isZL = circleCheck[(json?.approval?.indexOf("战略")>-1)?1:0];
  111. let is101 = circleCheck[(json?.approval?.indexOf("101计划")>-1)?1:0]; // 2024新重点
  112. let isZY = circleCheck[(json?.approval?.indexOf("中央")>-1)?1:0];
  113. let isSX = circleCheck[(json?.approval?.indexOf("四新")>-1)?1:0];
  114. let isJS = circleCheck[(json?.importantProject?.indexOf("建设")>-1)?1:0];
  115. let isNotImpt = (json?.approval?.indexOf("101计划")==-1) && (json?.approval?.indexOf("中央")==-1) && (json?.approval?.indexOf("四新")==-1) && (json?.approval?.indexOf("建设")==-1)
  116. isNotImpt = circleCheck[isNotImpt?1:0];
  117. // 初版时间
  118. let firstDate = new Date(textbook?.get("editionFirst"));
  119. let firstYear = firstDate?.getFullYear();
  120. let firstMonth = firstDate?.getMonth()+1;
  121. // 本版时间印次
  122. let currentDate = new Date(textbook?.get("editionDate"));
  123. let currentYear = currentDate?.getFullYear();
  124. let currentMonth = currentDate?.getMonth()+1;
  125. // 最新时间印次
  126. let latestDate = new Date(textbook?.get("printDate"));
  127. let latestYear = latestDate?.getFullYear();
  128. let latestMonth = latestDate?.getMonth()+1;
  129. // 初版至今重点项目
  130. let isBSQT = !((json?.importantProject?.indexOf("建设")>-1) || (json?.importantProject?.indexOf("本科国家")>-1) || (json?.importantProject?.indexOf("省级优秀")>-1) || (json?.importantProject?.indexOf("省级规划")>-1))// 是否其他省级奖项
  131. let bookData = {
  132. // 封面信息
  133. titlePad:padString(json?.title,21),
  134. ISBN:padString(json?.ISBN,21),
  135. one:squareCheck[(json?.type=="单本"||json?.type=="单册")?1:0], // 单本/单册 方框
  136. full:squareCheck[json?.type=="全册"?1:0], // 全册
  137. oneCircle:circleCheck[(json?.type=="单本"||json?.type=="单册")?1:0], // 单本/单册 圆圈
  138. fullCircle:circleCheck[json?.type=="全册"?1:0], // 全册
  139. tn:json?.typeNumber,
  140. authorPad:padString(json?.author,21),
  141. mobile:padString(mobile,21),
  142. authorUnit:padString(json?.unit,21),
  143. publisherPad:padString(json?.editionUnit,21),
  144. recommandUnit:padString("",14), // 未找到
  145. majorCodePad:padString((majorCode),14),
  146. createdDate:padString(createdDate,21),
  147. // 基本信息
  148. title:json?.title,
  149. author:json?.author,
  150. unit:json?.unit,
  151. mc:majorCode,
  152. mn:majorName,
  153. lCN:circleCheck[(json?.lang=="中文")?1:0],
  154. lEN:circleCheck[(json?.lang=="英文")?1:0],
  155. lOT:circleCheck[(json?.lang?.indexOf("其他")>-1)?1:0],
  156. lSS:circleCheck[(json?.lang?.indexOf("少数")>-1)?1:0],
  157. authors:json?.authors, // 其他主编
  158. editor:json?.editor, // 其他编者
  159. isJC:isJC,
  160. isZL:isZL,
  161. isSX:isSX,
  162. is101:is101,
  163. isZY:isZY,
  164. isJS:isJS,
  165. isNotImpt:isNotImpt,
  166. publisher:json?.editionUnit,
  167. firstYear:firstYear,
  168. firstMonth:firstMonth,
  169. isZZ:circleCheck[(json?.carrierShape?.indexOf("纸质")>-1)?1:0],
  170. isDZ:circleCheck[(json?.carrierShape?.indexOf("电子")>-1)?1:0],
  171. isSZ:circleCheck[(json?.carrierShape?.indexOf("数字")>-1)?1:0],
  172. isQT:circleCheck[(json?.carrierShape?.indexOf("附带")>-1)?1:0],
  173. isFD:circleCheck[(json?.carrierShape?.indexOf("其他")>-1)?1:0],
  174. latestY:latestYear,
  175. latestM:latestMonth,
  176. latestNum:json?.printNumber || "",
  177. currentY:currentYear,
  178. currentM:currentMonth,
  179. currentNum:json?.editionNumber || "",
  180. printSum:json?.printSum || "",
  181. isBGJ:circleCheck[(json?.importantProject?.indexOf("本科国家")>-1)?1:0],
  182. isBSYX:circleCheck[(json?.importantProject?.indexOf("省级优秀")>-1)?1:0],
  183. isBSGH:circleCheck[(json?.importantProject?.indexOf("省级规划")>-1)?1:0],
  184. isBSQT:circleCheck[isBSQT?1:0],
  185. bsqtName:isBSQT?(json?.importantProject || ""):"",
  186. isFirstNot:circleCheck[json?.importantProject?0:1],
  187. }
  188. console.log(bookData)
  189. let bookid = json.code || json?.objectId;
  190. let tempFileName = path.join(`${bookid}${json.title}.docx`)
  191. return new Promise((resolve)=>{
  192. replaceDocx(TemplateDocxPath,tempFileName,bookData,{onDocxComplete:async (filePath)=>{
  193. // 需要API支持
  194. // docsToPdf(filePath)
  195. let url = (await uploadFileToOSS(filePath))?.url || null
  196. resolve({
  197. code:bookid,
  198. title:json?.title,
  199. filePath,
  200. url
  201. })
  202. }})
  203. })
  204. }
  205. function padString(str,width) {
  206. str = str || ""
  207. str = String(str)
  208. width = width || 21 // 目标宽度为21个单位
  209. let spaceChar = "&#160;" // 占位符
  210. // 计算字符串的宽度
  211. const charWidth = {
  212. 'space': 1, // 空格占用1个单位
  213. 'zh': 2, // 汉字占用2个单位
  214. 'en': 1, // 英文字母占用1个单位
  215. 'other': 1 // 其他字符(如标点符号)占用1个单位
  216. };
  217. let strWidth = 0;
  218. console.log(str)
  219. // 遍历文本中的每个字符
  220. for (let char of str) {
  221. if (/\s/.test(char)) {
  222. strWidth += charWidth.space; // 空格
  223. } else if (/[\u4e00-\u9fa5]/.test(char)) {
  224. strWidth += charWidth.zh; // 汉字
  225. } else if (/[a-zA-Z]/.test(char)) {
  226. strWidth += charWidth.en; // 英文字母
  227. } else {
  228. strWidth += charWidth.other; // 其他字符
  229. }
  230. }
  231. // for (let char of str) {
  232. // // 判断字符是否为中文
  233. // if (char.match(/[\u4e00-\u9fa5]/)) {
  234. // strWidth += 2; // 中文字符占4个单位
  235. // } else {
  236. // strWidth += 1; // 英文字符占1个单位
  237. // }
  238. // }
  239. const totalPadding = width - strWidth;
  240. // 如果已经达到或超过目标宽度,直接返回原字符串
  241. if (totalPadding <= 0) {
  242. return str;
  243. }
  244. // 计算左右两侧的空格数
  245. const leftPadding = Math.floor(totalPadding / 2) * 3;
  246. const rightPadding = Math.ceil(totalPadding / 2) * 3;
  247. // 生成填充空格的字符串
  248. const leftSpaces = spaceChar.repeat(leftPadding);
  249. const rightSpaces = spaceChar.repeat(rightPadding);
  250. // 返回补充后的字符串
  251. return leftSpaces + str + rightSpaces;
  252. }