PreviewDesignModal.tsx 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. import { IconExit, IconNext, IconPrev } from "@/assets/icons";
  2. import { initEditor } from "@/modules/editor";
  3. import { Design_Page_Size } from "@/modules/editor/dicts/CompOptions";
  4. import { DesignComp } from "@/modules/editor/objects/DesignTemp/DesignComp";
  5. import { Button, Dropdown } from "ant-design-vue";
  6. import { cloneDeep } from "lodash";
  7. import Modal from "queenjs/adapter/vue/components/modal";
  8. import { defineComponent, nextTick, provide, ref } from "vue";
  9. import { any } from "vue-types";
  10. import { ShareBox } from "./ShareBox";
  11. const NotFoundComp = () => <div>无效的组件</div>;
  12. export default defineComponent({
  13. props: {
  14. data: any(),
  15. },
  16. setup(props) {
  17. const editor = initEditor();
  18. const { helper, controls } = editor;
  19. const page = controls.pageCtrl;
  20. controls.pageCtrl.setDesignData(cloneDeep(props.data));
  21. const pageRef = ref();
  22. const compRef = ref();
  23. provide("isPreview", true);
  24. function getPageH() {
  25. const rootPage = controls.pageCtrl.rootPage;
  26. const pageH = rootPage?.layout.size?.[1] || Design_Page_Size[1];
  27. return helper.designToNaturalSize(pageH, {
  28. adaptiveH: true,
  29. });
  30. }
  31. function getPageStyles() {
  32. const pageRoot = page.rootPage;
  33. const isPcDesign = pageRoot?.value.useFor == "pc";
  34. const isLongPage = pageRoot?.value.pageMode == "long";
  35. if (!isPcDesign) {
  36. let mobileStle: any = {
  37. height: getPageH(),
  38. width: "375px",
  39. margin: "0 auto",
  40. };
  41. if (isLongPage) {
  42. mobileStle.overflowY = "auto";
  43. } else {
  44. mobileStle.marginBottom = "70px";
  45. }
  46. return mobileStle;
  47. }
  48. // pc
  49. let s: any = { height: "100%", margin: "0 auto" };
  50. if (!isLongPage) {
  51. const pageLayout = {
  52. height: pageRef.value?.clientHeight,
  53. width: pageRef.value?.clientWidth,
  54. };
  55. const ph = pageLayout.height - 170;
  56. const width = (ph * 16) / 9;
  57. s = {
  58. ...s,
  59. width: width + "px",
  60. padding: "40px 0 70px",
  61. };
  62. } else {
  63. s = {
  64. ...s,
  65. width: "100%",
  66. overflowY: "auto",
  67. height: "calc(100% + 40px)",
  68. };
  69. }
  70. return s;
  71. }
  72. nextTick(() => {
  73. const pageRoot = page.rootPage;
  74. const isShort = pageRoot?.value.pageMode == "short";
  75. if (isShort) controls.previewCtrl.initSwiper(compRef.value);
  76. });
  77. const Content = () => {
  78. const pageRoot = page.rootPage;
  79. const isShort = pageRoot?.value.pageMode == "short";
  80. const pageStyles = getPageStyles();
  81. return (
  82. <>
  83. <div ref={compRef} style={pageStyles} class="overflow-hidden">
  84. <div class={["page", isShort && "swiper-wrapper relative "]}>
  85. {controls.pageCtrl.streamCardIds.map((item) => {
  86. const c = helper.findComp(item) as DesignComp;
  87. const Comp =
  88. controls.compUICtrl.state.components.get(c.compKey)
  89. ?.Component || NotFoundComp;
  90. return (
  91. <Comp
  92. compId={c.id}
  93. class="swiper-slide relative !flex items-center justify-center"
  94. />
  95. );
  96. })}
  97. </div>
  98. </div>
  99. {/* buttons */}
  100. {isShort && (
  101. <div class="absolute bottom-30px left-1/2 w-130px h-44px px-4px flex items-center justify-between bg-[#262626] rounded-4px transform -translate-x-1/2 z-1">
  102. <IconPrev class="swiper-button-prev rounded-4px text-36px cursor-pointer transition hover:bg-dark-600 active:bg-dark-900" />
  103. <span class="select-none">
  104. {controls.previewCtrl.state.activeIndex}/
  105. {pageRoot.children.default.length}
  106. </span>
  107. <IconNext class="swiper-button-next rounded-4px text-36px cursor-pointer transition hover:bg-dark-600 active:bg-dark-900" />
  108. </div>
  109. )}
  110. </>
  111. );
  112. };
  113. return () => {
  114. const { data } = props;
  115. return (
  116. <div class="h-100vh flex flex-col bg-[#0B0B0B] overflow-hidden">
  117. <div class="h-64px flex items-center justify-between px-20px bg-[#262626] flex-shrink-0">
  118. <Button
  119. class="flex items-center text-12px bg-dark-50 hover:(bg-dark-100 border-transparent text-light-50)"
  120. onClick={() => Modal.clear()}
  121. >
  122. <IconExit class="text-20px mr-5px" /> 退出预览
  123. </Button>
  124. {/* <span class="text-14px select-none">{data.title}</span> */}
  125. {/* <Dropdown
  126. overlay={<ShareBox />}
  127. trigger="click"
  128. placement="bottomRight"
  129. >
  130. <Button class="text-dark-500 hover:text-dark-500" type="primary">
  131. 分享
  132. </Button>
  133. </Dropdown> */}
  134. </div>
  135. <div
  136. ref={pageRef}
  137. class="flex-1 h-0 pb-30px pt-30px flex items-center"
  138. >
  139. <Content />
  140. </div>
  141. </div>
  142. );
  143. };
  144. },
  145. });