CardDesign.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573
  1. <template>
  2. <div class="card-design">
  3. <div class="design-header">
  4. <div class="design-back" @click="toExit">
  5. <i class="el-icon-d-arrow-left" title="退出"></i>
  6. 返回
  7. </div>
  8. </div>
  9. <!-- actions -->
  10. <div class="design-action">
  11. <div class="design-logo">
  12. <h1>
  13. <i class="el-icon-d-arrow-left" @click="toExit" title="退出"></i>
  14. 答题卡制作
  15. </h1>
  16. </div>
  17. <div class="action-part">
  18. <div class="action-part-title"><h2>基本设置</h2></div>
  19. <div class="action-part-body">
  20. <page-prop-edit @init-page="initPageData"></page-prop-edit>
  21. </div>
  22. </div>
  23. <div class="action-part">
  24. <div class="action-part-title"><h2>试题配置</h2></div>
  25. <div class="action-part-body">
  26. <div class="type-list">
  27. <div class="type-item" v-for="item in TOPIC_LIST" :key="item.type">
  28. <el-button @click="addNewTopic(item)"
  29. ><i class="el-icon-plus"></i>{{ item.name }}</el-button
  30. >
  31. </div>
  32. <div
  33. class="type-item"
  34. v-for="item in NOT_TOPIC_LIST"
  35. :key="item.type"
  36. >
  37. <el-button @click="addNewTopic(item)"
  38. ><i class="el-icon-plus"></i>{{ item.name }}</el-button
  39. >
  40. </div>
  41. </div>
  42. <p class="tips-info">提示:点击创建试题</p>
  43. </div>
  44. </div>
  45. <div class="action-part">
  46. <div class="action-part-title"><h2>插入元素</h2></div>
  47. <div class="action-part-body">
  48. <div class="type-list">
  49. <div
  50. class="type-item"
  51. v-for="(item, index) in ELEMENT_LIST"
  52. :key="index"
  53. draggable="true"
  54. @dragstart="dragstart(item)"
  55. >
  56. <el-button><i class="el-icon-plus"></i>{{ item.name }}</el-button>
  57. </div>
  58. <p class="tips-info">提示:拖动插入元素</p>
  59. </div>
  60. <!-- card config edit -->
  61. <card-config-prop-edit
  62. v-if="isFromCardManage"
  63. class="mt-2"
  64. ></card-config-prop-edit>
  65. </div>
  66. </div>
  67. </div>
  68. <div id="design-main" class="design-main">
  69. <!-- menus -->
  70. <div class="design-control">
  71. <div class="control-left tab-btns">
  72. <el-button
  73. v-for="(page, pageNo) in pages"
  74. :key="pageNo"
  75. :type="curPageNo === pageNo ? 'primary' : 'default'"
  76. @click="swithPage(pageNo)"
  77. >第{{ pageNo + 1 }}页</el-button
  78. >
  79. </div>
  80. <div class="control-right">
  81. <div v-if="hasUnsubmitCont" class="tips-info tips-error">
  82. <i class="el-icon-warning"></i>有未提交数据
  83. </div>
  84. <el-button
  85. type="success"
  86. :loading="isSubmit"
  87. :disabled="!pages.length"
  88. @click="toPreview"
  89. >预览</el-button
  90. >
  91. <!-- <el-button
  92. v-if="showSaveBtn"
  93. type="primary"
  94. :loading="isSubmit"
  95. :disabled="canSave || !pages.length"
  96. @click="toSave"
  97. >暂存</el-button
  98. > -->
  99. <el-button type="primary" :loading="isSubmit" @click="toSubmit"
  100. >提交</el-button
  101. >
  102. </div>
  103. </div>
  104. <!-- edit body -->
  105. <div class="design-body">
  106. <!-- 注意:后台要替换内容,改类名时,要注意 -->
  107. <div
  108. v-for="(page, pageNo) in pages"
  109. :key="pageNo"
  110. :id="`edit-page-box-${pageNo}`"
  111. :class="[
  112. 'page-box',
  113. `page-box-${cardConfig.pageSize}`,
  114. `page-box-${pageNo % 2}`,
  115. { 'page-box-less': pages.length <= 2 },
  116. ]"
  117. >
  118. <!-- locator -->
  119. <div class="page-locator page-locator-top">
  120. <div
  121. v-for="elem in page.locators.top"
  122. :key="elem.id"
  123. :id="elem.id"
  124. class="page-locator-item"
  125. ></div>
  126. </div>
  127. <div class="page-locator page-locator-bottom">
  128. <div
  129. v-for="elem in page.locators.bottom"
  130. :key="elem.id"
  131. :id="elem.id"
  132. class="page-locator-item"
  133. ></div>
  134. </div>
  135. <!-- inner edit area -->
  136. <div class="page-main-inner">
  137. <div
  138. :class="['page-main', `page-main-${page.columns.length}`]"
  139. :style="{ margin: `0 -${page.columnGap / 2}px` }"
  140. >
  141. <div
  142. class="page-column"
  143. v-for="(column, columnNo) in page.columns"
  144. :key="columnNo"
  145. :style="{ padding: `0 ${page.columnGap / 2}px` }"
  146. >
  147. <div
  148. class="page-column-main"
  149. :id="[`column-${pageNo}-${columnNo}`]"
  150. >
  151. <div class="page-column-body" v-if="column.elements.length">
  152. <topic-element-edit
  153. class="page-column-element"
  154. :data-h="element.h"
  155. v-for="element in column.elements"
  156. :key="element.id"
  157. :data="element"
  158. ></topic-element-edit>
  159. </div>
  160. <div class="page-column-body" v-else>
  161. <div
  162. class="page-column-forbid-area"
  163. v-if="cardConfig.showForbidArea"
  164. >
  165. <p>该区域严禁作答</p>
  166. </div>
  167. </div>
  168. </div>
  169. </div>
  170. </div>
  171. </div>
  172. <!-- outer edit area -->
  173. <div class="page-main-outer">
  174. <page-number
  175. type="rect"
  176. :total="pages.length"
  177. :current="pageNo + 1"
  178. ></page-number>
  179. <page-number
  180. type="text"
  181. :total="pages.length"
  182. :current="pageNo + 1"
  183. ></page-number>
  184. <elem-undertaking
  185. v-if="cardConfig.undertakingEnable && pageNo === pages.length - 1"
  186. :content="cardConfig.undertakingBody"
  187. ></elem-undertaking>
  188. <course-barcode
  189. v-if="pageNo % 2 === 0"
  190. :data="cardConfig"
  191. ></course-barcode>
  192. </div>
  193. </div>
  194. </div>
  195. </div>
  196. <!-- all topics -->
  197. <div v-if="!isSubmit" class="topic-list">
  198. <div :class="['page-box', `page-box-${cardConfig.pageSize}`]">
  199. <div class="page-main-inner">
  200. <div
  201. :class="['page-main', `page-main-${cardConfig.columnNumber}`]"
  202. :style="{ margin: `0 -${cardConfig.columnGap / 2}px` }"
  203. >
  204. <div
  205. class="page-column"
  206. :style="{ padding: `0 ${cardConfig.columnGap / 2}px` }"
  207. >
  208. <div class="page-column-main" id="topic-column">
  209. <div class="page-column-body">
  210. <!-- card-head-sample -->
  211. <card-head-sample
  212. :data="cardHeadSampleData"
  213. id="simple-card-head"
  214. v-if="topics.length && cardHeadSampleData"
  215. ></card-head-sample>
  216. <!-- topic element -->
  217. <topic-element-preview
  218. class="page-column-element"
  219. v-for="element in topics"
  220. :key="element.id"
  221. :data="element"
  222. ></topic-element-preview>
  223. </div>
  224. </div>
  225. </div>
  226. </div>
  227. </div>
  228. </div>
  229. </div>
  230. <!-- element-prop-edit -->
  231. <element-prop-edit ref="ElementPropEdit"></element-prop-edit>
  232. <!-- right-click-menu -->
  233. <right-click-menu @inset-topic="insetNewTopic"></right-click-menu>
  234. <!-- topic select dialog -->
  235. <topic-select-dialog
  236. ref="TopicSelectDialog"
  237. :topics="topicList"
  238. :insert-topic="preInsertTopic"
  239. @confirm="addNewTopic"
  240. ></topic-select-dialog>
  241. </div>
  242. </template>
  243. <script>
  244. import { mapState, mapMutations, mapActions } from "vuex";
  245. import {
  246. getElementModel,
  247. getCardHeadModel,
  248. ELEMENT_LIST,
  249. TOPIC_LIST,
  250. NOT_TOPIC_LIST,
  251. OTHER_ELEMENT,
  252. } from "../elementModel";
  253. import { CARD_VERSION } from "../enumerate";
  254. import CardConfigPropEdit from "../components/CardConfigPropEdit";
  255. import TopicElementEdit from "../components/TopicElementEdit";
  256. import TopicElementPreview from "../components/TopicElementPreview";
  257. import PagePropEdit from "../components/PagePropEdit";
  258. import ElementPropEdit from "../components/ElementPropEdit";
  259. import RightClickMenu from "../components/RightClickMenu";
  260. import PageNumber from "../components/PageNumber";
  261. import CardHeadSample from "../elements/card-head/CardHead";
  262. import TopicSelectDialog from "../components/TopicSelectDialog";
  263. import ElemUndertaking from "../elements/undertaking/ElemUndertaking.vue";
  264. import CourseBarcode from "./CourseBarcode.vue";
  265. export default {
  266. name: "card-design",
  267. props: {
  268. content: {
  269. type: Object,
  270. default() {
  271. return {
  272. pages: [],
  273. cardConfig: {},
  274. answers: {},
  275. };
  276. },
  277. },
  278. showSaveBtn: {
  279. type: Boolean,
  280. default: true,
  281. },
  282. },
  283. components: {
  284. CardConfigPropEdit,
  285. TopicElementEdit,
  286. TopicElementPreview,
  287. PagePropEdit,
  288. ElementPropEdit,
  289. RightClickMenu,
  290. CardHeadSample,
  291. PageNumber,
  292. TopicSelectDialog,
  293. ElemUndertaking,
  294. CourseBarcode,
  295. },
  296. data() {
  297. return {
  298. ELEMENT_LIST,
  299. TOPIC_LIST,
  300. NOT_TOPIC_LIST,
  301. topicList: [],
  302. columnWidth: 0,
  303. isSubmit: false,
  304. canSave: false,
  305. hasUnsubmitCont: false,
  306. preInsertTopic: {},
  307. };
  308. },
  309. computed: {
  310. ...mapState("card", [
  311. "cardConfig",
  312. "topics",
  313. "pages",
  314. "curElement",
  315. "curPage",
  316. "curPageNo",
  317. ]),
  318. cardHeadSampleData() {
  319. if (!this.cardConfig["pageSize"]) return;
  320. const data = getCardHeadModel(this.cardConfig);
  321. data.isSimple = true;
  322. return data;
  323. },
  324. isFromCardManage() {
  325. return this.$route.name === "CardManage";
  326. },
  327. },
  328. mounted() {
  329. this.initCard();
  330. },
  331. beforeDestroy() {
  332. this.initState();
  333. },
  334. methods: {
  335. ...mapMutations("card", [
  336. "addPage",
  337. "setCurPage",
  338. "setCurElement",
  339. "setCardConfig",
  340. "setOpenElementEditDialog",
  341. "setCurDragElement",
  342. "setPages",
  343. "initState",
  344. ]),
  345. ...mapActions("card", [
  346. "resetTopicSeries",
  347. "removePage",
  348. "addElement",
  349. "modifyCardHead",
  350. "modifyElement",
  351. "rebuildPages",
  352. "initTopicsFromPages",
  353. "scrollToElementPage",
  354. ]),
  355. async initCard() {
  356. const { cardConfig, pages } = this.content;
  357. this.setCardConfig(cardConfig);
  358. if (pages && pages.length) {
  359. this.setPages(pages);
  360. this.initTopicsFromPages();
  361. this.resetTopicSeries();
  362. this.setCurPage(0);
  363. } else {
  364. this.initPageData();
  365. }
  366. this.addWatch();
  367. this.$nextTick(() => {
  368. this.registSrollEvent();
  369. });
  370. },
  371. initPageData() {
  372. this.modifyCardHead({
  373. ...getCardHeadModel(this.cardConfig),
  374. });
  375. this.$nextTick(() => {
  376. this.rebuildPages();
  377. this.setCurPage(0);
  378. });
  379. },
  380. addNewTopic(item) {
  381. if (!item.type) {
  382. this.$message.error("试题类型丢失!");
  383. return;
  384. }
  385. let element = getElementModel(item.type);
  386. element.w = document.getElementById("topic-column").offsetWidth;
  387. if (
  388. ["FILL_QUESTION_SIMPLE", "FILL_QUESTION_MULTIPLY"].includes(item.type)
  389. ) {
  390. element.optionCount = this.cardConfig.defaultOptionNumber;
  391. }
  392. if (item.type === "FORBID_AREA") {
  393. const lastTopicElement = this.topics.findLast(
  394. (item) => !OTHER_ELEMENT.includes(item.type)
  395. );
  396. element.sign = lastTopicElement ? lastTopicElement.sign : "objective";
  397. this.addElement(element);
  398. this.setCurElement(element);
  399. this.$nextTick(() => {
  400. this.rebuildPages();
  401. this.$nextTick(() => {
  402. this.scrollToElementPage(element);
  403. });
  404. });
  405. } else {
  406. if (item.insertTopic && item.insertTopic.id) {
  407. element.insertTopic = item.insertTopic;
  408. }
  409. this.setCurElement(element);
  410. this.$refs.ElementPropEdit.open();
  411. // to elementPropEdit/ElementPropEdit open topic edit dialog
  412. }
  413. },
  414. insetNewTopic({ id, type }) {
  415. this.preInsertTopic = { id, type };
  416. this.topicList = this.TOPIC_LIST;
  417. this.$refs.TopicSelectDialog.open();
  418. },
  419. // 元件编辑
  420. dragstart(element) {
  421. this.setCurDragElement(getElementModel(element.type));
  422. },
  423. addWatch() {
  424. this.$watch("cardConfig", (val) => {
  425. const element = getCardHeadModel(val);
  426. this.modifyCardHead(element);
  427. this.$nextTick(() => {
  428. this.rebuildPages();
  429. });
  430. });
  431. },
  432. // 页面切换
  433. swithPage(pindex) {
  434. if (this.curPageNo === pindex) return;
  435. let pageTops = this.pages.map((page, pageNo) => {
  436. return document.getElementById(`edit-page-box-${pageNo}`).offsetTop;
  437. });
  438. pageTops = pageTops.map((item) => item - pageTops[0]);
  439. document.getElementById("design-main").scrollTop = pageTops[pindex];
  440. this.setCurPage(pindex);
  441. this.setCurElement({});
  442. },
  443. registSrollEvent() {
  444. document.getElementById("design-main").addEventListener("scroll", (e) => {
  445. e.preventDefault();
  446. e.stopPropagation();
  447. let pageTops = this.pages.map((page, pageNo) => {
  448. return document.getElementById(`edit-page-box-${pageNo}`).offsetTop;
  449. });
  450. const pageRangeHeight =
  451. document.getElementById(`edit-page-box-0`).offsetHeight * 0.2;
  452. pageTops = pageTops.map((item) => item - pageTops[0] - pageRangeHeight);
  453. const scrollTop = e.target.scrollTop;
  454. // console.log(pageTops, scrollTop);
  455. const targePageTop = pageTops.find((item) => scrollTop < item);
  456. const pageNo = targePageTop
  457. ? pageTops.indexOf(targePageTop) - 1
  458. : pageTops.length - 1;
  459. if (this.curPageNo === pageNo) return;
  460. this.setCurPage(pageNo);
  461. });
  462. },
  463. // save
  464. getCardData(htmlContent = "", model = "") {
  465. const data = {
  466. title: this.cardConfig.cardTitle,
  467. content: model,
  468. htmlContent,
  469. };
  470. return data;
  471. },
  472. checkElementCovered() {
  473. let elements = [];
  474. this.pages.forEach((page) => {
  475. page.columns.forEach((column) => {
  476. column.elements.forEach((element) => {
  477. if (element.isCovered) {
  478. elements.push(element.id);
  479. }
  480. });
  481. });
  482. });
  483. return elements.length;
  484. },
  485. checkCardValid() {
  486. if (!this.cardConfig.cardTitle) {
  487. this.$message.error("题卡标题不能为空!");
  488. this.setCurPageNo(0);
  489. setTimeout(() => {
  490. document.getElementById("cardTitleInput").focus();
  491. });
  492. return false;
  493. }
  494. if (this.checkElementCovered()) {
  495. this.$message.error("题卡中存在被遮挡的元件,请注意调整!");
  496. return false;
  497. }
  498. return true;
  499. },
  500. getCardJson() {
  501. // 防止页面未渲染完成,各试题高度未及时更新,保存数据有误的问题
  502. return new Promise((resolve) => {
  503. setTimeout(() => {
  504. const data = JSON.stringify(
  505. {
  506. version: CARD_VERSION,
  507. cardType: "STANDARD",
  508. cardConfig: this.cardConfig,
  509. pages: this.pages,
  510. // 来自题库的试卷生成的题卡会有答案信息
  511. answers: this.content.answers || {},
  512. },
  513. (k, v) => (k.startsWith("_") ? undefined : v)
  514. );
  515. resolve(data);
  516. }, 100);
  517. });
  518. },
  519. async toPreview() {
  520. this.$emit("on-preview", {
  521. cardConfig: this.cardConfig,
  522. pages: this.pages,
  523. });
  524. },
  525. async toSave() {
  526. const model = await this.getCardJson();
  527. const datas = this.getCardData("", model);
  528. this.$emit("on-save", datas);
  529. },
  530. toSubmit() {
  531. if (this.isSubmit) return;
  532. if (!this.checkCardValid()) return;
  533. this.$emit("on-submit", {
  534. cardConfig: {
  535. ...this.cardConfig,
  536. courseCodeBarcodeSrc: "",
  537. courseCodeBarcodeName: "",
  538. },
  539. pages: this.pages,
  540. answers: this.content.answers || {},
  541. });
  542. },
  543. toExit() {
  544. this.$emit("on-exit");
  545. },
  546. loading() {
  547. this.isSubmit = true;
  548. },
  549. unloading() {
  550. this.isSubmit = false;
  551. },
  552. updateUnsubmitStatus(status) {
  553. this.hasUnsubmitCont = status;
  554. },
  555. },
  556. };
  557. </script>