diary.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697
  1. <template>
  2. <div class="diary">
  3. <div
  4. style="background-color: #fff;width: 100%;height: 100%;box-sizing: border-box; padding: 15px;"
  5. >
  6. <!-- 分类与导出 -->
  7. <div class="top">
  8. <div class="oneType">
  9. <div
  10. v-for="(i, index) in VeidooList"
  11. :key="index + 'a'"
  12. :class="Veidoo == i.id ? 'onTypeTxt' : 'onTypeTxt2'"
  13. @click="changeType(i)"
  14. >
  15. {{ i.name }}
  16. <div
  17. class="line"
  18. :style="{ display: Veidoo == i.id ? 'block' : 'none' }"
  19. ></div>
  20. </div>
  21. </div>
  22. <div class="wordBtn" @click="judgeExport">
  23. 导出Excel
  24. </div>
  25. <div class="twoType">
  26. <div
  27. v-for="(i, index) in VeidooListChild"
  28. :key="index + 'b'"
  29. :class="VeidooChild == i.id ? 'twoTypeTxt' : 'twoTypeTxt2'"
  30. @click="changeTypeChild(i)"
  31. >
  32. {{ i.name }}
  33. </div>
  34. </div>
  35. </div>
  36. <div class="selectSty">
  37. <div style="margin-right: 10px;">时间筛选</div>
  38. <el-select v-model="termId" placeholder="请选择">
  39. <el-option
  40. label="全部"
  41. @click.native="changeTerm('')"
  42. value=""
  43. ></el-option>
  44. <el-option
  45. v-for="item in termList"
  46. :key="item.id"
  47. :label="item.name"
  48. :value="item.id"
  49. @click.native="changeTerm(item.id)"
  50. >
  51. </el-option>
  52. </el-select>
  53. </div>
  54. <el-table
  55. ref="multipleTable"
  56. :data="tableData"
  57. tooltip-effect="dark"
  58. style="width: 100%"
  59. border
  60. header-align="center"
  61. :header-cell-style="{
  62. background: '#E0EAFB',
  63. color: 'rgba(0, 0, 0, 0.90)'
  64. }"
  65. @selection-change="handleSelectionChange"
  66. >
  67. <el-table-column
  68. type="selection"
  69. align="center"
  70. label="全选"
  71. :show-overflow-tooltip="true"
  72. width="55"
  73. >
  74. </el-table-column>
  75. <el-table-column
  76. prop="create_at"
  77. label="创建时间"
  78. align="center"
  79. width="120"
  80. >
  81. <!-- <template slot-scope="scope">{{ scope.row.date }}</template> -->
  82. </el-table-column>
  83. <el-table-column
  84. prop="recordTit"
  85. label="观察内容"
  86. align="center"
  87. width="120"
  88. >
  89. </el-table-column>
  90. <el-table-column
  91. prop="place"
  92. label="观察地点"
  93. align="center"
  94. show-overflow-tooltip
  95. >
  96. </el-table-column>
  97. <el-table-column
  98. prop="tname"
  99. label="维度"
  100. align="center"
  101. show-overflow-tooltip
  102. >
  103. </el-table-column>
  104. <el-table-column
  105. prop="recordContent"
  106. label="内容"
  107. align="center"
  108. show-overflow-tooltip
  109. >
  110. </el-table-column>
  111. <el-table-column
  112. label="操作"
  113. align="center"
  114. width="175px"
  115. show-overflow-tooltip
  116. >
  117. <template slot-scope="scope">
  118. <div class="evaluate">
  119. <div
  120. class="TableBtn"
  121. style="color: #3681FC;"
  122. @click="lookRecord(scope.row)"
  123. >
  124. 查看
  125. </div>
  126. <div
  127. class="TableBtn"
  128. style="color: #3681FC;"
  129. @click="updateCred(scope.row)"
  130. >
  131. 修改
  132. </div>
  133. <div
  134. class="TableBtn"
  135. style="color: #EE3E3E;"
  136. @click="delRecord(scope.row)"
  137. >
  138. 删除
  139. </div>
  140. </div>
  141. </template>
  142. </el-table-column>
  143. </el-table>
  144. <!-- 分页 -->
  145. <el-pagination
  146. @current-change="handleCurrentChange"
  147. background
  148. :page-size="8"
  149. layout="prev, pager, next"
  150. :total="total"
  151. class="pagination"
  152. >
  153. </el-pagination>
  154. </div>
  155. <div v-if="isPop" @click="closePop" class="shade"></div>
  156. <popbox
  157. ref="refPop"
  158. v-if="isPop"
  159. :judgeNum="judgeNum"
  160. :recordDataCopy="recordData"
  161. :termList="termList"
  162. :weiList="weiList"
  163. :studentList="studentList"
  164. @selectData="selectData"
  165. @closePop="closePop"
  166. ></popbox>
  167. </div>
  168. </template>
  169. <script>
  170. import popbox from "./component/popbox";
  171. export default {
  172. components: {
  173. popbox
  174. },
  175. data() {
  176. return {
  177. // 筛选数据
  178. VeidooList: [],
  179. VeidooListChild: [],
  180. // 默认选择项
  181. // VeidooValue:{},
  182. // 默认选择项
  183. termValue: "",
  184. // 学期数据
  185. termList: [],
  186. // 要提交的筛选数据项
  187. Veidoo: "",
  188. VeidooChild: "",
  189. termId: "",
  190. page: 1, //当前页数,默认为第一页
  191. // 本页多少条数
  192. total: 0,
  193. tableData: [],
  194. multipleSelection: [],
  195. // 组件信息数据
  196. isPop: false,
  197. recordData: {},
  198. recordDataCopy: {},
  199. weiList: [],
  200. studentList: [],
  201. judgeNum: 0
  202. };
  203. },
  204. methods: {
  205. closePop() {
  206. this.isPop = false;
  207. // console.log('关了');
  208. // this.$refs.refPop.fuClick(this.recordData);
  209. },
  210. // 切换页
  211. handleCurrentChange(val) {
  212. //当页数发生改变的时候调用获取列表数据请求
  213. // console.log(`当前页: ${val}`);
  214. this.Page = val;
  215. this.selectData();
  216. },
  217. // 获取数据
  218. getData() {
  219. // 获取筛选框数据
  220. let params = {
  221. uid: this.$route.query.userid,
  222. cid: this.$route.query.cid,
  223. org: this.$route.query.org,
  224. cu: ""
  225. };
  226. this.ajax
  227. .get(this.$store.state.api + "selectSETable", params)
  228. .then(res => {
  229. this.isLoading = false;
  230. var ftype = res.data[1]; //公共父级分类
  231. var stype = res.data[2]; //公共子级分类
  232. var sctype = res.data[3]; //该学校子级分类
  233. var fctype = res.data[4]; //该学校父级分类
  234. var fotype = res.data[5]; //组织父级分类
  235. var sotype = res.data[6]; //组织子级分类
  236. var allfType = [];
  237. var allsType = [];
  238. if (fotype.length == 0 && sotype.length == 0) {
  239. if (fctype.length == 0 && sctype.length == 0) {
  240. for (var i = 0; i < ftype.length; i++) {
  241. allfType.push(ftype[i]);
  242. }
  243. for (var i = 0; i < stype.length; i++) {
  244. allsType.push(stype[[i]]);
  245. }
  246. } else {
  247. for (var i = 0; i < fctype.length; i++) {
  248. allfType.push(fctype[i]);
  249. }
  250. for (var i = 0; i < sctype.length; i++) {
  251. allsType.push(sctype[[i]]);
  252. }
  253. }
  254. } else {
  255. for (var i = 0; i < fotype.length; i++) {
  256. allfType.push(fotype[i]);
  257. }
  258. for (var i = 0; i < sotype.length; i++) {
  259. allsType.push(sctysotypepe[[i]]);
  260. }
  261. }
  262. allfType.forEach((e, index) => {
  263. allfType[index][e.id] = [];
  264. });
  265. // 大数组套小数组循环,进行push操作,最后手动添加艺术
  266. allfType.forEach((e, w) => {
  267. allsType.forEach((k, index) => {
  268. if (e.id == k.pid) {
  269. // console.log("e.id", ftype[w][e.id]);
  270. allfType[w][e.id].push(k);
  271. }
  272. });
  273. });
  274. this.VeidooList = allfType;
  275. // 渲染完数据后立即查询一次默认筛选项
  276. this.changeType(this.VeidooList[0]);
  277. // 学期数据
  278. this.ajax
  279. .get(this.$store.state.api + "selectTerm", {})
  280. .then(res => {
  281. this.isLoading = false;
  282. console.log("学期数据", res.data[0]);
  283. this.termList = res.data[0];
  284. this.termList.forEach(e => {
  285. if (e.defaultC == 1) {
  286. this.termId = e.id;
  287. // this.termValue = e.name;
  288. }
  289. });
  290. // 调用获取数据方法
  291. this.selectData();
  292. })
  293. .catch(err => {
  294. this.isLoading = false;
  295. console.error(err);
  296. });
  297. // console.log("ftype1", ftype);
  298. })
  299. .catch(err => {
  300. this.isLoading = false;
  301. console.error(err);
  302. });
  303. // 获取纬度筛选框
  304. const data = {
  305. oid: this.$route.query.org,
  306. cla: 0
  307. };
  308. this.ajax.get(this.$store.state.api + "selectVeiDoo", data).then(res => {
  309. // console.log(res);
  310. this.weiList = res.data[0];
  311. console.log("this.weiList", res);
  312. });
  313. // 获取班学生列表
  314. const data2 = {
  315. cid: this.$route.query.cid,
  316. uid: this.$route.query.suid
  317. };
  318. this.ajax
  319. .get(this.$store.state.api + "selectManyClassStudent", data2)
  320. .then(res => {
  321. // console.log(res);
  322. this.studentList = res.data[0];
  323. console.log("this.studentList", res);
  324. });
  325. },
  326. // 选择导出哪些记录
  327. handleSelectionChange(val) {
  328. this.multipleSelection = val;
  329. console.log(this.multipleSelection);
  330. },
  331. // 点击大筛选项
  332. changeType(e) {
  333. // console.log(e[e.id]);
  334. // 点击大筛选项,清除小筛选项
  335. this.VeidooChild = "";
  336. this.Veidoo = e.id;
  337. this.VeidooListChild = e[e.id];
  338. this.selectData();
  339. },
  340. // 点击小选项
  341. changeTypeChild(e) {
  342. // 重复点击取消选择
  343. if (this.VeidooChild == e.id) {
  344. this.VeidooChild = "";
  345. } else {
  346. this.VeidooChild = e.id;
  347. }
  348. this.selectData();
  349. },
  350. // 点击学期
  351. changeTerm(e) {
  352. console.log(e);
  353. this.termId = e;
  354. this.selectData();
  355. },
  356. // 获取筛选数据
  357. selectData() {
  358. let data = {
  359. uid: this.$route.query.suid,
  360. cid: this.$route.query.cid,
  361. cu: this.Veidoo,
  362. cn: this.VeidooChild,
  363. cm: this.termId,
  364. page: this.page,
  365. lim: 8
  366. };
  367. // return console.log("selectData", data);
  368. this.ajax
  369. .get(this.$store.state.api + "selectVeidooType", data)
  370. .then(res => {
  371. this.isLoading = false;
  372. this.tableData = res.data[0];
  373. this.total = res.data[0].length > 0 ? res.data[0][0].num : 0;
  374. console.log(" 获取筛选数据", res);
  375. });
  376. },
  377. // 修改记录
  378. updateCred(e) {
  379. this.judgeNum = 1;
  380. this.isPop = true;
  381. if (!Array.isArray(e.type)) {
  382. if (!e.type) {
  383. e.type = [];
  384. } else {
  385. e.type = e.type.split(",");
  386. }
  387. }
  388. if (!Array.isArray(e.contact)) {
  389. if (!e.contact) {
  390. e.contact = [];
  391. } else {
  392. e.contact = e.contact.split(",");
  393. }
  394. }
  395. if (!Array.isArray(e.recordImg)) {
  396. if (!e.recordImg) {
  397. e.recordImg = [];
  398. } else {
  399. e.recordImg = e.recordImg.split(",");
  400. }
  401. }
  402. this.recordData = e;
  403. },
  404. // 判断导出数据
  405. judgeExport() {
  406. if (this.multipleSelection.length) {
  407. this.exportExcel();
  408. } else {
  409. console.log(2);
  410. }
  411. },
  412. // exportExcelTwo() {
  413. // let data = {
  414. // uid: this.$route.query.suid,
  415. // cid: this.$route.query.cid,
  416. // cu: this.Veidoo,
  417. // cn: this.VeidooChild,
  418. // cm: this.termId,
  419. // page: this.page,
  420. // lim: 8
  421. // };
  422. // // return console.log("selectData", data);
  423. // this.ajax
  424. // .get(this.$store.state.api + "selectVeidooType", data)
  425. // .then(res => {
  426. // this.isLoading = false;
  427. // this.tableData = res.data[0];
  428. // this.total = res.data[0].length > 0 ? res.data[0][0].num : 0;
  429. // console.log(" 获取筛选数据", res);
  430. // });
  431. // },
  432. exportExcel() {
  433. // var res = res.data[0];
  434. var res = this.multipleSelection;
  435. //如果value的json字段的key值和想要的headers值不一致时,可做如下更改
  436. //将和下面的Object.fromEntries结合,将json字段的key值改变为要求的excel的header值
  437. var array = [];
  438. for (var i = 0; i < res.length; i++) {
  439. var _json = {};
  440. _json["创建时间"] = res[i].create_at;
  441. _json["班级"] = res[i].className;
  442. _json["记录时间"] = res[i].recordDate;
  443. // _json["用户名"] = res[i].studentName;
  444. // _json["用户名"] = res[i].studentName;
  445. _json["观察地点"] = res[i].place;
  446. _json["观察内容"] = res[i].recordTit;
  447. _json["内容"] = res[i].recordContent;
  448. array.push(_json);
  449. }
  450. var XLSX = require("xlsx");
  451. const workbook = XLSX.utils.book_new(); //创建一个新的工作簿对象
  452. let ws = XLSX.utils.json_to_sheet(array); //将json对象数组转化成工作表
  453. // ws["!cols"] = [
  454. // //设置每一列的宽度
  455. // { wch: 50 },
  456. // { wch: 50 },
  457. // { wch: 50 }
  458. // ];
  459. XLSX.utils.book_append_sheet(workbook, ws, "sheet1"); //把sheet添加到workbook里,第三个参数是sheet名
  460. XLSX.writeFile(workbook, "观察日记.xlsx");
  461. // const wopts = { bookType: "xlsx", bookSST: false, type: "array" };//写入的样式bookType:输出的文件类型,type:输出的数据类型,bookSST: 是否生成Shared String Table,官方解释是,如果开启生成速度会下降,但在低版本IOS设备上有更好的兼容性
  462. // const wbout = XLSX.write(workbook, wopts);// 浏览器端和node共有的API,实际上node可以直接使用xlsx.writeFile来写入文件,但是浏览器没有该API
  463. // FileSaver.saveAs(new Blob([wbout], { type: "application/octet-stream" }), `${title} demo.xlsx`);//保存文件
  464. this.$message({
  465. message: "导出成功",
  466. type: "success"
  467. });
  468. },
  469. // 删除记录
  470. delRecord(e) {
  471. this.$confirm("是否删除?", "提示", {
  472. confirmButtonText: "确定",
  473. cancelButtonText: "取消"
  474. })
  475. .then(() => {
  476. this.ajax
  477. .post(this.$store.state.api + "updateRecord", [{ rid: e.rid }])
  478. .then(res => {
  479. console.log(res);
  480. this.selectData();
  481. this.$message({
  482. type: "success",
  483. message: "已删除"
  484. });
  485. // this.isLoading = false; updateRecord
  486. });
  487. })
  488. .catch(() => {
  489. this.$message({
  490. type: "info",
  491. message: "已取消删除"
  492. });
  493. });
  494. // return console.log(e);
  495. },
  496. lookRecord(e) {
  497. this.judgeNum = 0;
  498. console.log("eeeeeeeeeeeeeeeeeee", e);
  499. this.isPop = true;
  500. if (!Array.isArray(e.type)) {
  501. if (!e.type) {
  502. e.type = [];
  503. } else {
  504. e.type = e.type.split(",");
  505. }
  506. }
  507. if (!Array.isArray(e.contact)) {
  508. if (!e.contact) {
  509. e.contact = [];
  510. } else {
  511. e.contact = e.contact.split(",");
  512. }
  513. }
  514. if (!Array.isArray(e.recordImg)) {
  515. if (!e.recordImg) {
  516. e.recordImg = [];
  517. } else {
  518. e.recordImg = e.recordImg.split(",");
  519. }
  520. }
  521. this.recordData = e;
  522. }
  523. },
  524. created() {
  525. this.getData();
  526. },
  527. mounted() {
  528. // this.$nextTick(() => {
  529. // this.selectData();
  530. // });
  531. }
  532. };
  533. </script>
  534. <style scoped>
  535. .shade {
  536. position: fixed;
  537. top: 0;
  538. left: 0;
  539. z-index: 3;
  540. width: 100%;
  541. height: 100%;
  542. background-color: rgba(0, 0, 0, 0.5);
  543. }
  544. /* .el-pager li{
  545. background-color: #fff !important;
  546. border: 1px solid #DCDCDC !important;
  547. }
  548. .el-pager li:nth-child(1){
  549. background-color: #fff !important;
  550. border: none !important;
  551. } */
  552. .el-table__cell {
  553. text-align: center !important;
  554. }
  555. .diary {
  556. box-sizing: border-box;
  557. padding: 15px;
  558. height: 100%;
  559. }
  560. .top {
  561. width: 100%;
  562. display: flex;
  563. position: relative;
  564. justify-content: space-between;
  565. align-items: flex-end;
  566. }
  567. .oneType {
  568. display: flex;
  569. justify-content: flex-start;
  570. /* flex-direction: column; */
  571. }
  572. .onTypeTxt {
  573. cursor: pointer;
  574. display: flex;
  575. font-size: 16px;
  576. position: relative;
  577. align-items: center;
  578. justify-content: center;
  579. padding: 16px;
  580. color: rgba(54, 129, 252, 1);
  581. }
  582. .onTypeTxt2 {
  583. cursor: pointer;
  584. display: flex;
  585. font-size: 16px;
  586. position: relative;
  587. align-items: center;
  588. justify-content: center;
  589. padding: 16px;
  590. color: rgba(0, 0, 0, 0.9);
  591. }
  592. .line {
  593. position: absolute;
  594. left: 50%;
  595. bottom: 0;
  596. transform: translate(-50%, 0);
  597. height: 3px;
  598. background-color: #3681fc;
  599. width: 50%;
  600. }
  601. .wordBtn {
  602. cursor: pointer;
  603. background-color: #3681fc;
  604. color: #fff;
  605. width: 100px;
  606. height: 34px;
  607. border-radius: 3px;
  608. display: flex;
  609. justify-content: center;
  610. align-items: center;
  611. }
  612. .twoType {
  613. position: absolute;
  614. display: flex;
  615. top: 120%;
  616. left: 0;
  617. }
  618. .twoTypeTxt {
  619. cursor: pointer;
  620. box-sizing: border-box;
  621. min-width: 76px;
  622. padding: 0 10px;
  623. height: 34px;
  624. overflow: hidden;
  625. white-space: nowrap;
  626. text-overflow: ellipsis;
  627. color: #fff;
  628. background-color: #3681fc;
  629. border-radius: 3px;
  630. border: 1px solid #e7e7e7;
  631. display: flex;
  632. justify-content: center;
  633. align-items: center;
  634. margin-right: 15px;
  635. }
  636. .twoTypeTxt2 {
  637. cursor: pointer;
  638. box-sizing: border-box;
  639. min-width: 76px;
  640. padding: 0 10px;
  641. height: 34px;
  642. overflow: hidden;
  643. white-space: nowrap;
  644. text-overflow: ellipsis;
  645. color: #000;
  646. border-radius: 3px;
  647. border: 1px solid #e7e7e7;
  648. border-radius: 3px;
  649. display: flex;
  650. justify-content: center;
  651. align-items: center;
  652. margin-right: 15px;
  653. }
  654. .selectSty {
  655. display: flex;
  656. justify-content: flex-start;
  657. align-items: center;
  658. /* margin-top: 60px; */
  659. margin: 80px 10px 40px 0;
  660. }
  661. .evaluate {
  662. text-align: center;
  663. font-family: "Microsoft YaHei";
  664. font-size: 14px;
  665. font-style: normal;
  666. font-weight: 400;
  667. line-height: 22px; /* 157.143% */
  668. display: flex;
  669. justify-content: space-around;
  670. }
  671. .TableBtn {
  672. cursor: pointer;
  673. }
  674. .pagination {
  675. margin-top: 20px;
  676. float: right;
  677. }
  678. </style>