CardFreeDesign.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536
  1. <template>
  2. <div class="card-design card-free-design">
  3. <div class="design-header">
  4. <div class="design-header-cont box-justify">
  5. <div></div>
  6. <el-button
  7. class="btn-help"
  8. icon="el-icon-question"
  9. type="text"
  10. @click="showHelp"
  11. ></el-button>
  12. </div>
  13. </div>
  14. <!-- actions -->
  15. <div class="design-action">
  16. <div class="design-logo">
  17. <h1>
  18. <i class="el-icon-d-arrow-left" @click="toExit" title="退出"></i>
  19. 答题卡制作
  20. </h1>
  21. </div>
  22. <div class="action-part">
  23. <div class="action-part-title"><h2>基本设置</h2></div>
  24. <div class="action-part-body">
  25. <page-prop-edit></page-prop-edit>
  26. </div>
  27. </div>
  28. <div class="action-part">
  29. <div class="action-part-title"><h2>当前页面设置</h2></div>
  30. <div class="action-part-body">
  31. <edit-page></edit-page>
  32. </div>
  33. </div>
  34. <div class="action-part">
  35. <div class="action-part-title"><h2>复杂元素</h2></div>
  36. <div class="action-part-body">
  37. <div class="type-list">
  38. <div
  39. class="type-item"
  40. v-for="(item, index) in TOPIC_LIST"
  41. :key="index"
  42. draggable="true"
  43. @dragstart="dragstart(item)"
  44. >
  45. <el-button><i class="el-icon-plus"></i>{{ item.name }}</el-button>
  46. </div>
  47. </div>
  48. <p class="tips-info">提示:拖动插入元素</p>
  49. </div>
  50. </div>
  51. <div class="action-part">
  52. <div class="action-part-title"><h2>简单元素</h2></div>
  53. <div class="action-part-body">
  54. <div class="type-list">
  55. <div
  56. class="type-item"
  57. v-for="(item, index) in ELEMENT_LIST"
  58. :key="index"
  59. draggable="true"
  60. @dragstart="dragstart(item)"
  61. >
  62. <el-button><i class="el-icon-plus"></i>{{ item.name }}</el-button>
  63. </div>
  64. <p class="tips-info">提示:拖动插入元素</p>
  65. </div>
  66. </div>
  67. </div>
  68. <div class="action-part">
  69. <div class="action-part-title"><h2>元素面板</h2></div>
  70. <div class="action-part-body">
  71. <!-- element-tier-edit -->
  72. <element-tier-edit ref="ElementTierEdit"></element-tier-edit>
  73. </div>
  74. </div>
  75. </div>
  76. <div class="design-main">
  77. <!-- menus -->
  78. <div class="design-control">
  79. <div class="control-left tab-btns">
  80. <el-button
  81. v-for="(page, pageNo) in pages"
  82. :key="pageNo"
  83. :type="curPageNo === pageNo ? 'primary' : 'default'"
  84. @click="swithPage(pageNo)"
  85. >
  86. <span>第{{ pageNo + 1 }}页</span>
  87. <span class="page-delete" @click.stop="toDeletePage(page)"
  88. ><i class="el-icon-error"></i
  89. ></span>
  90. </el-button>
  91. <el-button icon="el-icon-plus" @click="toAddPage"></el-button>
  92. </div>
  93. <div class="control-right">
  94. <el-button
  95. type="success"
  96. :loading="isSubmit"
  97. :disabled="!pages.length"
  98. @click="toPreview"
  99. >预览</el-button
  100. >
  101. <el-button
  102. type="primary"
  103. :loading="isSubmit"
  104. :disabled="canSave || !pages.length"
  105. @click="toSave"
  106. >暂存</el-button
  107. >
  108. <el-button type="primary" :loading="isSubmit" @click="toSubmit"
  109. >提交</el-button
  110. >
  111. </div>
  112. </div>
  113. <!-- edit body -->
  114. <div class="design-body">
  115. <div
  116. :class="[
  117. 'page-box',
  118. `page-box-${curPage.pageSize}`,
  119. `page-box-${curPageNo % 2}`
  120. ]"
  121. v-if="curPage.id"
  122. >
  123. <div
  124. :class="[
  125. 'page-locators',
  126. `page-locators-${curPage.locators.length}`
  127. ]"
  128. >
  129. <ul
  130. class="page-locator-group"
  131. v-for="(locator, iind) in curPage.locators"
  132. :key="iind"
  133. >
  134. <li
  135. v-for="(elem, eindex) in locator"
  136. :key="eindex"
  137. :id="elem.id"
  138. ></li>
  139. </ul>
  140. </div>
  141. <!-- inner edit area -->
  142. <div class="page-main-inner">
  143. <div
  144. :class="['page-main', `page-main-${curPage.columns.length}`]"
  145. :style="{ margin: `0 -${curPage.columnGap / 2}px` }"
  146. >
  147. <div
  148. class="page-column"
  149. v-for="(column, columnNo) in curPage.columns"
  150. :key="columnNo"
  151. :style="{ padding: `0 ${curPage.columnGap / 2}px` }"
  152. >
  153. <topic-column-edit
  154. class="page-column-main"
  155. :data="column"
  156. ></topic-column-edit>
  157. </div>
  158. </div>
  159. </div>
  160. <!-- outer edit area -->
  161. <div class="page-main-outer">
  162. <page-number
  163. type="rect"
  164. :total="pages.length"
  165. :current="curPageNo + 1"
  166. ></page-number>
  167. <page-number
  168. type="text"
  169. :total="pages.length"
  170. :current="curPageNo + 1"
  171. ></page-number>
  172. </div>
  173. </div>
  174. </div>
  175. </div>
  176. <!-- element-prop-edit -->
  177. <element-prop-edit ref="ElementPropEdit"></element-prop-edit>
  178. <!-- right-click-menu -->
  179. <right-click-menu></right-click-menu>
  180. <!-- shortcut-key -->
  181. <shortcut-key
  182. @sk-save="skSave"
  183. @sk-submit="skSubmit"
  184. @sk-preview="skPreview"
  185. ></shortcut-key>
  186. <!-- help-dialog -->
  187. <help-dialog ref="HelpDialog"></help-dialog>
  188. <!-- card-view-frame -->
  189. <div class="design-preview-frame" v-if="cardPreviewUrl">
  190. <iframe :src="cardPreviewUrl" frameborder="0"></iframe>
  191. </div>
  192. </div>
  193. </template>
  194. <script>
  195. import { mapState, mapMutations, mapActions } from "vuex";
  196. import { cardConfigInfos, cardDetail, saveCard } from "../../../api";
  197. import { getElementModel, ELEMENT_LIST, TOPIC_LIST } from "../elements/model";
  198. import { getModel as getPageModel } from "../../../elements/page/model";
  199. import { CARD_VERSION } from "../../../enumerate";
  200. import TopicColumnEdit from "../components/TopicColumnEdit";
  201. import ElementPropEdit from "../components/ElementPropEdit";
  202. import ElementTierEdit from "../components/ElementTierEdit";
  203. import PagePropEdit from "../components/PagePropEdit";
  204. import RightClickMenu from "../components/RightClickMenu";
  205. import ShortcutKey from "../components/ShortcutKey";
  206. import HelpDialog from "../components/HelpDialog";
  207. import EditPage from "../../../elements/page/EditPage";
  208. import PageNumber from "../../../components/PageNumber";
  209. export default {
  210. name: "card-free-design",
  211. components: {
  212. TopicColumnEdit,
  213. PagePropEdit,
  214. EditPage,
  215. ElementPropEdit,
  216. ElementTierEdit,
  217. RightClickMenu,
  218. ShortcutKey,
  219. HelpDialog,
  220. PageNumber
  221. },
  222. data() {
  223. return {
  224. cardId: this.$route.params.cardId || this.$ls.get("cardId"),
  225. prepareTcPCard: this.$ls.get("prepareTcPCard", {
  226. examTaskId: "",
  227. courseCode: "",
  228. courseName: "",
  229. makeMethod: "SELF",
  230. cardRuleId: ""
  231. }),
  232. ELEMENT_LIST,
  233. TOPIC_LIST,
  234. topicList: [],
  235. columnWidth: 0,
  236. cardPreviewUrl: "",
  237. isSubmit: false,
  238. canSave: false
  239. };
  240. },
  241. computed: {
  242. ...mapState("free", [
  243. "cardConfig",
  244. "pages",
  245. "curElement",
  246. "curPage",
  247. "curPageNo",
  248. "curDragElement",
  249. "curColumnId"
  250. ]),
  251. isEdit() {
  252. return !!this.cardId;
  253. }
  254. },
  255. mounted() {
  256. // if (!this.prepareTcPCard.examTaskId && !this.isEdit) {
  257. // this.$message.error("找不到命题任务,请退出题卡制作!");
  258. // return;
  259. // }
  260. this.initCard();
  261. this.registWindowSubmit();
  262. },
  263. methods: {
  264. ...mapMutations("free", [
  265. "setCurPageNo",
  266. "setCurElement",
  267. "setCardConfig",
  268. "setOpenElementEditDialog",
  269. "setCurDragElement",
  270. "setPages",
  271. "initState"
  272. ]),
  273. ...mapActions("free", [
  274. "addPage",
  275. "removePage",
  276. "addElement",
  277. "modifyElement"
  278. ]),
  279. async initCard() {
  280. if (this.isEdit) {
  281. await this.getCardTempDetail();
  282. } else {
  283. await this.getCardConfig();
  284. this.initPageData();
  285. }
  286. },
  287. getCardTitle(titleRule) {
  288. const fieldMap = {
  289. courseCode: this.prepareTcPCard.courseCode,
  290. courseName: this.prepareTcPCard.courseName,
  291. schoolName: this.prepareTcPCard.schoolName
  292. };
  293. Object.entries(fieldMap).forEach(([key, val]) => {
  294. titleRule = titleRule.replace("${" + key + "}", val);
  295. });
  296. return titleRule;
  297. },
  298. async getCardTempDetail() {
  299. const detData = await cardDetail(this.cardId);
  300. // this.canSave = !detData.operateStatus;
  301. // 可能存在题卡内容没有记录的情况
  302. if (detData.content) {
  303. const cont = JSON.parse(detData.content);
  304. this.setPages(cont.pages);
  305. this.setCardConfig(cont.cardConfig);
  306. this.setCurPageNo(0);
  307. } else {
  308. await this.getCardConfig();
  309. // 没有题卡内容时,直接创建新的内容
  310. if (detData.makeMethod === "CUST") {
  311. this.setCardConfig({ cardTitle: detData.title });
  312. }
  313. this.initPageData();
  314. }
  315. },
  316. initPageData() {
  317. const page = getPageModel(this.cardConfig);
  318. this.addPage(page);
  319. this.setCurPageNo(0);
  320. },
  321. toAddPage() {
  322. const page = getPageModel(this.cardConfig);
  323. this.addPage(page);
  324. },
  325. async getCardConfig() {
  326. const data = await cardConfigInfos(this.prepareTcPCard.cardRuleId);
  327. if (!data) {
  328. this.$message.error("找不到题卡规则!");
  329. return;
  330. }
  331. let config = {
  332. ...data,
  333. ...{
  334. pageSize: "A3",
  335. columnNumber: 2,
  336. columnGap: 20,
  337. showForbidArea: true,
  338. cardDesc: "",
  339. makeMethod: this.prepareTcPCard.makeMethod
  340. }
  341. };
  342. config.aOrB = true; // 默认开启A/B卷型
  343. config.requiredFields = JSON.parse(config.requiredFields);
  344. config.extendFields = JSON.parse(config.extendFields);
  345. config.cardTitle = this.getCardTitle(config.titleRule);
  346. this.setCardConfig(config);
  347. },
  348. addNewTopic(item) {
  349. let element = getElementModel(item.type);
  350. this.setCurElement(element);
  351. this.$refs.ElementPropEdit.open();
  352. // to elementPropEdit/ElementPropEdit open topic edit dialog
  353. },
  354. // 元件编辑
  355. dragstart(element) {
  356. this.setCurDragElement(getElementModel(element.type));
  357. },
  358. // 操作
  359. skSave() {
  360. this.toSave();
  361. },
  362. skSubmit() {
  363. this.toSubmit();
  364. },
  365. skPreview() {
  366. this.toPreview();
  367. },
  368. async toPreview() {
  369. if (this.isSubmit) return;
  370. this.isSubmit = true;
  371. const result = await this.save().catch(() => {});
  372. this.isSubmit = false;
  373. if (!result) return;
  374. const { href } = this.$router.resolve({
  375. name: "CardFreePreview",
  376. params: {
  377. cardId: this.cardId,
  378. viewType: "view"
  379. }
  380. });
  381. window.open(href);
  382. },
  383. swithPage(pindex) {
  384. if (this.curPageNo === pindex) return;
  385. this.setCurPageNo(pindex);
  386. this.setCurElement({});
  387. },
  388. toDeletePage(page) {
  389. if (this.pages.length === 1) {
  390. this.$message.error("只剩最后一页,不能再删除了");
  391. return;
  392. }
  393. this.removePage(page);
  394. },
  395. // save
  396. getCardData(htmlContent = "", model = "") {
  397. let data = {
  398. title: this.cardConfig.cardTitle,
  399. content: model,
  400. htmlContent,
  401. type: "CUSTOM",
  402. ...this.prepareTcPCard
  403. };
  404. if (this.cardId) data.id = this.cardId;
  405. return data;
  406. },
  407. getRequestConfig() {
  408. return this.prepareTcPCard.makeMethod === "CUST"
  409. ? {
  410. headers: {
  411. schoolId: this.prepareTcPCard.schoolId
  412. }
  413. }
  414. : {};
  415. },
  416. getCardJson() {
  417. // 防止页面未渲染完成,各试题高度未及时更新,保存数据有误的问题
  418. return new Promise(resolve => {
  419. setTimeout(() => {
  420. const data = JSON.stringify(
  421. {
  422. version: CARD_VERSION,
  423. cardConfig: this.cardConfig,
  424. paperParams: this.paperParams,
  425. pages: this.pages
  426. },
  427. (k, v) => (k.startsWith("_") ? undefined : v)
  428. );
  429. resolve(data);
  430. }, 100);
  431. });
  432. },
  433. async save() {
  434. const cardJson = await this.getCardJson();
  435. let datas = this.getCardData("", cardJson);
  436. datas.status = "STAGE";
  437. const result = await saveCard(datas, this.getRequestConfig());
  438. this.cardId = result;
  439. this.$ls.set("cardId", this.cardId);
  440. return true;
  441. },
  442. async toSave() {
  443. if (this.isSubmit) return;
  444. this.isSubmit = true;
  445. const result = await this.save().catch(() => {});
  446. this.isSubmit = false;
  447. if (result) this.$message.success("保存成功!");
  448. },
  449. toSubmit() {
  450. if (this.isSubmit) return;
  451. if (this.pages.length % 2) {
  452. this.$message.error("请确保题卡页数是偶数");
  453. return;
  454. }
  455. this.$confirm("确定要提交当前题卡吗?", "提示", {
  456. type: "warning"
  457. })
  458. .then(() => {
  459. window.cardData = {
  460. cardConfig: this.cardConfig,
  461. pages: this.pages,
  462. paperParams: this.paperParams
  463. };
  464. this.isSubmit = true;
  465. const { href } = this.$router.resolve({
  466. name: "CardPreview",
  467. params: {
  468. cardId: 1,
  469. viewType: "frame"
  470. }
  471. });
  472. this.cardPreviewUrl = href;
  473. })
  474. .catch(() => {});
  475. },
  476. registWindowSubmit() {
  477. window.submitCardTemp = async (htmlContent, model) => {
  478. const datas = this.getCardData(htmlContent, model);
  479. datas.status = "SUBMIT";
  480. const result = await saveCard(
  481. datas,
  482. this.getRequestConfig()
  483. ).catch(() => {});
  484. this.cardPreviewUrl = "";
  485. this.isSubmit = false;
  486. window.cardData = null;
  487. if (result) {
  488. this.cardId = result;
  489. this.$ls.set("cardId", this.cardId);
  490. this.canSave = false;
  491. this.$message.success("提交成功!");
  492. this.goback();
  493. } else {
  494. this.$message.error("提交失败,请重新尝试!");
  495. }
  496. };
  497. },
  498. toExit() {
  499. this.$confirm(
  500. "请确保当前题卡已经正常保存,确定要退出当前题卡编辑吗?",
  501. "提示",
  502. {
  503. type: "warning"
  504. }
  505. )
  506. .then(() => {
  507. this.goback();
  508. })
  509. .catch(() => {});
  510. },
  511. showHelp() {
  512. this.$refs.HelpDialog.open();
  513. }
  514. },
  515. beforeDestroy() {
  516. this.$ls.remove("cardId");
  517. this.$ls.remove("prepareTcPCard");
  518. this.initState();
  519. delete window.submitCardTemp;
  520. }
  521. };
  522. </script>