gizemo.ts 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509
  1. import { Bounds } from './objects/bounds';
  2. import { Rectangle } from './objects/rectangle';
  3. import { Container } from './objects/container';
  4. import { CompObject } from './compObj';
  5. import { Matrix } from './matrix';
  6. import { Events } from "queenjs"
  7. import { RxValue } from '../ReactCtrl/rxValue';
  8. import { DesignComp } from '../../objects/DesignTemp/DesignComp';
  9. import { Transform } from './objects/transform';
  10. import { history } from '../../objects/DesignTemp/factory';
  11. import { designSizeToPx } from '../../module/utils';
  12. type FnGetCompObj = (id:string)=>DesignComp;
  13. const AngleNames = [
  14. "scaleBottomright",
  15. "scaleBottomleft",
  16. "scaleTopleft",
  17. "scaleTopright",
  18. ];
  19. const EdgeNames = ["scaleright", "scaleleft", "scalebottom", "scaletop"];
  20. export class Gizemo extends Events {
  21. aabb = new Bounds();
  22. tempBound = new Bounds();
  23. parent = new Container();
  24. rect = new Rectangle();
  25. pivotIndex = 0;
  26. selected:CompObject[] = [];
  27. history = history;
  28. state = RxValue.create({
  29. transform: {
  30. selected: [] as string[],
  31. x: 0,
  32. y: 0,
  33. w: 0,
  34. h: 0,
  35. sx: 1,
  36. sy: 1,
  37. px: 0,
  38. py: 0,
  39. r: 0,
  40. },
  41. lastId: "",
  42. scaleWidth: 1,
  43. scaleHeight: 1,
  44. mouse: ""
  45. }, this.history)
  46. lastSelChanged = false;
  47. getCompObj: FnGetCompObj;
  48. constructor(getCompObj: FnGetCompObj) {
  49. super()
  50. this.getCompObj = getCompObj;
  51. this.parent.sortableChildren = false;
  52. this.initGizmoEvent();
  53. }
  54. get selectedIds() {
  55. return this.selected.map(item=>item.comp.id);
  56. }
  57. getBound() {
  58. //加上parent的旋转和平移
  59. this.tempBound.clear();
  60. this.tempBound.addFrame(this.parent.transform, 0, 0, this.width, this.height);
  61. return this.tempBound.getRectangle();
  62. }
  63. get width() {
  64. return this.rect.width
  65. }
  66. get height() {
  67. return this.rect.height;
  68. }
  69. childSubs = [] as any[];
  70. addChildWorldNoChange(child: CompObject) {
  71. const m = this.parent.worldTransform.clone();
  72. m.invert();
  73. const wm = child.worldTransform.clone();
  74. wm.prepend(m);
  75. child.setMatrix(wm);
  76. this.parent.addChild(child);
  77. child.updateTransform();
  78. }
  79. initGizmoEvent() {
  80. this.state.onTransformChanged((t, ot)=>{
  81. if (t.selected != ot.selected) {
  82. this.destroy();
  83. this.rect.width = t.w;
  84. this.rect.height = t.h;
  85. this.parent.width = t.w;
  86. this.parent.height = t.h;
  87. this.parent.setTransform(t.x, t.y, t.sx, t.sy, t.r, 0, 0, t.px, t.py)
  88. this.parent.updateTransform();
  89. const objs :any= [];
  90. t.selected.forEach((oid) => {
  91. const obj = new CompObject(this.getCompObj(oid), true)
  92. let box = obj.calculateBounds();
  93. this.aabb.addBounds(box);
  94. objs.push(obj);
  95. this.addChildWorldNoChange(obj);
  96. let first = true;
  97. const s= obj.state.onTransformChanged(()=>{
  98. if (!first) this.updateSize();
  99. first = false;
  100. })
  101. this.childSubs.push(s);
  102. });
  103. this.selected = objs;
  104. this.parent.updateTransform();
  105. this.updateCompState();
  106. this.emit("change");
  107. return;
  108. }
  109. if (t.selected.length < 1) {
  110. this.destroy();
  111. this.emit("change");
  112. return;
  113. }
  114. //只是移动操作
  115. this.parent.x = t.x;
  116. this.parent.y = t.y;
  117. this.rect.width = t.w;
  118. this.rect.height = t.h;
  119. this.parent.scale.x = t.sx;
  120. this.parent.scale.y = t.sy;
  121. this.parent.rotation = t.r;
  122. this.parent.pivot.x = t.px;
  123. this.parent.pivot.y = t.py;
  124. this.parent._boundsID++;
  125. this.parent.updateTransform();
  126. this.updateCompState();
  127. this.emit("change");
  128. })
  129. }
  130. initScaleSizeEvent() {
  131. this.state.onScaleWidthChanged((x)=>{
  132. const prew = this.parent.scale.x * this.rect.width;
  133. //怎么改对应的偏移值
  134. this.parent.scale.x = x
  135. const w = this.parent.scale.x * this.rect.width;
  136. this.parent.updateTransform();
  137. this.applyChildWidth({scaleX: w / prew })
  138. this.updateCompState();
  139. this.emit("change");
  140. })
  141. }
  142. selectObjs(ids: string[], force=false) {
  143. let noChange = false;
  144. if (!force && this.selected.length == ids.length ) {
  145. if (ids.length == 0) {
  146. noChange = true;
  147. } else if (ids.length == 1) {
  148. if (ids[0] == this.selected[0].comp.id) {
  149. noChange = true;
  150. }
  151. } else {
  152. const id1 = ids.sort().join("");
  153. const id2 = this.selected.map(item=>item.comp.id).sort().join("");
  154. if (id1 == id2) {
  155. noChange = true;
  156. }
  157. }
  158. }
  159. this.lastSelChanged = !noChange;
  160. console.log("changing....", this.lastSelChanged);
  161. if (noChange) {
  162. return;
  163. }
  164. let n = ids.length;
  165. const selected = ids;
  166. let w = 0, h = 0, x = 0, y = 0, sx = 1 , sy = 1, r = 0, px=0, py=0;
  167. if (n == 1) { //单对象
  168. let obj = new CompObject(this.getCompObj(selected[0]));
  169. w = obj.width;
  170. h = obj.height;
  171. const transform = new Transform();
  172. transform.setFromMatrix(obj.worldTransform);
  173. x = transform.position.x;
  174. y = transform.position.y;
  175. sx = transform.scale.x;
  176. sy = transform.scale.y;
  177. r = transform.rotation;
  178. } else if (n>1) {
  179. const bound = new Bounds();
  180. while (n--) {
  181. let obj = new CompObject(this.getCompObj(selected[n]) );
  182. let box = obj.calculateBounds();
  183. bound.addBounds(box);
  184. }
  185. //构建当前对象的转换矩阵
  186. let rect = new Rectangle();
  187. bound.getRectangle(rect);
  188. this.rect = rect;
  189. w = rect.width;
  190. h = rect.height;
  191. x = rect.x;
  192. y = rect.y;
  193. }
  194. this.state.setTransform({selected: ids, x, y, w, h, sx,sy, r, px, py})
  195. }
  196. testClick(sx:number,sy:number)
  197. {
  198. let w = this.width;
  199. let h = this.height;
  200. let local = {x:0,y:0} as any;
  201. this.parent.worldTransform.applyInverse({x:sx,y:sy} as any, local);
  202. if( local.x < 0 || local.x > w ) return false;
  203. if( local.y < 0 || local.y > h ) return false;
  204. return true;
  205. }
  206. rotate(r:number, submit = false) {
  207. const t = {...this.state.transform};
  208. t.r = r;
  209. this.state.setTransform(t);
  210. if (submit) this.history.submit();
  211. }
  212. //index
  213. // 0 --5-- 1
  214. // | |
  215. // 8 4 6
  216. // | |
  217. // 3 --7---2
  218. setPivot(index:number) {
  219. let rect = this.rect;
  220. let pivots = [ { x: 0, y: 0 }, { x: rect.width, y: 0 },
  221. { x: rect.width, y: rect.height },
  222. { x: 0, y: rect.height },
  223. { x: rect.width / 2.0, y: rect.height / 2.0 },
  224. { x: rect.width / 2.0, y: 0 },
  225. { x: rect.width , y: rect.height / 2.0 },
  226. { x: rect.width / 2.0 , y: rect.height},
  227. { x: 0, y: rect.height / 2.0},
  228. ];
  229. let targetPivot = pivots[index];
  230. let point = { x: targetPivot.x, y: targetPivot.y } as any;
  231. this.parent.worldTransform.apply(point, point);
  232. const t = {...this.state.transform};
  233. t.px = targetPivot.x;
  234. t.py = targetPivot.y;
  235. t.x = point.x;
  236. t.y = point.y;
  237. this.pivotIndex = index;
  238. this.state.setTransform(t);
  239. return { x: point.x, y: point.y};
  240. }
  241. setPivot2(x:number, y:number) {
  242. let targetPivot = {x, y}
  243. let point = { x, y } as any;
  244. this.parent.worldTransform.apply(point, point);
  245. const t = {...this.state.transform};
  246. t.px = targetPivot.x;
  247. t.py = targetPivot.y;
  248. t.x = point.x;
  249. t.y = point.y;
  250. this.state.setTransform(t);
  251. return { x: point.x, y: point.y};
  252. }
  253. getPivotXY(index:number) {
  254. let rect = this.rect;
  255. let pivots = [ { x: 0, y: 0 }, { x: rect.width, y: 0 },
  256. { x: rect.width, y: rect.height },
  257. { x: 0, y: rect.height },
  258. { x: rect.width / 2.0, y: rect.height / 2.0 },
  259. { x: rect.width / 2.0, y: 0 },
  260. { x: rect.width , y: rect.height / 2.0 },
  261. { x: rect.width / 2.0 , y: rect.height},
  262. { x: 0, y: rect.height / 2.0},
  263. ];
  264. let targetPivot = pivots[index];
  265. let point = { x: targetPivot.x, y: targetPivot.y } as any;
  266. this.parent.worldTransform.apply(point, point);
  267. let yIndex = 0, xIndex = 0;
  268. if (index == 3) {
  269. yIndex = 0;
  270. xIndex = 2;
  271. } else if (index == 0) {
  272. yIndex = 3;
  273. xIndex = 1;
  274. } else if (index == 1) {
  275. yIndex = 2;
  276. xIndex = 0;
  277. } else if (index == 2) {
  278. yIndex = 1;
  279. xIndex = 3;
  280. }
  281. let pointY = pivots[yIndex];
  282. let pY = { x: pointY.x, y: pointY.y } as any;
  283. this.parent.worldTransform.apply(pY, pY);
  284. let pointX = pivots[xIndex];
  285. let pX = { x: pointX.x, y: pointX.y } as any;
  286. this.parent.worldTransform.apply(pX, pX);
  287. let xVec = { x: (pX.x - point.x), y: (pX.y - point.y) };
  288. let yVec = { x: (pY.x - point.x), y: (pY.y - point.y) };
  289. return { x: xVec, y: yVec };
  290. }
  291. scale(x:number, y:number, submit = false) {
  292. const t = {...this.state.transform};
  293. t.sx = x;
  294. t.sy = y;
  295. this.state.setTransform(t);
  296. if (submit) this.history.submit();
  297. }
  298. scaleX(x:number, submit= false) {
  299. const t = {...this.state.transform};
  300. t.sx = x;
  301. this.state.setTransform(t);
  302. if (submit) this.history.submit();
  303. }
  304. scaleY(y:number, submit= false) {
  305. const t = {...this.state.transform};
  306. t.sy = y;
  307. this.state.setTransform(t);
  308. if (submit) this.history.submit();
  309. }
  310. applyChildWidth(option:{scaleX?:number, scaleY?:number}) {
  311. if (this.selected.length < 1) return;
  312. const obj = this.selected[0];
  313. //先移除
  314. this.parent.removeChildWorldNoChange(obj);
  315. const m = new Matrix();
  316. m.scale(option.scaleX ? option.scaleX : 1, option.scaleY? option.scaleY : 1)
  317. m.invert();
  318. m.prepend(obj.worldTransform)
  319. obj.transform.setFromMatrix(m)
  320. if (option.scaleX) {
  321. obj.width = option.scaleX * obj.width;
  322. }
  323. if (option.scaleY) {
  324. obj.height = option.scaleY * obj.height;
  325. }
  326. obj.updateTransform();
  327. this.parent.addChildWorldNoChange(obj);
  328. }
  329. scaleSize(x:number, y:number) {
  330. let preW = this.parent.scale.x * this.rect.width;
  331. let preH = this.parent.scale.y * this.rect.height;
  332. this.parent.scale.y = y
  333. this.parent.scale.x = x
  334. const Width = this.parent.scale.x * this.rect.width;
  335. const Height = this.parent.scale.y * this.rect.height;
  336. this.parent.updateTransform();
  337. this.applyChildWidth({scaleX: Width / preW, scaleY: Height/preH})
  338. this.updateCompState();
  339. }
  340. scaleWidth(x:number) {
  341. this.state.setScaleWidth(x);
  342. }
  343. scaleHeight(y:number) {
  344. const preH = this.parent.scale.y * this.rect.height;
  345. this.parent.scale.y = y
  346. const h = this.parent.scale.y * this.rect.height;
  347. this.parent.updateTransform();
  348. this.applyChildWidth({scaleY: h / preH});
  349. this.updateCompState();
  350. }
  351. translate(xOffset:number, yOffset:number, submit=false) {
  352. const t = {...this.state.transform};
  353. t.x += xOffset;
  354. t.y += yOffset;
  355. this.state.setTransform(t);
  356. if (submit) this.history.submit();
  357. }
  358. destroy() {//选中的对象坐标转到画布空间坐标
  359. let selected = this.selected;
  360. let n = selected.length;
  361. while (n--) {
  362. let child = selected[n];
  363. this.parent.removeChildWorldNoChange(child)
  364. child.comp.layout.transformMatrix = child.worldTransform.getMatrixStr();
  365. }
  366. this.selected = [];
  367. this.parent.removeChildren(0, this.parent.children.length);
  368. //获取选择对象的aabb(画布空间)
  369. this.aabb.clear();
  370. this.childSubs.forEach(s=>{
  371. s.unsubscribe();
  372. })
  373. this.childSubs = [];
  374. }
  375. updateCompState() {
  376. let n = this.selected.length;
  377. while (n--) {
  378. let child = this.selected[n];
  379. if (child.comp) child.comp.layout.transformMatrix = child.worldTransform.getMatrixStr();
  380. }
  381. }
  382. reSelectCurrElemnts() {
  383. let t = {...this.state.transform}
  384. t.selected = this.state.transform.selected.slice(0)
  385. this.state.setTransform(t);
  386. }
  387. updateSize() {
  388. let selected = this.selected;
  389. let n = selected.length;
  390. this.setPivot(0)
  391. this.aabb.clear();
  392. this.selected.forEach((obj) => {
  393. let box = obj.calculateBounds();
  394. this.aabb.addBounds(box);
  395. });
  396. this.aabb.getRectangle(this.rect);
  397. let x = this.rect.x, y = this.rect.y, sx = 1, sy= 1;
  398. if ( n == 1) {
  399. const s = selected[0].worldTransform.getScale();
  400. sx = s.x;
  401. sy = s.y;
  402. }
  403. const r = n == 1 ?selected[0].rotation : 0;
  404. this.parent.setTransform(x, y, sx, sy, r, 0, 0, 0, 0);
  405. this.parent.updateTransform();
  406. const w = n == 1 ? selected[0].width : this.rect.width;
  407. const h = n == 1 ? selected[0].height : this.rect.height;
  408. this.rect.width = w;
  409. this.rect.height = h;
  410. this.state.transform.w = w
  411. this.state.transform.h = h;
  412. this.state.transform.x = x;
  413. this.state.transform.y = y;
  414. this.state.transform.sx = sx;
  415. this.state.transform.sy = sy;
  416. this.state.transform.r = r;
  417. this.state.transform.px = 0;
  418. this.state.transform.py = 0;
  419. this.emit("change");
  420. }
  421. }