index.ts 10 KB

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