ServiceDialog.vue 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831
  1. <script setup>
  2. import html2canvas from "html2canvas";
  3. import { contractInterface } from "@/api/api.js";
  4. import { ref,reactive,watch } from "vue";
  5. import { ElMessage } from "element-plus";
  6. const props = defineProps({
  7. serviceCon: null,
  8. serviceShow: {
  9. type: Boolean,
  10. default: false,
  11. },
  12. contractType:''
  13. });
  14. const emits=defineEmits(['serviceSave','serviceClose'])
  15. const showOpt=ref(false)//选择权限弹窗
  16. const optionLists=ref([]) //权限列表数据
  17. const tableHeadData=ref([]) //表头数据
  18. const tableData=ref([]) //表格数据
  19. const roleType=ref(localStorage.getItem("Role"))
  20. const tableSelectList=reactive({
  21. 商品复盘: [
  22. { id: 2, name: "能化专栏《化里化外》", select: false },
  23. { id: 4, name: "黑色专栏《知白守黑》", select: false },
  24. { id: 5, name: "有色专栏《有声有色》", select: false },
  25. // { id: 1, name: "《每日宏观商品复盘》", select: false },
  26. { id: 3, name: "《股债日评》", select: false },
  27. ],
  28. 深度月报:[
  29. { id: 6, name: "宏观经济", select: false },
  30. { id: 7, name: "草根调研", select: false },
  31. // { id: 8, name: "PVC月报", select: false },
  32. ]
  33. }) //表格中自带选择的数据
  34. const tableSelectCheck=reactive({
  35. 商品复盘: [],
  36. }) //表格中自带选择的数据选中项
  37. const position=reactive({
  38. top: 0,
  39. left: 0,
  40. }) //鼠标右击时的位置
  41. const showRightClickMenu=ref(false) //显示右键菜单
  42. const rightClickCon=ref([
  43. { type: "selectRow", name: "选择该行" },
  44. { type: "insertColumn", name: "插入一列" },
  45. { type: "insertRow", name: "插入一行" },
  46. { type: "deleteColumn", name: "删除该列" },
  47. { type: "deleteRow", name: "删除该行" },
  48. ]) //右键菜单内容
  49. const showInputHead=ref(false) //显示填写表头
  50. const form=reactive({
  51. title: "",
  52. }) //插入一列填写的表头数据
  53. const formRule=reactive({
  54. title: [{ required: true, message: "请填写名称", trigger: "blur" }],
  55. })
  56. const selectIndex=reactive({
  57. rindex: 0, //行序号
  58. cindex: 0, //列序号
  59. })
  60. const selectRowIndex=ref(null) //选择该行的序列
  61. // 初始化表格数据
  62. watch(()=>props.serviceCon,()=>{
  63. initData();
  64. },{
  65. // 开启深度监听,监听到‘补充协议’与其他合同类型切换时,对象里面客观经济的选中变化
  66. deep:true
  67. })
  68. //选择上面选项更新表格数据
  69. // optionLists: {
  70. // handler(nval, oval) {
  71. // if (this.selectRowIndex === null) return;
  72. // let arr = [];
  73. // let arrId = [];
  74. // let count = 0;
  75. // nval.forEach((item) => {
  76. // if (item.CheckAll) {
  77. // count++;
  78. // }
  79. // });
  80. // //全选
  81. // if (count === nval.length) {
  82. // arr = ["全品种"];
  83. // nval.forEach((item) => {
  84. // arrId = [...arrId, ...item.CheckList];
  85. // });
  86. // } else {
  87. // nval.forEach((item) => {
  88. // item.Items.forEach((item2) => {
  89. // let flag = item.CheckList.indexOf(item2.ChartPermissionId);
  90. // if (flag != -1) {
  91. // arr.push(item2.PermissionName);
  92. // arrId.push(item2.ChartPermissionId);
  93. // }
  94. // });
  95. // });
  96. // }
  97. // // console.log(arr);
  98. // this.tableData[this.selectRowIndex].forEach((item) => {
  99. // if (item.HeadName === "品种") {
  100. // item.Value = arr.join("、");
  101. // item.ValueId = arrId;
  102. // }
  103. // });
  104. // },
  105. // deep: true,
  106. // },
  107. // 如果没有选择任何行则重置顶部选项数据
  108. watch(selectRowIndex,(nval, oval)=>{
  109. if (nval === null) {
  110. optionLists.value.forEach((item) => {
  111. item.CheckList = [];
  112. });
  113. }
  114. })
  115. const initData=()=>{
  116. console.log("初始化");
  117. if (!props.serviceCon.Detail) return;
  118. let temarr = props.serviceCon.Detail.map((rowItem) => {
  119. let rowArr = [];
  120. for (let key in rowItem) {
  121. if (key.substr(0, 3) === "Col" && rowItem[key] !== "") {
  122. rowArr.push(JSON.parse(rowItem[key]));
  123. }
  124. }
  125. return rowArr;
  126. });
  127. tableHeadData.value = temarr[0];
  128. tableData.value = temarr.slice(1);
  129. // 回显表格内部选择的
  130. tableData.value.forEach((item) => {
  131. item.forEach((e) => {
  132. if (e.Type === "select") {
  133. let key = e.Value;
  134. item.forEach((e2) => {
  135. if (e2.HeadName === "品种") {
  136. if(e2.Value){
  137. tableSelectCheck[key] = e2.ValueId;
  138. }else{
  139. tableSelectCheck[key] = [];
  140. }
  141. }
  142. });
  143. }
  144. });
  145. });
  146. }
  147. //点击保存选择的权限数据
  148. const handleSaveOpt=()=>{
  149. if (selectRowIndex.value === null) return;
  150. let arr=[]
  151. let arrId=[]
  152. let count = 0
  153. optionLists.value.forEach((item) => {
  154. if (item.CheckAll) {
  155. count++;
  156. }
  157. });
  158. // 全选
  159. if(count===optionLists.value.length){
  160. arr = ["全品种"];
  161. optionLists.value.forEach((item) => {
  162. arrId = [...arrId, ...item.CheckList];
  163. });
  164. }else{
  165. optionLists.value.forEach((item) => {
  166. item.Items.forEach((item2) => {
  167. let flag = item.CheckList.indexOf(item2.ChartPermissionId);
  168. if (flag != -1) {
  169. arr.push(item2.PermissionName);
  170. arrId.push(item2.ChartPermissionId);
  171. }
  172. });
  173. });
  174. }
  175. // 找出和该行关联的行 同时修改品种栏数据
  176. let rowIndexarr=[]
  177. if(!tableData.value[selectRowIndex.value][0].RowName){
  178. rowIndexarr.push(selectRowIndex.value)
  179. }else{
  180. tableData.value.forEach((item,index)=>{
  181. if(item[0].RowName==tableData.value[selectRowIndex.value][0].RowName){
  182. rowIndexarr.push(index)
  183. }
  184. })
  185. }
  186. console.log(rowIndexarr);
  187. rowIndexarr.forEach(index=>{
  188. tableData.value[index].forEach((item) => {
  189. if (item.HeadName === "品种") {
  190. item.Value = arr.join("、");
  191. item.ValueId = arrId;
  192. }
  193. });
  194. })
  195. // this.tableData[this.selectRowIndex].forEach((item) => {
  196. // if (item.HeadName === "品种") {
  197. // item.Value = arr.join("、");
  198. // item.ValueId = arrId;
  199. // }
  200. // });
  201. showOpt.value=false
  202. }
  203. // 点击关闭选择权限弹窗
  204. const closeOpt=()=>{
  205. showOpt.value=false
  206. selectRowIndex.value=null
  207. }
  208. // 左键选择行
  209. const handleSelectRow=(rindex,data)=>{
  210. // 如果选择的行存在 选项数据则不允许选择 或者 该行不可选择 即RowDisable=true
  211. if(data.HeadName!='品种') return
  212. let tag = false;
  213. tableData.value[rindex].forEach((item) => {
  214. if (item.Type === "select"||item.RowDisable) {
  215. tag = true;
  216. }
  217. });
  218. if (tag) return;
  219. selectRowIndex.value = rindex;
  220. formatOptionList();
  221. showOpt.value=true
  222. }
  223. //获取顶部权限选项
  224. const getPermissionOpt=()=>{
  225. let CompanyType = localStorage.getItem("Role") === "ficc_seller" ? "ficc" : "权益";
  226. contractInterface.getPermissionList({ CompanyType }).then((res) => {
  227. if (res.Ret === 200) {
  228. let arr = res.Data.List;
  229. arr.forEach((item) => {
  230. item.CheckList = [];
  231. item.CheckAll = false;
  232. item.indeterminate = false;
  233. });
  234. optionLists.value = arr;
  235. }
  236. });
  237. }
  238. //每次选中一行时格式化顶部选择数据
  239. const formatOptionList=()=> {
  240. let valueId = [];
  241. tableData.value[selectRowIndex.value].forEach((item) => {
  242. if (item.HeadName === "品种") {
  243. valueId = item.ValueId;
  244. }
  245. });
  246. optionLists.value.forEach((item) => {
  247. item.CheckList = [];
  248. item.Items.forEach((e) => {
  249. let flag = valueId.indexOf(e.ChartPermissionId);
  250. if (flag != -1) {
  251. item.CheckList.push(e.ChartPermissionId);
  252. }
  253. });
  254. if (item.CheckList.length === item.Items.length) {
  255. item.CheckAll = true;
  256. } else {
  257. item.CheckAll = false;
  258. }
  259. });
  260. }
  261. //全选
  262. const handleCheckAllChange=(e)=>{
  263. let arr = e.Items.map((item) => {
  264. return item.ChartPermissionId;
  265. });
  266. e.CheckList = e.CheckAll ? arr : [];
  267. e.indeterminate = false;
  268. // 需求更改,合同类型为补充协议时不默认勾选宏观经济
  269. if(e.ClassifyName==='宏观经济'&&props.contractType!=='补充协议'){
  270. e.CheckList.push(1)
  271. }
  272. }
  273. //选择单个
  274. const handleCheckChange=(e)=>{
  275. e.CheckAll = e.CheckList.length === e.Items.length;
  276. e.indeterminate = e.CheckList.length > 0 && e.CheckList.length < e.Items.length;
  277. }
  278. //点击表格中选项数据
  279. const handleTableSelectChange=(title, rindex)=>{
  280. let arr = tableSelectCheck[title];
  281. let arr2 = [];
  282. tableSelectList[title].forEach((item) => {
  283. arr.forEach((e) => {
  284. if (e == item.id) {
  285. arr2.push(item.name);
  286. }
  287. });
  288. });
  289. tableData.value[rindex].forEach((item) => {
  290. if (item.HeadName === "品种") {
  291. item.Value = arr2.join("、");
  292. item.ValueId = arr;
  293. }
  294. });
  295. }
  296. //鼠标右键
  297. const rightClick=(rindex, cindex, event)=>{
  298. selectIndex.rindex = Number(rindex);
  299. selectIndex.cindex = Number(cindex);
  300. formatRightClickCon(rindex, cindex);
  301. showRightClickMenu.value = true;
  302. position.top = event.clientY + "px";
  303. position.left = event.clientX + "px";
  304. }
  305. //格式化鼠标右击显示内容
  306. const formatRightClickCon=(rindex, cindex)=>{
  307. let arr = [
  308. { type: "insertColumn", name: "向后插入一列", show: true },
  309. { type: "insertRow", name: "向后插入一行", show: true },
  310. { type: "deleteColumn", name: "删除该列", show: true },
  311. { type: "deleteRow", name: "删除该行", show: true },
  312. ];
  313. // 品种一列不允许删除
  314. if (tableHeadData.value[cindex].Tag === "品种"||tableHeadData.value[cindex].Tag === "FICC小套餐服务内容") {
  315. arr[2].show = false;
  316. }
  317. //删除列 不能少于两列
  318. if (tableHeadData.value.length <= 2) {
  319. arr[2].show = false;
  320. }
  321. //删除行 不能少于两行 并且商品复盘、周报、双周报、数据点评、月报必须有一个存在
  322. if (tableData.value.length <= 2) {
  323. arr[3].show = false;
  324. }
  325. let temarr=[]
  326. tableData.value.forEach(item=>{
  327. item.forEach(item2=>{
  328. if(item2.HeadName==='FICC小套餐服务内容'){
  329. if(item2.Value==='数据点评'||
  330. item2.Value==='FICC周报'||
  331. item2.Value==='商品双周报+线上电话会讨论会<br/>(由弘则的研究员主持线上讨论)'){
  332. temarr.push(item2.Value)
  333. }
  334. }
  335. })
  336. })
  337. if(temarr.length<2){
  338. tableData.value[selectIndex.rindex].forEach(item=>{
  339. if(item.Value==='数据点评'||
  340. item.Value==='FICC周报'||
  341. item.Value==='商品双周报+线上电话会讨论会<br/>(由弘则的研究员主持线上讨论)'){
  342. arr[3].show = false;
  343. }
  344. })
  345. }
  346. //新增列 不能超过6列
  347. if (tableHeadData.value.length >= 6) {
  348. arr[0].show = false;
  349. }
  350. rightClickCon.value = arr.filter((item) => {
  351. return item.show;
  352. });
  353. }
  354. // 点击 新增/删除 行/列 选择该行
  355. const handleUpdateTable=(type)=>{
  356. //删除行
  357. if (type === "deleteRow") {
  358. selectRowIndex.value = null;
  359. tableData.value.splice(selectIndex.rindex, 1);
  360. }
  361. //新增行
  362. if (type === "insertRow") {
  363. selectRowIndex.value = null;
  364. let arr = tableHeadData.value.map((item) => {
  365. let obj = { CanEdit: true, Type: "text", Value: "", ValueId: [], HeadName: item.Tag,RowDisable:false,RowName:''};
  366. if (item.Tag === "品种") {
  367. obj.CanEdit = false;
  368. }
  369. return obj;
  370. });
  371. tableData.value.splice(selectIndex.rindex + 1, 0, arr);
  372. }
  373. //删除列
  374. if (type === "deleteColumn") {
  375. selectRowIndex.value = null;
  376. tableHeadData.value.splice(selectIndex.cindex, 1);
  377. tableData.value = tableData.value.map((item) => {
  378. item.splice(selectIndex.cindex, 1);
  379. return item;
  380. });
  381. }
  382. //新增列
  383. if (type === "insertColumn") {
  384. selectRowIndex.value = null;
  385. showInputHead.value = true;
  386. }
  387. showRightClickMenu.value = false;
  388. }
  389. const headTitleForm=ref(null)
  390. //点击确定新增表头内容
  391. const handleConfirmHead=()=>{
  392. headTitleForm.value.validate((valid) => {
  393. if (!valid) return;
  394. tableHeadData.value.splice(selectIndex.cindex + 1, 0, { CanEdit: false, Type: "text", Value: form.title, Tag: form.title });
  395. tableData.value = tableData.value.map((item) => {
  396. item.splice(selectIndex.cindex + 1, 0, { CanEdit: true, Type: "text", Value: "", HeadName: form.title,RowDisable:false,RowName:'' });
  397. return item;
  398. });
  399. showInputHead.value = false;
  400. form.title = "";
  401. });
  402. }
  403. //点击取消填写新增列头弹窗
  404. const handleCancelHead=()=>{
  405. headTitleForm.value.resetFields();
  406. showInputHead.value = false;
  407. }
  408. //点击保存
  409. const handleSave=()=>{
  410. let flag = validateData();
  411. if (!flag) return;
  412. // 生成表格图片
  413. html2canvas(document.getElementById("table-png"), {
  414. backgroundColor: "#ffffff",
  415. useCORS: true, // 允许图片跨域
  416. allowTaint: true, // 在渲染前测试图片
  417. imageTimeout: 0, // 加载延时
  418. scale:3,
  419. }).then((res) => {
  420. let img = res.toDataURL("image/png");
  421. emits("serviceSave", {
  422. Value: img,
  423. ServiceTemplateId: props.serviceCon.ServiceTemplateId,
  424. tableData: tableData.value,
  425. tableHeadData: tableHeadData.value,
  426. });
  427. handleServiceClose();
  428. });
  429. }
  430. //点击取消
  431. const handleCancel=()=>{
  432. initData()
  433. handleServiceClose();
  434. }
  435. //校验数据
  436. const validateData=()=>{
  437. //判断品种列是否有值
  438. let arr = [];
  439. tableData.value.forEach((item) => {
  440. item.forEach((item2) => {
  441. if (item2.HeadName === "品种") {
  442. arr.push(...item2.ValueId);
  443. }
  444. });
  445. });
  446. // 判断行是否有值
  447. let arr2 = [];
  448. tableData.value.forEach((item, index) => {
  449. let arr = item.filter((item2) => {
  450. if (!item2.Value) return item2;
  451. });
  452. if (arr.length === tableHeadData.value.length) {
  453. arr2.push(index);
  454. }
  455. });
  456. // 商品复盘、周报、双周报、数据点评、月报 品种必须有一个选择了
  457. let arr3=[]
  458. tableData.value.forEach(item=>{
  459. item.forEach(item2=>{
  460. if(item2.Value==='数据点评'||
  461. item2.Value==='FICC周报'||
  462. item2.Value==='商品双周报+线上电话会讨论会<br/>(由弘则的研究员主持线上讨论)'){
  463. item.forEach(item3=>{
  464. if (item3.HeadName === "品种") {
  465. arr3.push(...item3.ValueId);
  466. }
  467. })
  468. }
  469. })
  470. })
  471. let constractType=props.contractType
  472. console.log(constractType);
  473. if (!arr.length&&constractType!=='补充协议') {
  474. ElMessage.warning("至少选择一个品种");
  475. return false;
  476. }
  477. if (arr2.length) {
  478. ElMessage.warning("行内至少填一项");
  479. return false;
  480. }
  481. if(!arr3.length&&constractType!=='补充协议'){
  482. ElMessage.warning("周报、双周报、数据点评至少选择一个品种");
  483. return false;
  484. }
  485. return true;
  486. }
  487. //关闭弹窗
  488. const handleServiceClose=()=>{
  489. showRightClickMenu.value = false;
  490. selectRowIndex.value = null;
  491. emits("serviceClose");
  492. }
  493. defineExpose({
  494. tableHeadData,tableData
  495. })
  496. getPermissionOpt();
  497. </script>
  498. <template>
  499. <div class="service-content-wrap">
  500. <!-- 大套餐或其他等直接查看图片类型的 -->
  501. <el-dialog v-dialogDrag top="5vh" :model-value="props.serviceShow" class="self-dialog-c" :modal-append-to-body="false" @close="handleServiceClose" :show-close="false" v-if="!serviceCon.Detail">
  502. <!-- <h2 class="title">{{ serviceCon.Title }}</h2> -->
  503. <template #header>
  504. <!-- 权益合同和ficc合同有别 -->
  505. {{roleType}}
  506. <h2 class="title">{{roleType == 'ficc_seller'?serviceCon.Title:serviceCon.showTitle }}</h2>
  507. </template>
  508. <img class="img" style="width: 100%; display: block; margin: 0 auto 20px auto"
  509. :src="roleType == 'ficc_seller'?serviceCon.Value:serviceCon.showValue" />
  510. <div style="text-align: center; margin-bottom: 30px">
  511. <el-button type="primary" @click="handleServiceClose">知道了</el-button>
  512. </div>
  513. </el-dialog>
  514. <!-- 小套餐 -->
  515. <el-dialog v-dialogDrag title="提示" width="65%" top="5vh" :model-value="props.serviceShow" :modal-append-to-body="false" @close="handleCancel" v-else>
  516. <template #header>
  517. <img width="20" src="../../../assets/img/constract/select.png" style="vertical-align: middle;margin-right: 4px" />
  518. <span style="vertical-align: middle">选择品种</span>
  519. </template>
  520. <!-- 选择种类部分 -->
  521. <!-- <div class="type-select-wrap">
  522. <h2>权限设置</h2>
  523. <div class="type-select-box">
  524. <div class="item flex" v-for="item in optionLists" :key="item.ClassifyName" style="margin-bottom: 20px">
  525. <el-checkbox
  526. :disabled="selectRowIndex === null"
  527. :indeterminate="item.CheckList.length > 0 && item.CheckList.length < item.Items.length"
  528. v-model="item.CheckAll"
  529. @change="handleCheckAllChange(item)"
  530. >{{ item.ClassifyName }}:</el-checkbox
  531. >
  532. <el-checkbox-group :disabled="selectRowIndex === null" v-model="item.CheckList" @change="handleCheckChange(item)">
  533. <el-checkbox :label="tag.ChartPermissionId" v-for="tag in item.Items" :key="tag.ChartPermissionId">{{ tag.PermissionName }}</el-checkbox>
  534. </el-checkbox-group>
  535. </div>
  536. </div>
  537. </div> -->
  538. <!-- 表格部分 -->
  539. <div style="max-height: 70vh; overflow-y: auto; padding: 10px 0">
  540. <table id="table" ref="table" class="table-wrap" @click="showRightClickMenu = false">
  541. <thead>
  542. <tr style="background: #f0f2f5">
  543. <th class="table-item" v-for="item in tableHeadData" :key="item.Value">{{ item.Value }}</th>
  544. </tr>
  545. </thead>
  546. <tbody>
  547. <tr v-for="(row, rindex) in tableData" :key="rindex" :class="rindex === selectRowIndex ? 'row-dark' : null" >
  548. <td
  549. class="table-item"
  550. v-for="(item, cindex) in row"
  551. :key="item"
  552. @click="handleSelectRow(rindex,item)"
  553. @contextmenu.prevent="rightClick(rindex, cindex, $event)"
  554. :data-rindex="rindex"
  555. :data-cindex="cindex"
  556. >
  557. <div v-if="item.Type === 'text'">
  558. <img
  559. class="choose-icon"
  560. width="14"
  561. src="../../../assets/img/constract/icon-1.png"
  562. alt=""
  563. v-if="item.HeadName=='品种'&&!item.RowDisable"
  564. />
  565. <span
  566. v-if="!item.CanEdit"
  567. v-html="item.Value"
  568. :style="{paddingLeft:item.HeadName=='品种'&&!item.RowDisable?'15px':null,display:'inline-block'}"
  569. ></span>
  570. <el-input v-else type="textarea" placeholder="请输入" v-model="item.Value"></el-input>
  571. </div>
  572. <div v-if="item.Type === 'select'">
  573. <div style="margin-bottom: 10px;font-size:15px">{{ item.Value }}</div>
  574. <div style="text-align: left">
  575. <el-checkbox-group v-model="tableSelectCheck[item.Value]">
  576. <el-checkbox style="font-size:14px;height: 19px" :label="opt.id"
  577. v-for="opt in tableSelectList[item.Value]" :key="opt.id" @change="handleTableSelectChange(item.Value, rindex, cindex)">{{ opt.name }}</el-checkbox>
  578. </el-checkbox-group>
  579. </div>
  580. </div>
  581. </td>
  582. </tr>
  583. </tbody>
  584. </table>
  585. </div>
  586. <div style="margin: 40px 0; text-align: center">
  587. <el-button type="primary" style="margin-right: 44px" @click="handleSave" size="large">保存</el-button>
  588. <el-button type="primary" plain @click="handleCancel" size="large">取消</el-button>
  589. </div>
  590. </el-dialog>
  591. <!-- 权限选择弹窗 -->
  592. <el-dialog v-model="showOpt" :modal-append-to-body="false" @close="closeOpt" v-dialogDrag>
  593. <template #header>
  594. <img style="vertical-align: middle;margin-right: 4px;" width="20" src="../../../assets/img/icons/add_icon.png" alt="" />
  595. <span style="vertical-align: middle">选择品种</span>
  596. </template>
  597. <div class="type-select-wrap">
  598. <div class="type-select-box">
  599. <div class="item flex" v-for="item in optionLists" :key="item.ClassifyName" style="margin-bottom: 30px">
  600. <el-checkbox
  601. :disabled="selectRowIndex === null"
  602. :indeterminate="item.CheckList.length > 0 && item.CheckList.length < item.Items.length"
  603. v-model="item.CheckAll"
  604. @change="handleCheckAllChange(item)"
  605. >{{ item.ClassifyName }}:</el-checkbox
  606. >
  607. <el-checkbox-group :disabled="selectRowIndex === null" v-model="item.CheckList" @change="handleCheckChange(item)">
  608. <el-checkbox :disabled="contractType!=='补充协议'&&tag.PermissionName==='宏观经济'" :label="tag.ChartPermissionId" v-for="tag in item.Items" :key="tag.ChartPermissionId">{{ tag.PermissionName }}</el-checkbox>
  609. </el-checkbox-group>
  610. </div>
  611. </div>
  612. <div style="text-align:center">
  613. <el-button type="primary" @click="handleSaveOpt" size="large">保存</el-button>
  614. <el-button type="primary" plain @click="closeOpt" size="large">取消</el-button>
  615. </div>
  616. </div>
  617. </el-dialog>
  618. <!-- 新增列填写表头弹窗 -->
  619. <el-dialog v-model="showInputHead" :modal-append-to-body="false" width="560px" top="20vh" v-dialogDrag>
  620. <template #header>
  621. <img style="vertical-align: middle" width="20" src="../../../assets/img/icons/add_icon.png" alt="" />
  622. <span style="vertical-align: middle">添加列</span>
  623. </template>
  624. <el-form :model="form" ref="headTitleForm" :rules="formRule" label-width="100px" @submit.native.prevent>
  625. <el-form-item label="表格名称" prop="title">
  626. <el-input v-model="form.title"></el-input>
  627. </el-form-item>
  628. <div style="text-align: center; margin-top: 70px; margin-bottom: 30px">
  629. <el-button type="primary" @click="handleConfirmHead" size="large">保存</el-button>
  630. <el-button type="primary" plain @click="handleCancelHead" size="large">取消</el-button>
  631. </div>
  632. </el-form>
  633. </el-dialog>
  634. <!-- 右键选项弹框 -->
  635. <div class="right-click-wrap" :style="position" v-show="showRightClickMenu">
  636. <div @click="handleUpdateTable(item.type)" v-for="item in rightClickCon" :key="item.type">{{ item.name }}</div>
  637. </div>
  638. <!-- 生成表格图片dom -->
  639. <table id="table-png" class="table-wrap">
  640. <thead>
  641. <tr style="background: #f0f2f5">
  642. <th class="table-item" v-for="item in tableHeadData" :key="item.Value" style="color: #333;font-size:15px">{{ item.Value }}</th>
  643. </tr>
  644. </thead>
  645. <tbody>
  646. <tr v-for="(row, rindex) in tableData" :key="rindex">
  647. <td class="table-item" v-for="item in row" :key="item" style="max-width: 300px">
  648. <div v-html="item.Value" style="color: #333;font-size:14px"></div>
  649. </td>
  650. </tr>
  651. </tbody>
  652. </table>
  653. </div>
  654. </template>
  655. <style>
  656. .self-dialog-c .el-dialog__header {
  657. background-color: #fff;
  658. padding: 15px 20px 0 20px !important;
  659. }
  660. .self-dialog-c{
  661. width: 60%;
  662. }
  663. /* .el-checkbox{
  664. margin-bottom: 20px;
  665. } */
  666. .table-item .el-checkbox__label {
  667. font-size: 14px !important;
  668. }
  669. .type-select-wrap .el-checkbox{
  670. margin-bottom: 10px;
  671. }
  672. .type-select-wrap .el-checkbox__label {
  673. font-size: 14px !important;
  674. }
  675. </style>
  676. <style lang="scss" scoped>
  677. .img{
  678. image-rendering: -moz-crisp-edges;
  679. image-rendering: -o-crisp-edges;
  680. image-rendering: -webkit-optimize-contrast;
  681. image-rendering: crisp-edges;
  682. -ms-interpolation-mode: nearest-neighbor;
  683. }
  684. .self-dialog-c .title {
  685. text-align: center;
  686. font-size: 18px;
  687. color: #333;
  688. }
  689. .flex {
  690. display: flex;
  691. }
  692. .table-wrap {
  693. position: relative;
  694. width: 100%;
  695. text-align: center;
  696. border-top: 1px solid #ebeef5;
  697. border-left: 1px solid #ebeef5;
  698. .row-dark {
  699. background-color: #deedff;
  700. }
  701. .table-item {
  702. padding: 20px 10px;
  703. border-right: 1px solid #ebeef5;
  704. border-bottom: 1px solid #ebeef5;
  705. min-width: 160px;
  706. position: relative;
  707. .table-item-mask {
  708. position: absolute;
  709. top: 0;
  710. left: 0;
  711. width: 100%;
  712. height: 100%;
  713. background-color: rgba(0, 0, 0, 0);
  714. z-index: 1000;
  715. }
  716. .choose-icon{
  717. position: absolute;
  718. top: 50%;
  719. left: 10px;
  720. transform:translateY(-50%);
  721. }
  722. }
  723. }
  724. .right-click-wrap {
  725. border-radius: 4px;
  726. background-color: #fff;
  727. padding: 10px 0;
  728. position: fixed;
  729. z-index: 5000;
  730. box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);
  731. line-height: 2;
  732. div {
  733. cursor: pointer;
  734. padding: 0 20px;
  735. &:hover {
  736. background-color: #ebeef5;
  737. }
  738. }
  739. }
  740. .type-select-wrap {
  741. margin-bottom: 30px;
  742. .type-select-box {
  743. // margin-top: 20px;
  744. padding: 20px 30px 0 30px;
  745. border: 1px dashed #aab4cc;
  746. border-radius: 4px;
  747. margin-bottom: 20px;
  748. }
  749. }
  750. #table-png {
  751. position: absolute;
  752. z-index: -10;
  753. top: -100%;
  754. left: 0;
  755. width: 1000px;
  756. }
  757. </style>