component.tsx 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251
  1. import { defineComponent, onMounted , ref, effect} from "vue";
  2. import { string } from "vue-types";
  3. import { useEditor } from "../../../..";
  4. import { View } from "../View";
  5. import { CompUI } from "../..";
  6. import { values } from "lodash";
  7. import { Angle } from "@/modules/editor/controllers/SelectCtrl/objects/mathUtils";
  8. import {editState} from "./toolbar"
  9. import { useCompData } from "../../defines/hook";
  10. import { CompPolygonObj } from ".";
  11. function findNearestPoint(points: number[][], w:number, h:number, sx:number, sy:number) {
  12. const n = points.length;
  13. let minv = 100000;
  14. let minIndex = -1;
  15. const positions = []
  16. for(let i=0; i<n; i++) {
  17. const p = points[i];
  18. const x = w *p[0];
  19. const y = h *p[1];
  20. positions.push([x, y]);
  21. const ln = (sx-x)*(sx-x) + (sy-y)*(sy-y);
  22. if ( ln < minv ) {
  23. minIndex = i;
  24. minv = ln;
  25. if(!(sx < (x -10) || sx > (x + 10) || sy < (y -10) || sy > (y+10) ) ) {
  26. return {clicked: true, index: i};
  27. }
  28. }
  29. }
  30. let preIndex = minIndex -1;
  31. if (preIndex < 0 ) preIndex = n-1;
  32. let afterIndex = minIndex + 1;
  33. if ( afterIndex >= n) afterIndex = 0;
  34. const currV = {x:sx-positions[minIndex][0], y:sy-positions[minIndex][1]};
  35. const preV = {x:positions[preIndex][0]-positions[minIndex][0], y:positions[preIndex][1]-positions[minIndex][1]}
  36. const afterV = {x:positions[afterIndex][0]-positions[minIndex][0], y:positions[afterIndex][1]-positions[minIndex][1]}
  37. const a1 = Angle(currV, preV)
  38. const a2 = Angle(currV, afterV);
  39. if (a1 < a2) {
  40. return {index: preIndex};
  41. }
  42. return {index: minIndex};
  43. }
  44. export const Component = defineComponent({
  45. props: {
  46. compId: string().isRequired,
  47. },
  48. setup(props) {
  49. const { helper, controls , store} = useEditor();
  50. const data = useCompData<CompPolygonObj>(props.compId);
  51. const canvasRef = ref<HTMLCanvasElement>();
  52. onMounted(()=>{
  53. // draw();
  54. let isDragingIndex = -1;
  55. canvasRef.value?.addEventListener("dblclick", function(e:MouseEvent){
  56. const x = helper.pxToDesignSize(e.offsetX )
  57. const y = helper.pxToDesignSize(e.offsetY)
  58. const points = data.value.points as number[][];
  59. const w = data.layout.size[0];
  60. const h = data.layout.size[1];
  61. //判断直线相交求解点到直线的距离
  62. const ret = findNearestPoint(data.value.points, w , h, x, y);
  63. //判断是否
  64. console.log("dbclick=>xxxxx", ret);
  65. if (ret.clicked) {//点击删除
  66. points.splice(ret.index, 1);
  67. if (points.length < 4) {
  68. return;
  69. }
  70. editState.index = 0;
  71. } else {
  72. points.splice(ret.index+1, 0, [x / w, y / h]);
  73. editState.index = ret.index + 1;
  74. }
  75. draw();
  76. })
  77. canvasRef.value?.addEventListener("mousedown", function(e:MouseEvent){
  78. if (store.currCompId != props.compId) return;
  79. const el = canvasRef.value as HTMLCanvasElement;
  80. const x = helper.pxToDesignSize(e.offsetX )
  81. const y = helper.pxToDesignSize(e.offsetY)
  82. console.log(x , y);
  83. const points = data.value.points as number[][];
  84. const width = data.layout.size[0];
  85. const height = data.layout.size[1];
  86. let n = points.length;
  87. isDragingIndex = -1;
  88. let initX = 0;
  89. let initY = 0;
  90. while(n--) {
  91. const p = points[n];
  92. const px = width * p[0];
  93. const py = height * p[1];
  94. if(!(x < (px -10) || x > (px + 10) || y < (py -10) || y > (py+10) ) ) {
  95. isDragingIndex = n;
  96. initX = p[0]
  97. initY = p[1]
  98. break;
  99. }
  100. }
  101. const dragingX = x;
  102. const dragingY = y;
  103. if (isDragingIndex != -1) {
  104. e.preventDefault();
  105. e.stopPropagation();
  106. editState.index = isDragingIndex;
  107. const move = function (e:MouseEvent){
  108. if ( isDragingIndex == -1) return;
  109. const offx = (helper.pxToDesignSize(e.offsetX ) - dragingX) / width;
  110. const offy = (helper.pxToDesignSize(e.offsetY ) - dragingY) / height;
  111. points[isDragingIndex][0] = initX + offx;
  112. points[isDragingIndex][1] = initY + offy;
  113. draw();
  114. }
  115. el.addEventListener("mousemove", move)
  116. el.addEventListener("mouseup", function(e:MouseEvent){
  117. el.removeEventListener("mousemove", move);
  118. });
  119. el.addEventListener("mouseleave", function(e:MouseEvent){
  120. el.removeEventListener("mousemove", move);
  121. })
  122. }
  123. }, )
  124. effect(draw);
  125. })
  126. function draw() {
  127. const canvas = canvasRef.value as HTMLCanvasElement;
  128. if (!canvas) return;
  129. const ctx = canvasRef.value?.getContext("2d") as CanvasRenderingContext2D;
  130. const width = data.layout.size[0];
  131. const height = data.layout.size[1];
  132. canvas.width = Math.ceil(Math.max(1, width));
  133. canvas.height = Math.ceil(Math.max(1, height));
  134. ctx.clearRect(0, 0, canvas.width, canvas.height);
  135. ctx.lineWidth = data.value.lineWidth;
  136. ctx.save();
  137. if (data.value.reverseFill) {
  138. ctx.fillStyle = data.value.fillColor;
  139. ctx.fillRect(0, 0, canvas.width, canvas.height);
  140. }
  141. // if (data.value.res)
  142. ctx.strokeStyle = data.value.lineColor;
  143. const padding = 0.2;
  144. ctx.lineJoin = "round";
  145. ctx.beginPath();
  146. let points = data.value.points as number[][];
  147. if (data.value.points.length == 0) {
  148. data.value.points = [[padding, padding], [1 - padding, padding], [1 - padding, 1 - padding] , [padding, 1 - padding]];
  149. points = data.value.points;
  150. }
  151. points.forEach((p, index)=>{
  152. const x = width * p[0];
  153. const y = height * p[1];
  154. index == 0 ? ctx.moveTo(x, y) : ctx.lineTo(x, y);
  155. })
  156. if (data.value.isClose) {
  157. ctx.closePath();
  158. }
  159. if (data.value.lineWidth !== 0) {
  160. ctx.stroke();
  161. }
  162. if (data.value.isFill) {
  163. let bColor = data.value.fillColor;
  164. if (!bColor) bColor = data.value.lineColor;
  165. ctx.fillStyle = bColor;
  166. if (data.value.reverseFill) {
  167. ctx.fillStyle = "red";
  168. // 设置剪切区域
  169. ctx.clip();
  170. // 将当前路径指定的区域颜色扣除
  171. ctx.globalCompositeOperation = "destination-out";
  172. }
  173. ctx.fill();
  174. }
  175. ctx.restore();
  176. if (store.isEditMode && store.currCompId == props.compId) {
  177. points.forEach((p, index)=>{
  178. const x = width * p[0];
  179. const y = height * p[1];
  180. ctx.fillStyle = "red"
  181. ctx.beginPath();
  182. ctx.arc(x, y, 5, 0, Math.PI*2);
  183. ctx.fill();
  184. })
  185. }
  186. }
  187. return () => (
  188. <View compId={props.compId}>
  189. <canvas ref={canvasRef} style={{width:"100%", height: "100%"}}> </canvas>
  190. </View>
  191. );
  192. },
  193. });