index.js 67 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777
  1. import mitt from 'mitt'
  2. import axios from 'axios' //{ axios } from '@/api'
  3. let requestLoadCount = 0
  4. let maxLoadingCount = 2; //正在加载模型的最大数目
  5. //0看看,1看见,2深时,3用户上传三维模型,4深时mesh,5深光点云,6深光mesh
  6. const ModelTypes = {
  7. 0 : {name:'看看(八目)', panos4dkk:true},
  8. 1 : {name:'看见(双目转台)', panos4dkk:true, rot90:true},
  9. 2 : {name:'深时', },
  10. 3 : {name:'用户上传三维模型'},
  11. 4 : {name:'深时mesh(激光转台)',panos4dkk:true, rot90:true},//3dtiles
  12. 5 : {name:'深光点云' },
  13. 6 : {name:'深光mesh',panos4dkk:true, rot90:true},//3dtiles
  14. }
  15. let cesAspect
  16. export const enter = ({ dom, mapDom, isLocal, lonlat, scenes }) => {
  17. console.warn('新的页面')
  18. Potree.settings.isOfficial = true //标记为正式、非测试版本
  19. //Potree.fileServer = axios
  20. Potree.settings.libsUrl = './lib/'
  21. let loadStartTime = Date.now()
  22. //正式环境(本地调试会打不开)
  23. if (location.host === 'mix3d.4dkankan.com') {
  24. Potree.settings.urls.prefix = Potree.settings.urls.prefix6
  25. Potree.settings.webSite = 'datav1'
  26. } else if (location.host === 'xfhd.4dkankan.com') {
  27. Potree.settings.urls.prefix = Potree.settings.urls.prefix7
  28. Potree.settings.webSite = 'datav1'
  29. }
  30. const mapBus = mitt(), sceneBus = mitt()
  31. const tagLimitDis = 8;
  32. Potree.settings.showCompass = true
  33. Potree.settings.compassDom = dom.querySelector('#direction')
  34. Potree.settings.mergeType2 = true //标识新版
  35. Potree.settings.modelSkybox = true //是否将全景图贴在模型上(会导致卡顿)。若不显示模型将不显示Reticule
  36. Potree.settings.tiles3DMaxMemory = 300 //稍微增加点
  37. Potree.settings.mergeTransCtlOnClick = true
  38. Potree.settings.canWalkThroughModel = true
  39. let { THREE } = Potree.mergeEditStart(dom, mapDom)
  40. let MergeEditor = viewer.modules.MergeEditor
  41. Potree.settings.unableNavigate = true
  42. /* Potree.settings.showCesium = !!lonlat
  43. Potree.settings.showCesium && buildMap()
  44. */
  45. //因为getPose里用的是target,俯视的yaw不准,所以限制一下不要完全俯视
  46. viewer.mainViewport.view.maxPitch-=0.001
  47. viewer.mainViewport.view.minPitch+=0.001
  48. viewer.addEventListener('camera_changed', e => {
  49. var camera = e.viewport.camera
  50. var pos = camera.position
  51. if (e.viewport.name == 'MainView') {
  52. sceneBus.emit('cameraChange', { x: pos.x, y: pos.y, z: pos.z, rotate: camera.rotation })
  53. updateMap()
  54. Potree.Common.intervalTool.isWaiting('updateCamNear', ()=>{
  55. updateCamNear()
  56. }, 1000)
  57. //viewer.scene.tags.children.forEach(tag=>tag.functions.updateVisiFar())
  58. }
  59. })
  60. //-------------------------------------
  61. let modelAinB = (A,B)=>{ //B的expand(5m) bound完全包含A
  62. let boundB = B.boundingBox.clone().expandByVector(new THREE.Vector3(5,5,5)).applyMatrix4(B.matrixWorld)
  63. let boundA = A.boundingBox.clone().applyMatrix4(A.matrixWorld)
  64. return boundB.containsBox(boundA)
  65. }
  66. let changeMeshVisi = (object, show) => {
  67. if(show == void 0) show = Potree.settings.displayMode == 'showPointCloud' || object == viewer.images360.currentPano.pointcloud && Potree.settings.modelSkybox || object.showInPano //showInPano: 装饰物,一直显示
  68. || !object.panos && modelAinB(object, viewer.images360.currentPano.pointcloud) //装饰物
  69. Potree.Utils.updateVisible(object, 'showPanos', show)
  70. }
  71. if(Potree.settings.canWalkThroughModel){
  72. let lastModel
  73. viewer.images360.addEventListener('flyToPano',(e)=>{//开始漫游 漫游到另一个模型就要选中这个模型?
  74. let model = e.toPano.pano.pointcloud
  75. if(lastModel != model){
  76. changeMeshVisi(model, true)
  77. //MergeEditor.selectModel(model)
  78. //model.result_.flyInPano(e.toPano.pano, {dontFly:true}) //切换模型显示,因为flyInPano有事件怕乱所以统一用这个函数
  79. updateCamNear()
  80. }
  81. })
  82. viewer.images360.addEventListener('flyToPanoDone',(e)=>{
  83. if(!e.makeIt)return
  84. let model = viewer.images360.currentPano.pointcloud
  85. if(lastModel != model){
  86. lastModel?.isModel && changeMeshVisi(lastModel, false)
  87. sceneBus.emit('panoModelChange', model.result_ )
  88. }
  89. lastModel = model
  90. })
  91. }
  92. viewer.images360.addEventListener('endChangeMode',(e)=>{
  93. sceneBus.emit('modeChange', {mode: e.mode == 'showPanos' ? 'pano' : 'fuse', model : e.mode == 'showPanos' && viewer.images360.currentPano.pointcloud.result_} )
  94. Potree.Utils.updateVisible(MergeEditor.transformControls, 'showPanos', e.mode == 'showPointCloud')
  95. Potree.Utils.updateVisible(MergeEditor.boxHelper, 'showPanos', e.mode == 'showPointCloud')
  96. if(e.mode == 'showPanos'){
  97. viewer.setControls( viewer.fpControls )
  98. viewer.removeEventListener('camera_changed', camera_changed)
  99. }else{
  100. viewer.addEventListener('camera_changed', camera_changed)
  101. }
  102. viewer.objs.children.forEach((e)=>{changeMeshVisi(e)})
  103. Potree.settings.canWalkThroughModel || viewer.images360.panos.forEach(pano => {
  104. pano.setEnable(e.mode == 'showPanos' ? pano.pointcloud == viewer.images360.currentPano.model : true)
  105. })
  106. Potree.settings.unableNavigate = e.mode == 'showPointCloud'
  107. updateCamNear()
  108. })
  109. let camera_changed = (e) => {
  110. if (e.viewport.name == 'MainView' && e.changeInfo.positionChanged) {
  111. //viewer.mainViewport.camera.position
  112. viewer.mainViewport.view.radius = 0.1 //使pivot在面前一丢丢距离
  113. viewer.setControls(viewer.orbitControls)
  114. viewer.removeEventListener('camera_changed', camera_changed)
  115. }
  116. }
  117. let requestInPano = false
  118. //-------------------------------------
  119. /* viewer.inputHandler.addEventListener('keydown', (e)=>{
  120. if(e.event.key == "e" ){
  121. MergeEditor.transformControls.mode = 'rotate'
  122. }else if(e.event.key == "w"){
  123. MergeEditor.transformControls.mode = 'translate'
  124. }else if(e.event.key == "s"){
  125. MergeEditor.transformControls.mode = 'scale'
  126. }
  127. }) */
  128. viewer.addEventListener('webglError', e => {
  129. console.error('viewer webglError: ' + e)
  130. let memory = '. \n jsHeapSizeLimit:'+ performance.memory.jsHeapSizeLimit/ 1e6 + ', usedJSHeapSize: '+performance.memory.usedJSHeapSize/ 1e6 + '(M)'
  131. sceneBus.emit('webglError', { msg: e.msg + memory })
  132. })
  133. viewer.compass.setAutoDisplay(true)
  134. /* mapBus.on('visible', v => {
  135. //console.log('mapBus visible', v)
  136. viewer.mapViewer.visible = v
  137. if (v) {
  138. viewer.mapViewer.mapLayer.needUpdate = true
  139. }
  140. viewer.mapViewer.dispatchEvent({type:'forceVisible',visible:v})
  141. }) */
  142. {
  143. let index = 1;
  144. //let setDisplay()
  145. if (!Potree.isIframeChild) {
  146. /* viewer.addEventListener('createIframe',(e)=>{//创建了子页面
  147. }) */
  148. window.winIndex = 0;
  149. window.iframeCreated = function (iframe) {
  150. let child = iframe.contentWindow
  151. child.winIndex = index++
  152. //案件里视图提取页面子页面覆盖了父级页面,父级的模型可以隐藏以释放内存
  153. console.error('createdIframe', child.winIndex, child.location.href)
  154. viewer.setDisplay(false)
  155. child.beforeDestroy = function () { //注:在前端仍会找不到beforeDestroy,可能contentWindow变更??所以手动调用setDisplay
  156. console.warn('beforeDestroy', child.winIndex)
  157. child.viewer && child.viewer.setDisplay(false)
  158. //如果是四维看看的场景,先不管了,页面被销毁应该就没了吧
  159. viewer.setDisplay(true)//恢复主页的模型显示
  160. if (!child.viewer) {
  161. try {
  162. let player = child.__sdk.core.get('Player')
  163. /* let runtime = player.model._3dTilesRuntime
  164. let tileset = runtime.getTileset()
  165. tileset._cache.trim(); //使下一次update时dispose所有不可见的tiles
  166. let sceneRenderer = child.__sdk.core.get('SceneRenderer')
  167. player.model.visible = false
  168. runtime.update(16, sceneRenderer.renderer, sceneRenderer.camera, true) //没用,为何_trimTiles的while无法进入
  169. */
  170. player.model.traverse(e => {
  171. e.geometry && e.geometry.dispose()
  172. if (e.material) {
  173. e.material.map && e.material.map.dispose()
  174. if (e.material.uniforms && e.material.uniforms.map && e.material.uniforms.map.value) {
  175. e.material.uniforms.map.value.dispose()
  176. }
  177. }
  178. }) //效果甚微
  179. /* let sceneRenderer = child.__sdk.core.get('SceneRenderer')
  180. sceneRenderer.renderer.render(sceneRenderer.scene, sceneRenderer.camera)
  181. */
  182. } catch (e) {
  183. console.log(e)
  184. }
  185. }
  186. }
  187. }
  188. //不知道删除iframe时是否那些模型还在内存里,需要释放吗? 如果要需要加一个事件
  189. } else {
  190. }
  191. }
  192. window.THREE = THREE
  193. //isLocal = false
  194. let autoLoads = /* window.autoLoads = */ []
  195. let readyToAddModel
  196. let mainBackground = viewer.background
  197. const units = { 1: 'metric', 2: 'imperial' }
  198. let getMeasureType = function (type, unit = 1) {
  199. let info
  200. switch (type) {
  201. case 'free':
  202. info = { measureType: 'Distance' }
  203. break
  204. case 'area':
  205. info = { measureType: 'Area' }
  206. break
  207. case 'vertical':
  208. info = { measureType: 'Ver Distance' }
  209. break
  210. default:
  211. console.error('无此 measure type')
  212. }
  213. info.unit = units[unit]
  214. return info
  215. }
  216. let getMeasureFunction = function (measure, bus) {
  217. measure.addEventListener('highlight', (e) => {
  218. //console.log('3d->2d highlight',e.state)
  219. bus.emit('highlight', e.state)
  220. })
  221. measure.addEventListener('marker_dropped', (e) => {//拖拽结束后发送changeCallBack
  222. if (measure.parent) {
  223. //未被删除
  224. bus.emit('update', [
  225. measure.dataset_points.map(p => p.clone()),
  226. measure.points_datasets
  227. ])
  228. }
  229. })
  230. return {
  231. /* quit: () => {
  232. Potree.Log('quit结束且删除: ' + measure.id, '#00c7b2')
  233. viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure })
  234. }, //触发结束。退出测量模式,清除之前操作 */
  235. destroy: () => {
  236. viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure })
  237. viewer.scene.removeMeasurement(measure)
  238. },
  239. /* getPoints: () => {
  240. return measure.points
  241. },
  242. getDatasetLocations: () => {
  243. return measure.dataset_points
  244. },
  245. getDatasets: () => {
  246. return measure.points_datasets
  247. },
  248. getDatasetId: () => {
  249. return measure.datasetId
  250. }, */
  251. getArea: () => {
  252. return measure.area //{value:area, string:..}
  253. },
  254. getDistance: () => {
  255. if (measure.points.length < 2) return 0
  256. var value = measure.points[0].distanceTo(measure.points[1])
  257. return {
  258. value, //米
  259. string: measure.getConvertString(value, 'distance')
  260. }
  261. },
  262. //手动开启或关闭:
  263. show: () => {
  264. Potree.Utils.updateVisible(measure, 'inListByUser', true)
  265. },
  266. hide: () => {
  267. Potree.Utils.updateVisible(measure, 'inListByUser', false)
  268. },
  269. fly() {
  270. let result = viewer.focusOnObject(measure, 'measure', 1200)
  271. return result.msg ? result.msg : result.promise
  272. //返回值 1 deferred 表示即将位移 2 'posNoChange' 表示已在最佳位置 3 'tooFar' 表示距离最佳位置太远
  273. },
  274. changeSelect(isHight) {
  275. console.log('2d->3d isHight ', isHight)
  276. measure.setSelected(isHight, 'byList')
  277. },
  278. }
  279. }
  280. let sdk = {
  281. sceneBus, mapBus,
  282. canTurnToPanoMode(pos) {
  283. pos = pos ? new THREE.Vector3().copy(pos) : viewer.images360.position
  284. let pano = viewer.images360.findNearestPano(pos)
  285. if (pano && pano.position.distanceTo(pos) < Potree.config.panoFieldRadius * pano.pointcloud.scale.x) {
  286. return {model:pano.pointcloud.result_}
  287. }
  288. //poschange后会调用这个,如果返回false会变为点云模式,且不会自动变回原先的模式
  289. },
  290. getPositionByScreen(pos2d, hopeModelId) {//通过屏幕坐标获取真实坐标 . hopeModelId: 如果指定了模型,优先返回hopeModelId上的intersect
  291. //console.log('getPositionByScreen',hopeModelId)
  292. hopeModelId = null
  293. let worldPos, localPos, modelId, intersect, normal, localNormal
  294. let Handler = viewer.inputHandler
  295. let reGet = () => {//不使用当前鼠标所在位置的intersect,单独算
  296. pos2d.clientX = pos2d.x
  297. pos2d.clientY = pos2d.y
  298. pos2d.onlyGetIntersect = true
  299. pos2d.whichPointcloud = true
  300. if (hopeModelId != void 0) {//隐藏其他的模型
  301. let models = MergeEditor.getAllObjects()
  302. models.forEach(model => {
  303. Potree.Utils.updateVisible(model, 'forPick', model.dataset_id == hopeModelId)
  304. })
  305. }
  306. let intersect2 = Handler.onMouseMove(pos2d)
  307. if (hopeModelId != void 0) {//恢复
  308. let models = MergeEditor.getAllObjects()
  309. models.forEach(model => {
  310. Potree.Utils.updateVisible(model, 'forPick', true)
  311. })
  312. }
  313. if (intersect2 && intersect2.location) {
  314. intersect = intersect2
  315. }
  316. }
  317. if (pos2d && pos2d.inDrag) {
  318. reGet()
  319. } else {
  320. intersect = Handler.intersect
  321. if (intersect) {
  322. modelId = intersect.pointcloud ? intersect.pointcloud.dataset_id : intersect.object.dataset_id
  323. if (hopeModelId != void 0 && modelId != hopeModelId) {
  324. reGet()
  325. }
  326. }
  327. }
  328. if (intersect && intersect.location) {
  329. modelId = intersect.pointcloud ? intersect.pointcloud.dataset_id : intersect.object.dataset_id
  330. /* if(hopeModelId != void 0 && modelId != hopeModelId){
  331. return null
  332. } */
  333. worldPos = intersect.location.clone()
  334. localPos = Potree.Utils.datasetPosTransform({ toDataset: true, datasetId: modelId, position: worldPos })
  335. normal = intersect.normal
  336. localNormal = intersect.localNormal
  337. } else return null
  338. return { worldPos, modelId, normal, localPos, localNormal }
  339. },
  340. getScreenByPosition(pos3d, modelId, canShelter/* , disToCameraLimit */) {//通过模型局部坐标获取屏幕坐标
  341. //console.log('getScreenByPoint ')
  342. let isLocal = modelId != void 0
  343. pos3d = new THREE.Vector3().copy(pos3d)
  344. let worldPos = isLocal ? Potree.Utils.datasetPosTransform({ fromDataset: true, datasetId: modelId, position: pos3d }) : pos3d
  345. if (!worldPos) return
  346. if (canShelter) {
  347. if (viewer.inputHandler.ifBlockedByIntersect(worldPos, 0.1, true)) return { trueSide: false };
  348. }
  349. var viewport = viewer.mainViewport
  350. var camera = viewport.camera
  351. var dom = viewer.renderArea
  352. if (tagLimitDis != void 0) {
  353. if (camera.position.distanceToSquared(worldPos) > Math.pow(tagLimitDis, 2)) return false
  354. }
  355. //console.log('getScreenByPoint ' + pos3d.toArray())
  356. return Potree.Utils.getPos2d(worldPos, viewport, dom)
  357. },
  358. setCameraFov(fov) {
  359. viewer.setFOV(fov)
  360. },
  361. screenshot: (width, height/* , bgOpacity=0 */ ) => {//
  362. //截图
  363. let bgOpacity = Potree.settings.showCesium ? 0 : 1 /* viewer.background == 'skybox' */ //因为要画map底图所以上层只能透明。之后需要的话再改
  364. console.log('bgOpacity', bgOpacity)
  365. Potree.Utils.updateVisible(MergeEditor.boxHelper, 'screenshot', false)
  366. var { getImagePromise, finishPromise } = viewer.startScreenshot({ type: 'default', /* useRenderTarget:true, */bgOpacity }, width, height)
  367. var deferred = $.Deferred();
  368. finishPromise.done(({ dataUrl }) => {
  369. if(Potree.settings.displayMode != 'showPanos' && Potree.settings.showCesium){//need map background
  370. Potree.cesScreenshot(width, height).done((mapBGurl)=>{
  371. let img = new Image(); img.src = dataUrl
  372. let imgBG = new Image(); imgBG.src = mapBGurl
  373. let loadCount = 0
  374. img.onload = imgBG.onload = ()=>{
  375. loadCount++;
  376. if(loadCount == 2){
  377. let url = Potree.Common.imgAddLabel(imgBG,img,{leftRatioToImg:0,topRatioToImg:0})
  378. deferred.resolve(url)
  379. }
  380. }
  381. })
  382. }else{
  383. deferred.resolve(dataUrl)
  384. }
  385. Potree.Utils.updateVisible(MergeEditor.boxHelper, 'screenshot', true)
  386. })
  387. return deferred.promise()
  388. },
  389. getPose() {//获取当前点位和朝向
  390. const camera = viewer.scene.getActiveCamera()
  391. const target = viewer.scene.view.getPivot().clone()
  392. const position = viewer.scene.view.position.clone()
  393. const pose = { position, target, displayMode:Potree.settings.displayMode }
  394. if(Potree.settings.displayMode == 'showPanos'){
  395. let model = viewer.images360.currentPano.pointcloud
  396. pose.panoId = viewer.images360.currentPano.originID
  397. pose.model = model.result_
  398. pose.posInModel = Potree.Utils.datasetPosTransform({ toDataset: true, position: camera.position.clone(), object:model })
  399. pose.rotInModel = Potree.Utils.datasetRotTransform({ toDataset: true, quaternion: camera.quaternion.clone(), getQuaternion: true, pointcloud:model }).toArray() //拿第一个数据集
  400. }
  401. //console.log('getPose',position, target)
  402. return pose
  403. },
  404. comeTo(o = {}) {
  405. //console.log('comeTo',o.position, o.target)
  406. //飞到某个点
  407. let deferred = $.Deferred()
  408. if(o.panoId != void 0){
  409. let model = o.model.model
  410. let pano = model.panos.find(a=>a.originID == o.panoId)
  411. if(pano){
  412. o.rotInModel = new THREE.Quaternion().fromArray(o.rotInModel)
  413. let quaternion = Potree.Utils.datasetRotTransform({ fromDataset: true, quaternion: o.rotInModel, getQuaternion: true, object:model})
  414. o.model.flyInPano(pano, {quaternion, duration:0, callback(){
  415. o.callback && o.callback()
  416. deferred.resolve(true)
  417. }})
  418. return deferred.promise()
  419. }else{
  420. console.warn('没有找到漫游点',o)
  421. }
  422. }else if(requestInPano){
  423. requestInPano.result_.flyOutPano()
  424. }else{
  425. if (o.modelId != void 0) {
  426. ['position', 'target'].forEach(e => {
  427. if (o[e]) {
  428. o[e] = Potree.Utils.datasetPosTransform({ fromDataset: true, datasetId: o.modelId, position: o[e] })
  429. }
  430. })
  431. }
  432. }
  433. if (o.distance) {
  434. let position = o.target || o.position
  435. return viewer.focusOnObject({ position }, 'tag', null, { distance: o.distance }).promise
  436. }
  437. viewer.scene.view.setView($.extend({}, o, {
  438. duration: o.dur,
  439. callback: () => {
  440. o.callback && o.callback()
  441. deferred.resolve(true)
  442. }
  443. }))
  444. return deferred.promise()
  445. },
  446. setBackdrop(sky, type, { scale, rotate }) {//天空盒背景
  447. //console.log('天空盒背景', sky,type)
  448. let setGroundAndText = (color) => {
  449. MergeEditor.secondCompass.dom.find(".dirText").css({ 'color': color })
  450. viewer.compass.dom.find(".dirText").css({ 'color': color })
  451. MergeEditor.ground.material.uniforms.uColor.value.set(color)
  452. //MergeEditor.ground.children[0].material.color.set(color)
  453. }
  454. viewer.dispatchEvent('content_changed')
  455. if(type == 'map'){
  456. MergeEditor.setGroundPlaneImg(null)
  457. viewer.setBackground(mainBackground)
  458. Potree.settings.showCesium = true
  459. buildMap()
  460. viewer.backgroundOpacity = 0
  461. return
  462. }else{
  463. Potree.settings.showCesium = false
  464. }
  465. if (type == 'bimg') {//地面图
  466. MergeEditor.setGroundPlaneImg(sky, scale, rotate)
  467. setGroundAndText('#e0e0e0')
  468. viewer.setBackground(mainBackground)
  469. } else {
  470. MergeEditor.setGroundPlaneImg(null)
  471. if (sky == 'none') {
  472. viewer.setBackground(mainBackground)
  473. setGroundAndText('#eee')
  474. } else if (sky[0] == '#') {
  475. viewer.setBackground(new THREE.Color(sky))
  476. let color = sky == '#fff' ? '#666' : sky == '#333' ? '#eee' : '#bbb' //反相
  477. setGroundAndText(color)
  478. } else if (type == 'image-map' || type == 'vector-map') {//影像|矢量 地图
  479. } else {//环境
  480. viewer.setBackground('skybox', sky)
  481. setGroundAndText('#e0e0e0')
  482. }
  483. }
  484. },
  485. switchMapType(type) {
  486. let map = viewer.mapViewer.mapLayer.maps.find(e => e.name == 'map')
  487. map.switchStyle(type/* map.style == 'satellite' ? 'standard' : 'satellite' */)
  488. },
  489. enableMap(mapArea, latlng) {
  490. if (!viewer.mapViewer) {
  491. //--------------------------------
  492. viewer.mapViewer = new Potree.MapViewer(mapArea)
  493. viewer.mapViewer.initProjection()
  494. //focus
  495. let boundSize = new THREE.Vector3(200, 150, 1).max(viewer.bound.boundSize)
  496. viewer.mapViewer.addEventListener('viewerResize', () => {
  497. viewer.mapViewer.moveTo(viewer.bound.center, boundSize, 0)
  498. }, { once: true })
  499. }
  500. },
  501. enterSceneGuide(pathArr) {//导览 (不需要修改参数)
  502. let editor = viewer.modules.CamAniEditor
  503. console.log('pathArr', pathArr)
  504. //console.log('enterSceneGuide',pathArr)
  505. pathArr.forEach(e=>{
  506. if(e.panoId != void 0){
  507. e.model = e.model.model
  508. }
  509. })
  510. let data = {
  511. //duration: pathArr.slice(0, pathArr.length - 1).reduce(function (total, currentValue) { return total + currentValue.time }, 0), //总时长(要去掉最后一个,因为已到终点,该点time无意义)
  512. points: pathArr,
  513. useDurSlice: true
  514. }
  515. let ani = editor.createMulAnimation(data)
  516. //注:最多只存在一条导览
  517. let bus = mitt()
  518. //播放完成
  519. ani.event_.addEventListener('playDone', () => {
  520. bus.emit('playComplete')
  521. viewer.images360.panos.forEach(e=>e.marker && Potree.Utils.updateVisible(e, 'playAni', true))
  522. })
  523. //切换点
  524. ani.event_.addEventListener('updateCurrentIndex', e => {
  525. bus.emit('changePoint', e.currentIndex + 1)
  526. })
  527. return {
  528. bus,
  529. play() {
  530. MergeEditor.selectModel(null)
  531. viewer.images360.panos.forEach(e=>e.marker && Potree.Utils.updateVisible(e, 'playAni', false))
  532. ani.play()
  533. },
  534. pause() {
  535. viewer.images360.panos.forEach(e=>e.marker && Potree.Utils.updateVisible(e, 'playAni', true))
  536. ani.stop()
  537. },
  538. clear() {
  539. ani.remove()
  540. },
  541. }
  542. },
  543. //[path1, paht2], { time, speed }
  544. calcPathInfo(paths, info) { //传入的time, speed仅有一个。返回完整的 time, speed
  545. //这一版的control似乎无法在某个位置上改变角度,位置和角度一般都是一起变的,所以先不增加单位更换功能。
  546. let pos1 = new THREE.Vector3().copy(paths[0].position)
  547. let pos2 = new THREE.Vector3().copy(paths[1].position)
  548. let dis = pos1.distanceTo(pos2)
  549. if (info.time != void 0) {
  550. info.speed = dis / info.time
  551. } else {
  552. info.time = dis / info.speed
  553. }
  554. return info
  555. },
  556. addModel(props) {
  557. let bus = props.bus = mitt()
  558. //console.log('addModel',props)
  559. props.isFirstLoad = isLocal ? props.bottom == void 0 : (props.isDynamicAdded || props.mode == 'single') // 在编辑时用户添加的 或 展示单个模型 (props.mode='single'模型展示页, props.mode='many'融合页)
  560. if (props.opacity == void 0) props.opacity = 1
  561. if (props.type == 'obj') props.type = 'glb'
  562. props.scale /= 100
  563. if (props.rotation) {
  564. if (props.rotation._x == void 0 && props.rotation.x != void 0) {
  565. props.rotation = new THREE.Euler().setFromVector3(props.rotation)
  566. }
  567. }
  568. let getDefaultRotation = () => {
  569. if(ModelTypes[props.fromType]?.rot90 && props.type != 'glb'){
  570. return new THREE.Euler(Math.PI / 2, 0, 0)
  571. } else return new THREE.Euler(0, 0, 0)
  572. }
  573. if (!props.isFirstLoad) {
  574. if (autoLoads.length == 0) { //首次加载
  575. setTimeout(() => {
  576. let sizes = autoLoads.map(e => e.size || 0)
  577. console.log('需要请求加载的模型大小为', sizes, '总大小', sizes.reduce(function (total, currentValue) {
  578. let current = parseFloat(currentValue)
  579. return total + ((typeof currentValue == 'number' || currentValue.includes('M')) ? current : current / 1024)
  580. }, 0))
  581. readyToAddModel = true //准备开始加载
  582. loadNext()//startLoad(autoLoads[0])
  583. }, 30)
  584. }
  585. autoLoads.push(props)
  586. readyToAddModel = false
  587. } else {
  588. readyToAddModel = true
  589. props.rotation = getDefaultRotation()
  590. }
  591. let model
  592. let done = (model_) => {
  593. model = model_
  594. model.result_ = result
  595. model.props = props
  596. result.model = model
  597. model.fromType = ModelTypes[props.fromType].name
  598. if (!props.isFirstLoad) {
  599. model.visible = false//先不显示,防止卡顿
  600. }
  601. model.showInPano = props.raw.showInPano
  602. props.opacity < 100 && result.changeOpacity(props.opacity)
  603. model.addEventListener('changeSelect', (e) => {
  604. bus.emit('changeSelect', e.selected)
  605. })
  606. let lastState = {}
  607. model.addEventListener('transformChanged', (e) => {
  608. let msg = {}
  609. if (!lastState.position || !model.position.equals(lastState.position)) {
  610. lastState.position = msg.position = model.position.clone()
  611. }
  612. if (!lastState.rotation || !model.rotation.equals(lastState.rotation)) {
  613. lastState.rotation = msg.rotation = model.rotation.clone()
  614. }
  615. if (lastState.scale == void 0 || model.scale.x * 100 != lastState.scale) {
  616. lastState.scale = msg.scale = model.scale.x * 100
  617. }
  618. msg = Potree.Common.CloneObject(msg)
  619. //console.log(msg)
  620. bus.emit('transformChanged', msg)
  621. })
  622. spliceFromArr(model, props, true)
  623. model.addEventListener('changeSelect', (e) => {
  624. MergeEditor.transformControls.visible && e.selected && MergeEditor.transformControls.attach(model, e.clickPos) //: MergeEditor.transformControls.detach()
  625. })
  626. MergeEditor.modelAdded(model)
  627. if (props.mode == 'single') {//模型查看页
  628. MergeEditor.noNeedSelection = true
  629. setTimeout(() => {
  630. MergeEditor.focusOn([model], 1000, true, true)
  631. }, 1)
  632. }
  633. if(ModelTypes[props.fromType].panos4dkk){
  634. Potree.load4dkkPanos(props.raw.num, model, getDefaultRotation(), () => {
  635. bus.emit('loadDone')
  636. }, props.fromType == 0 ? '2k' : '4k' ) //看看场景是2k
  637. } else {
  638. bus.emit('loadDone')
  639. }
  640. //console.log('loadDone' )
  641. }
  642. let progressFun = (progress) => {
  643. bus.emit('loadProgress', progress)
  644. }
  645. let onError = function (xhr) {
  646. bus.emit('loadError', xhr)
  647. console.log('loadError!!!!!!!!!', Potree.Common.getNameFromURL(props.url), props.size, xhr)
  648. spliceFromArr(model, props, false)
  649. }
  650. try {
  651. props.url = JSON.parse(props.url) //去掉 '\'
  652. } catch (e) { }
  653. props.done = done; props.progressFun = progressFun; props.onError = onError
  654. if (readyToAddModel) {
  655. if (autoLoads.filter(e => e.loading).length < maxLoadingCount) {
  656. startLoad(props)
  657. }
  658. }
  659. let scaleMeasure
  660. let result = {
  661. bus,
  662. model,
  663. getDefaultRotation,
  664. supportPano() { //是否支持全景图
  665. return model?.panos?.length > 0
  666. },
  667. flyInPano(pano, {dontFly, quaternion, duration}={}) {// 飞入全景图
  668. requestInPano = model
  669. pano = pano || viewer.images360.findNearestPano(null, model.panos)
  670. if (pano) {
  671. dontFly || viewer.images360.flyToPano({ pano, canCancelLast: true, quaternion, duration})
  672. Potree.settings.displayMode = 'showPanos'
  673. }
  674. },
  675. flyOutPano() {// 飞出全景图(就是切换到正常融合视角)
  676. requestInPano = false
  677. Potree.settings.displayMode = 'showPointCloud'
  678. /* setTimeout(() => {//在下一帧再变,因为3dtiles需要更新一下才会显示tiles
  679. if (!requestInPano) {
  680. Potree.settings.displayMode = 'showPointCloud'
  681. Potree.Utils.updateVisible(MergeEditor.boxHelper, 'showPanos', true)
  682. }
  683. }, 50) */
  684. },
  685. changeShow(show) {
  686. props.show = show //for autoLoads show model
  687. if (model) {
  688. Potree.Utils.updateVisible(model, 'datasetSelection', show)
  689. if (model.panos) {
  690. model.panos.forEach(e => e.setEnable(show))
  691. }
  692. viewer.dispatchEvent('content_changed')
  693. }
  694. },
  695. changeSelect(state) {
  696. console.error('select', state)
  697. if (model) {
  698. let fly = viewer.images360.latestRequestMode != 'showPanos'
  699. MergeEditor.selectModel(model, state, fly, true)
  700. updateCamNear()
  701. //console.log('changeSelect', props.id, state)
  702. }
  703. },
  704. changeScale(s) {
  705. if (model) {
  706. s /= 100
  707. if (model.scale.x == s) return
  708. //MergeEditor.history.beforeChange(model)//但不知道什么时候结束拖拽
  709. model.scale.set(s, s, s)
  710. model.isPointcloud && model.changePointSize(/* Potree.config.material.realPointSize * s */)
  711. model.dispatchEvent("scale_changed")
  712. }
  713. },
  714. changeOpacity(opacity) { //见笔记:透明物体的材质设置
  715. if (opacity == void 0) opacity = 100
  716. opacity /= 100
  717. MergeEditor.changeOpacity(model, opacity)
  718. },
  719. changeBottom(z) {
  720. /* model && MergeEditor.setModelBtmHeight(model,z)
  721. model.dispatchEvent('transformChanged') //改了position */
  722. },
  723. changePosition(pos) {//校准取消时执行
  724. console.log('changePosition', pos.x, pos.y, pos.z)
  725. model && model.position.copy(pos)
  726. model.dispatchEvent({ type: 'position_changed' })
  727. },
  728. changeRotation(rot) {//校准取消时执行
  729. console.log('changeRotation', rot.x, rot.y, rot.z)
  730. model && model.rotation.setFromVector3(rot)
  731. model.dispatchEvent({ type: 'rotation_changed' })
  732. },
  733. enterRotateMode() {
  734. if (model) {
  735. if (MergeEditor.split) {//分屏校准
  736. MergeEditor.setTransformState('rotate')
  737. MergeEditor.transformControls2.attach(model)
  738. MergeEditor.transformControls2.mode = 'rotate'
  739. }
  740. MergeEditor.transformControls.attach(model)
  741. MergeEditor.transformControls.mode = 'rotate'
  742. }
  743. },
  744. enterMoveMode() {
  745. console.log('enterMoveMode')
  746. if (model) {
  747. if (MergeEditor.split) {//分屏校准
  748. MergeEditor.setTransformState('translate')
  749. MergeEditor.transformControls2.attach(model)
  750. MergeEditor.transformControls2.mode = 'translate'
  751. }
  752. MergeEditor.transformControls.attach(model)
  753. MergeEditor.transformControls.mode = 'translate'
  754. }
  755. },
  756. leaveTransform() {
  757. console.log('leaveTransform')
  758. if (MergeEditor.split) {//分屏校准
  759. MergeEditor.setTransformState(null)
  760. } else {
  761. MergeEditor.transformControls.detach()
  762. MergeEditor.transformControls2.detach()
  763. }
  764. MergeEditor.history.clear()
  765. },
  766. enterAlignment() {//开始校准
  767. result.leaveTransform()
  768. MergeEditor.enterSplit()
  769. //console.log('enterAlignment',model.position, model.rotation)
  770. let bus = new mitt()
  771. /* MergeEditor.transformControls.attach(model)
  772. MergeEditor.transformControls.mode = 'translate' */
  773. return {
  774. bus
  775. }
  776. },
  777. leaveAlignment() {
  778. //console.log('leaveAlignment',model.position, model.rotation)
  779. MergeEditor.leaveSplit()
  780. MergeEditor.transformControls.detach()
  781. MergeEditor.transformControls2.detach()
  782. },
  783. enterScaleSet() {//设置比例
  784. let bus = new mitt()
  785. let length, measureBuilded;
  786. //viewer.outlinePass.selectedObjects = []
  787. if (!Potree.Utils.isInsideFrustum(model.boundingBox.clone().applyMatrix4(model.matrixWorld), viewer.scene.getActiveCamera())) {
  788. MergeEditor.focusOn(model, 600)
  789. }
  790. MergeEditor.getAllObjects().forEach(m => {//隐藏其他的模型
  791. if (m != model) Potree.Utils.updateVisible(m, 'enterScaleSet', false)
  792. })
  793. let setScale = () => {
  794. if (length == void 0 || !measureBuilded) return
  795. let vec = new THREE.Vector3().subVectors(viewer.mainViewport.camera.position, scaleMeasure.points[1])
  796. let s = length / (scaleMeasure.points[0].distanceTo(scaleMeasure.points[1]))
  797. result.changeScale(model.scale.x * s * 100)
  798. /* setTimeout(()=>{
  799. viewer.focusOnObject(scaleMeasure , 'measure', 500)
  800. },1) */
  801. let newCamPos = new THREE.Vector3().addVectors(scaleMeasure.points[1], vec.multiplyScalar(s))
  802. viewer.scene.view.setView({
  803. position: newCamPos, target: scaleMeasure.getCenter(), duration: 0, callback: () => {
  804. //更改target到measure中心的好处就是可以让相机绕measure中心转,坏处是每次更改都会变一下画面
  805. }
  806. })
  807. }
  808. return {
  809. bus,
  810. setLength(v) {
  811. if (!v) return
  812. length = v
  813. setScale()
  814. },
  815. startMeasure() {
  816. if (scaleMeasure) {
  817. viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure: scaleMeasure })
  818. viewer.scene.removeMeasurement(scaleMeasure)
  819. }
  820. measureBuilded = false
  821. scaleMeasure = viewer.measuringTool.startInsertion(
  822. { measureType: "Distance", unit: "metric" },
  823. () => {
  824. //done:
  825. //bus.emit('end' ) //完成
  826. measureBuilded = true
  827. setScale()
  828. },
  829. () => {
  830. //cancel
  831. //bus.emit('quit') //删除
  832. }
  833. )
  834. scaleMeasure.addEventListener('marker_dropped', (e) => {//拖拽结束后发送changeCallBack
  835. if (scaleMeasure.parent) {
  836. //未被删除
  837. measureBuilded && setScale()
  838. }
  839. })
  840. }
  841. }
  842. },
  843. leaveScaleSet() {
  844. if (scaleMeasure) {
  845. viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure: scaleMeasure })
  846. viewer.scene.removeMeasurement(scaleMeasure)
  847. scaleMeasure = null
  848. }
  849. //viewer.outlinePass.selectedObjects = [model];
  850. MergeEditor.getAllObjects().forEach(m => {//恢复其他的模型
  851. if (m != model) Potree.Utils.updateVisible(m, 'enterScaleSet', true)
  852. })
  853. },
  854. destroy() {
  855. model && MergeEditor.removeModel(model)
  856. result.changeSelect(false)
  857. viewer.dispatchEvent('content_changed')
  858. }
  859. }
  860. return result
  861. },
  862. //测量线的点都附着于各个模型,当模型变化时,点跟着变化。
  863. // 新的测量创建方法,传入type 返回新测量对象
  864. startMeasure(type) {
  865. // 寻创建的测量对象有上面绘画测量对象的所有方法
  866. const bus = mitt()
  867. let info = getMeasureType(type)
  868. let measure = viewer.measuringTool.startInsertion(
  869. info,
  870. () => {
  871. //done:
  872. /* bus.emit('submit', {
  873. dataset_points: measure.dataset_points.map(p=>p.clone()) ,
  874. points_datasets: measure.points_datasets
  875. } ) //完成 */
  876. bus.emit('submit')
  877. bus.emit('update', [
  878. measure.dataset_points.map(p => p.clone()),
  879. measure.points_datasets
  880. ])
  881. },
  882. () => {
  883. //cancel
  884. bus.emit('cancel'/* , ret */) //删除
  885. }
  886. )
  887. Potree.Log('startMeasure: ' + measure.id, '#00c7b2')
  888. /* let cancel = ()=>{
  889. Potree.Log('clear删除: ' + measure.id, '#00c7b2')
  890. viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure })
  891. viewer.scene.removeMeasurement(measure)
  892. } */
  893. let result = {
  894. bus,
  895. ...getMeasureFunction(measure, bus),
  896. }
  897. /* StartMeasure = Measure & {
  898. // 多了cancel 取消测量的事件,没有参数
  899. // 多了invalidPoint 当用户测量了无效点时的事件,抛出无效原因
  900. bus: Emitter<{ cancel: void; invalidPoint: string }>
  901. } */
  902. return result
  903. },
  904. // 绘画测量线(非新增使用)
  905. // type = 'free' (自由) || 'vertical' (垂直) || 'area' (面积)
  906. // positions 点数组 构成如下 [{ point: {x,y,z}, modelId: 1 }]
  907. drawMeasure(type, dataset_points, points_datasets) {
  908. // 返回测量对象有如下
  909. const bus = mitt()
  910. let info = getMeasureType(type /* , unit */)
  911. //info.points = positions
  912. info.dataset_points = dataset_points
  913. info.points_datasets = points_datasets
  914. //info.sid = sid
  915. info.bus = bus
  916. let measure = viewer.measuringTool.createMeasureFromData(info)
  917. if (!measure) return { bus }
  918. Potree.Log('drawMeasure由数据新建: ' + measure.id, '#00c7b2')
  919. let result = {
  920. bus,
  921. setPositions(dataset_points, points_datasets) {//用于恢复measure的点,不会修改点的个数
  922. measure.dataset_points = dataset_points.map(e => {
  923. return e && new THREE.Vector3().copy(e)
  924. })
  925. measure.points_datasets = points_datasets
  926. measure.points = measure.dataset_points.map((p, i) => {
  927. return Potree.Utils.datasetPosTransform({ fromDataset: true, datasetId: measure.points_datasets[i], position: p })
  928. })
  929. measure.getPoint2dInfo(measure.points)
  930. measure.update({ ifUpdateMarkers: true })
  931. measure.setSelected(false)//隐藏edgelabel
  932. },
  933. ...getMeasureFunction(measure, bus),
  934. }
  935. return result
  936. },
  937. createPath(props){//路线
  938. /* export type Path = {
  939. bus: Emitter<{
  940. // 标注点击事件
  941. click: void;
  942. // 鼠标移入标注事件
  943. enter: void;
  944. // 鼠标移出标注事件
  945. leave: void;
  946. // 顶部标注中心点像素位置更改事件, 传出像素位置
  947. lineTopPositionChange: SceneLocalPos
  948. // 路径点位置变更
  949. changePoints: PathProps['points']
  950. // 距离相机位置变更
  951. toCameraDistanceChange: number
  952. }>;
  953. // 是否可编辑
  954. changeCanEdit: (canMove: boolean) => void
  955. // 标注可见性
  956. visibility: (visibility: boolean) => void
  957. // 更改标题气泡属性
  958. changeLine: (props: Partial<PathProps['line']>) => void
  959. // 顶标注中心像素位置
  960. lineTopPosition: ScreenLocalPos
  961. // 距离相机位置
  962. toCameraDistance: number
  963. // 线段销毁
  964. destory: () => void */
  965. },
  966. /* createTagging: (props: Tagging3DProps) => Tagging3D
  967. export type Tagging3DProps = {
  968. // 标注类型 2d | 3d
  969. type: TaggingPositionType,
  970. // 模型id
  971. modelId: string,
  972. // 贴地射线获取的位置
  973. position: SceneLocalPos
  974. // 是否可以移动
  975. canMove: boolean,
  976. // 贴地图片url
  977. image: string
  978. // 离地高度
  979. altitudeAboveGround: number,
  980. // 贴地图片的变换
  981. mat: {
  982. scale: SceneLocalPos,
  983. rotation: Rotation,
  984. }
  985. }
  986. export type Tagging3D = {
  987. bus: Emitter<{
  988. // 标注点击事件
  989. click: void;
  990. // 鼠标移入标注事件
  991. enter: void;
  992. // 鼠标移出标注事件
  993. leave: void;
  994. // 顶部标注中心点像素位置更改事件, 传出像素位置
  995. topPositionChange: SceneLocalPos
  996. // 位置变更
  997. changePosition: {pos: SceneLocalPos, modelId: string}
  998. // 距离相机位置变更
  999. toCameraDistanceChange: number
  1000. }>;
  1001. changeCanMove: (canMove: boolean) => void
  1002. // 更改图标
  1003. changeImage: (url: string) => void
  1004. // 标注可见性
  1005. visibility: (visibility: boolean) => void
  1006. // 标注图片变换,传入变换
  1007. changeMat: (props: Tagging3DProps['mat']) => void
  1008. // 更改离地高度
  1009. changeAltitudeAboveGround: (val: number) => void
  1010. // 顶标注中心像素位置 (2d要加上图片大小)
  1011. topPosition: SceneLocalPos
  1012. // 距离相机位置
  1013. toCameraDistance: number
  1014. // 标注销毁
  1015. destory: () => void
  1016. }
  1017. */
  1018. createTagging(props){
  1019. let bus = mitt()
  1020. let labelHide = false
  1021. let root = viewer.scene.pointclouds.concat(viewer.objs.children).find(e=>e.dataset_id == props.modelId)
  1022. if(!root){
  1023. return console.error('热点没有找到该modelId,模型是否已经删除?')
  1024. }
  1025. let info = {
  1026. position: new THREE.Vector3().copy(props.position), //局部坐标
  1027. normal: new THREE.Vector3().copy(props.normal),
  1028. root
  1029. }
  1030. let tag = viewer.tagTool.createTagFromData(info)
  1031. tag.addEventListener('mouseover',()=>{
  1032. bus.emit('enter')
  1033. })
  1034. tag.addEventListener('mouseleave',()=>{
  1035. bus.emit('leave')
  1036. })
  1037. tag.addEventListener('click',()=>{
  1038. bus.emit('click')
  1039. })
  1040. tag.addEventListener('posChanged',(e)=>{
  1041. bus.emit(changePosition, {
  1042. modelId: tag.root.dataset_id
  1043. })
  1044. })
  1045. tag.functions = {
  1046. bus,
  1047. changeType(type){
  1048. let onMesh = type == '3d'
  1049. if(tag.onMesh != onMesh){
  1050. tag.changeOnMesh(onMesh)
  1051. Potree.Utils.updateVisible(tag.line,'hideTitle', labelHide && onMesh ? false : true)
  1052. }
  1053. },
  1054. visibility(v){// 标注可见性
  1055. Potree.Utils.updateVisible(tag,'force', v)
  1056. },
  1057. visibilityTitle(v){
  1058. labelHide = !v
  1059. Potree.Utils.updateVisible(tag.label,'hideTitle', v)
  1060. tag.onMesh && Potree.Utils.updateVisible(tag.line,'hideTitle', v)
  1061. },
  1062. changeImage(url){
  1063. tag.changeMap(url)
  1064. },
  1065. changeTitle(title){
  1066. tag.changeTitle(title)
  1067. },
  1068. matChange(o){//大小旋转 贴墙时
  1069. console.log(o)
  1070. //tag.setFaceAngle()
  1071. //tag.spot.scale.set()
  1072. },
  1073. // 更改离地高度
  1074. changeAltitudeAboveGround(height){//线长
  1075. tag.changeLineLen(height)
  1076. },
  1077. changeCanMove(canMove){
  1078. tag.dragEnable = canMove
  1079. },
  1080. /* // 顶标注中心像素位置 (2d要加上图片大小)
  1081. topPosition: SceneLocalPos
  1082. // 距离相机位置
  1083. */
  1084. /* topPosition(){//label的位置 暂定 spot上方
  1085. const disToSpot = 0.2;
  1086. tag.lineLength + disToSpot
  1087. }, */
  1088. getImageCenter(){
  1089. tag.titleLabel.parent.updateMatrix()
  1090. //tag.titleLabel.parent.updateMatrixWorld()
  1091. let pos = tag.titleLabel.parent.getWorldPosition(new THREE.Vector3)
  1092. console.log(pos)
  1093. return pos
  1094. },
  1095. /* toCameraDistance(far){//多远会消失
  1096. tag.farSquared = far * far
  1097. this.updateVisiFar(dis)
  1098. },
  1099. updateVisiFar(){//我自己调用
  1100. if(tag.farSquared){
  1101. let v = viewer.mainViewport.camera.position.distanceToSquared(tag.position) < tag.farSquared
  1102. Potree.Utils.updateVisible(tag,'updateVisiFar',v)
  1103. }
  1104. }, */
  1105. getCameraDisSquared(){
  1106. return viewer.mainViewport.camera.position.distanceToSquared(tag.position) /* < tag.farSquared */
  1107. },
  1108. destory(){
  1109. tag.dispose()
  1110. },
  1111. }
  1112. /*
  1113. tag.functions.changeType(props.type)
  1114. tag.functions.changeImage(props.image)
  1115. */
  1116. return tag.functions
  1117. },
  1118. /*
  1119. addTag(info) {//加标签
  1120. let bus = mitt()
  1121. let tag
  1122. let done = () => {
  1123. bus.emit('added')
  1124. bus.emit('update', { position: tag.position.clone(), normal: o.normal.clone(), modelId: tag.root.dataset_id })
  1125. tag = tag_
  1126. tag.spot.addEventListener('mouseover', () => {
  1127. bus.emit('hoverState', true)
  1128. })
  1129. tag.spot.addEventListener('mouseout', () => {
  1130. bus.emit('hoverState', false)
  1131. })
  1132. }
  1133. if (!info.position) {
  1134. viewer.tagTool.startInsertion().done(tag_ => {
  1135. done()
  1136. })
  1137. } else {
  1138. info.root = MergeEditor.getAllObjects().find(e => e.dataset_id == info.modelId)
  1139. if (!info.root) {
  1140. console.error('没有找到该modelId')
  1141. }
  1142. tag = viewer.tagTool.createTagFromData(info)
  1143. done()
  1144. }
  1145. let result = {
  1146. bus,
  1147. getScreenPos() {
  1148. let pos3d = new THREE.Vector3().setFromMatrixPosition(tag.matrixWorld)
  1149. return sdk.getScreenByPosition(pos3d)
  1150. },
  1151. show() {
  1152. Potree.Utils.updateVisible(tag, 'byList', true)
  1153. },
  1154. hide() {
  1155. Potree.Utils.updateVisible(tag, 'byList', false)
  1156. },
  1157. destroy() {
  1158. if (tag) {
  1159. tag.dispose()
  1160. }
  1161. viewer.dispatchEvent({ type: 'cancel_insertions', remove: true })
  1162. },
  1163. changeTitle(title) {
  1164. tag.changeTitle(title)
  1165. }
  1166. }
  1167. return result
  1168. }, */
  1169. /* export type PathProps = {
  1170. line: {
  1171. width: number,
  1172. color: string,
  1173. altitudeAboveGround: number
  1174. position: SceneLocalPos,
  1175. modelId: string
  1176. },
  1177. points: {
  1178. position: SceneLocalPos,
  1179. modelId: string,
  1180. }[]
  1181. } */
  1182. showGrid() {
  1183. Potree.Utils.updateVisible(viewer.modules.MergeEditor.ground, 'hideGrid', true)
  1184. viewer.dispatchEvent('content_changed')
  1185. },
  1186. hideGrid() {
  1187. Potree.Utils.updateVisible(viewer.modules.MergeEditor.ground, 'hideGrid', false)
  1188. viewer.dispatchEvent('content_changed')
  1189. }
  1190. }
  1191. function spliceFromArr(model, props, loaded){
  1192. //let autoLoads.find()
  1193. props.loadFinish = true
  1194. props.loading = false
  1195. if (loaded) {
  1196. props.loaded = true
  1197. props.model = model
  1198. } else {
  1199. props.error = true
  1200. }
  1201. /* let haventLoad = autoLoads.filter(e=>!e.loading && !e.loadFinish);
  1202. if( haventLoad[0]){
  1203. startLoad(haventLoad[0])
  1204. */
  1205. if (!loadNext()) {
  1206. if (autoLoads.filter(e => !e.loadFinish).length == 0 && autoLoads.filter(e => e.loaded).length > 0 && !props.isFirstLoad) {//设置相机位置:当自动开始加载第一个模型时(其余的也跟着自动加载),等这批加载完后;
  1207. let autoLoadsDone = autoLoads.filter(e => e.loaded).map(e => e.model)
  1208. let loadTimeCost = Date.now() - loadStartTime
  1209. console.log('所有模型加载完毕, 耗时', parseInt(loadTimeCost) )
  1210. autoLoads.filter(e => e.loaded && e.show).forEach(e => e.model.visible = true)
  1211. MergeEditor.focusOn(autoLoadsDone, 1000, true, true)
  1212. autoLoads.length = 0
  1213. }
  1214. }
  1215. }
  1216. function loadNext(){
  1217. let haventLoad = autoLoads.filter(e => !e.loading && !e.loadFinish);
  1218. let loading = autoLoads.filter(e => e.loading);
  1219. let needLoad = haventLoad.slice(0, maxLoadingCount - loading.length)
  1220. needLoad.forEach(e => startLoad(e))
  1221. return haventLoad.length > 0
  1222. }
  1223. function startLoad(prop){
  1224. /* if(prop.raw.visible !== 1){//用于临时隐藏
  1225. setTimeout(()=>{
  1226. spliceFromArr(null, prop, false)
  1227. prop.bus.emit('loadError' )
  1228. },1)
  1229. return
  1230. } */
  1231. if(prop.loading || prop.loadFinish)return
  1232. Potree.Log(`--开始加载--`, { font: { color: '#f68' } });
  1233. console.log('id:', prop.id, ', title:', prop.title, ', filename:', Potree.Common.getNameFromURL(prop.url), ', type:', prop.type, prop)
  1234. prop.unlit = prop.renderType != 'normal'
  1235. prop.maximumScreenSpaceError = 70
  1236. prop.prefix = prop.raw.prefix
  1237. Potree.addModel(prop, prop.done, prop.progressFun, prop.onError)
  1238. prop.loading = true
  1239. }
  1240. function buildMap(){
  1241. if (Potree.settings.showCesium && !window.cesiumViewer) {
  1242. viewer.backgroundOpacity = 0
  1243. Cesium.Ion.defaultAccessToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJqdGkiOiI2ZGM2YzY0ZC1kNWE0LTRiYTgtYTkwNS1kYmJiODRjMWUwMmQiLCJpZCI6MjMzMTQ1LCJpYXQiOjE3MjI5OTUwNTB9.niqpkl6xOkQ2KeJjelyDDDydmSGqKXKb5cX2NyxSNAw'
  1244. window.cesiumViewer = new Cesium.Viewer('app', {
  1245. useDefaultRenderLoop: true,
  1246. requestRenderMode: true, //add 只有需要render时才会render,如tile加载完后、镜头移动后
  1247. animation: false,
  1248. baseLayerPicker: false,
  1249. fullscreenButton: false,
  1250. geocoder: false,
  1251. homeButton: false,
  1252. infoBox: false,
  1253. sceneModePicker: false,
  1254. selectionIndicator: false,
  1255. timeline: false,
  1256. navigationHelpButton: false,
  1257. //imageryProvider : Cesium.createOpenStreetMapImageryProvider({url : 'https://a.tile.openstreetmap.org/'}),
  1258. imageryProvider: Cesium.UrlTemplateImageryProvider({ //直接用84坐标,不用转高德
  1259. //"https://wprd04.is.autonavi.com/appmaptile?lang=zh_cn&style=7&yrs=m&x=${x}&y=${y}&z=${z}" //
  1260. //url : 'https://webst0{0-7}.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=7&x={x}&y={y}&z={z}&token=YOUR_API_KEY',
  1261. url: 'https://wprd04.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=7&x={x}&y={y}&z={z}&token=YOUR_API_KEY',
  1262. minimumLevel: 0,
  1263. maximumLevel: 19
  1264. }),
  1265. //高德秘钥版 imageryProvider: new Cesium.AmapImageryProvider({key, mapStyle: 'normal'})
  1266. //报错 401 (Unauthorized) 的方法 https://blog.csdn.net/LBY_XK/article/details/121992641
  1267. terrainShadows: Cesium.ShadowMode.DISABLED, //terrain地形
  1268. });
  1269. //lonlat = [113.595236803415,22.3665168584444]//[113.600356,22.364093]
  1270. Potree.setLonlat(lonlat[0], lonlat[1])
  1271. Potree.cesScreenshot = (w,h)=>{
  1272. console.log('cesScreenshot',w,h)
  1273. cesiumViewer.scene.canvas.style.width = w+'px'
  1274. cesiumViewer.scene.canvas.style.height = h+'px'
  1275. cesiumViewer.scene.canvas.style.visibility = 'hidden'
  1276. cesiumViewer.resize()
  1277. cesAspect = w/h
  1278. let deferred = $.Deferred();
  1279. updateMap(w/h)//hfov可能改变了需要update。
  1280. setTimeout(()=>{ //延迟是似乎还要做别的处理,否则立即截图的话可能得到绿色底图(俯视状态容易触发)
  1281. let oldMode = window.cesiumViewer._cesiumWidget._scene.requestRenderMode
  1282. window.cesiumViewer._cesiumWidget._scene.requestRenderMode = 0 //强制render,否则会黑屏
  1283. cesiumViewer.render();
  1284. let dataUrl = window.cesiumViewer.scene.canvas.toDataURL('image/png')
  1285. window.cesiumViewer._cesiumWidget._scene.requestRenderMode = oldMode
  1286. //Potree.Common.downloadFile(dataUrl, 'screenshot.png')
  1287. cesAspect = null
  1288. cesiumViewer.scene.canvas.style.width = ''
  1289. cesiumViewer.scene.canvas.style.height = ''
  1290. cesiumViewer.scene.canvas.style.visibility = ''
  1291. deferred.resolve(dataUrl)
  1292. },200) //时间短了容易黑屏
  1293. return deferred.promise()
  1294. }
  1295. }
  1296. updateMap()
  1297. }
  1298. function updateCamNear(type){// 有的漫游场景模型缩放的很小(0.1%),需要缩小near才能看见, 但会造成z-fighting, 离远了看大模型会闪烁
  1299. const min = 0.0001, max = 0.1
  1300. let near , bigScale = 0.2
  1301. if(Potree.settings.displayMode == 'showPanos'/* && type != 'cameraMove' */){
  1302. let currentModel = viewer.images360.currentPano.pointcloud
  1303. near = Potree.math.linearClamp(currentModel.scale.x, [0, 1], [min, max])
  1304. }else/* if(type == 'cameraMove') */{
  1305. //没有完美的解决方式,优先考虑选中的模型。否则优先考虑离得近的。 另外尽量用大的,因为缩很小的情况很少。
  1306. //虽然案例说应该看model.bound.size,但如果非场景模型,缩很小的话也不需要凑近看。
  1307. let allModels = viewer.objs.children.concat(viewer.scene.pointclouds)
  1308. if(allModels.length == 0)return
  1309. allModels.sort((a,b)=>{return a.scale.x - b.scale.x})
  1310. let minS = allModels[0].scale.x, maxS = allModels.pop().scale.x
  1311. let considerModel
  1312. if(minS>bigScale) near = max
  1313. else{
  1314. if(MergeEditor.selected){
  1315. considerModel = MergeEditor.selected
  1316. near = Potree.math.linearClamp(considerModel.scale.x, [0, bigScale], [min, max])
  1317. }else{
  1318. //allModels = allModels.filter() //写不下去了好难,就算了吧, 折中
  1319. near = Potree.math.linearClamp(minS, [0, bigScale], [max/4, max])
  1320. }
  1321. }
  1322. }
  1323. if(near != viewer.mainViewport.camera.near){
  1324. console.log('updateNear',near)
  1325. viewer.mainViewport.camera.near = near
  1326. viewer.mainViewport.camera.updateProjectionMatrix()
  1327. viewer.dispatchEvent('content_changed')
  1328. }
  1329. }
  1330. function updateMap( ){
  1331. if (Potree.settings.showCesium && Potree.settings.displayMode == 'showPointCloud') {
  1332. let camera = viewer.mainViewport.camera
  1333. let pPos = new THREE.Vector3(0, 0, 0).applyMatrix4(camera.matrixWorld);
  1334. let pRight = new THREE.Vector3(600, 0, 0).applyMatrix4(camera.matrixWorld);
  1335. let pUp = new THREE.Vector3(0, 600, 0).applyMatrix4(camera.matrixWorld);
  1336. let pTarget = viewer.scene.view.getPivot();
  1337. let toCes = (pos) => {
  1338. let xy = [pos.x, pos.y];
  1339. let height = pos.z;
  1340. let deg = viewer.transform.lonlatToLocal.inverse(xy) // toMap.forward(xy);
  1341. let cPos = Cesium.Cartesian3.fromDegrees(...deg, height);
  1342. return cPos;
  1343. };
  1344. let cPos = toCes(pPos);
  1345. let cUpTarget = toCes(pUp);
  1346. let cTarget = toCes(pTarget);
  1347. let cDir = Cesium.Cartesian3.subtract(cTarget, cPos, new Cesium.Cartesian3());
  1348. let cUp = Cesium.Cartesian3.subtract(cUpTarget, cPos, new Cesium.Cartesian3());
  1349. cDir = Cesium.Cartesian3.normalize(cDir, new Cesium.Cartesian3());
  1350. cUp = Cesium.Cartesian3.normalize(cUp, new Cesium.Cartesian3());
  1351. cesiumViewer.camera.setView({
  1352. destination: cPos,
  1353. orientation: {
  1354. direction: cDir,
  1355. up: cUp
  1356. }
  1357. });
  1358. let aspect = cesAspect || viewer.scene.getActiveCamera().aspect;
  1359. //console.log('updateMap', aspect)
  1360. if (aspect < 1) {
  1361. let fovy = Math.PI * (viewer.scene.getActiveCamera().fov / 180);
  1362. cesiumViewer.camera.frustum.fov = fovy;
  1363. } else {
  1364. let fovy = Math.PI * (viewer.scene.getActiveCamera().fov / 180);
  1365. let fovx = Math.atan(Math.tan(0.5 * fovy) * aspect) * 2
  1366. cesiumViewer.camera.frustum.fov = fovx;
  1367. }
  1368. cesiumViewer.render(); //立即render,否则会和点云render不同步而错位
  1369. }
  1370. }
  1371. return sdk
  1372. }
  1373. /* class path = */
  1374. /*
  1375. 暂定不同场景间的漫游点不能互通。虽然它们可能是摆放正确的,如果是组成一整个场景的话还是要打通……
  1376. 不互通的方法是设置pano.enable
  1377. 现在需要互通了。但是还需要设置neibgbours, 有点麻烦,暂时没写。
  1378. */
  1379. export default enter