assistMagnetCtrl.ts 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. import { SelectCtrl } from ".";
  2. import { ObjsContainer } from "./ObjsContainer";
  3. import { CompObject } from "./compObj";
  4. /**
  5. * 吸附功能实现
  6. */
  7. export class AssistMagnetCtrl {
  8. ctrl: SelectCtrl
  9. enable = true;
  10. constructor(ctrl: SelectCtrl) {
  11. this.ctrl = ctrl;
  12. }
  13. rulerYs:number[]= [];
  14. rulerXs:number[]= [];
  15. clientX = 0;
  16. clientY = 0;
  17. currRefYs:number[] = [];
  18. currRefXs:number[] = []
  19. otherSlides : CompObject[] = [];
  20. getMinValues(srcYs:number[], refYs: number[]) {
  21. //找出最近的参考线
  22. let n = srcYs.length;
  23. let out :{min: number, v:number, o:number}[] = [];
  24. while(n--) {
  25. const srcY = srcYs[n];
  26. let k = refYs.length;
  27. let currMinY = 10000;
  28. let currMinRefY = -10000;
  29. let lastOffset = 0;
  30. for (let i=0; i<k; i++) {
  31. const o = srcY - refYs[i]
  32. if (Math.abs(o) <= currMinY) {
  33. currMinY = Math.abs(o);
  34. currMinRefY = refYs[i]
  35. lastOffset = o;
  36. continue;
  37. }
  38. break
  39. }
  40. out.push({min: currMinY, v: currMinRefY, o: lastOffset})
  41. }
  42. out = out.sort((a, b)=>(a.min-b.min))
  43. let min = out[0].min;
  44. const sames = [out[0].v];
  45. let m = out.length;
  46. for(let i=1; i<m; i++) {
  47. if (Math.abs(out[i].min- min) < 1) {
  48. sames.push(out[i].v);
  49. } else {
  50. break;
  51. }
  52. }
  53. return {offset: out[0].o, min: min, values: sames};
  54. }
  55. test(e:MouseEvent) {
  56. this.clientX = e.clientX;
  57. this.clientY = e.clientY;
  58. if ( !this.enable ) return;
  59. const eps = 2;
  60. //在屏幕空间中计算
  61. const bund = this.ctrl.getSelectBound();
  62. const box = this.ctrl.getCurrCardViewPortBox();
  63. //包围盒的上 中 下 三条横线
  64. const srcY1 = (bund.y + box.top)*2.0
  65. const srcYs = [srcY1, srcY1 + bund.height, srcY1 + bund.height*2];
  66. if (this.rulerYs.length < 1) {
  67. this.rulerYs.push(box.top *2.0) //card上边线
  68. this.rulerYs.push(box.top *2.0 + box.height )//中线
  69. this.rulerYs.push((box.top + box.height) * 2.0) //下边线
  70. //所有参考线
  71. const rulers = this.ctrl.assistRuler?.rulers || [];
  72. let n = rulers.length;
  73. while(n--) {
  74. if (rulers[n].horz != undefined)
  75. this.rulerYs.push(box.top *2 + (rulers[n].horz as number));
  76. }
  77. this.otherSlides.forEach(item=>{
  78. const b = item.getBox();
  79. const y = (b.y + box.top) *2.0
  80. if (this.rulerYs.indexOf(y) == -1) {
  81. this.rulerYs.push(y)
  82. }
  83. const y1 = y + b.h
  84. if (this.rulerYs.indexOf(y1) == -1) {
  85. this.rulerYs.push(y1)
  86. }
  87. const y2 = y + b.h *2
  88. if (this.rulerYs.indexOf(y2) == -1) {
  89. this.rulerYs.push(y2)
  90. }
  91. })
  92. this.rulerYs = this.rulerYs.sort((a,b)=>a-b)
  93. }
  94. this.currRefYs = [];
  95. const {offset, min, values} = this.getMinValues(srcYs, this.rulerYs)
  96. if (min < eps) {//找到了最近点,进行磁铁吸附
  97. this.clientY = e.clientY - offset / 2.0;
  98. this.currRefYs = values;
  99. }
  100. //包围盒的左 中 右三条横线
  101. const srcX1 = (bund.x + box.left)*2.0
  102. const srcXs = [srcX1, srcX1 + bund.width, srcX1 + bund.width*2];
  103. if (this.rulerXs.length < 1) {
  104. this.rulerXs.push(box.left *2.0) //card上边线
  105. this.rulerXs.push(box.left *2.0 + box.width )//中线
  106. this.rulerXs.push((box.left + box.width) * 2.0) //下边线
  107. //所有参考线
  108. const rulers = this.ctrl.assistRuler?.rulers || [];
  109. let n = rulers.length;
  110. while(n--) {
  111. if (rulers[n].verz != undefined)
  112. this.rulerXs.push(box.left *2 + (rulers[n].verz as number));
  113. }
  114. //添加card内的其他对象
  115. this.otherSlides.forEach(item=>{
  116. const b = item.getBox();
  117. const x = (b.x + box.left) *2.0
  118. if (this.rulerXs.indexOf(x) == -1) {
  119. this.rulerXs.push(x)
  120. }
  121. const x1 = x + b.w
  122. if (this.rulerXs.indexOf(x1) == -1) {
  123. this.rulerXs.push(x1)
  124. }
  125. const x2 = x + b.w *2
  126. if (this.rulerXs.indexOf(x2) == -1) {
  127. this.rulerXs.push(x2)
  128. }
  129. })
  130. this.rulerXs = this.rulerXs.sort((a,b)=>a-b)
  131. }
  132. this.currRefXs = [];
  133. const ret = this.getMinValues(srcXs, this.rulerXs)
  134. if (ret.min < eps) {//找到了最近点,进行磁铁吸附
  135. this.clientX = e.clientX - ret.offset / 2.0;
  136. this.currRefXs = ret.values;
  137. }
  138. }
  139. onBeforeTest() {
  140. this.otherSlides = this.ctrl.getUnSelectChilds();
  141. }
  142. onMouseUp() {
  143. this.rulerYs = [];
  144. this.rulerXs = [];
  145. this.currRefXs = [];
  146. this.currRefYs = [];
  147. this.otherSlides = [];
  148. }
  149. draw() {
  150. if (!this.enable || !(this.currRefXs.length > 0 || this.currRefYs.length > 0) ) {
  151. return;
  152. }
  153. const ctx = this.ctrl._selCtx;
  154. let n = this.currRefXs.length;
  155. ctx.lineWidth = 2;
  156. ctx.strokeStyle = "orange";
  157. // 清除虚线样式
  158. // ctx.setLineDash([]);
  159. while(n--) {
  160. let x = this.currRefXs[n]
  161. ctx.beginPath();
  162. ctx.moveTo(x, 0)
  163. ctx.lineTo(x, this.ctrl._selCanvaseSize.h);
  164. ctx.stroke();
  165. ctx.closePath();
  166. }
  167. n = this.currRefYs.length;
  168. while(n--) {
  169. let y = this.currRefYs[n]
  170. ctx.beginPath();
  171. ctx.moveTo(0, y)
  172. ctx.lineTo(this.ctrl._selCanvaseSize.h, y);
  173. ctx.stroke();
  174. ctx.closePath();
  175. }
  176. }
  177. }