useResizeTable.js 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. import { ref,nextTick } from 'vue';
  2. import { isMobile } from '@/utils/utils';
  3. import _ from 'lodash';
  4. import { useRoute } from 'vue-router';
  5. import { ElMessage } from 'element-plus';
  6. export function useResizeTable(refName) {
  7. const route = useRoute();
  8. const minThresholdColWid = 30,
  9. minThresholdColH = 30,
  10. moveToDistance=5;
  11. const MobileCellWID = '120px',MobileCellH = '40px';
  12. const columnsWArr = ref([])
  13. const rowsHArr = ref([])
  14. const dragging = ref(false)
  15. const dragState=ref({})
  16. const dragStateHeight = ref({})
  17. const handleMouseMove = _.throttle(function(e,rIndex,cIndex) {
  18. if(isMobile()) return
  19. // if(rIndex!==0 && cIndex!==0) return
  20. let target = e.target
  21. if (!dragging.value) {
  22. let rect = target.getBoundingClientRect();
  23. if (rect.width > 12 && rect.right - e.pageX < moveToDistance /* && rIndex===0 */) {
  24. e.target.style.cursor = 'col-resize'
  25. }else if (rect.height > 12 && rect.bottom - e.pageY < moveToDistance /* && cIndex===0 */) {
  26. e.target.style.cursor = 'row-resize'
  27. } else if (!dragging.value) {
  28. e.target.style.cursor = ''
  29. }
  30. }
  31. },20)
  32. const handleMouseDown = _.throttle(function(e,rIndex,cIndex) {
  33. if(isMobile()) return
  34. document.onselectstart = function() { return false; };//解决拖动会选中文字的问题
  35. if(e.button!=0){
  36. return
  37. }
  38. let rect = e.target.getBoundingClientRect();
  39. if (rect.width > 12 && rect.right - e.pageX < moveToDistance /* && rIndex===0 */) {
  40. // 开始拖拽
  41. // 当前拖拽的列所在的表格
  42. // 当前所在列(单元格)
  43. let thEL = e.target
  44. dragging.value = true
  45. thEL.classList.add('noclick')
  46. dragState.value = {
  47. startMouseLeft: e.clientX, // 鼠标开始的地方
  48. columnWidth: rect.width, // th开始拖拽的宽度
  49. }
  50. document.onselectstart = function () {
  51. return false
  52. }
  53. document.ondragstart = function () {
  54. return false
  55. }
  56. const endResize = (event) => {
  57. if(dragging.value){
  58. // 拖拽完毕
  59. const { startMouseLeft, columnWidth } = dragState.value;
  60. const columnWidthDiff = event.clientX - startMouseLeft;
  61. columnsWArr.value[cIndex] = Math.max(minThresholdColWid,columnWidthDiff + columnWidth)
  62. // console.log(columnsWArr.value[cIndex])
  63. // console.log(e.target.offsetWidth)
  64. e.target.style.width = columnsWArr.value[cIndex]
  65. let widthTotal=0
  66. columnsWArr.value.forEach((item)=>{
  67. widthTotal+=item
  68. })
  69. // console.log(widthTotal)
  70. //多出来的宽度
  71. let otherWidth=refName.value.offsetWidth-widthTotal;
  72. columnsWArr.value.forEach((item,colIndex)=>{
  73. if(colIndex!=cIndex){
  74. columnsWArr.value[colIndex]= Number(parseFloat(item + otherWidth/columnsWArr.value.length-1).toFixed(2))
  75. }
  76. })
  77. setTimeout(()=>{
  78. initTableCellsWid('change')
  79. },500)
  80. // event.target.style.cursor = ''
  81. dragging.value = false
  82. dragState.value = {}
  83. }
  84. document.removeEventListener('mouseup', endResize);
  85. document.onselectstart = null
  86. document.ondragstart = null
  87. thEL.classList.remove('noclick')
  88. }
  89. document.addEventListener('mouseup', endResize);
  90. }
  91. if (rect.height > 12 && rect.bottom - e.pageY < moveToDistance /* && cIndex===0 */) {
  92. let thEL = e.target
  93. dragging.value = true
  94. thEL.classList.add('noclick')
  95. dragStateHeight.value = {
  96. startMouseTop: e.clientY,
  97. columnHeight: rect.height,
  98. }
  99. document.onselectstart = function () {
  100. return false
  101. }
  102. document.ondragstart = function () {
  103. return false
  104. }
  105. const endResize = (event) => {
  106. if(dragging.value){
  107. // 拖拽完毕
  108. const { startMouseTop, columnHeight } = dragStateHeight.value;
  109. const columnHeightDiff = event.clientY - startMouseTop
  110. rowsHArr.value[rIndex] = Math.max(minThresholdColH,columnHeightDiff + columnHeight)
  111. setTimeout(()=>{
  112. initTableCellsWid('change')
  113. },500)
  114. event.target.style.cursor = ''
  115. dragging.value = false
  116. dragStateHeight.value = {}
  117. }
  118. document.removeEventListener('mouseup', endResize);
  119. document.onselectstart = null
  120. document.ondragstart = null
  121. setTimeout(function () {
  122. thEL.classList.remove('noclick')
  123. }, 0)
  124. }
  125. document.addEventListener('mouseup', endResize);
  126. }
  127. },20)
  128. function initTableCellsWid(type='init',sceneConfig) {
  129. if(type=='init'){
  130. const { HeightList,WidthList } = sceneConfig;
  131. nextTick(()=>{
  132. const table = refName.value;
  133. if (table) {
  134. if(HeightList&&WidthList) {
  135. columnsWArr.value = WidthList.split(',').map(_ =>Number(_));
  136. rowsHArr.value = HeightList.split(',').map(_ =>Number(_));
  137. nextTick(() => {
  138. postSheetHeightMsg()
  139. })
  140. return
  141. }
  142. const rows=table.rows;
  143. const cells = table.rows[0].cells;
  144. let widthArr= Array.from(cells).map(cell => cell.offsetWidth);
  145. console.log(widthArr)
  146. let heightArr=Array.from(rows).map(row => row.offsetHeight);
  147. console.log(heightArr)
  148. columnsWArr.value=widthArr
  149. rowsHArr.value=heightArr
  150. nextTick(() => {
  151. postSheetHeightMsg()
  152. })
  153. }
  154. })
  155. }else if(type === 'change') {
  156. changeTableCellWidSave()
  157. }
  158. }
  159. function getSize(index,type) {
  160. if(type==='width'){
  161. return isMobile() ? MobileCellWID : (columnsWArr.value[index]?`${columnsWArr.value[index]}px`:'')
  162. }else{
  163. return isMobile() ? MobileCellH : (rowsHArr.value[index]?`${rowsHArr.value[index]}px`:'20px')
  164. }
  165. }
  166. /* 初始化抛出渲染后的表格高度 */
  167. function postSheetHeightMsg(type="") {
  168. let ele = document.getElementsByClassName('sheet-show-wrapper')[0];
  169. let params = {
  170. height: ele.offsetHeight,
  171. code: route.query.code,
  172. uid: route.query.uid||""
  173. }
  174. window.parent.postMessage(params,'*')
  175. type==='update'&&ElMessage.success('更新样式成功')
  176. }
  177. /* 拖拽后抛出参数到外面存储 只在报告编辑页可保存*/
  178. function changeTableCellWidSave() {
  179. if(!route.query.sourceId || !route.query.uid) return
  180. let params = {
  181. type:'changeCol',
  182. code: route.query.code,
  183. sourceId: route.query.sourceId,
  184. uid: route.query.uid,
  185. columnsWArr:columnsWArr.value.map(_ =>_),
  186. rowsHArr:rowsHArr.value.map(_=>_)
  187. }
  188. console.log(params)
  189. window.parent.postMessage(params,'*')
  190. }
  191. return {
  192. columnsWArr,
  193. rowsHArr,
  194. handleMouseMove,
  195. handleMouseDown,
  196. initTableCellsWid,
  197. getSize,
  198. postSheetHeightMsg
  199. }
  200. }