index.js 75 KB

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