CompTree.tsx 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300
  1. import { TreeToolbars } from "@/modules/editor/objects/Toolbars";
  2. import { css } from "@linaria/core";
  3. import { Image } from "@queenjs/ui";
  4. import { useReactive } from "@queenjs/use";
  5. import { Tree, TreeProps } from "ant-design-vue";
  6. import { defineComponent, nextTick, watch } from "vue";
  7. import { string, bool } from "vue-types";
  8. import { useEditor } from "../../../..";
  9. import { DesignComp } from "../../../../objects/DesignTemp/DesignComp";
  10. import { TreeDataItem } from "ant-design-vue/lib/tree";
  11. import { CompObject } from "@/modules/editor/controllers/SelectCtrl/compObj";
  12. type TreeItem = {
  13. key: string;
  14. title: string;
  15. value: string;
  16. children: TreeItem[];
  17. };
  18. export const CompTree = defineComponent({
  19. setup() {
  20. const { store, actions, helper, controls } = useEditor();
  21. const { compUICtrl } = controls;
  22. const state = useReactive(() => ({
  23. expandedKeys: [] as string[],
  24. treeData() {
  25. const rootComp = helper.findRootComp();
  26. let deep = 0;
  27. function getCompChildren(ids: string[]): TreeItem[] {
  28. deep += 1;
  29. if (deep > 2) ids = [...ids].reverse();
  30. return ids.map((id) => {
  31. const comp = helper.findComp(id) as DesignComp;
  32. return {
  33. key: comp.id,
  34. title:
  35. comp.title ||
  36. compUICtrl.state.components.get(comp.compKey)?.name ||
  37. "未命名",
  38. value: comp.id,
  39. children: comp?.getChildIds
  40. ? getCompChildren(comp?.getChildIds())
  41. : [],
  42. };
  43. });
  44. }
  45. return getCompChildren(rootComp?.id ? [rootComp.id] : []);
  46. },
  47. }));
  48. watch(
  49. () => store.currCompId,
  50. () => {
  51. // expandedKeys需要等Tree的内部状态更新后再赋值
  52. nextTick(() => {
  53. state.expandedKeys = store.currCompId
  54. ? helper.getCompTrees(store.currCompId).map((comp) => comp.id)
  55. : [];
  56. });
  57. }
  58. );
  59. const onDrop = (info: any) => {
  60. const dragNode = info.dragNode; //被拖拽
  61. const dropNode = info.node; //拖拽落下
  62. const dragKey = dragNode.key;
  63. const dropKey = dropNode.key;
  64. const dragPos = dragNode.pos.split("-");
  65. const dropPos = dropNode.pos.split("-");
  66. const dragLevel = dragPos.length;
  67. const dropLevel = dropPos.length;
  68. if (
  69. dragLevel >= 5 ||
  70. dragKey == "root" ||
  71. dragLevel < dropLevel ||
  72. dragLevel - dropLevel > 1
  73. ) {
  74. return;
  75. }
  76. let parentComp: any = null;
  77. if (dragLevel > dropLevel) {
  78. parentComp = helper.findComp(dropKey);
  79. } else {
  80. parentComp = helper.findParentComp(dropKey);
  81. }
  82. const currComp = helper.findComp(dragKey);
  83. const currParent = helper.findParentComp(dragKey);
  84. if (!currComp) return;
  85. const currParentComp = helper.findParentComp(dragKey);
  86. const index = currParentComp?.children.default?.indexOf(currComp.id);
  87. if (index != -1) {
  88. currParentComp?.children.default?.splice(index as number, 1);
  89. helper.extendStreamCard(currParentComp?.id || "");
  90. }
  91. if (!info.dropToGap) {
  92. parentComp?.children.default == parentComp?.children.default || [];
  93. if (dragLevel <= 3) {
  94. parentComp?.children.default?.unshift(currComp?.id);
  95. } else {
  96. parentComp?.children.default?.push(currComp?.id);
  97. }
  98. if (currParent?.id != parentComp.id) {
  99. const len = parentComp?.children.default?.length;
  100. const index = dragLevel > 3 ? len - 1 : 0;
  101. const revert = dragLevel > 3 ? true : false;
  102. sortCompInCard(parentComp, index, revert);
  103. }
  104. } else {
  105. if (dragLevel <= 3) {
  106. parentComp?.children.default?.splice(
  107. info.dropPosition,
  108. 0,
  109. currComp.id
  110. );
  111. } else {
  112. const len = parentComp?.children.default.length;
  113. parentComp?.children.default?.splice(
  114. len - info.dropPosition,
  115. 0,
  116. currComp.id
  117. );
  118. }
  119. if (currParent?.id != parentComp.id) {
  120. const revert = dragLevel > 3 ? true : false;
  121. const len = parentComp?.children.default.length - 1;
  122. const index = revert ? len - info.dropPosition : info.dropPosition;
  123. sortCompInCard(parentComp, index, revert);
  124. }
  125. }
  126. nextTick(() => {
  127. controls.editorCtrl.clickPickComp(currComp.id);
  128. helper.extendStreamCard(parentComp.id);
  129. });
  130. };
  131. const sortCompInCard = (
  132. parentComp: DesignComp,
  133. index: number,
  134. revert = false
  135. ) => {
  136. if (parentComp?.compKey != "Container") {
  137. return;
  138. }
  139. const children = parentComp.children.default || [];
  140. if (children?.length > 0) {
  141. const prveCompId = revert ? children[index + 1] : children[index - 1];
  142. const nowCompId = children[index];
  143. const nowComp = helper.findComp(nowCompId);
  144. if (!prveCompId) {
  145. const nowCompBound = helper.getCardCompBound(nowCompId);
  146. if (!nowComp) {
  147. return;
  148. }
  149. const nowObj = new CompObject(nowComp);
  150. nowObj.worldTransform.translate(-nowCompBound.x, -nowCompBound.y);
  151. nowComp.layout.transformMatrix = nowObj.worldTransform.getMatrixStr();
  152. return;
  153. }
  154. const prevCompBound = helper.getCardCompBound(prveCompId);
  155. const prevComp = helper.findComp(prveCompId);
  156. if (!prevComp || !nowComp) {
  157. return;
  158. }
  159. const obj = new CompObject(prevComp);
  160. const yOffset = prevCompBound.h;
  161. obj.worldTransform.translate(0, yOffset);
  162. nowComp.layout.transformMatrix = obj.worldTransform.getMatrixStr();
  163. }
  164. };
  165. return () => (
  166. <div class="!overflow-x-auto">
  167. <Tree
  168. class={treeStyle}
  169. treeData={state.treeData}
  170. v-model={[state.expandedKeys, "expandedKeys"]}
  171. selectedKeys={[store.currCompId]}
  172. blockNode={true}
  173. draggable={true}
  174. onDrop={onDrop}
  175. onSelect={(ids) => {
  176. const id =
  177. (ids[0] as string) || state.expandedKeys.at(-2) || "root";
  178. controls.editorCtrl.clickPickComp(id);
  179. }}
  180. >
  181. {{
  182. title: (data: any) => {
  183. return <CompNode title={data.title} id={data.key} />;
  184. },
  185. }}
  186. </Tree>
  187. </div>
  188. );
  189. },
  190. });
  191. const CompNode = defineComponent({
  192. props: {
  193. id: string().isRequired,
  194. title: string().isRequired,
  195. },
  196. setup(props) {
  197. const editor = useEditor();
  198. const comp = editor.helper.findComp(props.id);
  199. const textTitle = (value: any) => {
  200. const reg = /<[^>]+>/gi;
  201. const text = value.text.replace(reg, "");
  202. return text.substring(0, 10);
  203. };
  204. return () => {
  205. if (!comp) return;
  206. const compOpts = editor.controls.compUICtrl.state.components.get(
  207. comp.compKey
  208. );
  209. const actions = TreeToolbars[comp.compKey] || TreeToolbars.default;
  210. const thumbnail =
  211. comp.compKey === "Image"
  212. ? comp.value.url
  213. : comp.thumbnail || compOpts?.thumbnail;
  214. const title =
  215. comp.compKey === "Text" ? textTitle(comp.value) : props.title;
  216. return (
  217. <div class={[nodeStyle, "flex"]}>
  218. <Image src={thumbnail} size={240} />
  219. <span class="flex-1 w-0 text-12px whitespace-nowrap overflow-hidden overflow-ellipsis pr-6px">
  220. {title}
  221. </span>
  222. <span class="actions space-x-2px whitespace-nowrap transition">
  223. {actions.map((action, i) => {
  224. return action.getVisible.call(editor, comp) ? (
  225. <action.component
  226. key={i}
  227. class={["p-4px", action.getValue?.(comp) ? "active" : ""]}
  228. value={action.getValue?.(comp)}
  229. onClick={(e: MouseEvent) => {
  230. e.stopPropagation();
  231. action.onClick.call(editor, comp);
  232. }}
  233. />
  234. ) : null;
  235. })}
  236. </span>
  237. </div>
  238. );
  239. };
  240. },
  241. });
  242. const treeStyle = css`
  243. .ant-tree-switcher {
  244. display: flex;
  245. width: 14px;
  246. padding-right: 4px;
  247. justify-content: center;
  248. align-items: center;
  249. }
  250. .ant-tree-indent-unit {
  251. width: 4px;
  252. }
  253. `;
  254. const nodeStyle = css`
  255. display: flex;
  256. align-items: center;
  257. padding: 6px 2px;
  258. @apply rounded;
  259. > img {
  260. width: 20px;
  261. height: 20px;
  262. margin-right: 4px;
  263. object-fit: contain;
  264. }
  265. .actions {
  266. .inficon {
  267. opacity: 0;
  268. font-size: 14px;
  269. &.active {
  270. opacity: 1;
  271. }
  272. }
  273. }
  274. &:hover {
  275. .actions {
  276. .inficon {
  277. opacity: 1;
  278. }
  279. }
  280. }
  281. `;