association.ts 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  1. import { sdk } from './sdk'
  2. import { toRaw, ref, watch, nextTick } from 'vue'
  3. import {
  4. viewModeStack,
  5. showLeftPanoStack,
  6. custom,
  7. getResource
  8. } from '@/env'
  9. import {
  10. mount,
  11. diffArrayChange,
  12. shallowWatchArray,
  13. arrayChildEffectScope,
  14. showLoad,
  15. hideLoad,
  16. deepIsRevise,
  17. round,
  18. togetherCallback
  19. } from '@/utils'
  20. import {
  21. fuseModels,
  22. taggings,
  23. isEdit,
  24. sysBus,
  25. getFuseModelShowVariable,
  26. SceneType,
  27. backupFuseModels,
  28. MeasureType,
  29. measures
  30. } from '@/store'
  31. import TaggingComponent from '@/components/tagging/list.vue'
  32. import type { FuseModel, Tagging, Measure } from '@/store'
  33. import type {
  34. SDK,
  35. SceneModel,
  36. SceneGuidePath,
  37. ModelAttrRange,
  38. Measure as SceneMeasure
  39. } from '.'
  40. let isUnSet = false
  41. const unSet = ((fn: () => void) => {
  42. nextTick(() => {
  43. isUnSet = true
  44. fn()
  45. nextTick(() => isUnSet = false)
  46. })
  47. })
  48. // -----------------模型关联--------------------
  49. export const modelRange: ModelAttrRange = {
  50. opacityRange: { min: 0, max: 100, step: 0.1 },
  51. bottomRange: { min: -30, max: 70, step: 0.1 },
  52. scaleRange: { min: 0, max: 200, step: 0.1 }
  53. }
  54. const sceneModelMap = new WeakMap<FuseModel, SceneModel>()
  55. export const getSceneModel = (model?: FuseModel | null) => model && sceneModelMap.get(toRaw(model))
  56. const associationModels = (sdk: SDK) => {
  57. const getModels = () => fuseModels.value
  58. shallowWatchArray(getModels, (models, oldModels) => {
  59. const { added, deleted } = diffArrayChange(models, oldModels)
  60. for (const item of added) {
  61. if (getSceneModel(item)) {
  62. continue;
  63. }
  64. const itemRaw = toRaw(item)
  65. const sceneModel = sdk.addModel({
  66. ...itemRaw,
  67. ...modelRange,
  68. type: item.type === SceneType.SWSS ? 'laser' : 'glb',
  69. url: item.type === SceneType.SWSS ? item.url : getResource(item.url)
  70. })
  71. sceneModelMap.set(itemRaw, sceneModel)
  72. sceneModel.bus.on('transformChanged', transform => {
  73. transform = { ...transform }
  74. if (transform.rotation) {
  75. transform.rotation = {
  76. x: round(transform.rotation.x, 5),
  77. y: round(transform.rotation.y, 5),
  78. z: round(transform.rotation.z, 5),
  79. }
  80. }
  81. if (transform.position) {
  82. transform.position = {
  83. x: round(transform.position.x, 5),
  84. y: round(transform.position.y, 5),
  85. z: round(transform.position.z, 5),
  86. }
  87. }
  88. const updateKeys = Object.keys(transform)
  89. const update: any = {}
  90. for (const key of updateKeys) {
  91. update[key] = (item as any)[key]
  92. }
  93. if (deepIsRevise(update, transform)) {
  94. unSet(() => Object.assign(item, transform))
  95. }
  96. })
  97. sceneModel.bus.on('changeSelect', select => {
  98. if (custom.currentModel === item && !select) {
  99. custom.currentModel = null
  100. } else if (custom.currentModel !== item && select) {
  101. custom.currentModel = item
  102. }
  103. })
  104. showLoad()
  105. sceneModel.bus.on('loadDone', () => {
  106. item.loaded = true
  107. hideLoad()
  108. backupFuseModels()
  109. })
  110. sceneModel.bus.on('loadError', () => {
  111. item.error = true
  112. item.show = false
  113. custom.showModelsMap.delete(item)
  114. hideLoad()
  115. backupFuseModels()
  116. })
  117. sceneModel.bus.on('loadProgress', progress => item.progress = progress)
  118. }
  119. for (const item of deleted) {
  120. getSceneModel(item)?.destroy()
  121. }
  122. })
  123. arrayChildEffectScope(getModels, item => {
  124. const stopLoadedWatch = watch(
  125. () => item.loaded,
  126. (loaded) => {
  127. if (loaded) {
  128. const modelShow = getFuseModelShowVariable(item)
  129. watch(
  130. () => item.bottom,
  131. () => isUnSet || getSceneModel(item)?.changeBottom(item.bottom),
  132. { immediate: true }
  133. )
  134. watch(
  135. () => item.opacity,
  136. () => isUnSet || getSceneModel(item)?.changeOpacity(item.opacity),
  137. { immediate: true }
  138. )
  139. watch(
  140. () => item.scale,
  141. () => isUnSet || getSceneModel(item)?.changeScale(item.scale),
  142. { immediate: true }
  143. )
  144. watch(
  145. () => item.position,
  146. () => isUnSet || getSceneModel(item)?.changePosition(item.position),
  147. { immediate: true }
  148. )
  149. watch(
  150. () => item.rotation,
  151. () => isUnSet || getSceneModel(item)?.changeRotation(item.rotation),
  152. { immediate: true }
  153. )
  154. watch(
  155. () => modelShow.value,
  156. () => isUnSet || getSceneModel(item)?.changeShow(modelShow.value),
  157. { immediate: true }
  158. )
  159. stopLoadedWatch()
  160. }
  161. }
  162. )
  163. })
  164. }
  165. // -----------------热点关联--------------------
  166. const associationTaggings = (el: HTMLDivElement) => {
  167. const getTaggings = () => taggings.value
  168. const taggingVMs = new WeakMap<Tagging, ReturnType<typeof mount>>()
  169. shallowWatchArray(getTaggings, (taggings, oldTaggings) => {
  170. const { added, deleted } = diffArrayChange(taggings, oldTaggings)
  171. for (const item of added) {
  172. taggingVMs.set(toRaw(item), mount(el, TaggingComponent, { tagging: item }))
  173. }
  174. for (const item of deleted) {
  175. const unMount = taggingVMs.get(toRaw(item))
  176. unMount && unMount()
  177. }
  178. })
  179. }
  180. // -----------------测量关联--------------------
  181. const sceneMeasureMap = new WeakMap<Measure , SceneMeasure>()
  182. export const getSceneMeasure = (measure?: Measure | null) => measure && sceneMeasureMap.get(toRaw(measure))
  183. export const associationMessaure = <T extends MeasureType>(smMeasure: SceneMeasure<T>, measure: Measure<T>) => {
  184. smMeasure.bus.on('update', ([points, modelIds]) => {
  185. unSet(() => {
  186. measure.positions = points.map((point, i) => ({ point, modelId: modelIds[i] }))
  187. measure.desc = measure.type === MeasureType.area
  188. ? (smMeasure as unknown as SceneMeasure<MeasureType.area>).getArea().toString()
  189. : (smMeasure as unknown as SceneMeasure<MeasureType.free>).getDistance().toString()
  190. })
  191. })
  192. smMeasure.bus.on('highlight', selected => unSet(() => measure.selected = selected))
  193. }
  194. const associationMessaures = (sdk: SDK) => {
  195. const getMeasures = () => measures.value
  196. shallowWatchArray(getMeasures, (measures, oldMeasures) => {
  197. const { added, deleted } = diffArrayChange(measures, oldMeasures)
  198. for (const item of added) {
  199. const sceneMeasure = sdk.drawMeasure(
  200. item.type,
  201. item.positions.map(position => position.point),
  202. item.positions.map(position => position.modelId),
  203. )
  204. associationMessaure(sceneMeasure, item)
  205. sceneMeasureMap.set(item, sceneMeasure)
  206. }
  207. for (const item of deleted) {
  208. const sceneMeasure = getSceneMeasure(item)
  209. sceneMeasure && sceneMeasure.destroy()
  210. }
  211. })
  212. arrayChildEffectScope(getMeasures, measure => {
  213. watch(
  214. () => measure.selected,
  215. (selected = false) => isUnSet || getSceneMeasure(measure)?.changeSelect(selected)
  216. )
  217. watch(
  218. () => custom.showMeasures,
  219. (show) => {
  220. if (!isUnSet) {
  221. const smMeasure = getSceneMeasure(measure)
  222. if (show) {
  223. smMeasure?.show()
  224. } else {
  225. smMeasure?.hide()
  226. }
  227. }
  228. },
  229. { immediate: true }
  230. )
  231. })
  232. }
  233. // -----------------导览关联--------------------
  234. const fullView = async (fn: () => void) => {
  235. const popViewMode = togetherCallback([
  236. viewModeStack.push(ref('full')),
  237. showLeftPanoStack.push(ref(false))
  238. ])
  239. await document.documentElement.requestFullscreen()
  240. const driving = () => document.fullscreenElement || fn()
  241. document.addEventListener('fullscreenchange', driving)
  242. document.addEventListener('fullscreenerror', fn)
  243. return () => {
  244. popViewMode()
  245. document.fullscreenElement && document.exitFullscreen()
  246. document.removeEventListener('fullscreenchange', driving)
  247. document.removeEventListener('fullscreenerror', fn)
  248. }
  249. }
  250. export const isScenePlayIng = ref(false)
  251. export const playSceneGuide = async (paths: SceneGuidePath[], changeIndexCallback?: (index: number) => void) => {
  252. if (isScenePlayIng.value) {
  253. throw new Error('导览正在播放')
  254. }
  255. isScenePlayIng.value = true
  256. const sceneGuide = sdk.enterSceneGuide(paths)
  257. changeIndexCallback && sceneGuide.bus.on('changePoint', changeIndexCallback)
  258. const quitHandler = () => (isScenePlayIng.value = false)
  259. const clearHandler = isEdit.value ? null : await fullView(quitHandler)
  260. if (!clearHandler) {
  261. sysBus.on('leave', quitHandler, { last: true })
  262. sysBus.on('save', quitHandler, { last: true })
  263. }
  264. sceneGuide.play()
  265. const reces = [
  266. new Promise(resolve => sceneGuide.bus.on('playComplete', resolve)),
  267. new Promise<void>(resolve => {
  268. const stop = watch(isScenePlayIng, () => {
  269. if (!isScenePlayIng.value) {
  270. resolve()
  271. sceneGuide.pause()
  272. stop()
  273. }
  274. })
  275. }),
  276. ]
  277. await Promise.race(reces)
  278. isScenePlayIng.value = false
  279. if (clearHandler) {
  280. clearHandler()
  281. } else {
  282. sysBus.off('leave', quitHandler)
  283. sysBus.off('save', quitHandler)
  284. }
  285. sceneGuide.clear()
  286. sceneGuide.bus.off('changePoint')
  287. }
  288. export const pauseSceneGuide = () => isScenePlayIng.value = false
  289. // -----------------启动关联--------------------
  290. export const setupAssociation = (mountEl: HTMLDivElement) => {
  291. associationModels(sdk)
  292. associationTaggings(mountEl)
  293. associationMessaures(sdk)
  294. }