scene-select.vue 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. <template>
  2. <Modal
  3. width="800px"
  4. title="添加场景"
  5. :visible="visible"
  6. @ok="okHandler"
  7. @cancel="visible = false"
  8. okText="确定"
  9. cancelText="取消"
  10. class="model-table"
  11. >
  12. <div>
  13. <div className="model-header">
  14. <p class="header-desc">
  15. 已选择数据<span>( {{ rowSelection.selectedRowKeys.length }} )</span>
  16. </p>
  17. <Search
  18. className="content-header-search"
  19. placeholder="输入名称搜索"
  20. v-model:value="keyword"
  21. allow-clear
  22. style="width: 244px"
  23. />
  24. </div>
  25. <div class="table-layout" v-if="origin.length">
  26. <Tabs v-model:activeKey="type">
  27. <TabPane v-for="current in types" :key="current" :tab="current">
  28. <Table
  29. v-if="typeFilterScenes[current].length"
  30. :row-key="(record: Scene) => record.modelId"
  31. :columns="cloumns"
  32. :rowSelection="rowSelection"
  33. :data-source="typeFilterScenes[current]"
  34. :pagination="false"
  35. />
  36. <div style="padding: 1px" v-else>
  37. <Empty
  38. description="暂无搜索结果"
  39. :image="Empty.PRESENTED_IMAGE_SIMPLE"
  40. className="ant-empty ant-empty-normal"
  41. />
  42. </div>
  43. </TabPane>
  44. </Tabs>
  45. </div>
  46. <div style="padding: 1px" v-else>
  47. <Empty
  48. :description="keyword.length ? '暂无搜索结果' : '暂无结果'"
  49. :image="Empty.PRESENTED_IMAGE_SIMPLE"
  50. className="ant-empty ant-empty-normal"
  51. />
  52. </div>
  53. </div>
  54. </Modal>
  55. <div class="slot-layout">
  56. <Dropdown placement="bottom">
  57. <slot></slot>
  58. <template #overlay>
  59. <Menu>
  60. <MenuItem @click="visible = true">场景管理</MenuItem>
  61. <MenuItem @click="selectModel">媒体库</MenuItem>
  62. </Menu>
  63. </template>
  64. </Dropdown>
  65. </div>
  66. </template>
  67. <script lang="ts" setup>
  68. import {
  69. Modal,
  70. Input,
  71. Table,
  72. Empty,
  73. Tabs,
  74. TabPane,
  75. Dropdown,
  76. Menu,
  77. MenuItem,
  78. } from "ant-design-vue";
  79. import { computed, nextTick, ref, watch, watchEffect } from "vue";
  80. import { scenes, save, SceneTypeDesc } from "@/store";
  81. import { asyncTimeout, createLoadPack, diffArrayChange } from "@/utils";
  82. import {
  83. fuseModels,
  84. createFuseModels,
  85. addFuseModel,
  86. fuseModelsLoaded,
  87. initialScenes,
  88. } from "@/store";
  89. import { SceneType, uploadMaterialToModel, type Scene } from "@/api";
  90. import { activeModel, getSceneModel } from "@/sdk";
  91. import { selectMaterials } from "@/components/materials/quisk";
  92. import { custom } from "@/env";
  93. import { actionItems, currentItem } from "@/views/merge";
  94. type Key = Scene["modelId"];
  95. const Search = Input.Search;
  96. const selectIds = computed(() => fuseModels.value.filter(item => item.type !== SceneType.SWMX).map((item) => item.modelId));
  97. const visible = ref(false);
  98. const keyword = ref("");
  99. const SceneGroupTypeDesc: any = {
  100. [SceneType.SWKK]: 'Mesh场景',
  101. [SceneType.SWKJ]: 'Mesh场景',
  102. [SceneType.SWSS]: '点云场景',
  103. [SceneType.SWSSMX]: 'Mesh场景',
  104. [SceneType.SWYDSS]: '点云场景',
  105. [SceneType.SWYDMX]: 'Mesh场景',
  106. [SceneType.DSFXJ]: 'Mesh场景'
  107. }
  108. const origin = computed(() =>
  109. scenes.value.filter(scene => scene.type !== SceneType.SWMX).map((scene) => ({
  110. ...scene,
  111. createTime: scene.createTime.substr(0, 16),
  112. type: SceneGroupTypeDesc[scene.type],
  113. }))
  114. );
  115. const typeFilterScenes = computed(() => {
  116. const typeScenes: any = {};
  117. for (const type of types.value) {
  118. typeScenes[type] = origin.value
  119. .filter((item) => item.name && item.modelId && item.name.includes(keyword.value))
  120. .filter((item) => item.type === type);
  121. }
  122. return typeScenes;
  123. });
  124. const types = computed(() => [
  125. ...new Set(origin.value.map((item) => item.type)).values(),
  126. ]);
  127. const type = ref(types.value[0]);
  128. const cache = {} as any
  129. const selects = ref<Key[]>(selectIds.value);
  130. const rowSelection: any = ref({
  131. selectedRowKeys: selects,
  132. onChange: (ids: number[]) => {
  133. ids = ids.filter(id => !selectIds.value.includes(id))
  134. cache[type.value] = ids
  135. const curIds = [...selectIds.value]
  136. for (const key in cache) {
  137. curIds.push(...cache[key])
  138. }
  139. selects.value = curIds
  140. },
  141. getCheckboxProps: (record: Scene) => ({
  142. disabled: selectIds.value.includes(record.modelId),
  143. }),
  144. });
  145. const cloumns = [
  146. {
  147. width: "400px",
  148. title: "名称",
  149. dataIndex: "name",
  150. key: "name",
  151. },
  152. {
  153. title: "类型",
  154. dataIndex: "type",
  155. key: "type",
  156. },
  157. {
  158. title: "拍摄/创建时间",
  159. dataIndex: "createTime",
  160. key: "createTime",
  161. },
  162. ];
  163. const addModelHandler = createLoadPack(async (modelIds: number[]) => {
  164. const models = modelIds.map((modelId) => createFuseModels({ modelId }));
  165. const addPromises = models.map(addFuseModel);
  166. const addModels = await Promise.all(addPromises);
  167. await new Promise<void>((resolve) => {
  168. nextTick(() => {
  169. const stop = watchEffect(() => {
  170. if (fuseModelsLoaded.value) {
  171. nextTick(() => {
  172. stop();
  173. resolve();
  174. });
  175. }
  176. });
  177. });
  178. });
  179. models.forEach((model) => {
  180. if (getSceneModel(model)) {
  181. model.rotation = getSceneModel(model)!.getDefaultRotation();
  182. }
  183. });
  184. await asyncTimeout(100);
  185. await save();
  186. activeModel({ showMode: 'fuse', active: addModels[addModels.length - 1] })
  187. // custom.currentModel = addModels[addModels.length - 1]
  188. currentItem.value = actionItems[0]
  189. });
  190. const okHandler = createLoadPack(async () => {
  191. console.log(selects.value);
  192. const models = selects.value.filter(
  193. (modelId) => !fuseModels.value.some((model) => model.modelId === modelId)
  194. );
  195. await addModelHandler(models);
  196. visible.value = false;
  197. });
  198. watch(visible, (visible, oldvisible) => {
  199. if (visible !== oldvisible) {
  200. keyword.value = "";
  201. selects.value = selectIds.value;
  202. visible && initialScenes();
  203. }
  204. });
  205. const selectModel = async () => {
  206. const list = await selectMaterials({
  207. uploadFormat: ["zip"],
  208. format: ["obj", "ply", "las", "laz", "b3dm", "shp", "osgb", "glb"],
  209. maxSize: 2 * 1024 * 1024 * 1024,
  210. });
  211. if (!list?.length) return;
  212. const modelList = await Promise.all(list.filter(item => item.uploadId).map(item => uploadMaterialToModel(item.uploadId!)))
  213. const modelIds = modelList
  214. .map((item) => item.modelId!)
  215. .filter(
  216. (modelId) => modelId && !fuseModels.value.some((model) => model.modelId === modelId)
  217. );
  218. await addModelHandler(modelIds);
  219. };
  220. </script>
  221. <style lang="less" scoped>
  222. .model-header {
  223. display: flex;
  224. justify-content: space-between;
  225. padding-bottom: 24px;
  226. align-items: center;
  227. }
  228. .table-layout {
  229. max-height: 500px;
  230. overflow-y: auto;
  231. }
  232. .slot-layout {
  233. display: flex;
  234. align-items: center;
  235. height: 100%;
  236. }
  237. </style>
  238. <style lang="less">
  239. .model-header .header-desc {
  240. margin-bottom: 0;
  241. }
  242. .ant-modal-root .ant-table-tbody > tr > td {
  243. word-break: break-all;
  244. }
  245. </style>