CardDesign.vue 16 KB

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