component.tsx 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. import { IconPlay2 } from "@/assets/icons";
  2. import { useEditor } from "@/modules/editor";
  3. import { css } from "@linaria/core";
  4. import { IconClose } from "@queenjs/icons";
  5. import { Image } from "@queenjs/ui";
  6. import { queenApi, useModal } from "queenjs";
  7. import { defineComponent, reactive, watch } from "vue";
  8. import { string } from "vue-types";
  9. import { View } from "../View";
  10. import { useCompData } from "../../defines/hook";
  11. import { CompWeb3DObj } from ".";
  12. export const Component = defineComponent({
  13. props: {
  14. compId: string().isRequired,
  15. },
  16. setup(props) {
  17. const { store, actions } = useEditor();
  18. const comp = useCompData<CompWeb3DObj>(props.compId);
  19. const { value } = comp;
  20. async function pickPack() {
  21. // await controls.pickCtrl.onPickPack();
  22. // value.url =
  23. // "https://www.sku3d.com/share.html?id=6478676ca494a3ea15a6fa82";
  24. }
  25. function showWeb3D() {
  26. if (value.inline) {
  27. state.show3d = true;
  28. } else {
  29. queenApi.dialog(<Iframe3D url={value.url} />, {
  30. fullscreen: true,
  31. closable: false,
  32. });
  33. }
  34. }
  35. const state = reactive({
  36. show3d: false,
  37. });
  38. watch(
  39. () => [value.ratio],
  40. () => {
  41. comp.setH(comp.getW() / value.ratio);
  42. actions.onCompLayoutUpdated(comp);
  43. }
  44. );
  45. return () => {
  46. return (
  47. <View
  48. compId={props.compId}
  49. onDblclick={store.isEditMode ? pickPack : undefined}
  50. >
  51. {state.show3d ? (
  52. <iframe class="w-full h-full border-none" src={value.url} />
  53. ) : (
  54. <>
  55. <Image
  56. class="w-full h-full pointer-events-none object-cover"
  57. size={480}
  58. src={value.poster}
  59. />
  60. <IconPlay2
  61. class={iconCls}
  62. onClick={!store.isEditMode ? showWeb3D : undefined}
  63. />
  64. </>
  65. )}
  66. </View>
  67. );
  68. };
  69. },
  70. });
  71. const Iframe3D = defineComponent({
  72. props: {
  73. url: string().isRequired,
  74. },
  75. setup(props) {
  76. const modal = useModal();
  77. return () => (
  78. <div class="w-full h-full overflow-hidden relative">
  79. <IconClose class={closeCls} onClick={() => modal.cancel()} />
  80. <iframe class="w-full h-full border-none" src={props.url} />
  81. </div>
  82. );
  83. },
  84. });
  85. const iconCls = css`
  86. position: absolute;
  87. top: 50%;
  88. left: 50%;
  89. padding: 15px;
  90. font-size: 36px;
  91. color: rgba(255, 255, 255, 0.8);
  92. transform: translate(-50%, -50%);
  93. border-radius: 50%;
  94. background-color: rgba(0, 0, 0, 0.6);
  95. `;
  96. const closeCls = css`
  97. position: absolute;
  98. top: 15px;
  99. left: 15px;
  100. font-size: 15px;
  101. color: #666;
  102. padding: 10px;
  103. border-radius: 50%;
  104. background-color: #fff;
  105. @apply shadow;
  106. `;