ObjsContainer.ts 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  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. export class ObjsContainer {
  7. aabb = new Bounds();
  8. tempBound = new Bounds();
  9. parent = new Container();
  10. rect = new Rectangle();
  11. pivotIndex = 0;
  12. selected:CompObject[] = [];
  13. constructor(selected:any[]) {
  14. this.selected = selected;
  15. this.parent.sortableChildren = false;
  16. if (selected.length > 0) {
  17. this.init();
  18. }
  19. }
  20. clone() {
  21. const ret = new ObjsContainer([]);
  22. const parent = this.parent;
  23. ret.parent.pivot = parent.pivot;
  24. ret.parent.skew = parent.skew;
  25. ret.parent.scale = parent.scale;
  26. ret.parent.rotation = parent.rotation;
  27. ret.parent.position = parent.position;
  28. ret.rect.copyFrom(this.rect);
  29. ret.parent.updateTransform();
  30. ret.setPivot(this.pivotIndex);
  31. return ret;
  32. }
  33. getBound() {
  34. //加上parent的旋转和平移
  35. this.tempBound.clear();
  36. this.tempBound.addFrame(this.parent.transform, 0, 0, this.rect.width, this.rect.height);
  37. return this.tempBound.getRectangle();
  38. }
  39. get width() {
  40. return this.rect.width
  41. }
  42. get height() {
  43. return this.rect.height;
  44. }
  45. testClick(sx:number,sy:number)
  46. {
  47. let w = this.width;
  48. let h = this.height;
  49. let local = {x:0,y:0} as any;
  50. this.parent.worldTransform.applyInverse({x:sx,y:sy} as any, local);
  51. if( local.x < 0 || local.x > w ) return false;
  52. if( local.y < 0 || local.y > h ) return false;
  53. return true;
  54. }
  55. init() {
  56. this.parent.removeChildren(0, this.parent.children.length);
  57. //获取选择对象的aabb(画布空间)
  58. this.aabb.clear();
  59. let n = this.selected.length;
  60. const selected = this.selected;
  61. const selectedRotation = 0;
  62. if (n == 1) { //单对象
  63. let obj = selected[0];
  64. let rect = new Rectangle(0, 0, obj.width , obj.height);
  65. this.rect = rect;
  66. this.parent.transform.setFromMatrix(obj.worldTransform);
  67. this.parent.updateTransform();
  68. this.parent.addChildWorldNoChange(obj);
  69. this.parent.updateTransform();
  70. return;
  71. }
  72. while (n--) {
  73. let obj = selected[n];
  74. let box = obj.calculateBounds();
  75. this.aabb.addBounds(box);
  76. }
  77. //构建当前对象的转换矩阵
  78. let rect = new Rectangle();
  79. this.aabb.getRectangle(rect);
  80. let center = rect.center;
  81. this.rect = rect;
  82. //设置位置
  83. let p = this.parent.position;
  84. p.x = center.x;
  85. p.y = center.y;
  86. //设置旋转中心点
  87. let pivot = this.parent.pivot;
  88. pivot.x = rect.width / 2;
  89. pivot.y = rect.height / 2;
  90. //设置旋转
  91. this.parent.scale = { x: 1, y: 1 } as any;
  92. this.parent.width = rect.width;
  93. this.parent.height = rect.height;
  94. //设置旋转
  95. this.parent.rotation = selectedRotation;
  96. // this.parent.scale = {x:1.5, y:1.5};
  97. this.parent.updateTransform();
  98. //选择的对象坐标从画布空间转到选框坐标,
  99. n = selected.length;
  100. for (let i = 0; i < n; i++) {
  101. let obj = selected[i];
  102. this.parent.addChildWorldNoChange(obj);
  103. }
  104. this.parent.updateTransform();
  105. }
  106. rotate(r:number) {
  107. this.parent.rotation = r;
  108. this.parent._boundsID++;
  109. this.parent.updateTransform();
  110. this.updateCompState();
  111. }
  112. //index
  113. // 0 ---- 1
  114. // | 4 |
  115. // 3 -----2
  116. setPivot(index:number) {
  117. let rect = this.rect;
  118. let pivots = [{ x: 0, y: 0 }, { x: rect.width, y: 0 }, { x: rect.width, y: rect.height }, { x: 0, y: rect.height }, { x: rect.width / 2, y: rect.height / 2 }];
  119. let targetPivot = pivots[index];
  120. let point = { x: targetPivot.x, y: targetPivot.y } as any;
  121. this.parent.worldTransform.apply(point, point);
  122. this.parent.pivot = targetPivot as any;
  123. this.parent.position.x = point.x;
  124. this.parent.position.y = point.y;
  125. this.parent.updateTransform();
  126. this.pivotIndex = index;
  127. return { x: point.x, y: point.y};
  128. }
  129. setPivot2(x:number, y:number) {
  130. let targetPivot = {x, y}
  131. let point = { x, y } as any;
  132. this.parent.worldTransform.apply(point, point);
  133. this.parent.pivot = targetPivot as any;
  134. this.parent.position.x = point.x;
  135. this.parent.position.y = point.y;
  136. this.parent.updateTransform();
  137. return { x: point.x, y: point.y};
  138. }
  139. getPivotXY(index:number) {
  140. let rect = this.rect;
  141. let pivots = [{ x: 0, y: 0 }, { x: rect.width, y: 0 }, { x: rect.width, y: rect.height }, { x: 0, y: rect.height }, { x: rect.width / 2, y: rect.height / 2 }];
  142. let targetPivot = pivots[index];
  143. let point = { x: targetPivot.x, y: targetPivot.y } as any;
  144. this.parent.worldTransform.apply(point, point);
  145. let yIndex = 0, xIndex = 0;
  146. if (index == 3) {
  147. yIndex = 0;
  148. xIndex = 2;
  149. } else if (index == 0) {
  150. yIndex = 3;
  151. xIndex = 1;
  152. } else if (index == 1) {
  153. yIndex = 2;
  154. xIndex = 0;
  155. } else if (index == 2) {
  156. yIndex = 1;
  157. xIndex = 3;
  158. }
  159. let pointY = pivots[yIndex];
  160. let pY = { x: pointY.x, y: pointY.y } as any;
  161. this.parent.worldTransform.apply(pY, pY);
  162. let pointX = pivots[xIndex];
  163. let pX = { x: pointX.x, y: pointX.y } as any;
  164. this.parent.worldTransform.apply(pX, pX);
  165. let xVec = { x: (pX.x - point.x), y: (pX.y - point.y) };
  166. let yVec = { x: (pY.x - point.x), y: (pY.y - point.y) };
  167. return { x: xVec, y: yVec };
  168. }
  169. scale(x:number, y:number) {
  170. this.parent.scale.x = x;
  171. this.parent.scale.y = y;
  172. this.parent.updateTransform();
  173. this.updateCompState();
  174. }
  175. scaleX(x:number) {
  176. this.parent.scale.x = x;
  177. this.parent.updateTransform();
  178. this.updateCompState();
  179. }
  180. scaleY(y:number) {
  181. this.parent.scale.y = y;
  182. this.parent.updateTransform();
  183. this.updateCompState();
  184. }
  185. applyChildWidth(option:{scaleX?:number, scaleY?:number}) {
  186. if (this.selected.length < 1) return;
  187. const obj = this.selected[0];
  188. //先移除
  189. this.parent.removeChildWorldNoChange(obj);
  190. const m = new Matrix();
  191. m.scale(option.scaleX ? option.scaleX : 1, option.scaleY? option.scaleY : 1)
  192. m.invert();
  193. m.prepend(obj.worldTransform)
  194. obj.transform.setFromMatrix(m)
  195. if (option.scaleX) {
  196. obj.width = option.scaleX * obj.width;
  197. }
  198. if (option.scaleY) {
  199. obj.height = option.scaleY * obj.height;
  200. }
  201. obj.updateTransform();
  202. this.parent.addChildWorldNoChange(obj);
  203. }
  204. scaleSize(x:number, y:number) {
  205. let preW = this.parent.scale.x * this.rect.width;
  206. let preH = this.parent.scale.y * this.rect.height;
  207. this.parent.scale.y = y
  208. this.parent.scale.x = x
  209. const Width = this.parent.scale.x * this.rect.width;
  210. const Height = this.parent.scale.y * this.rect.height;
  211. this.parent.updateTransform();
  212. this.applyChildWidth({scaleX: Width / preW, scaleY: Height/preH})
  213. this.updateCompState();
  214. }
  215. scaleWidth(x:number) {
  216. const prew = this.parent.scale.x * this.rect.width;
  217. //怎么改对应的偏移值
  218. this.parent.scale.x = x
  219. const w = this.parent.scale.x * this.rect.width;
  220. this.parent.updateTransform();
  221. this.applyChildWidth({scaleX: w / prew })
  222. this.updateCompState();
  223. }
  224. scaleHeight(y:number) {
  225. const preH = this.parent.scale.y * this.rect.height;
  226. this.parent.scale.y = y
  227. const h = this.parent.scale.y * this.rect.height;
  228. this.parent.updateTransform();
  229. this.applyChildWidth({scaleY: h / preH});
  230. this.updateCompState();
  231. }
  232. translate(x:number, y:number) {
  233. this.parent.x += x;
  234. this.parent.y += y;
  235. this.parent.updateTransform();
  236. this.updateCompState();
  237. }
  238. destroy() {//选中的对象坐标转到画布空间坐标
  239. let selected = this.selected;
  240. let n = selected.length;
  241. while (n--) {
  242. let child = selected[n];
  243. this.parent.removeChildWorldNoChange(child)
  244. }
  245. this.selected = [];
  246. }
  247. updateCompState() {
  248. let n = this.selected.length;
  249. while (n--) {
  250. let child = this.selected[n];
  251. if (child.comp) child.comp.layout.transformMatrix = child.worldTransform.getMatrixStr();
  252. }
  253. }
  254. updateSize() {
  255. this.init();
  256. }
  257. }