index.ts 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. import { isPc } from "@queenjs/utils";
  2. import { nanoid } from "nanoid";
  3. import { EditorModule } from "..";
  4. import { CompObject } from "../../controllers/SelectCtrl/compObj";
  5. import { Matrix } from "../../controllers/SelectCtrl/matrix";
  6. import { Design_Page_Size, Design_Pixel_Ratio } from "../../dicts/CompOptions";
  7. import { DesignComp } from "../../objects/DesignTemp/DesignComp";
  8. import { createCompStyle } from "../../objects/DesignTemp/creates/createCompStyle";
  9. import { Layout } from "../../typings";
  10. export const helpers = EditorModule.helper({
  11. pxToDesignSize(value: number) {
  12. return value * Design_Pixel_Ratio;
  13. },
  14. designSizeToPx(value: number) {
  15. return value / Design_Pixel_Ratio;
  16. },
  17. designToNaturalSize(
  18. value: number,
  19. options?: { adaptiveH?: boolean }
  20. ): string {
  21. if (options?.adaptiveH && !isPc()) {
  22. value =
  23. value *
  24. ((window.innerHeight * Design_Pixel_Ratio) / Design_Page_Size[1]);
  25. }
  26. return this.helper.designSizeToPx(value) + "px";
  27. },
  28. findComp(compId: string) {
  29. const compMap = this.controls.pageCtrl.compMap;
  30. if (!compMap) return;
  31. const comp = compMap[compId];
  32. if (comp) return comp;
  33. },
  34. isStreamCard(compId: string) {
  35. return this.controls.pageCtrl.streamCardIds.indexOf(compId) > -1;
  36. },
  37. isGroupCompChild(compId: string) {
  38. const comps = this.helper.getCompTrees(compId);
  39. comps.pop();
  40. while (comps.length) {
  41. const comp = comps.pop() as DesignComp;
  42. if (comp.compKey === "Group") {
  43. return true;
  44. }
  45. }
  46. return false;
  47. },
  48. isStreamCardChild(compId: string) {
  49. if (compId == "root" || this.helper.isStreamCard(compId)) {
  50. return false;
  51. }
  52. const cards = this.controls.pageCtrl.streamCardIds;
  53. let n = cards.length;
  54. const ctrl = this.controls.pageCtrl
  55. const compMap = ctrl.designData.compMap;
  56. while (n--) {
  57. const childs = compMap[cards[n]].children.default || [];
  58. if (childs.indexOf(compId) > -1) return true;
  59. }
  60. return false;
  61. },
  62. findParentComp(compId: string): DesignComp | undefined {
  63. const comp = this.helper.findComp(compId);
  64. if (comp) return this.helper.findComp(this.controls.pageCtrl.compPids[compId]);
  65. },
  66. findRootComp(): DesignComp | undefined {
  67. return this.controls.pageCtrl.designData.compMap?.["root"];
  68. },
  69. findCardAllChildren(index:number) {
  70. const ctrl = this.controls.pageCtrl
  71. const cardId = ctrl.streamCardIds[index];
  72. return ctrl.designData.compMap[cardId].children.default || [] as string[];
  73. },
  74. getCompCard(compId: string) {
  75. const paths: DesignComp[] = this.helper.getCompTrees(compId);
  76. return paths[1];
  77. },
  78. getCompWorlParentPos(compId: string, vx: number, vy: number) {
  79. const paths: DesignComp[] = this.helper.getCompTrees(compId);
  80. const m = new Matrix();
  81. let n = paths.length;
  82. let box = paths[1].$el.getBoundingClientRect();
  83. const s = this.controls.editorCtrl.state.scale;
  84. m.translate((box.left - vx) / s, (box.top - vy)/s);
  85. for (let i = 2; i < n - 1; i++) {
  86. //card开始遍历
  87. const m1 = new Matrix();
  88. m1.setMatrixStr(paths[i].layout.transformMatrix || "matrix(1,0,0,1,0,0)");
  89. m.append(m1);
  90. }
  91. return m;
  92. },
  93. getCompTrees(compId: string) {
  94. const ctrl = this.controls.pageCtrl;
  95. const comps: DesignComp[] = [];
  96. const getParentComp = (compId: string) => {
  97. const comp = this.helper.findComp(compId);
  98. if (comp) {
  99. comps.unshift(comp);
  100. getParentComp(ctrl.compPids[comp.id]);
  101. }
  102. };
  103. getParentComp(compId);
  104. return comps;
  105. },
  106. createStyle(layout: Partial<Layout> & { size: any[] }, comp: DesignComp) {
  107. return createCompStyle(this, layout, comp);
  108. },
  109. isCurrComp(compId: string) {
  110. return this.store.currCompId === compId;
  111. },
  112. isCustomChildComp(comp: DesignComp): boolean {
  113. const parentComp = this.helper.findParentComp(comp.id);
  114. if (!parentComp) return false;
  115. const i =
  116. parentComp.children.default?.findIndex((d) => d === comp.id) ?? -1;
  117. return i >= 0;
  118. },
  119. isCompCanDelete(compId: string): boolean {
  120. const page = this.controls.pageCtrl;
  121. return (
  122. this.helper.isStreamCardChild(compId) ||
  123. (this.helper.isStreamCard(compId) && page.streamCardIds.length > 1)
  124. );
  125. },
  126. isShowSaveComp(comp: DesignComp): boolean {
  127. if (!this.helper.isStreamCardChild(comp.id)) return false;
  128. return true;
  129. },
  130. clearUnusedComps(compMap: Record<string, DesignComp>, rootId = "root") {
  131. const used = new Set<string>();
  132. const getUsedIds = (ids: string[]) => {
  133. ids.forEach((id) => {
  134. const comp = compMap[id];
  135. if (!comp) return;
  136. used.add(id);
  137. getUsedIds(comp.getChildIds());
  138. });
  139. return used;
  140. };
  141. getUsedIds([rootId]);
  142. Object.keys(compMap).forEach((compId) => {
  143. if (!used.has(compId)) {
  144. delete compMap[compId];
  145. }
  146. });
  147. },
  148. clearProjectUnusedComps(compMap: Record<string, DesignComp>) {
  149. const used = new Set<string>();
  150. const getUsedIds = (ids: string[]) => {
  151. ids.forEach((id) => {
  152. const comp = compMap[id];
  153. if (!comp) return;
  154. used.add(id);
  155. getUsedIds(comp.getChildIds());
  156. });
  157. return used;
  158. };
  159. getUsedIds(["root"]);
  160. const ctrl = this.controls.pageCtrl;
  161. const compScreenMap = ctrl.designData.compScreenMap
  162. const keys = Object.keys(compScreenMap)
  163. keys.forEach(k=>{
  164. const card = compScreenMap[k];
  165. card.forEach(c=>{
  166. c.children.forEach(item=>{
  167. const comp = compMap[item.id];
  168. if (!comp) return;
  169. used.add(item.id);
  170. getUsedIds([item.id])
  171. })
  172. })
  173. })
  174. Object.keys(compMap).forEach((compId) => {
  175. if (!used.has(compId)) {
  176. delete compMap[compId];
  177. }
  178. });
  179. },
  180. getPointOffsetWith(e: MouseEvent, dom: HTMLElement) {
  181. const domRect = dom.getBoundingClientRect();
  182. return {
  183. x: e.clientX - domRect.left,
  184. y: e.clientY - domRect.top,
  185. };
  186. },
  187. getCardCompBound(compId: string) {
  188. const ctrl = this.controls.pageCtrl;
  189. const compMap = ctrl.designData.compMap;
  190. const c = compMap[compId];
  191. const obj = new CompObject(c);
  192. const bound = obj.getBox();
  193. return bound;
  194. },
  195. extendStreamCard(streamCardId: string) {
  196. if (!streamCardId) return 0;
  197. const ctrl = this.controls.pageCtrl;
  198. const compMap = ctrl.designData.compMap;
  199. const card = compMap[streamCardId];
  200. const size:any = card.layout.size.slice(0)
  201. if (this.controls.screenCtrl.isShortPage) {
  202. const h = this.controls.screenCtrl.getCurrScreenHeight();
  203. size[1] = h;
  204. card.layout.setSize(size);
  205. return h;
  206. }
  207. const childs = card.children.default || [];
  208. let maxH = 0,
  209. n = childs.length;
  210. while (n--) {
  211. const c = childs[n];
  212. const aabb = this.helper.getCardCompBound(c);
  213. maxH = Math.max(maxH, aabb.y + aabb.h);
  214. }
  215. maxH = this.helper.pxToDesignSize(maxH);
  216. if (childs.length < 1) {
  217. maxH = 200;
  218. }
  219. size[1] = maxH;
  220. card.layout.setSize(size);
  221. return maxH;
  222. },
  223. getCardNextPosY(cardId:string, itemWidth:number) {
  224. let yOffset = 0;
  225. if (cardId != this.store.currStreamCardId) {
  226. //const paths = this.helper.getCompTrees(cardId);
  227. const bound = this.helper.getCardCompBound(cardId);
  228. yOffset = bound.y + bound.h;
  229. } else {
  230. const currCard = this.helper.findComp(cardId) as DesignComp;
  231. //没有选中组件添加到当前卡片最后
  232. const children = currCard.children.default || [];
  233. let prevCompIndex = -1;
  234. if (this.store.currCompId != "" && this.store.currCompId != "root") {
  235. prevCompIndex = children.indexOf(this.store.currCompId);
  236. }
  237. if (prevCompIndex != -1) {
  238. const bound = this.helper.getCardCompBound(children[prevCompIndex]);
  239. yOffset = bound.y + bound.h;
  240. }
  241. }
  242. const w = this.controls.screenCtrl.getCurrScreenWidth();
  243. //添加组件到当前选中的组件下面
  244. const xOffset:number = this.helper.designSizeToPx(
  245. w / 2.0 - (itemWidth || w) / 2
  246. );
  247. return {x: xOffset, y: yOffset};
  248. },
  249. getClientId() {
  250. let clientId = undefined;
  251. try {
  252. const userInfo = JSON.parse(localStorage.getItem("userInfo") || "{}");
  253. clientId = userInfo._id || localStorage.getItem("clientId");
  254. } catch (error) {
  255. console.error(error);
  256. }
  257. if (!clientId) {
  258. clientId = nanoid();
  259. localStorage.setItem("clientId", clientId);
  260. }
  261. return clientId;
  262. },
  263. loadImage(url ) {
  264. return new Promise((res, rej)=>{
  265. let t = setTimeout(() => {
  266. t = null as any;
  267. rej("下载超时")
  268. }, 1000 * 10);
  269. const img = new Image();
  270. img.onload = ()=>{
  271. res(img);
  272. if (t) clearTimeout(t);
  273. }
  274. img.onerror = (e)=>{
  275. rej("下载失败!")
  276. }
  277. img.src = url;
  278. })
  279. }
  280. });