index.tsx 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492
  1. import {
  2. NButton,
  3. NCascader,
  4. NForm,
  5. NFormItem,
  6. NInput,
  7. NScrollbar,
  8. NSelect,
  9. NSpace,
  10. useMessage
  11. } from 'naive-ui';
  12. import {
  13. TransitionGroup,
  14. defineComponent,
  15. onMounted,
  16. reactive,
  17. ref
  18. } from 'vue';
  19. import styles from './index.module.less';
  20. import UploadFile from '@/components/upload-file';
  21. import { nextTick } from 'vue';
  22. import { scrollToErrorForm } from '@/utils';
  23. import { api_lessonCoursewareSave } from '../../api';
  24. import iconUpload from '../../images/icon-upload.png';
  25. import iconUpload2 from '../../images/icon-upload2.png';
  26. import iconAdd from '../../images/icon-add.png';
  27. import iconMenu from '../../images/icon-menu.png';
  28. import btnAdd from '../../images/btn-add.png';
  29. import btnDelete from '../../images/btn-delete.png';
  30. import btnUp from '../../images/btn-up.png';
  31. import btnDown from '../../images/btn-down.png';
  32. import btnRemove from '../../images/btn-remove.png';
  33. import { lessonCoursewareDetail } from '/src/views/prepare-lessons/api';
  34. import { useCatchStore } from '/src/store/modules/catchData';
  35. export const BOOK_DATA = {
  36. grades: [
  37. { label: '一年级', value: 1 },
  38. { label: '二年级', value: 2 },
  39. { label: '三年级', value: 3 },
  40. { label: '四年级', value: 4 },
  41. { label: '五年级', value: 5 },
  42. { label: '六年级', value: 6 },
  43. { label: '七年级', value: 7 },
  44. { label: '八年级', value: 8 },
  45. { label: '九年级', value: 9 }
  46. ],
  47. bookTypes: [
  48. { label: '上册', value: 'LAST' },
  49. { label: '下册', value: 'NEXT' }
  50. ]
  51. };
  52. /** 添加单元 */
  53. const createLesson = () => {
  54. return {
  55. key: 'item' + Date.now(),
  56. name: '', // 单元名称
  57. lessonTargetDesc: '', // 课时目标描述
  58. knowledgeList: [
  59. {
  60. key: Date.now() + '' + 0,
  61. name: '' // 章节名称
  62. }
  63. ] // 章节信息
  64. };
  65. };
  66. const initState = () => ({
  67. id: null, // 教材id
  68. name: '',
  69. currentGradeNum: null as any,
  70. instrumentIds: null as any,
  71. // bookType: null, // 上下册
  72. coverImg: '', // 封面
  73. instruemntIds: [] as any,
  74. enableFlag: true, // 状态
  75. bookTypes: [], // 册别
  76. type: 'COURSEWARE', // 教材类型:COURSEWARE,THEORY,可用值:COURSEWARE,THEORY
  77. lessonList: [] as any // 单元列表
  78. });
  79. export default defineComponent({
  80. name: 'addNatural',
  81. props: {
  82. item: {
  83. type: Object,
  84. default: () => ({})
  85. }
  86. },
  87. emits: ['close', 'confirm'],
  88. setup(props, { emit }) {
  89. const catchStore = useCatchStore();
  90. const message = useMessage();
  91. const data = reactive({
  92. uploading: false,
  93. subjectList: [] as any
  94. });
  95. const formRef = ref();
  96. const form = reactive(initState());
  97. const handleSave = () => {
  98. formRef.value?.validate((err: any) => {
  99. if (err) {
  100. nextTick(scrollToErrorForm);
  101. return;
  102. }
  103. handleSubmit();
  104. });
  105. };
  106. const handleSubmit = async () => {
  107. data.uploading = true;
  108. try {
  109. const { currentGradeNum, bookTypes, instrumentIds, ...more } = form;
  110. await api_lessonCoursewareSave({
  111. currentGradeNum: currentGradeNum?.join(','),
  112. bookTypes: bookTypes?.join(','),
  113. instrumentIds: instrumentIds?.join(','),
  114. ...more
  115. });
  116. Object.assign(form, initState());
  117. message.success(props.item.id ? '保存成功' : '添加成功');
  118. emit('close', true);
  119. emit('confirm');
  120. } catch {
  121. //
  122. }
  123. data.uploading = false;
  124. };
  125. onMounted(async () => {
  126. if (!props.item.id) {
  127. form.lessonList = [createLesson()];
  128. }
  129. await catchStore.getSubjects();
  130. if (props.item.id) {
  131. // form.lessonList = [];
  132. const { data } = await lessonCoursewareDetail({ id: props.item.id });
  133. form.id = data.id;
  134. form.name = data.name;
  135. form.currentGradeNum = data.currentGradeNum
  136. ? data.currentGradeNum.split(',').map((item: any) => Number(item))
  137. : null;
  138. form.bookTypes = data.bookTypes ? data.bookTypes.split(',') : null
  139. form.instrumentIds = data.instrumentIds
  140. ? data.instrumentIds.split(',').map((item: any) => item)
  141. : null;
  142. // form.bookType = data.bookType;
  143. form.coverImg = data.coverImg;
  144. form.lessonList = [];
  145. const lessonList = data.lessonList || [];
  146. const tempLesson: any[] = [];
  147. lessonList.forEach((item: any) => {
  148. const tmpItem: any = {
  149. id: item.id,
  150. key: 'item' + Date.now() + '-' + Math.random() * 100,
  151. name: item.name,
  152. lessonTargetDesc: item.lessonTargetDesc,
  153. knowledgeList: [] as any
  154. };
  155. if (item.knowledgeList && item.knowledgeList.length) {
  156. item.knowledgeList.forEach((knowledge: any) => {
  157. tmpItem.knowledgeList.push({
  158. id: knowledge.id,
  159. key: Date.now() + '-' + Math.random() * 100,
  160. name: knowledge.name
  161. });
  162. });
  163. }
  164. tempLesson.push(tmpItem);
  165. });
  166. form.lessonList = tempLesson;
  167. }
  168. data.subjectList = catchStore.getEnableSingleAllSubjects(
  169. form.instrumentIds
  170. );
  171. });
  172. return () => (
  173. <div class={styles.container}>
  174. <NScrollbar style={{ 'max-height': '65vh' }}>
  175. <NForm
  176. ref={formRef}
  177. labelPlacement="left"
  178. labelWidth={120}
  179. model={form}>
  180. <div class={styles.topForms}>
  181. <NFormItem
  182. path="coverImg"
  183. rule={[
  184. {
  185. required: true,
  186. message: '请上传教材封面',
  187. trigger: ['change']
  188. }
  189. ]}>
  190. <UploadFile
  191. cropper
  192. // tips="建议尺寸:210*297, 文件大小: 5MB以内;"
  193. v-model:fileList={form.coverImg}
  194. showType="custom"
  195. size={2}
  196. accept=".jpg,jpeg,.png"
  197. options={{
  198. autoCropWidth: 210,
  199. autoCropHeight: 297,
  200. fixedBox: true
  201. }}>
  202. {{
  203. custom: () => (
  204. <div class={styles.uploadContent}>
  205. <img src={iconUpload2} class={styles.iconUpload} />
  206. <p>请上传教材封面</p>
  207. </div>
  208. )
  209. }}
  210. </UploadFile>
  211. </NFormItem>
  212. <div class={styles.topFormInput}>
  213. <NFormItem
  214. style={{ minWidth: '360px' }}
  215. path="name"
  216. rule={[
  217. {
  218. required: true,
  219. message: '请输入教材名称',
  220. trigger: ['blur', 'change']
  221. }
  222. ]}>
  223. <NInput
  224. placeholder="请输入教材名称"
  225. maxlength={25}
  226. v-model:value={form.name}
  227. clearable></NInput>
  228. </NFormItem>
  229. <NFormItem
  230. path="currentGradeNum"
  231. rule={{
  232. required: true,
  233. message: '请选择年级',
  234. trigger: 'change',
  235. type: 'array'
  236. }}>
  237. <NSelect
  238. style={{ minWidth: '360px' }}
  239. placeholder="请选择年级"
  240. options={BOOK_DATA.grades}
  241. v-model:value={form.currentGradeNum}
  242. clearable
  243. multiple
  244. filterable
  245. maxTagCount={3}
  246. />
  247. </NFormItem>
  248. <NFormItem
  249. path="bookTypes">
  250. <NSelect
  251. style={{ minWidth: '360px' }}
  252. placeholder="请选择册别(选填)"
  253. options={[{
  254. label: '上册',
  255. value: 'LAST'
  256. }, {
  257. label: '下册',
  258. value: 'NEXT'
  259. }]}
  260. v-model:value={form.bookTypes}
  261. clearable
  262. multiple
  263. filterable
  264. />
  265. </NFormItem>
  266. <NFormItem
  267. path="instrumentIds"
  268. style={{ width: '360px' }}
  269. rule={{
  270. required: true,
  271. message: '请选择乐器',
  272. trigger: 'change',
  273. type: 'array'
  274. }}>
  275. <NCascader
  276. placeholder="请选择乐器"
  277. options={data.subjectList}
  278. v-model:value={form.instrumentIds}
  279. checkStrategy="child"
  280. showPath={false}
  281. childrenField="instruments"
  282. expandTrigger="hover"
  283. labelField="name"
  284. valueField="id"
  285. clearable
  286. filterable
  287. multiple
  288. maxTagCount="responsive"
  289. style={{ width: '400px' }}
  290. />
  291. </NFormItem>
  292. {/* <NFormItem
  293. path="bookType"
  294. style={{ width: '360px' }}
  295. rule={{
  296. required: true,
  297. message: '请选择册别',
  298. trigger: 'change'
  299. }}>
  300. <NSelect
  301. placeholder="请选择册别"
  302. options={BOOK_DATA.bookTypes}
  303. v-model:value={form.bookType}
  304. clearable
  305. />
  306. </NFormItem> */}
  307. </div>
  308. </div>
  309. <div class={styles.menuTitle}>
  310. <img src={iconMenu} class={styles.iconMenu} />
  311. 目录
  312. </div>
  313. <TransitionGroup name="list" tag="div">
  314. {form.lessonList.map((item: any, index: number) => {
  315. return (
  316. <NSpace
  317. class={styles.lessonItem}
  318. wrap={false}
  319. wrapItem={false}
  320. align="start"
  321. key={item.key}>
  322. <NFormItem
  323. label="单元名称"
  324. labelPlacement="top"
  325. path={`lessonList[${index}].name`}
  326. rule={{
  327. required: true,
  328. message: '填写单元名称',
  329. trigger: ['blur', 'change']
  330. }}>
  331. <NInput
  332. placeholder="填写单元名称"
  333. maxlength={25}
  334. v-model:value={item.name}
  335. clearable></NInput>
  336. </NFormItem>
  337. <TransitionGroup name="list" tag="div">
  338. {item.knowledgeList.map((know: any, knowIndex: number) => {
  339. return (
  340. <NFormItem
  341. style={{
  342. '--n-label-height': knowIndex === 0 ? '26px' : '0'
  343. }}
  344. labelPlacement="top"
  345. label={knowIndex === 0 ? '章节名称' : ''}
  346. key={know.key}
  347. path={`lessonList[${index}].knowledgeList[${knowIndex}].name`}
  348. rule={{
  349. required: true,
  350. message: '填写章节名称',
  351. trigger: ['blur', 'change']
  352. }}>
  353. <NSpace
  354. wrap={false}
  355. align="center"
  356. class={styles.btnGroupAll}
  357. wrapItem={false}>
  358. <NInput
  359. maxlength={25}
  360. placeholder="填写章节名称"
  361. v-model:value={know.name}
  362. clearable></NInput>
  363. <NButton
  364. quaternary
  365. circle
  366. onClick={() => {
  367. item.knowledgeList.splice(knowIndex + 1, 0, {
  368. name: '',
  369. key: Date.now() + '' + knowIndex
  370. });
  371. }}>
  372. {{
  373. icon: () => (
  374. <img src={btnAdd} class={styles.btnImg} />
  375. )
  376. }}
  377. </NButton>
  378. <NButton
  379. quaternary
  380. circle
  381. disabled={item.knowledgeList.length < 2}
  382. onClick={() => {
  383. item.knowledgeList.splice(knowIndex, 1);
  384. }}>
  385. {{
  386. icon: () => (
  387. <img
  388. src={btnDelete}
  389. class={styles.btnImg}
  390. />
  391. )
  392. }}
  393. </NButton>
  394. <NButton
  395. quaternary
  396. circle
  397. disabled={knowIndex === 0}
  398. onClick={() => {
  399. if (knowIndex === 0) return;
  400. const tmp = item.knowledgeList[knowIndex - 1];
  401. item.knowledgeList[knowIndex - 1] =
  402. item.knowledgeList[knowIndex];
  403. item.knowledgeList[knowIndex] = tmp;
  404. }}>
  405. {{
  406. icon: () => (
  407. <img src={btnUp} class={styles.btnImg} />
  408. )
  409. }}
  410. </NButton>
  411. <NButton
  412. quaternary
  413. circle
  414. disabled={
  415. knowIndex === item.knowledgeList.length - 1
  416. }
  417. onClick={() => {
  418. if (
  419. knowIndex ===
  420. item.knowledgeList.length - 1
  421. )
  422. return;
  423. const tmp = item.knowledgeList[knowIndex + 1];
  424. item.knowledgeList[knowIndex + 1] =
  425. item.knowledgeList[knowIndex];
  426. item.knowledgeList[knowIndex] = tmp;
  427. }}>
  428. {{
  429. icon: () => (
  430. <img src={btnDown} class={styles.btnImg} />
  431. )
  432. }}
  433. </NButton>
  434. </NSpace>
  435. </NFormItem>
  436. );
  437. })}
  438. </TransitionGroup>
  439. <NButton
  440. class={styles.closeBtn}
  441. secondary
  442. circle
  443. size="small"
  444. disabled={form.lessonList.length < 2}
  445. onClick={() => {
  446. form.lessonList.splice(index, 1);
  447. }}>
  448. <img src={btnRemove} />
  449. </NButton>
  450. </NSpace>
  451. );
  452. })}
  453. </TransitionGroup>
  454. <div class={styles.line}></div>
  455. <NButton
  456. block
  457. class={styles.addUnitBtn}
  458. ghost
  459. color="#198CFE"
  460. onClick={() => {
  461. form.lessonList.push(createLesson());
  462. }}>
  463. {{
  464. icon: () => <img src={iconAdd} />,
  465. default: () => '新增单元'
  466. }}
  467. </NButton>
  468. </NForm>
  469. </NScrollbar>
  470. <NSpace class={styles.btnGroup} justify="center">
  471. <NButton round onClick={() => emit('close')}>
  472. 取消
  473. </NButton>
  474. <NButton
  475. round
  476. loading={data.uploading}
  477. type="primary"
  478. onClick={() => handleSave()}>
  479. 保存
  480. </NButton>
  481. </NSpace>
  482. </div>
  483. );
  484. }
  485. });