index.js 116 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729
  1. import mitt from 'mitt'
  2. import libTransform from 'coordtransform';
  3. import axios from 'axios' //{ axios } from '@/api'
  4. let requestLoadCount = 0
  5. let maxLoadingCount = 2; //正在加载模型的最大数目
  6. //0看看,1看见,2深时,3用户上传三维模型,4深时mesh,5深光点云,6深光mesh
  7. const ModelTypes = {
  8. 0 : {name:'看看(八目)', panos4dkk:true},
  9. 1 : {name:'看见(双目转台)', panos4dkk:true, rot90:true},
  10. 2 : {name:'深时', },
  11. 3 : {name:'用户上传三维模型'},
  12. 4 : {name:'深时mesh(激光转台)',panos4dkk:true, rot90:true},//3dtiles or obj
  13. 5 : {name:'深光点云' },
  14. 6 : {name:'深光mesh',panos4dkk:true, rot90:true},//3dtiles
  15. 7 : {name:'圆周率相机' },//圆周率相机场景
  16. 8 : {name:'动画模型'}
  17. }
  18. let satellite = true
  19. let defaultMapProps = satellite ? [
  20. {url: `//wprd04.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=6&x={x}&y={y}&z={z}&layer=6&token=YOUR_API_KEY`, //style=6是卫星,7是标准
  21. maximumLevel: 18 ,
  22. name:'高德baseLayer'
  23. },{
  24. url: `//wprd04.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=8&x={x}&y={y}&z={z}&layer=6&token=YOUR_API_KEY`, //style=6是卫星,7是标准
  25. maximumLevel: 18,
  26. name:'高德textLayer'
  27. }] : [
  28. {url: `//wprd04.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=7&x={x}&y={y}&z={z}&layer=6&token=YOUR_API_KEY`, //style=6是卫星,7是标准
  29. maximumLevel: 19 ,
  30. name:'高德baseLayer'
  31. },
  32. ]
  33. let cesAspect , cesImageryProvider, mapProps = defaultMapProps
  34. /*
  35. let Potree.math = {
  36. fromCes(cartesian, getLonlat){
  37. var cartographic = Cesium.Cartographic.fromCartesian(cartesian);
  38. var lng = Cesium.Math.toDegrees(cartographic.longitude);
  39. var lat = Cesium.Math.toDegrees(cartographic.latitude);
  40. var height = cartographic.height;
  41. let loc = new THREE.Vector3(lng, lat, height)
  42. if(getLonlat)return loc
  43. return viewer.transform.lonlatToLocal.forward(loc)
  44. },
  45. toCes(pos){
  46. let xy = [pos.x, pos.y];
  47. let height = pos.z;
  48. let deg = viewer.transform.lonlatToLocal.inverse(xy) // toMap.forward(xy);
  49. let cPos = Cesium.Cartesian3.fromDegrees(...deg, height);//console.log('toCes',cPos,height) //数字过大如e35会崩溃
  50. return cPos;
  51. }
  52. }
  53. */
  54. const Id_noIntersect = -100 //path绘制在地图上的点,modelId传这个值,勿更改
  55. let isValidPoint = (modelId)=>{//所存的modelId没被删或者它本身不在模型上
  56. return modelId == Id_noIntersect || viewer.objs.children.concat(viewer.scene.pointclouds).some(e=>e.dataset_id == modelId )
  57. }
  58. let curSelectPath
  59. {
  60. // 84坐标转高德 (国外地区用84,所以地理注册时填的是84,我这需要转成高德)
  61. const wgs84ToAMap = (pos ) => {
  62. const latlng = libTransform.wgs84togcj02(pos.x, pos.y)
  63. return {
  64. x: latlng[0],
  65. y: latlng[1]
  66. }
  67. }
  68. // 高德坐标转84
  69. const aMapToWgs84 = (pos ) => {
  70. const latlng = libTransform.gcj02towgs84(pos.x, pos.y)
  71. return {
  72. x: latlng[0],
  73. y: latlng[1]
  74. }
  75. }
  76. window.AMapWith84__ = { //在Potree里setLonlat时不管转不转效果都一样 很奇怪,所以这里改名。只在ces
  77. aMapToWgs84, wgs84ToAMap
  78. }
  79. }
  80. //江门本地版本
  81. export const enter = ({ dom, mapDom, isLocal, lonlat, scenes, laserRoot, laserOSSRoot, panoOSSRoot,ossRoot }) => {
  82. console.warn('新的页面')
  83. Potree.settings.isOfficial = true //标记为正式、非测试版本
  84. //Potree.fileServer = axios
  85. Potree.settings.libsUrl = './lib/'
  86. let loadStartTime = Date.now()
  87. //正式环境(本地调试会打不开)
  88. if (location.host === 'mix3d.4dkankan.com') {
  89. Potree.settings.urls.prefix = Potree.settings.urls.prefix6
  90. Potree.settings.webSite = 'datav1'
  91. } else if (location.host === 'xfhd.4dkankan.com') {
  92. Potree.settings.urls.prefix = Potree.settings.urls.prefix7
  93. Potree.settings.webSite = 'datav1'
  94. }
  95. if(window.offline){//离线版
  96. Potree.settings.urls.templates = {
  97. depthTex : 'swss/{sceneCode}/env/www/{sceneCode}/wwwroot/{sceneCode}/data/{sceneCode}/depthmap',
  98. vision : '/swkk/{sceneCode}/env/wwwroot/scene_view_data/{sceneCode}/images/vision.txt'
  99. }
  100. Potree.fileStorage = {
  101. get(url){
  102. return new Promise(async function(resolve,reject){
  103. try {
  104. let data = await window.offlineData[url]
  105. if(data){
  106. resolve(data)
  107. }
  108. }catch(e){
  109. console.error('没找到', url, e)
  110. reject()
  111. }
  112. })
  113. }
  114. }
  115. }
  116. if(laserRoot != void 0){
  117. Potree.settings.urls.prefix = Potree.settings.urls.handlePrefix(laserRoot)
  118. }
  119. if(laserOSSRoot != void 0){
  120. Potree.settings.urls.prefix1 = laserOSSRoot
  121. }
  122. if(panoOSSRoot != void 0){
  123. Potree.settings.urls.prefix3 = panoOSSRoot //tile
  124. }
  125. if(ossRoot){
  126. //Potree.settings.urls.panoPrefix = ossRoot //vision.txt
  127. Potree.settings.urls.setPrefix(8, ossRoot)//vision.txt
  128. }
  129. const mapBus = mitt(), sceneBus = mitt()
  130. let isLocal2 = true
  131. if(isLocal2){//本地配置
  132. Potree.settings.isLocal = Potree.settings.tileOriginUrl = isLocal2
  133. }
  134. Potree.settings.showCompass = true
  135. Potree.settings.compassDom = dom.querySelector('#direction')
  136. Potree.settings.mergeType2 = true //标识新版
  137. Potree.settings.modelSkybox = true //是否将全景图贴在模型上(会导致卡顿)。若不显示模型将不显示Reticule
  138. Potree.settings.tiles3DMaxMemory = 300 //稍微增加点
  139. Potree.settings.mergeTransCtlOnClick = true
  140. Potree.settings.canWalkThroughModel = true
  141. window.cesErrorWords = '内存占用过高,建议关闭部分场景或升级显卡。'
  142. window.cesErrorCallback = ()=>{
  143. //sdk.setBackdrop('none')
  144. }
  145. let { THREE } = Potree.mergeEditStart(dom, mapDom, {
  146. queryCloudLonLatUrl:Potree.settings.urls.prefix+"/laser/4dage/{sceneCode}/getDataSetAndControlPoint"
  147. })
  148. let {MergeEditor, AnimationEditor} = viewer.modules
  149. Potree.settings.unableNavigate = true
  150. Potree.setLonlat(lonlat[0], lonlat[1])
  151. if(window.offline){//离线版 改目录
  152. viewer.images360.tileDownloader.getTiles = function(d, sceneNum, useV4url, model){
  153. let kankan = !model.isPointcloud //ModelTypes[model.props.fromType].panos4dkk
  154. if(kankan){
  155. return `/swkk/${sceneNum}/wwwroot/scene_view_data/${sceneNum}/images/${d}`
  156. }else{
  157. return `/swss/${sceneNum}/www/${sceneNum}/scene_view_data/${sceneNum}/images/${d}`
  158. }
  159. }
  160. }
  161. /* Potree.loadControlPoint = async function(callback,sceneCode,onError,prefix){//点云绑定地图
  162. let path = `/laser/jm/${sceneCode}/getDataSetAndControlPoint`
  163. return Potree.loadFile(path, {
  164. fetchMethod: 'post'
  165. }, callback,onError)
  166. } */
  167. //因为getPose里用的是target,俯视的yaw不准,所以限制一下不要完全俯视
  168. viewer.mainViewport.view.maxPitch-=0.001
  169. viewer.mainViewport.view.minPitch+=0.001
  170. viewer.addEventListener('camera_changed', e => {
  171. var camera = e.viewport.camera
  172. var pos = camera.position
  173. if (e.viewport.name == 'MainView' ) {
  174. sceneBus.emit('cameraChange', { x: pos.x, y: pos.y, z: pos.z, rotate: camera.rotation })
  175. Potree.Common.intervalTool.isWaiting('updateCamNear', ()=>{
  176. updateCamNear()
  177. }, 1000)
  178. updateCamFar()
  179. if(e.changeInfo.positionChanged){
  180. viewer.objs.children.forEach(model=>{
  181. model.result_.updateVisiByRange && model.result_.updateVisiByRange()
  182. })
  183. }
  184. }
  185. if (e.viewport.name == 'MainView' || e.viewport.name == 'top' ) {
  186. updateMap()
  187. }
  188. })
  189. viewer.addEventListener('shelterComputed', (e)=>{
  190. //console.log('shelterComputed')
  191. var camera = viewer.mainViewport.camera
  192. var pos = camera.position
  193. sceneBus.emit('cameraChange', { x: pos.x, y: pos.y, z: pos.z, rotate: camera.rotation })
  194. })
  195. //-------------------------------------
  196. let modelAinB = (A,B)=>{ //B的expand(5m) bound完全包含A
  197. let boundB = B.boundingBox.clone().expandByVector(new THREE.Vector3(5,5,5)).applyMatrix4(B.matrixWorld)
  198. let boundA = A.boundingBox.clone().applyMatrix4(A.matrixWorld)
  199. return boundB.containsBox(boundA)
  200. }
  201. let changeMeshVisi = (object, show) => {
  202. if(show == void 0) show = Potree.settings.displayMode == 'showPointCloud' || object == viewer.images360.currentPano.pointcloud && Potree.settings.modelSkybox || object.showInPano //showInPano: 装饰物,一直显示
  203. || !object.panos && modelAinB(object, viewer.images360.currentPano.pointcloud) //装饰物
  204. Potree.Utils.updateVisible(object, 'showPanos', show)
  205. }
  206. if(Potree.settings.canWalkThroughModel){
  207. let lastModel
  208. viewer.images360.addEventListener('flyToPano',(e)=>{//开始漫游 漫游到另一个模型就要选中这个模型?
  209. let model = e.toPano.pano.pointcloud
  210. if(lastModel != model){
  211. changeMeshVisi(model, true)
  212. //MergeEditor.selectModel(model)
  213. //model.result_.flyInPano(e.toPano.pano, {dontFly:true}) //切换模型显示,因为flyInPano有事件怕乱所以统一用这个函数
  214. updateCamNear()
  215. }
  216. })
  217. viewer.images360.addEventListener('flyToPanoDone',(e)=>{
  218. if(!e.makeIt)return
  219. let model = viewer.images360.currentPano.pointcloud
  220. if(lastModel != model){
  221. lastModel?.isModel && changeMeshVisi(lastModel, false)
  222. sceneBus.emit('panoModelChange', model.result_ )
  223. }
  224. lastModel = model
  225. })
  226. }
  227. viewer.images360.addEventListener('endChangeMode',(e)=>{
  228. sceneBus.emit('modeChange', {mode: e.mode == 'showPanos' ? 'pano' : 'fuse', model : e.mode == 'showPanos' && viewer.images360.currentPano.pointcloud.result_} )
  229. console.log('emit Changemode', e.mode )
  230. //Potree.Utils.updateVisible(MergeEditor.transformControls, 'showPanos', e.mode == 'showPointCloud')
  231. //Potree.Utils.updateVisible(MergeEditor.boxHelper, 'showPanos', e.mode == 'showPointCloud')
  232. if(e.mode == 'showPanos'){
  233. viewer.setControls( viewer.fpControls )
  234. viewer.removeEventListener('camera_changed', camera_changed)
  235. }else{
  236. viewer.addEventListener('camera_changed', camera_changed)
  237. }
  238. viewer.objs.children.forEach((e)=>{changeMeshVisi(e)})
  239. Potree.settings.canWalkThroughModel || viewer.images360.panos.forEach(pano => {
  240. pano.setEnable(e.mode == 'showPanos' ? pano.pointcloud == viewer.images360.currentPano.model : true)
  241. })
  242. Potree.settings.unableNavigate = e.mode == 'showPointCloud'
  243. updateCamNear()
  244. })
  245. let camera_changed = (e) => {
  246. if (e.viewport.name == 'MainView' && e.changeInfo.positionChanged) {
  247. //viewer.mainViewport.camera.position
  248. viewer.mainViewport.view.radius = 0.1 //使pivot在面前一丢丢距离
  249. viewer.setControls(viewer.orbitControls)
  250. viewer.removeEventListener('camera_changed', camera_changed)
  251. }
  252. }
  253. let requestInPano = false
  254. //-------------------------------------
  255. /* viewer.inputHandler.addEventListener('keydown', (e)=>{
  256. if(e.event.key == "e" ){
  257. MergeEditor.transformControls.mode = 'rotate'
  258. }else if(e.event.key == "w"){
  259. MergeEditor.transformControls.mode = 'translate'
  260. }else if(e.event.key == "s"){
  261. MergeEditor.transformControls.mode = 'scale'
  262. }
  263. }) */
  264. viewer.addEventListener('webglError', e => {
  265. console.error('viewer webglError: ' + e)
  266. let memory = '. \n jsHeapSizeLimit:'+ performance.memory.jsHeapSizeLimit/ 1e6 + ', usedJSHeapSize: '+performance.memory.usedJSHeapSize/ 1e6 + '(M)'
  267. sceneBus.emit('webglError', { msg: e.msg + memory })
  268. })
  269. viewer.compass.setAutoDisplay(true)
  270. viewer.addEventListener('watchMonitor',()=>{
  271. sceneBus.emit('watchMonitor')
  272. })
  273. viewer.addEventListener('monitorError',()=>{
  274. sceneBus.emit('monitorError')
  275. })
  276. /* mapBus.on('visible', v => {
  277. //console.log('mapBus visible', v)
  278. viewer.mapViewer.visible = v
  279. if (v) {
  280. viewer.mapViewer.mapLayer.needUpdate = true
  281. }
  282. viewer.mapViewer.dispatchEvent({type:'forceVisible',visible:v})
  283. }) */
  284. {
  285. let index = 1;
  286. //let setDisplay()
  287. if (!Potree.isIframeChild) {
  288. /* viewer.addEventListener('createIframe',(e)=>{//创建了子页面
  289. }) */
  290. window.winIndex = 0;
  291. window.iframeCreated = function (iframe) {
  292. let child = iframe.contentWindow
  293. child.winIndex = index++
  294. //案件里视图提取页面子页面覆盖了父级页面,父级的模型可以隐藏以释放内存
  295. console.error('createdIframe', child.winIndex, child.location.href)
  296. viewer.setDisplay(false)
  297. child.beforeDestroy = function () { //注:在前端仍会找不到beforeDestroy,可能contentWindow变更??所以手动调用setDisplay
  298. console.warn('beforeDestroy', child.winIndex)
  299. child.viewer && child.viewer.setDisplay(false)
  300. //如果是四维看看的场景,先不管了,页面被销毁应该就没了吧
  301. viewer.setDisplay(true)//恢复主页的模型显示
  302. if (!child.viewer) {
  303. try {
  304. let player = child.__sdk.core.get('Player')
  305. /* let runtime = player.model._3dTilesRuntime
  306. let tileset = runtime.getTileset()
  307. tileset._cache.trim(); //使下一次update时dispose所有不可见的tiles
  308. let sceneRenderer = child.__sdk.core.get('SceneRenderer')
  309. player.model.visible = false
  310. runtime.update(16, sceneRenderer.renderer, sceneRenderer.camera, true) //没用,为何_trimTiles的while无法进入
  311. */
  312. player.model.traverse(e => {
  313. e.geometry && e.geometry.dispose()
  314. if (e.material) {
  315. e.material.map && e.material.map.dispose()
  316. if (e.material.uniforms && e.material.uniforms.map && e.material.uniforms.map.value) {
  317. e.material.uniforms.map.value.dispose()
  318. }
  319. }
  320. }) //效果甚微
  321. /* let sceneRenderer = child.__sdk.core.get('SceneRenderer')
  322. sceneRenderer.renderer.render(sceneRenderer.scene, sceneRenderer.camera)
  323. */
  324. } catch (e) {
  325. console.log(e)
  326. }
  327. }
  328. }
  329. }
  330. //不知道删除iframe时是否那些模型还在内存里,需要释放吗? 如果要需要加一个事件
  331. } else {
  332. }
  333. }
  334. window.THREE = THREE
  335. //isLocal = false
  336. let autoLoads = /* window.autoLoads = */ []
  337. let readyToAddModel
  338. let mainBackground = viewer.background
  339. const units = { 1: 'metric', 2: 'imperial' }
  340. let getMeasureType = function (type, unit = 1) {
  341. let info
  342. switch (type) {
  343. case 'free':
  344. info = { measureType: 'Distance' }
  345. break
  346. case 'area':
  347. info = { measureType: 'Area' }
  348. break
  349. case 'vertical':
  350. info = { measureType: 'Ver Distance' }
  351. break
  352. default:
  353. console.error('无此 measure type')
  354. }
  355. info.unit = units[unit]
  356. return info
  357. }
  358. let getMeasureFunction = function (measure, bus) {
  359. measure.addEventListener('highlight', (e) => {
  360. if(measure.type == 'Path'){
  361. bus.emit(e.state ? 'enter' : 'leave')
  362. }else{
  363. bus.emit('highlight', e.state)
  364. }
  365. })
  366. let update = (e)=>{ //拖拽结束后发送changeCallBack
  367. if (measure.parent) {
  368. //未被删除
  369. console.warn('changePoints', measure.dataset_points.length )
  370. if(measure.type == 'Path'){
  371. bus.emit('changePoints', measure.dataset_points.map((p,i)=>{return {
  372. position: (p || measure.points[i]).clone(),
  373. modelId: measure.points_datasets[i] == void 0 ? Id_noIntersect : measure.points_datasets[i],
  374. name: measure.markerLabels[i].originText
  375. }}))
  376. }else{
  377. bus.emit('update', [
  378. measure.dataset_points.map(p => p.clone()),
  379. measure.points_datasets
  380. ])
  381. }
  382. }
  383. }
  384. measure.addEventListener('marker_dropped', update)
  385. measure.addEventListener('changed', update)
  386. measure.addEventListener('createDone', update)
  387. measure.addEventListener('changeByHistory', update);
  388. return {
  389. /* quit: () => {
  390. Potree.Log('quit结束且删除: ' + measure.id, '#00c7b2')
  391. viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure })
  392. }, //触发结束。退出测量模式,清除之前操作 */
  393. destroy: () => {
  394. viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure })
  395. viewer.scene.removeMeasurement(measure)
  396. },
  397. /* getPoints: () => {
  398. return measure.points
  399. },
  400. getDatasetLocations: () => {
  401. return measure.dataset_points
  402. },
  403. getDatasets: () => {
  404. return measure.points_datasets
  405. },
  406. getDatasetId: () => {
  407. return measure.datasetId
  408. }, */
  409. getArea: () => {
  410. return measure.area //{value:area, string:..}
  411. },
  412. getDistance: () => {
  413. if (measure.points.length < 2) return 0
  414. var value = measure.points[0].distanceTo(measure.points[1])
  415. return {
  416. value, //米
  417. string: measure.getConvertString(value, 'distance')
  418. }
  419. },
  420. //手动开启或关闭:
  421. show: () => {
  422. Potree.Utils.updateVisible(measure, 'inListByUser', true)
  423. },
  424. hide: () => {
  425. Potree.Utils.updateVisible(measure, 'inListByUser', false)
  426. },
  427. fly() {
  428. if(measure.type == 'Path') Potree.settings.displayMode = 'showPointCloud'
  429. let result = viewer.focusOnObject(measure, 'measure', 1200, {dontLookUp:measure.type == 'Path', maxDis: measure.fadeFar && measure.fadeFar*1.5})
  430. return result.msg ? result.msg : result.promise
  431. //返回值 1 deferred 表示即将位移 2 'posNoChange' 表示已在最佳位置 3 'tooFar' 表示距离最佳位置太远
  432. },
  433. changeSelect(isHight) {
  434. //console.log('2d->3d isHight ', isHight)
  435. measure.setSelected(isHight, 'byList')
  436. },
  437. }
  438. }
  439. let sdk = Potree.sdk = {
  440. sceneBus, mapBus,
  441. canTurnToPanoMode(pos, far=Potree.config.panoFieldRadius) {
  442. pos = pos ? new THREE.Vector3().copy(pos) : viewer.images360.position
  443. let pano = viewer.images360.findNearestPano(pos)
  444. if (pano && pano.position.distanceTo(pos) < far * pano.pointcloud.scale.x) {
  445. return {model:pano.pointcloud.result_}
  446. }
  447. //poschange后会调用这个,如果返回false会变为点云模式,且不会自动变回原先的模式
  448. },
  449. getPositionByScreen(pos2d, hopeModelId) {//通过屏幕坐标获取真实坐标 . hopeModelId: 如果指定了模型,优先返回hopeModelId上的intersect
  450. //console.log('getPositionByScreen',hopeModelId)
  451. hopeModelId = null
  452. let worldPos, localPos, modelId, intersect, normal, localNormal
  453. let Handler = viewer.inputHandler
  454. let reGet = () => {//不使用当前鼠标所在位置的intersect,单独算
  455. pos2d.clientX = pos2d.x
  456. pos2d.clientY = pos2d.y
  457. pos2d.onlyGetIntersect = true
  458. pos2d.whichPointcloud = true
  459. if (hopeModelId != void 0) {//隐藏其他的模型
  460. let models = MergeEditor.getAllObjects()
  461. models.forEach(model => {
  462. Potree.Utils.updateVisible(model, 'forPick', model.dataset_id == hopeModelId)
  463. })
  464. }
  465. let intersect2 = Handler.onMouseMove(pos2d)
  466. if (hopeModelId != void 0) {//恢复
  467. let models = MergeEditor.getAllObjects()
  468. models.forEach(model => {
  469. Potree.Utils.updateVisible(model, 'forPick', true)
  470. })
  471. }
  472. if (intersect2 && intersect2.location) {
  473. intersect = intersect2
  474. }
  475. }
  476. if (pos2d && pos2d.inDrag) {
  477. reGet()
  478. } else {
  479. intersect = Handler.intersect
  480. if (intersect) {
  481. modelId = intersect.pointcloud ? intersect.pointcloud.dataset_id : intersect.object.dataset_id
  482. if (hopeModelId != void 0 && modelId != hopeModelId) {
  483. reGet()
  484. }
  485. }
  486. }
  487. if (intersect && intersect.location) {
  488. modelId = intersect.pointcloud ? intersect.pointcloud.dataset_id : intersect.object.dataset_id
  489. /* if(hopeModelId != void 0 && modelId != hopeModelId){
  490. return null
  491. } */
  492. worldPos = intersect.location.clone()
  493. localPos = Potree.Utils.datasetPosTransform({ toDataset: true, datasetId: modelId, position: worldPos })
  494. normal = intersect.normal
  495. localNormal = intersect.localNormal
  496. } else return null
  497. return { worldPos, modelId, normal, localPos, localNormal }
  498. },
  499. getScreenByPosition(pos3d, modelId, canShelter/* , disToCameraLimit */) {//通过模型局部坐标获取屏幕坐标
  500. //console.log('getScreenByPoint ', pos3d.toArray())
  501. let isLocal = modelId != void 0
  502. pos3d = new THREE.Vector3().copy(pos3d)
  503. let worldPos = isLocal ? Potree.Utils.datasetPosTransform({ fromDataset: true, datasetId: modelId, position: pos3d }) : pos3d
  504. if (!worldPos) return
  505. if (canShelter) {
  506. if (viewer.inputHandler.ifBlockedByIntersect(worldPos, 0.1, true)) return { trueSide: false };
  507. }
  508. var viewport = viewer.mainViewport
  509. var camera = viewport.camera
  510. var dom = viewer.renderArea
  511. /* if (tagLimitDis != void 0) {
  512. if (camera.position.distanceToSquared(worldPos) > Math.pow(tagLimitDis, 2)) return false
  513. } */
  514. let result = Potree.Utils.getPos2d(worldPos, viewport, dom)
  515. if(!result.trueSide)return null
  516. return result
  517. },
  518. setCameraFov(fov) {
  519. viewer.setFOV(fov)
  520. },
  521. screenshot: (width, height/* , bgOpacity=0 */ ) => {//
  522. //截图
  523. let bgOpacity = Potree.settings.showCesium ? 0 : 1 /* viewer.background == 'skybox' */ //因为要画map底图所以上层只能透明。之后需要的话再改
  524. console.log('bgOpacity', bgOpacity)
  525. Potree.Utils.updateVisible(MergeEditor.boxHelper, 'screenshot', false)
  526. Potree.Utils.updateVisible(viewer.scene.overlayScene, 'screenshot', false) //hide all
  527. var { getImagePromise, finishPromise } = viewer.startScreenshot({ type: 'default', /* useRenderTarget:true, */bgOpacity }, width, height)
  528. var deferred = $.Deferred();
  529. finishPromise.done(({ dataUrl }) => {
  530. if(Potree.settings.displayMode != 'showPanos' && Potree.settings.showCesium){//need map background
  531. Potree.cesScreenshot(width, height).done((mapBGurl)=>{
  532. let img = new Image(); img.src = dataUrl
  533. let imgBG = new Image(); imgBG.src = mapBGurl
  534. let loadCount = 0
  535. img.onload = imgBG.onload = ()=>{
  536. loadCount++;
  537. if(loadCount == 2){
  538. let url = Potree.Common.imgAddLabel(imgBG,img,{leftRatioToImg:0,topRatioToImg:0})
  539. deferred.resolve(url)
  540. }
  541. }
  542. })
  543. }else{
  544. deferred.resolve(dataUrl)
  545. }
  546. Potree.Utils.updateVisible(MergeEditor.boxHelper, 'screenshot', true)
  547. Potree.Utils.updateVisible(viewer.scene.overlayScene, 'screenshot', true)
  548. })
  549. return deferred.promise()
  550. },
  551. getPose({modelId,isFlyToTag}={}) {//获取当前点位和朝向
  552. const camera = viewer.scene.getActiveCamera()
  553. const target = viewer.scene.view.getPivot().clone()
  554. const position = viewer.scene.view.position.clone()
  555. const pose = { position, target, displayMode:Potree.settings.displayMode }
  556. if(Potree.settings.displayMode == 'showPanos'){
  557. let model = viewer.images360.currentPano.pointcloud
  558. if(isFlyToTag && model.dataset_id != modelId){ //保存编辑热点时的视角,如果在全景模式,只有点位在所加模型上时才保存
  559. pose.noViewSaved = true //不保存视角
  560. }else{
  561. pose.modelId = model.dataset_id //pose.model = model.result_ //无法json化
  562. pose.panoId = viewer.images360.currentPano.originID
  563. pose.posInModel = Potree.Utils.datasetPosTransform({ toDataset: true, position: camera.position.clone(), object:model })
  564. pose.rotInModel = Potree.Utils.datasetRotTransform({ toDataset: true, quaternion: camera.quaternion.clone(), getQuaternion: true, pointcloud:model }).toArray() //拿第一个数据集
  565. }
  566. }else if(isFlyToTag){
  567. position.copy(Potree.Utils.datasetPosTransform({ toDataset: true, position: camera.position.clone(), datasetId: modelId, }))
  568. target.copy(Potree.Utils.datasetPosTransform({ toDataset: true, position: target.clone(), datasetId: modelId, }))
  569. }
  570. if(isFlyToTag){
  571. pose.maxDis = 15, pose.isFlyToTag = true, pose.modelId = modelId //补一下
  572. }
  573. return pose
  574. },
  575. comeTo(o = {}) {
  576. console.log('comeTo',o )
  577. //飞到某个点
  578. let deferred = $.Deferred()
  579. let fly = ()=>{
  580. if(o.panoId != void 0){
  581. let model = viewer.scene.pointclouds.concat(viewer.objs.children).find(e=>e.dataset_id == o.modelId)
  582. let pano = model.panos.find(a=>a.originID == o.panoId)
  583. if(pano){
  584. o.rotInModel = new THREE.Quaternion().fromArray(o.rotInModel)
  585. let quaternion = Potree.Utils.datasetRotTransform({ fromDataset: true, quaternion: o.rotInModel, getQuaternion: true, object:model})
  586. model.result_.flyInPano(pano, {quaternion, duration: o.dur||0, callback(){
  587. o.callback && o.callback()
  588. deferred.resolve(true)
  589. }})
  590. return deferred.promise()
  591. }else{
  592. console.warn('没有找到漫游点',o)
  593. }
  594. }/* else if(requestInPano){
  595. requestInPano.result_.flyOutPano()
  596. } */else{
  597. Potree.settings.displayMode = 'showPointCloud'
  598. }
  599. viewer.scene.view.setView($.extend({}, o, {
  600. duration: o.dur,
  601. callback: () => {
  602. o.callback && o.callback()
  603. deferred.resolve(true)
  604. }
  605. }))
  606. return deferred.promise()
  607. }
  608. if (o.modelId != void 0) {
  609. ['position', 'target', 'focusPos'].forEach(e => { //热点
  610. if (o[e]) {
  611. o[e] = Potree.Utils.datasetPosTransform({ fromDataset: true, datasetId: o.modelId, position: o[e] })
  612. }
  613. })
  614. }
  615. if (o.distance || o.maxDis) {
  616. //o.isFlyToTag = true
  617. let requestShowPano
  618. let position = o.focusPos || o.target || o.position
  619. if(o.isFlyToTag){
  620. let r = sdk.canTurnToPanoMode(position, 5)//热点新需求:如果附近有漫游点就飞到全景模式
  621. if(r){
  622. requestShowPano = true
  623. }else{
  624. if(!o.noViewSaved && (o.position || o.panoId != void 0)){
  625. return fly() //使用保存的视角
  626. }
  627. Potree.settings.displayMode = 'showPointCloud'
  628. }
  629. }
  630. let rusult = viewer.focusOnObject({ position }, 'tag', null, { distance: o.distance || 1, maxDis : o.maxDis, requestShowPano /* , checkIntersect:true */ })
  631. rusult.promise.then(()=>{
  632. if(o.isFlyToTag){
  633. Potree.settings.displayMode = requestShowPano ? 'showPanos' : 'showPointCloud'
  634. }
  635. })
  636. return rusult.promise
  637. }else{
  638. return fly()
  639. }
  640. },
  641. comeToByLatLng(lonlat){//飞到指定经纬度
  642. let pos = viewer.transform.lonlatToLocal.forward(lonlat)
  643. let location = viewer.mainViewport.view.position.clone().setX(pos[0]).setY(pos[1])
  644. viewer.scene.view.setView({position:location, duration:500});
  645. },
  646. setBackdrop(sky, type, { scale, rotate }={}) {//天空盒背景
  647. //console.log('天空盒背景', sky,type)
  648. let setGroundAndText = (color) => {
  649. MergeEditor.secondCompass.dom.find(".dirText").css({ 'color': color })
  650. viewer.compass.dom.find(".dirText").css({ 'color': color })
  651. MergeEditor.ground.material.uniforms.uColor.value.set(color)
  652. //MergeEditor.ground.children[0].material.color.set(color)
  653. }
  654. viewer.dispatchEvent('content_changed')
  655. if(type == 'map'){
  656. MergeEditor.setGroundPlaneImg(null)
  657. viewer.setBackground(mainBackground)
  658. Potree.settings.showCesium = true
  659. buildMap()
  660. viewer.backgroundOpacity = 0
  661. return
  662. }else{
  663. Potree.settings.showCesium = false
  664. }
  665. if (type == 'bimg') {//地面图
  666. MergeEditor.setGroundPlaneImg(sky, scale, rotate)
  667. setGroundAndText('#e0e0e0')
  668. viewer.setBackground(mainBackground)
  669. } else {
  670. MergeEditor.setGroundPlaneImg(null)
  671. if (sky == 'none') {
  672. viewer.setBackground(mainBackground)
  673. setGroundAndText('#eee')
  674. } else if (sky[0] == '#') {
  675. viewer.setBackground(new THREE.Color(sky))
  676. let color = sky == '#fff' ? '#666' : sky == '#333' ? '#eee' : '#bbb' //反相
  677. setGroundAndText(color)
  678. } else if (type == 'image-map' || type == 'vector-map') {//影像|矢量 地图
  679. } else {//环境
  680. viewer.setBackground('skybox', sky)
  681. setGroundAndText('#e0e0e0')
  682. }
  683. }
  684. },
  685. /* switchMapType(type) {
  686. let map = viewer.mapViewer.mapLayer.maps.find(e => e.name == 'map')
  687. map.switchStyle(type )
  688. }, */
  689. switchMapType(type){//切换成江门的卫星或标准
  690. if(window.location.href.includes('localhost:7173')/* || Potree.browser.urlHasValue('testMap') */ ) return
  691. console.log('switchMapType',type)
  692. mapProps = [{
  693. name:'江门',
  694. maximumLevel : type == 'satellite' ? 18 : 19,
  695. url: type == 'satellite' ? "//a.map.jms.gd/tile/weixing/{z}/{x}/{y}.png" : "//a.map.jms.gd/tile/gd_xiangtu/{z}/{x}/{y}.png"
  696. }]
  697. if(Potree.settings.showCesium){
  698. buildMapFromProp()
  699. }
  700. },
  701. changeMapTile(urls){
  702. //'http:/a.xxx.com/{x}/{y}/{z}.png'
  703. console.log('changeMapTile', urls )
  704. if(window.location.href.includes('localhost:7173') || window.location.href.includes('192.168.0.59') ) return
  705. mapProps = urls.map(e=>{
  706. return {
  707. maximumLevel : e.maximumLevel,
  708. url: e.tempUrl,
  709. name: e.name || ''
  710. }
  711. })
  712. if(Potree.settings.showCesium){
  713. buildMapFromProp()
  714. }
  715. },
  716. enableMap(mapArea, latlng) {
  717. if (!viewer.mapViewer) {
  718. //--------------------------------
  719. viewer.mapViewer = new Potree.MapViewer(mapArea)
  720. viewer.mapViewer.initProjection()
  721. //focus
  722. let boundSize = new THREE.Vector3(200, 150, 1).max(viewer.bound.boundSize)
  723. viewer.mapViewer.addEventListener('viewerResize', () => {
  724. viewer.mapViewer.moveTo(viewer.bound.center, boundSize, 0)
  725. }, { once: true })
  726. }
  727. },
  728. enterSceneGuide(pathArr) {//导览 (不需要修改参数)
  729. let editor = viewer.modules.CamAniEditor
  730. //console.log('pathArr', pathArr)
  731. //console.log('enterSceneGuide',pathArr)
  732. pathArr.forEach(p=>{
  733. if(p.panoId != void 0){
  734. p.model = viewer.scene.pointclouds.concat(viewer.objs.children).find(e=>e.dataset_id == p.modelId)//e.model.model
  735. }
  736. })
  737. let data = {
  738. //duration: pathArr.slice(0, pathArr.length - 1).reduce(function (total, currentValue) { return total + currentValue.time }, 0), //总时长(要去掉最后一个,因为已到终点,该点time无意义)
  739. points: pathArr,
  740. useDurSlice: true
  741. }
  742. let ani = editor.createMulAnimation(data)
  743. //注:最多只存在一条导览
  744. let bus = mitt()
  745. //播放完成
  746. ani.event_.addEventListener('playDone', () => {
  747. console.log('playComplete' )
  748. bus.emit('playComplete')
  749. viewer.images360.panos.forEach(e=>e.marker && Potree.Utils.updateVisible(e, 'playAni', true))
  750. })
  751. //切换点
  752. ani.event_.addEventListener('updateCurrentIndex', e => {
  753. console.log('updateCurrentIndex',e.currentIndex + 1)
  754. bus.emit('changePoint', e.currentIndex + 1)
  755. })
  756. return {
  757. bus,
  758. play() {
  759. MergeEditor.selectModel(null)
  760. viewer.images360.panos.forEach(e=>e.marker && Potree.Utils.updateVisible(e, 'playAni', false))
  761. ani.play()
  762. },
  763. pause() {
  764. viewer.images360.panos.forEach(e=>e.marker && Potree.Utils.updateVisible(e, 'playAni', true))
  765. ani.stop()
  766. },
  767. clear() {
  768. ani.remove()
  769. },
  770. }
  771. },
  772. //[path1, paht2], { time, speed }
  773. calcPathInfo(paths, info) { //传入的time, speed仅有一个。返回完整的 time, speed
  774. //这一版的control似乎无法在某个位置上改变角度,位置和角度一般都是一起变的,所以先不增加单位更换功能。
  775. let pos1 = new THREE.Vector3().copy(paths[0].position)
  776. let pos2 = new THREE.Vector3().copy(paths[1].position)
  777. let dis = pos1.distanceTo(pos2)
  778. if (info.time != void 0) {
  779. info.speed = dis / info.time
  780. } else {
  781. info.time = dis / info.speed
  782. }
  783. return info
  784. },
  785. addModel(props) {
  786. let model
  787. let bus = props.bus = mitt()
  788. //console.log('--addModel',props)
  789. props.isFirstLoad = isLocal ? props.bottom == void 0 : (props.isDynamicAdded || props.mode == 'single') // 在编辑时用户添加的 或 展示单个模型 (props.mode='single'模型展示页, props.mode='many'融合页)
  790. if (props.opacity == void 0) props.opacity = 100
  791. //if (props.type == 'obj') props.type = 'glb'
  792. props.scale && (props.scale /= 100)
  793. /* if(props.title == 'caijain'){
  794. props.url = [`${Potree.resourcePath}/models/tiles_2/tileset.json` ]
  795. } */
  796. let getBaseRotation = () => {
  797. if(ModelTypes[props.fromType]?.rot90 && props.type != 'obj'){
  798. return new THREE.Euler(Math.PI / 2, 0, 0)
  799. } else return new THREE.Euler(0, 0, 0)
  800. }
  801. let getDefaultRotation = () => {
  802. if(model.lonLatRot){
  803. return model.lonLatRot
  804. }else{
  805. return getBaseRotation()
  806. }
  807. }
  808. if (props.rotation) {
  809. if (props.rotation._x == void 0 && props.rotation.x != void 0) {
  810. props.rotation = new THREE.Euler().setFromVector3(props.rotation)
  811. }
  812. }
  813. props.baseRotation = getBaseRotation()
  814. props.is4dkkModel = ModelTypes[props.fromType].panos4dkk
  815. if (!props.isFirstLoad) {
  816. if (autoLoads.length == 0) { //首次加载
  817. setTimeout(() => {
  818. let sizes = autoLoads.map(e => e.size || 0)
  819. console.log('需要请求加载的模型为', autoLoads.map(e=>e.title) , '总大小', sizes.reduce(function (total, currentValue) {
  820. let current = parseFloat(currentValue)
  821. return total + ((typeof currentValue == 'number' || currentValue.includes('M')) ? current : current / 1024)
  822. }, 0))
  823. readyToAddModel = true //准备开始加载
  824. loadNext()//startLoad(autoLoads[0])
  825. }, 30)
  826. }
  827. autoLoads.push(props)
  828. readyToAddModel = false
  829. } else {
  830. readyToAddModel = true
  831. }
  832. let done = (model_) => {
  833. model = model_
  834. model.result_ = result
  835. model.props = props
  836. result.model = model
  837. model.fromType = ModelTypes[props.fromType].name
  838. if (!props.isFirstLoad) {
  839. model.visible = false//先不显示,防止卡顿
  840. }
  841. model.showInPano = /* !model.is4dkkModel// */props.raw.showInPano //现在不用这个,所有模型都可见,非is4dkkModel的还显示原本的贴图
  842. props.opacity < 100 && result.changeOpacity(props.opacity)
  843. model.addEventListener('changeSelect', (e) => {
  844. bus.emit('changeSelect', !!e.selected)
  845. })
  846. let lastState = {}
  847. model.addEventListener('transformChanged', (e) => {
  848. let msg = {byControl:!!e.byControl} //byControl代表是手动用控制轴修改 动画文件要改帧
  849. if (!lastState.position || !model.position.equals(lastState.position)) {
  850. lastState.position = msg.position = model.position.clone()
  851. //console.log('change pos', model.name, model.position.toArray())
  852. }
  853. if (!lastState.rotation || !model.rotation.equals(lastState.rotation)) {
  854. lastState.rotation = model.rotation.clone()
  855. msg.rotation = model.rotation.toObject()
  856. if(model.atPath && e.byControl){
  857. msg.quaAtPath = AnimationEditor.getModelQuaAtPath(model)
  858. msg.quaAtPath && (msg.quaAtPath = msg.quaAtPath.toObject())
  859. }
  860. }
  861. if (lastState.scale == void 0 || model.scale.x * 100 != lastState.scale) {
  862. lastState.scale = msg.scale = model.scale.x * 100
  863. }
  864. msg = Potree.Common.CloneObject(msg)
  865. //console.log(model.name, msg)
  866. bus.emit('transformChanged', msg)
  867. })
  868. spliceFromArr(model, props, true)
  869. model.addEventListener('changeSelect', (e) => {
  870. MergeEditor.transformControls.visible && e.selected && MergeEditor.transformControls.attach(model, e.clickPos) //: MergeEditor.transformControls.detach()
  871. })
  872. MergeEditor.modelAdded(model)
  873. load4dkkMedias(model)
  874. if (props.mode == 'single') {//模型查看页
  875. MergeEditor.noNeedSelection = true
  876. setTimeout(() => {
  877. MergeEditor.focusOn([model], 1000, true, true)
  878. }, 1)
  879. }
  880. if(ModelTypes[props.fromType].panos4dkk){
  881. Potree.load4dkkPanos(props.raw.num, model, getBaseRotation(), () => {
  882. bus.emit('loadDone',model)
  883. }, props.fromType == 0 ? '2k' : '4k' ) //看看场景是2k
  884. } else {
  885. bus.emit('loadDone',model)
  886. }
  887. //console.log('loadDone' )
  888. }
  889. let progressFun = (progress) => {
  890. bus.emit('loadProgress', progress)
  891. }
  892. let onError = function (xhr) {
  893. bus.emit('loadError', xhr)
  894. console.log('loadError!!!!!!!!!', Potree.Common.getNameFromURL(props.url), props.size, xhr)
  895. spliceFromArr(model, props, false)
  896. }
  897. try {
  898. props.url = JSON.parse(props.url) //去掉 '\'
  899. } catch (e) { }
  900. props.done = done; props.progressFun = progressFun; props.onError = onError
  901. if (readyToAddModel) {
  902. if (autoLoads.filter(e => e.loading).length < maxLoadingCount) {
  903. startLoad(props)
  904. }
  905. }
  906. let scaleMeasure
  907. let result = {
  908. bus,
  909. model,
  910. getDefaultRotation,
  911. supportPano() { //是否支持全景图
  912. return model?.panos?.length > 0
  913. },
  914. flyInPano(pano, {dontFly, quaternion, duration}={}) {// 飞入全景图
  915. requestInPano = model
  916. pano = pano || viewer.images360.findNearestPano(null, model.panos)
  917. if (pano) {
  918. dontFly || viewer.images360.flyToPano({ pano, canCancelLast: true, quaternion, duration})
  919. Potree.settings.displayMode = 'showPanos'
  920. }
  921. },
  922. flyOutPano() {// 飞出全景图(就是切换到正常融合视角)
  923. requestInPano = false
  924. Potree.settings.displayMode = 'showPointCloud'
  925. /* setTimeout(() => {//在下一帧再变,因为3dtiles需要更新一下才会显示tiles
  926. if (!requestInPano) {
  927. Potree.settings.displayMode = 'showPointCloud'
  928. Potree.Utils.updateVisible(MergeEditor.boxHelper, 'showPanos', true)
  929. }
  930. }, 50) */
  931. },
  932. moveModelTo(mouse, pos3d){//'移动到这里' //使模型中心底部的在鼠标所在位置
  933. console.log('moveModelTo', mouse)
  934. let viewport = viewer.mainViewport
  935. if(!pos3d){
  936. let width = viewport.width * viewer.renderArea.clientWidth
  937. let height = viewport.height * viewer.renderArea.clientHeight
  938. let pointer = Potree.Utils.convertScreenPositionToNDC(null, mouse, width, height);
  939. let {x,y} = Potree.Utils.getPointerPosAtHeight(model.bound.min.z, pointer)
  940. pos3d = new THREE.Vector3(x,y, model.bound.min.z)
  941. }
  942. MergeEditor.history.beforeChange(model)
  943. MergeEditor.moveBoundCenterTo(model, pos3d) //使模型中心的xy在鼠标所在位置
  944. model.position.z += model.boundSize.z / 2
  945. model.dispatchEvent({type:"position_changed", byControl:true}) //compute bound and emit
  946. MergeEditor.history.afterChange(model)
  947. },
  948. putInFrontOfCam(){//首次加载放在面前,高度和相机一致。(但不保证会不会被遮挡)
  949. let size = Math.max(1, new THREE.Vector2().copy(model.boundSize).length() * 0.7 )
  950. let vec = viewer.mainViewport.view.direction.clone().setZ(0).multiplyScalar(size)
  951. let pos = new THREE.Vector3().addVectors(viewer.mainViewport.view.position, vec)
  952. MergeEditor.moveBoundCenterTo(model, pos)
  953. model.dispatchEvent({type:"position_changed", byControl:true})
  954. },
  955. changeShow(show) {
  956. props.show = show //for autoLoads show model
  957. if (model) {
  958. Potree.Utils.updateVisible(model, 'datasetSelection', show)
  959. if (model.panos) {
  960. model.panos.forEach(e => e.setEnable(show))
  961. }
  962. }
  963. },
  964. changeSelect(state) {
  965. //console.error('changeSelect', model?.name, state)
  966. if (model) {
  967. let focus = !(viewer.images360.currentPano?.pointcloud == model && viewer.images360.isAtPano()) //viewer.images360.latestRequestMode != 'showPanos' //防止因走到漫游点后我发送panoModelChange后执行这个又飞了
  968. MergeEditor.selectModel(model, state, focus, true)
  969. updateCamNear()
  970. }
  971. },
  972. changeOpacity(opacity) { //见笔记:透明物体的材质设置
  973. if (opacity == void 0) opacity = 100
  974. opacity /= 100
  975. MergeEditor.changeOpacity(model, opacity)
  976. },
  977. changeBottom(z) {
  978. /* model && MergeEditor.setModelBtmHeight(model,z)
  979. model.dispatchEvent('transformChanged') //改了position */
  980. },
  981. changeScale(s) {
  982. if (model) {
  983. if(s.x != void 0){
  984. model.scale.copy(s) //animate model
  985. }else{
  986. s /= 100
  987. if (model.scale.x == s) return
  988. //MergeEditor.history.beforeChange(model)//但不知道什么时候结束拖拽
  989. model.scale.set(s, s, s)
  990. model.isPointcloud && model.changePointSize(/* Potree.config.material.realPointSize * s */)
  991. }
  992. model.dispatchEvent("scale_changed")
  993. }
  994. },
  995. changePosition(pos) {//校准取消时执行
  996. //console.log('changePosition', model.name, pos.x, pos.y, pos.z)
  997. if(pos.x == 0 && pos.y == 0 && pos.z == 0 && model.lonLatPos ){
  998. model && model.position.copy(model.lonLatPos)
  999. console.log('changePosition 使用经纬度坐标', model.name )
  1000. }else{
  1001. model && model.position.copy(pos)
  1002. }
  1003. model.dispatchEvent({ type: 'position_changed'/* , byControl:true */})
  1004. },
  1005. changeRotation(rot) {//校准取消时执行
  1006. //console.log('changeRotation', model.name, rot.x, rot.y, rot.z)
  1007. /* if(rot.x == 0 && rot.y == 0 && rot.z == 0 && model.lonLatRot ){
  1008. model && model.rotation.copy(model.lonLatRot)
  1009. console.log('changePosition 使用经纬度坐标', model.name )
  1010. }else{ */
  1011. model && model.rotation.setFromVector3(rot)
  1012. //}
  1013. model.dispatchEvent({ type: 'rotation_changed' , by2d:true/* , byControl:true */})
  1014. },
  1015. getModelPose(){
  1016. return {
  1017. position: model.position.clone(),
  1018. rotation: model.rotation.toObject(),
  1019. quaternion: model.quaternion.toObject(),
  1020. scale: model.scale.clone()
  1021. }
  1022. },
  1023. enterRotateMode() {
  1024. if (model) {
  1025. if (MergeEditor.split) {//分屏校准
  1026. MergeEditor.setTransformState('rotate')
  1027. MergeEditor.transformControls2.attach(model)
  1028. MergeEditor.transformControls2.mode = 'rotate'
  1029. }
  1030. MergeEditor.transformControls.attach(model)
  1031. MergeEditor.transformControls.mode = 'rotate'
  1032. }
  1033. },
  1034. enterMoveMode() {
  1035. console.log('enterMoveMode')
  1036. if (model) {
  1037. if (MergeEditor.split) {//分屏校准
  1038. MergeEditor.setTransformState('translate')
  1039. MergeEditor.transformControls2.attach(model)
  1040. MergeEditor.transformControls2.mode = 'translate'
  1041. }
  1042. MergeEditor.transformControls.attach(model)
  1043. MergeEditor.transformControls.mode = 'translate'
  1044. }
  1045. },
  1046. enterScaleMode(){
  1047. if (model) {
  1048. MergeEditor.transformControls.attach(model)
  1049. MergeEditor.transformControls.mode = 'scale'
  1050. }
  1051. },
  1052. leaveTransform() {
  1053. //console.log('leaveTransform')
  1054. if (MergeEditor.split) {//分屏校准
  1055. MergeEditor.setTransformState(null)
  1056. } else {
  1057. MergeEditor.transformControls.detach()
  1058. MergeEditor.transformControls2.detach()
  1059. }
  1060. },
  1061. enterAlignment() {//开始校准
  1062. result.leaveTransform()
  1063. MergeEditor.enterSplit()
  1064. if(Potree.settings.showCesium){
  1065. cesiumViewer.scene.canvas.style.width = '50%'
  1066. //cesiumViewer.resize()
  1067. }
  1068. let bus = new mitt()
  1069. return {
  1070. bus
  1071. }
  1072. },
  1073. leaveAlignment() {
  1074. //console.log('leaveAlignment',model.position, model.rotation)
  1075. MergeEditor.leaveSplit()
  1076. MergeEditor.transformControls.detach()
  1077. MergeEditor.transformControls2.detach()
  1078. if(Potree.settings.showCesium){
  1079. cesiumViewer.scene.canvas.style.width = ''
  1080. updateMap()
  1081. }
  1082. },
  1083. enterScaleSet() {//设置比例
  1084. let bus = new mitt()
  1085. let length, measureBuilded;
  1086. //viewer.outlinePass.selectedObjects = []
  1087. if (!Potree.Utils.isInsideFrustum(model.boundingBox.clone().applyMatrix4(model.matrixWorld), viewer.scene.getActiveCamera())) {
  1088. MergeEditor.focusOn(model, 600)
  1089. }
  1090. MergeEditor.getAllObjects().forEach(m => {//隐藏其他的模型
  1091. if (m != model) Potree.Utils.updateVisible(m, 'enterScaleSet', false)
  1092. })
  1093. result.oldFar = Potree.settings.cameraFar
  1094. model.enterScaleOldState_ = {
  1095. //view: viewer.mainViewport.view.clone(),
  1096. scale: model.scale.x,
  1097. far: Potree.settings.cameraFar
  1098. }
  1099. let setScale = () => {
  1100. if (length == void 0 || !measureBuilded) return
  1101. let vec = new THREE.Vector3().subVectors(viewer.mainViewport.camera.position, scaleMeasure.points[1])
  1102. let dis = scaleMeasure.points[0].distanceTo(scaleMeasure.points[1])
  1103. let s = length / Math.max(dis,0.00001)
  1104. result.changeScale(model.scale.x * s * 100)
  1105. /* setTimeout(()=>{
  1106. viewer.focusOnObject(scaleMeasure , 'measure', 500)
  1107. },1) */
  1108. let newCamPos = new THREE.Vector3().addVectors(scaleMeasure.points[1], vec.multiplyScalar(s))
  1109. viewer.scene.view.setView({
  1110. position: newCamPos, target: scaleMeasure.getCenter(), duration: 0, callback: () => {
  1111. //更改target到measure中心的好处就是可以让相机绕measure中心转,坏处是每次更改都会变一下画面
  1112. //Potree.settings.cameraFar = Math.max(model.enterScaleOldState_.far, viewer.scene.view.position.distanceTo(model.boundCenter) + model.boundingBox.clone().applyMatrix4(model.matrixWorld).getSize(new THREE.Vector3).length())
  1113. //use updateCamFar()
  1114. }
  1115. })
  1116. }
  1117. return {
  1118. bus,
  1119. setLength(v) {
  1120. if (!v) return
  1121. length = v
  1122. setScale()
  1123. },
  1124. startMeasure() {
  1125. if (scaleMeasure) {
  1126. viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure: scaleMeasure })
  1127. viewer.scene.removeMeasurement(scaleMeasure)
  1128. }
  1129. measureBuilded = false
  1130. scaleMeasure = viewer.measuringTool.startInsertion(
  1131. { measureType: "Distance", unit: "metric" },
  1132. () => {
  1133. //done:
  1134. //bus.emit('end' ) //完成
  1135. measureBuilded = true
  1136. setScale()
  1137. },
  1138. () => {
  1139. //cancel
  1140. //bus.emit('quit') //删除
  1141. }
  1142. )
  1143. scaleMeasure.forbitRepeatPoint = true //两个点不能相同,否则长度是0
  1144. scaleMeasure.addEventListener('marker_dropped', (e) => {//拖拽结束后发送changeCallBack
  1145. if (scaleMeasure.parent) {
  1146. //未被删除
  1147. measureBuilded && setScale()
  1148. }
  1149. })
  1150. }
  1151. }
  1152. },
  1153. leaveScaleSet() {
  1154. if (scaleMeasure) {
  1155. viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure: scaleMeasure })
  1156. viewer.scene.removeMeasurement(scaleMeasure)
  1157. scaleMeasure = null
  1158. }
  1159. //viewer.outlinePass.selectedObjects = [model];
  1160. MergeEditor.getAllObjects().forEach(m => {//恢复其他的模型
  1161. if (m != model) Potree.Utils.updateVisible(m, 'enterScaleSet', true)
  1162. })
  1163. setTimeout(()=>{//可能还原了 相机位置移动回去
  1164. if(model.scale.x == model.enterScaleOldState_.scale){
  1165. MergeEditor.focusOn(model, 0) //reset orbitcontrol's minRadius //viewer.mainViewport.view.copy(model.enterScaleOldState_.view)
  1166. //Potree.settings.cameraFar = model.enterScaleOldState_.far
  1167. }
  1168. },10)
  1169. },
  1170. destroy() {
  1171. model && MergeEditor.removeModel(model)
  1172. result.changeSelect(false)
  1173. viewer.dispatchEvent('content_changed')
  1174. }
  1175. }
  1176. return result
  1177. },
  1178. createAnimationGroup(){//创建动画模块
  1179. let bus = mitt()
  1180. AnimationEditor.addEventListener('atTime',e=>{//发送当前动画时间,同步字幕
  1181. //console.log('currentTime',e.time )
  1182. bus.emit('currentTime',e.time )
  1183. })
  1184. let funs = {
  1185. bus,
  1186. play(){
  1187. console.log('play ani, time:', AnimationEditor.cursorTime)
  1188. AnimationEditor.play({time:AnimationEditor.cursorTime })
  1189. },
  1190. pause(){
  1191. console.log('pause ani')
  1192. AnimationEditor.pause()
  1193. },
  1194. setCurrentTime(s){// 设置当前时间, 单位为秒
  1195. (!AnimationEditor.playing || s == 0) && AnimationEditor.at(s) //除非s==0否则一定要先pause后才能设置时间
  1196. //console.log('setCurrentTime',s, AnimationEditor.playing, AnimationEditor.cursorTime)
  1197. },
  1198. addAnimationModel(data){// 添加动画模型
  1199. /* title: string; 模型名称
  1200. url: string; 模型路径
  1201. showTitle: boolean; 是否显示名称
  1202. fontSize: number; 名称字体大小
  1203. globalVisibility: boolean; 是否全局可视
  1204. visibilityRange: number; */
  1205. console.log('添加动画文件',data)
  1206. let update = ()=>{
  1207. AnimationEditor.at(AnimationEditor.cursorTime || 0,null,true)
  1208. }
  1209. let model
  1210. let prop = {
  1211. url: data.url,
  1212. title: data.title,
  1213. fromType: 8, raw:{},
  1214. //renderType : 'normal', //色彩好些,否则MeshBasic颜色偏深
  1215. type:'obj', id:data.id, dontFocus:true
  1216. }
  1217. let modelFuns = sdk.addModel(prop)
  1218. let visibleRange
  1219. let updateVisiByRange = ()=>{//可见范围
  1220. let visi = true
  1221. if(visibleRange){
  1222. let cameraPos = viewer.mainViewport.view.position
  1223. let dis = model.boundCenter.distanceTo(cameraPos)
  1224. visi = dis < visibleRange
  1225. }
  1226. Potree.Utils.updateVisible(model,'outOfVisiRange', visi)
  1227. }
  1228. modelFuns.bus.on('loadDone',(model_)=>{
  1229. model = model_
  1230. MergeEditor.addTitleForModel(model)
  1231. model.addEventListener('transformChanged',updateVisiByRange)
  1232. model.result_ = result
  1233. if(data.quaAtPath){
  1234. model.quaAtPath = new THREE.Quaternion().copy(data.quaAtPath)
  1235. }
  1236. setTimeout(e=>{
  1237. update()//计算下默认姿态
  1238. },1)
  1239. })
  1240. let result = Object.assign(modelFuns, {
  1241. updateVisiByRange,
  1242. visibilityTitle(v){
  1243. Potree.Utils.updateVisible(model.titleLabel,'user', v)
  1244. },
  1245. // 更改动画可见范围 不传为全局可见
  1246. changeVisibilityRange(range){
  1247. visibleRange = range
  1248. updateVisiByRange()
  1249. },
  1250. // 更改模型名称
  1251. changeTitle(name){
  1252. model.name = name
  1253. if(model.titleLabel ){
  1254. model.titleLabel.setText(name)
  1255. Potree.Utils.updateVisible(model.titleLabel, 'noText', name.trim() != '')
  1256. viewer.dispatchEvent('content_changed')
  1257. }
  1258. },
  1259. changeFontSize(fontsize){
  1260. model.titleLabel.fontsize = fontsize
  1261. model.titleLabel.updateTexture();
  1262. viewer.dispatchEvent('content_changed')
  1263. },
  1264. // 添加模型帧
  1265. addFrame(frame){//pose 至少会有一个关键帧作为默认姿态
  1266. //console.log('addFrame',frame)
  1267. let key = {time: frame.time }
  1268. let getData = (data)=>{
  1269. let info = {}
  1270. info.pos = new THREE.Vector3().copy(data.position)
  1271. info.scale = (typeof data.scale == 'number') ? new THREE.Vector3(data.scale/100,data.scale/100,data.scale/100) : new THREE.Vector3().copy(data.scale)
  1272. info.qua = new THREE.Quaternion().copy(data.quaternion)
  1273. return info
  1274. }
  1275. Object.assign(key, getData(frame.mat))
  1276. AnimationEditor.addKey(model, 'pose', key )
  1277. return {
  1278. destroy(){
  1279. AnimationEditor.removeKey(model,'pose', key)
  1280. update()
  1281. },
  1282. changeTime(time){
  1283. if(time==key.time)return
  1284. key.time = time
  1285. AnimationEditor.reOrderKey(model, 'pose', key)
  1286. update()
  1287. },
  1288. setMat(data){//设置帧
  1289. //console.log('设置帧',data.scale)
  1290. Object.assign(key,getData(data))
  1291. //update() //更新的话会使transformControls打滑,因camFollowObject跟随该物体
  1292. }
  1293. }
  1294. },
  1295. getSupportActions(){
  1296. return model.actions?.map(e=>{return {name:e._clip.name, duration:e._clip.duration}}) //界面上的一次循环时长保留小数位方式为floor,为了避免超出时间使动作回到开头
  1297. },
  1298. setDefaultPose(pose){//没用了
  1299. model.defaultAniPose = pose //不在关键帧和路径时transformChanged后记录的
  1300. },
  1301. addAction(frame){// 添加模型动作
  1302. //console.log('addAction',frame)
  1303. let key = Object.assign({},frame)
  1304. key.weight = key.amplitude || 1 //幅度
  1305. key.speed = key.speed || 1
  1306. key.action = model.actions.find(e=>e._clip.name == key.key)
  1307. if(!key.action){
  1308. //return console.error('cannot find action', key.key)
  1309. let random = Math.floor(Math.random() * (model.actions.length))
  1310. random = Math.min(model.actions.length-1, random)
  1311. console.log('没找到动作 '+key.key+', 暂时先用第'+ random +'个')
  1312. key.action = model.actions[random]
  1313. }
  1314. delete key.amplitude
  1315. delete key.key
  1316. delete key.duration
  1317. AnimationEditor.addKey(model, 'clip', key)
  1318. let updateAction = ()=>{
  1319. if(key.chosed && AnimationEditor.cursorTime == key.time){//如果选择了该动作且在开头,就直接播放原动作
  1320. model.actions.forEach(a=>{
  1321. if(a == key.action){
  1322. a.play()
  1323. a.paused = false
  1324. a.setEffectiveTimeScale(key.speed)
  1325. a.setEffectiveWeight(key.weight);
  1326. }else{
  1327. a.stop()
  1328. }
  1329. })
  1330. }else update()
  1331. }
  1332. return {
  1333. destroy(){
  1334. AnimationEditor.removeKey(model,'clip', key)
  1335. update()
  1336. },
  1337. changeTime(time){
  1338. if(time==key.time)return
  1339. key.time = time
  1340. AnimationEditor.reOrderKey(model, 'clip', key)
  1341. updateAction()
  1342. },
  1343. changeDuration(dur){
  1344. key.dur = dur
  1345. AnimationEditor.updateTimeRange()
  1346. updateAction()
  1347. },
  1348. changeAmplitude(weight){//修改动作幅度
  1349. key.weight = weight
  1350. updateAction()
  1351. },
  1352. changeSpeed(speed){
  1353. key.speed = speed
  1354. updateAction()
  1355. },
  1356. chose(state){
  1357. key.chosed = state
  1358. updateAction() //选中后单独播放动作
  1359. }
  1360. }
  1361. },
  1362. addPath(frame){//走路路径
  1363. //console.log('addPath',frame)
  1364. let key = Object.assign({},frame)
  1365. key.path = key.path.path
  1366. key.dur = key.duration
  1367. delete key.duration
  1368. AnimationEditor.addKey(model, 'path', key)
  1369. return {
  1370. destroy(){
  1371. AnimationEditor.removeKey(model, 'path', key)
  1372. update()
  1373. },
  1374. changeTime(time){
  1375. if(time==key.time)return
  1376. key.time = time
  1377. AnimationEditor.reOrderKey(model, 'path', key)
  1378. update()
  1379. },
  1380. changeDuration(dur){
  1381. key.dur = dur
  1382. AnimationEditor.updateTimeRange()
  1383. update()
  1384. },
  1385. changeReverse(reverse){
  1386. key.reverse = reverse
  1387. update()
  1388. },
  1389. changePath(path){
  1390. key.path = path.path
  1391. update()
  1392. }
  1393. }
  1394. },
  1395. /* getCurrentMat(){// 获取当前时间改模型的姿态
  1396. return {
  1397. quaternion: model.quaternion.clone(),
  1398. scale: model.scale.clone(),
  1399. position: model.position.clone()
  1400. }
  1401. }, */
  1402. getCurrentSubtitlePixel({width , height}){// 获取当前模型字幕出现的适合位置,传入旁边dom的宽高,返回像素位置
  1403. if(!model.visible)return null
  1404. let boundPoints = Potree.Common.getBoundPoints(model.boundingBox, model.matrixWorld)
  1405. boundPoints.forEach(e=>{
  1406. e.pos2d = Potree.Utils.getPos2d(e, viewer.mainViewport , viewer.renderArea, viewer.renderer )
  1407. })
  1408. boundPoints = boundPoints.filter(e=>e.pos2d.inSight && e.pos2d.trueSide )
  1409. if(boundPoints.length == 0)return
  1410. // let startTime = performance.now()
  1411. {//判断遮挡 暂时计算全部八个顶点,以后再改。如延时(不过ifPointBlockedByIntersect本身就有延迟,每帧计算一点点),每帧判断不同的点,直到全部顶点都算完都被遮挡了才隐藏。且如果离得很远,pos2d像素差距很小的话,只判断中心点即可。
  1412. //let boundingBox = model.boundingBox.
  1413. let boundPointsShrink = [...boundPoints] //Potree.Common.getBoundPoints(boundingBox) 先用这个测试
  1414. let visi = boundPointsShrink.some(p=>!viewer.ifPointBlockedByIntersect( p )) //若模型刚好为长方体,自身可能遮挡顶点,可能需要先过滤自身
  1415. if(!visi)return //console.log('被遮挡')
  1416. }
  1417. //console.log('getCurrentSubtitlePixel')
  1418. /* let pos2ds = boundPoints.map((point)=>{
  1419. return Potree.Utils.getPos2d(point, viewer.mainViewport , viewer.renderArea, viewer.renderer )
  1420. }).filter(e=>e.inSight && e.trueSide )
  1421. if(pos2ds.length == 0)return */
  1422. let pos2ds = boundPoints.map(e=>e.pos2d)
  1423. pos2ds.sort((a,b)=>{return a.vector.y - b.vector.y})
  1424. let top = pos2ds[0], btm = pos2ds[pos2ds.length - 1]
  1425. let centerY = (top.pos.y + btm.pos.y) / 2
  1426. pos2ds.sort((a,b)=>{return a.vector.x - b.vector.x})
  1427. let left = pos2ds[0], right = pos2ds[pos2ds.length - 1]
  1428. let leftPad = left.vector.x - (-1) //模型左侧空出的宽度
  1429. let rightPad = 1 - right.vector.y
  1430. const margin = 10;
  1431. let x
  1432. let y = centerY - height/2 //返回字幕左上角位置
  1433. if(leftPad > rightPad){//向左侧扩展
  1434. x = left.pos.x - margin - width
  1435. }else{
  1436. x = right.pos.x + margin
  1437. }
  1438. //console.log('获取字幕位置', performance.now() - startTime )
  1439. return {x,y}
  1440. },
  1441. delayEndTime(){
  1442. return Potree.settings.maxClipFadeTime / 2
  1443. }
  1444. })
  1445. return result
  1446. },
  1447. enterEditPannel(){
  1448. MergeEditor.history.clear() //清除普通模型的
  1449. },
  1450. exitEditPannel(){
  1451. MergeEditor.history.clear() //清除动画模型的
  1452. }
  1453. }
  1454. return funs
  1455. },
  1456. //测量线的点都附着于各个模型,当模型变化时,点跟着变化。
  1457. // 新的测量创建方法,传入type 返回新测量对象
  1458. startMeasure(type) {
  1459. // 寻创建的测量对象有上面绘画测量对象的所有方法
  1460. const bus = mitt()
  1461. let info = getMeasureType(type)
  1462. let measure = viewer.measuringTool.startInsertion(
  1463. info,
  1464. () => {
  1465. //done:
  1466. bus.emit('submit')
  1467. },
  1468. () => {
  1469. //cancel
  1470. bus.emit('cancel'/* , ret */) //删除
  1471. }
  1472. )
  1473. Potree.Log('startMeasure: ' + measure.id, '#00c7b2')
  1474. /* let cancel = ()=>{
  1475. Potree.Log('clear删除: ' + measure.id, '#00c7b2')
  1476. viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure })
  1477. viewer.scene.removeMeasurement(measure)
  1478. } */
  1479. let result = {
  1480. bus,
  1481. ...getMeasureFunction(measure, bus),
  1482. }
  1483. /* StartMeasure = Measure & {
  1484. // 多了cancel 取消测量的事件,没有参数
  1485. // 多了invalidPoint 当用户测量了无效点时的事件,抛出无效原因
  1486. bus: Emitter<{ cancel: void; invalidPoint: string }>
  1487. } */
  1488. return result
  1489. },
  1490. // 绘画测量线(非新增使用)
  1491. // type = 'free' (自由) || 'vertical' (垂直) || 'area' (面积)
  1492. // positions 点数组 构成如下 [{ point: {x,y,z}, modelId: 1 }]
  1493. drawMeasure(type, dataset_points, points_datasets) {
  1494. // 返回测量对象有如下
  1495. const bus = mitt()
  1496. let info = getMeasureType(type /* , unit */)
  1497. //info.points = positions
  1498. info.dataset_points = dataset_points
  1499. info.points_datasets = points_datasets
  1500. //info.sid = sid
  1501. info.bus = bus
  1502. let measure = viewer.measuringTool.createMeasureFromData(info)
  1503. if (!measure) return { bus }
  1504. Potree.Log('drawMeasure由数据新建: ' + measure.id, '#00c7b2')
  1505. let result = {
  1506. bus,
  1507. setPositions(dataset_points, points_datasets) {//用于恢复measure的点,不会修改点的个数
  1508. measure.dataset_points = dataset_points.map(e => {
  1509. return e && new THREE.Vector3().copy(e)
  1510. })
  1511. measure.points_datasets = points_datasets
  1512. measure.points = measure.dataset_points.map((p, i) => {
  1513. return Potree.Utils.datasetPosTransform({ fromDataset: true, datasetId: measure.points_datasets[i], position: p })
  1514. })
  1515. measure.getPoint2dInfo(measure.points)
  1516. measure.update({ ifUpdateMarkers: true })
  1517. measure.setSelected(false)//隐藏edgelabel
  1518. },
  1519. ...getMeasureFunction(measure, bus),
  1520. }
  1521. return result
  1522. },
  1523. createPath(props){//路线
  1524. //console.log('createPath', props)
  1525. let bus = mitt()
  1526. let path
  1527. let info = {type : 'Path', minMarkers : 2, title:props.name, lineHeight : props.line.altitudeAboveGround }
  1528. if(props.points.length == 0){
  1529. path = viewer.measuringTool.startInsertion( info, () => {
  1530. bus.emit("drawed" ); //完成
  1531. })
  1532. viewer.dispatchEvent({ type: 'cancel_insertions', dontRemove: true, measure:path }) //要等进入编辑才能继续编辑
  1533. }else{
  1534. let originPointCount = props.points.length
  1535. props.points = props.points.filter(e=> isValidPoint(e.modelId))
  1536. info.points_datasets = props.points.map(e=> e.modelId == Id_noIntersect ? null : e.modelId)
  1537. info.dataset_points = info.points = props.points.map(e=>e.position)//当该点不在任何模型上时,记录的是世界坐标,所以两个都赋值,过后根据有无datasetID选择
  1538. path = viewer.measuringTool.createMeasureFromData(info);
  1539. if(props.line.position) {
  1540. if(isValidPoint(props.line.modelId)){
  1541. let pos = props.line.modelId == Id_noIntersect ? new THREE.Vector3().copy(props.line.position) :
  1542. Potree.Utils.datasetPosTransform({fromDataset:true, position: props.line.position, datasetId: props.line.modelId })
  1543. path.updateTitlePos(pos)
  1544. }else{
  1545. //console.log('path label pos 因模型被删而去除', info.title )
  1546. }
  1547. }
  1548. if(props.points.length < originPointCount ) {
  1549. path.dispatchEvent('createDone')
  1550. //console.log('path点因模型被删减少', info.title, originPointCount,'->',props.points.length)
  1551. }
  1552. }
  1553. {
  1554. let curSelectMarker
  1555. path.addEventListener('markerSelect',(e)=>{
  1556. let msg
  1557. if(e.cancel){
  1558. curSelectMarker == e.marker && (msg = -1) //是当前选中的marker就取消
  1559. }else{
  1560. curSelectMarker = e.marker
  1561. msg = path.markers.indexOf(e.marker)
  1562. }
  1563. //msg != void 0 && console.log('msg',msg)
  1564. msg != void 0 && bus.emit('activePoint', msg )
  1565. })
  1566. path.addEventListener('titlePosChanged',(e)=>{
  1567. //console.log('titlePosChanged',path.title, e.position.clone())
  1568. bus.emit('linePositionChange', {
  1569. modelId: e.root ? e.root.dataset_id : Id_noIntersect,
  1570. pos: e.root ? Potree.Utils.datasetPosTransform({toDataset:true, position: e.position.clone(), datasetId: e.root.dataset_id }) : e.position.clone()
  1571. })
  1572. })
  1573. path.addEventListener('chose',(e)=>{
  1574. switchSelect(e.state)
  1575. bus.emit('focus', e.state)
  1576. })
  1577. path.addEventListener('dragLineLen',(e)=>{
  1578. bus.emit('changeLineHeight', path.lineHeight)
  1579. })
  1580. }
  1581. let funs = getMeasureFunction(path, bus)
  1582. let switchSelect = (state)=>{//切换选择,最多一个选中
  1583. if(state){
  1584. curSelectPath && curSelectPath.setSelected('unclick' ) //取消上一个选中的
  1585. curSelectPath = path
  1586. }else{
  1587. curSelectPath == path && (curSelectPath = null)
  1588. }
  1589. }
  1590. let functions = Object.assign(funs,{
  1591. bus,
  1592. path,
  1593. changeEditMode(state){//进入编辑
  1594. if(!state){
  1595. viewer.dispatchEvent({ type: 'cancel_insertions', dontRemove: true, measure:path })
  1596. }
  1597. path.setEditEnable(state)
  1598. },
  1599. changeCanEdit(state){//是否点击pen图标以加点和删点
  1600. if(state){
  1601. if(path.points.length < 2){//继续绘制
  1602. info.resume = true, info.measure = path
  1603. path = viewer.measuringTool.startInsertion( info, () => {
  1604. bus.emit("drawed" ); //完成
  1605. })
  1606. }
  1607. }else{
  1608. viewer.dispatchEvent({ type: 'cancel_insertions', dontRemove: true, measure:path })
  1609. }
  1610. console.log('changeCanEdit',state)
  1611. path.setAddOrRemPoint(state)
  1612. },
  1613. visibility(v){
  1614. //console.log('visibility', path.title, v)
  1615. Potree.Utils.updateVisible(path,'user', v)
  1616. },
  1617. visibilityName(v){
  1618. path.setTitleVisi(path.titleLabel.parent, v, 'user')
  1619. },
  1620. changeName(name){
  1621. path.setTitle(name)
  1622. },
  1623. changePointName(index,name){
  1624. path.setMarkerTitle(index, name)
  1625. },
  1626. changePathPoints(points){
  1627. console.log('changePathPoints??????????',points)
  1628. },
  1629. deletePoint(index){
  1630. path.removePoint(index)
  1631. },
  1632. changeFontSize(fontsize){
  1633. path.setFontSize(fontsize)
  1634. },
  1635. changeLine({width,color,altitudeAboveGround}){
  1636. path.setPathWidth(width)
  1637. path.setColor(color)
  1638. path.setLineHeight(altitudeAboveGround)
  1639. },
  1640. changeVisibilityRange(far){//设置消失距离
  1641. path.setFadeFar(far== -1 ? 0 : far)
  1642. },
  1643. highlight(state){
  1644. path.setSelected(state?'hover':'unhover', true)
  1645. },
  1646. focus(state){
  1647. switchSelect(state)
  1648. path.setSelected(state?'click':'unclick', true)
  1649. },
  1650. changeDirection(show,reverse){
  1651. path.setArrowDisplay(show)
  1652. path.setReverse(reverse)
  1653. },
  1654. createAni(tension){
  1655. let distance = path.totalLength
  1656. let pathPoints = path.points.map(e=>e.clone().add(new THREE.Vector3(0,0,2))) //在地面之上一定高度
  1657. if(path.reverse) pathPoints.reverse()
  1658. const speed = 3, //m/s
  1659. turnDisPerRad = 1.5,
  1660. maxTurnDis = 3,//拐弯最大距离
  1661. maxRoadTurnRatio = 0.8 //每段路单次转弯最大比例,防止拐弯占据一整条路直到下一个点
  1662. let roadLens = []
  1663. let vecs = pathPoints.map((p,i)=>{
  1664. if(i==0)return
  1665. let last = pathPoints[i-1]
  1666. roadLens.push(p.distanceTo(last))
  1667. return new THREE.Vector3().subVectors(p,last).normalize()
  1668. })
  1669. let turnDis = vecs.map((vec,i)=>{//在每个转折点拐弯前后需要的米数
  1670. if(i==0 || i==vecs.length-1)return 0
  1671. let next = vecs[i+1]
  1672. let angle = next.angleTo(vec)
  1673. return Math.min(turnDisPerRad * angle / 2, maxTurnDis )
  1674. })
  1675. let points = []
  1676. let len = pathPoints.length
  1677. for(let i=0;i<len;i++){
  1678. let thisPoint = pathPoints[i]
  1679. let nextPoint = pathPoints[i+1]
  1680. if(i==0 || i==len-1){//首尾的点直接加入,其他点不加
  1681. points.push({
  1682. position: thisPoint.clone(),
  1683. target: i==0 ? nextPoint.clone() : pathPoints[len-1].clone().add(vecs[len-1])
  1684. })
  1685. }
  1686. if(i<len-1){ //加入每段边要加入的两个(or一个)拐点 ,拐点之间方向沿着路径
  1687. let turnDis1 = Math.min(turnDis[i], roadLens[i] * maxRoadTurnRatio)
  1688. let turnDis2 = Math.min(turnDis[i+1], roadLens[i] * maxRoadTurnRatio)
  1689. let turnDisSum = turnDis1 + turnDis2 //两端拐弯距离之和。
  1690. if(turnDisSum > roadLens[i]){//如果超过了路长度, 该条路就只有一个拐点
  1691. turnDis1 = turnDis1 / turnDisSum * roadLens[i]
  1692. let p = thisPoint.clone().add(vecs[i+1].clone().multiplyScalar(turnDis1))
  1693. points.push({position: p, target: nextPoint.clone()})
  1694. }else{
  1695. if(turnDis1>0){ //i==0时为0
  1696. let turnPoint1 = thisPoint.clone().add(vecs[i+1].clone().multiplyScalar(turnDis1))
  1697. points.push({position: turnPoint1, target: nextPoint.clone()})
  1698. }
  1699. if(turnDis2>0){//i==len-2时为0
  1700. let turnPoint2 = nextPoint.clone().sub(vecs[i+1].clone().multiplyScalar(turnDis2))
  1701. points.push({position:turnPoint2, target: nextPoint.clone()})
  1702. }
  1703. }
  1704. }
  1705. }
  1706. //加后缀&test以看路线
  1707. let data = {
  1708. name : 'path_guideTour',
  1709. duration : distance / speed,
  1710. points,
  1711. tension
  1712. }
  1713. path.animation_ = viewer.modules.CamAniEditor.createAnimation(data)
  1714. },
  1715. play(playDone){
  1716. if(path.points.length < 2)return playDone && playDone() //no points
  1717. Potree.settings.displayMode = 'showPointCloud'
  1718. let oldStates = {
  1719. editEnable: path.editEnable,
  1720. addOrRemovePoint: path.addOrRemovePoint
  1721. }
  1722. path.editEnable && functions.changeEditMode(false)
  1723. path.addOrRemovePoint && path.setAddOrRemPoint(false)
  1724. if(Potree.settings.pathSmooth){
  1725. let curve = path.curve.clone();
  1726. curve.points.forEach(e=>e.z += 2)
  1727. if(path.reverse) curve.points.reverse()
  1728. //let geoPoints = path.geoPoints.map(e=> e.clone().add(new THREE.Vector3(0,0,2)) )//height
  1729. let duration = path.totalLength / 3
  1730. //let tangentDt = path.totalLength * 0.0001
  1731. path.animation_ = viewer.modules.CamAniEditor.createCurveAni(curve, duration )
  1732. }else{
  1733. functions.createAni();//不传参数时路径最圆润缓和,但会脱离原路径。传参后除了拐弯都按路径,参数越大越圆润,但容易有折回的bug。 如果没有严格要求就不传参效果最佳。
  1734. }
  1735. path.animation_.play()
  1736. path.animation_.addEventListener('playDone', () => {
  1737. oldStates.editEnable && functions.changeEditMode(true)
  1738. oldStates.addOrRemovePoint && path.setAddOrRemPoint(true)
  1739. playDone && playDone()
  1740. },{once:true})
  1741. },
  1742. pause(){
  1743. path.animation_?.pause()
  1744. path.animation_ && viewer.modules.CamAniEditor.removeAnimation(path.animation_)
  1745. path.animation_ = null
  1746. }
  1747. })
  1748. /* for(let i in functions){
  1749. if(functions[i] instanceof Function){
  1750. let oldFun = functions[i]
  1751. functions[i] = function(){
  1752. console.warn('path', i, path.title, ...arguments)
  1753. oldFun.apply(this, arguments)
  1754. }
  1755. }
  1756. } */
  1757. path.functions = functions
  1758. props.line && functions.changeLine(props.line)
  1759. return functions
  1760. },
  1761. startAddSth(){//开始添加热点
  1762. Potree.settings.disableClick = true //禁止点击事件,尤其是全景模式下,否则会走到下一个点
  1763. viewer.dispatchEvent('start_inserting_tag')
  1764. },
  1765. endAddSth(){
  1766. Potree.settings.disableClick = false
  1767. viewer.dispatchEvent('endTagMove')
  1768. },
  1769. createTagging(props){
  1770. let bus = mitt()
  1771. //console.warn('createTagging', props)
  1772. let root = viewer.scene.pointclouds.concat(viewer.objs.children).find(e=>e.dataset_id == props.modelId)
  1773. if(!root){
  1774. return console.error('热点没有找到该modelId,模型是否已经删除?')
  1775. }
  1776. let info = {
  1777. position: new THREE.Vector3().copy(props.position), //局部坐标
  1778. normal: props.normal ? new THREE.Vector3().copy(props.normal) : new THREE.Vector3(0,0,1),
  1779. root, lineLength: props.altitudeAboveGround,
  1780. title: props.title, fontsize: props.fontSize
  1781. }
  1782. let tag = viewer.tagTool.createTagFromData(info)
  1783. tag.addEventListener('mouseover',()=>{
  1784. bus.emit('enter')
  1785. })
  1786. tag.addEventListener('mouseleave',()=>{
  1787. bus.emit('leave')
  1788. })
  1789. tag.addEventListener('click',()=>{
  1790. bus.emit('click')
  1791. })
  1792. tag.addEventListener('posChanged',(e)=>{
  1793. bus.emit('changePosition', {
  1794. modelId: tag.root.dataset_id,
  1795. normal: tag.normal.clone(),
  1796. pos: tag.position.clone()
  1797. })
  1798. })
  1799. tag.addEventListener('dragLineLen',(e)=>{
  1800. bus.emit('changeLineHeight', tag.lineLength)
  1801. })
  1802. tag.functions = {
  1803. bus,
  1804. changeType(type){
  1805. //console.log('changeType', tag.title, type)
  1806. let onMesh = type == '3d'
  1807. if(tag.onMesh != onMesh){
  1808. tag.changeOnMesh(onMesh)
  1809. }
  1810. },
  1811. visibility(v){// 标注可见性
  1812. //console.log('visibility', tag.title, v)
  1813. Potree.Utils.updateVisible(tag,'user', v)
  1814. },
  1815. visibilityTitle(v){
  1816. tag.setTitleVisi(v, 'user')
  1817. },
  1818. changePosition({modelId,position,normal}){
  1819. let root = viewer.scene.pointclouds.concat(viewer.objs.children).find(e=>e.dataset_id == props.modelId)
  1820. tag.changePos({root,position,normal})
  1821. },
  1822. changeImage(url){
  1823. tag.changeMap(url)
  1824. },
  1825. changeTitle(title){
  1826. tag.setTitle(title)
  1827. },
  1828. changeMat({scale,rotation}){//大小旋转 贴墙时
  1829. tag.setFaceAngle(rotation)
  1830. tag.changeSpotScale(scale)
  1831. },
  1832. changeFontSize(fontsize){
  1833. tag.setFontSize(fontsize)
  1834. },
  1835. // 更改离地高度
  1836. changeLineHeight(height){//线长
  1837. tag.changeLineLen(height)
  1838. },
  1839. changeCanMove(canMove){
  1840. //console.log('changeCanMove', tag.title, canMove)
  1841. tag.dragEnable = canMove
  1842. },
  1843. getImageCenter(){ //热点在模型的本地坐标
  1844. if(!tag.parent)return new THREE.Vector3
  1845. tag.titleLabel.sprite.update()
  1846. let pos = tag.onMesh ? tag.position : tag.titleLabel.parent.position.clone().applyMatrix4(tag.matrixWorld).applyMatrix4(tag.root.matrixWorld.clone().invert())
  1847. //console.log(props.title, 'getImageCenter', pos.toArray(), tag.lineLength)
  1848. return pos
  1849. },
  1850. getCameraDisSquared(){//距离intersect的位置
  1851. return viewer.mainViewport.camera.position.distanceToSquared(tag.getWorldPosition(new THREE.Vector3)) /* < tag.farSquared */
  1852. },
  1853. destroy(){
  1854. tag.dispose()
  1855. },
  1856. }
  1857. tag.functions.changeImage(props.image)
  1858. /*
  1859. tag.functions.changeType(props.type)
  1860. */
  1861. return tag.functions
  1862. },
  1863. showGrid() {
  1864. Potree.Utils.updateVisible(viewer.modules.MergeEditor.ground, 'hideGrid', true)
  1865. },
  1866. hideGrid() {
  1867. Potree.Utils.updateVisible(viewer.modules.MergeEditor.ground, 'hideGrid', false)
  1868. },
  1869. exitWatchMonitor(){
  1870. viewer.scene.monitors.find(e=>e.isWatching).leave()
  1871. }
  1872. }
  1873. function spliceFromArr(model, props, loaded){
  1874. //let autoLoads.find()
  1875. props.loadFinish = true
  1876. props.loading = false
  1877. if (loaded) {
  1878. props.loaded = true
  1879. props.model = model
  1880. } else {
  1881. props.error = true
  1882. }
  1883. /* let haventLoad = autoLoads.filter(e=>!e.loading && !e.loadFinish);
  1884. if( haventLoad[0]){
  1885. startLoad(haventLoad[0])
  1886. */
  1887. if (!loadNext()) {
  1888. if (autoLoads.filter(e => !e.loadFinish).length == 0 && autoLoads.filter(e => e.loaded).length > 0 && !props.isFirstLoad) {//设置相机位置:当自动开始加载第一个模型时(其余的也跟着自动加载),等这批加载完后;
  1889. let autoLoadsDone = autoLoads.filter(e => e.loaded).map(e => e.model)
  1890. autoLoads.filter(e => e.loaded && e.show).forEach(e => e.model.visible = true)
  1891. if(autoLoads.length > 1){
  1892. let loadTimeCost = Date.now() - loadStartTime
  1893. console.log('所有模型加载完毕, 耗时', parseInt(loadTimeCost) )
  1894. }
  1895. loadStartTime = Date.now()
  1896. if(!props.dontFocus){
  1897. MergeEditor.focusOn(autoLoadsDone, 1000, true, true)
  1898. }
  1899. autoLoads.length = 0
  1900. }
  1901. }
  1902. }
  1903. function loadNext(){
  1904. let haventLoad = autoLoads.filter(e => !e.loading && !e.loadFinish);
  1905. let loading = autoLoads.filter(e => e.loading);
  1906. let needLoad = haventLoad.slice(0, maxLoadingCount - loading.length)
  1907. needLoad.forEach(e => startLoad(e))
  1908. return haventLoad.length > 0
  1909. }
  1910. function startLoad(prop){
  1911. /* if(prop.raw.visible !== 1){//用于临时隐藏
  1912. setTimeout(()=>{
  1913. spliceFromArr(null, prop, false)
  1914. prop.bus.emit('loadError' )
  1915. },1)
  1916. return
  1917. } */
  1918. if(prop.loading || prop.loadFinish)return
  1919. Potree.Log(`--开始加载--`, { font: { color: '#f68' } });
  1920. console.log('id:', prop.id, ', title:', prop.title, ', filename:', Potree.Common.getNameFromURL(prop.url), ', type:', prop.type, prop)
  1921. prop.unlit = prop.renderType != 'normal'
  1922. prop.maximumScreenSpaceError = 70
  1923. prop.prefix = prop.raw.prefix
  1924. /* laserRoot != void 0 && (prop.prefix = laserRoot) //prefix for getdataset
  1925. //Potree.settings.urls.prefix = prop.prefix = '' */
  1926. Potree.addModel(prop, prop.done, prop.progressFun, prop.onError)
  1927. prop.loading = true
  1928. }
  1929. function buildMapFromProp(){
  1930. cesiumViewer.imageryLayers.removeAll();
  1931. mapProps.forEach(e=>{
  1932. let gaoDeImageryProvider = new Cesium.UrlTemplateImageryProvider({
  1933. url:e.url,
  1934. minimumLevel: 0,
  1935. maximumLevel: e.maximumLevel,
  1936. credit: new Cesium.Credit(e.name),
  1937. tilingScheme: new AmapMercatorTilingScheme(), //修改投影,从84->高德
  1938. crossOrigin: 'anonymous',
  1939. })
  1940. cesiumViewer.imageryLayers.addImageryProvider(gaoDeImageryProvider);
  1941. })
  1942. }
  1943. function buildMap(){
  1944. if (Potree.settings.showCesium && !window.cesiumViewer) {
  1945. viewer.backgroundOpacity = 0
  1946. //密钥
  1947. Cesium.Ion.defaultAccessToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJqdGkiOiI2ZGM2YzY0ZC1kNWE0LTRiYTgtYTkwNS1kYmJiODRjMWUwMmQiLCJpZCI6MjMzMTQ1LCJpYXQiOjE3MjI5OTUwNTB9.niqpkl6xOkQ2KeJjelyDDDydmSGqKXKb5cX2NyxSNAw'
  1948. window.cesiumViewer = new Cesium.Viewer('app', {
  1949. useDefaultRenderLoop: true,
  1950. requestRenderMode: true, //add 只有需要render时才会render,如tile加载完后、镜头移动后
  1951. animation: false,
  1952. baseLayerPicker: false,
  1953. fullscreenButton: false,
  1954. geocoder: false,
  1955. homeButton: false,
  1956. infoBox: false,
  1957. sceneModePicker: false,
  1958. selectionIndicator: false,
  1959. timeline: false,
  1960. navigationHelpButton: false,
  1961. //高德秘钥版 imageryProvider: new Cesium.AmapImageryProvider({key, mapStyle: 'normal'})
  1962. //报错 401 (Unauthorized) 的方法 https://blog.csdn.net/LBY_XK/article/details/121992641
  1963. //terrainShadows: Cesium.ShadowMode.DISABLED, //terrain地形 //自带的地图直接用84坐标
  1964. });
  1965. buildMapFromProp()
  1966. Potree.cesScreenshot = (w,h)=>{
  1967. console.log('cesScreenshot',w,h)
  1968. cesiumViewer.scene.canvas.style.width = w+'px'
  1969. cesiumViewer.scene.canvas.style.height = h+'px'
  1970. cesiumViewer.scene.canvas.style.visibility = 'hidden'
  1971. cesiumViewer.resize()
  1972. cesAspect = w/h
  1973. let deferred = $.Deferred();
  1974. updateMap(w/h)//hfov可能改变了需要update。
  1975. setTimeout(()=>{ //延迟是似乎还要做别的处理,否则立即截图的话可能得到绿色底图(俯视状态容易触发)
  1976. let oldMode = window.cesiumViewer._cesiumWidget._scene.requestRenderMode
  1977. window.cesiumViewer._cesiumWidget._scene.requestRenderMode = 0 //强制render,否则会黑屏
  1978. cesiumViewer.render();
  1979. let dataUrl = window.cesiumViewer.scene.canvas.toDataURL('image/png')
  1980. window.cesiumViewer._cesiumWidget._scene.requestRenderMode = oldMode
  1981. //Potree.Common.downloadFile(dataUrl, 'screenshot.png')
  1982. cesAspect = null
  1983. cesiumViewer.scene.canvas.style.width = ''
  1984. cesiumViewer.scene.canvas.style.height = ''
  1985. cesiumViewer.scene.canvas.style.visibility = ''
  1986. deferred.resolve(dataUrl)
  1987. },200) //时间短了容易黑屏
  1988. return deferred.promise()
  1989. }
  1990. }else{
  1991. buildMapFromProp()
  1992. }
  1993. updateMap()
  1994. }
  1995. function updateCamNear(type){// 有的漫游场景模型缩放的很小(0.1%),需要缩小near才能看见, 但会造成z-fighting, 离远了看大模型会闪烁
  1996. const min = 0.0001, max = 0.1
  1997. let near , bigScale = 0.2
  1998. if(Potree.settings.displayMode == 'showPanos'/* && type != 'cameraMove' */){
  1999. let currentModel = viewer.images360.currentPano.pointcloud
  2000. near = Potree.math.linearClamp(currentModel.scale.x, [0, 1], [min, max])
  2001. }else/* if(type == 'cameraMove') */{
  2002. //没有完美的解决方式,优先考虑选中的模型。否则优先考虑离得近的。 另外尽量用大的,因为缩很小的情况很少。
  2003. //虽然案例说应该看model.bound.size,但如果非场景模型,缩很小的话也不需要凑近看。
  2004. let allModels = viewer.objs.children.concat(viewer.scene.pointclouds)
  2005. if(allModels.length == 0)return
  2006. allModels.sort((a,b)=>{return a.scale.x - b.scale.x})
  2007. let minS = allModels[0].scale.x, maxS = allModels.pop().scale.x
  2008. let considerModel
  2009. if(minS>bigScale) near = max
  2010. else{
  2011. if(MergeEditor.selected){
  2012. considerModel = MergeEditor.selected
  2013. near = Potree.math.linearClamp(considerModel.scale.x, [0, bigScale], [min, max])
  2014. }else{
  2015. //allModels = allModels.filter() //写不下去了好难,就算了吧, 折中
  2016. near = Potree.math.linearClamp(minS, [0, bigScale], [max/4, max])
  2017. }
  2018. }
  2019. }
  2020. if(near != viewer.mainViewport.camera.near){
  2021. //console.log('updateNear',near)
  2022. viewer.mainViewport.camera.near = near
  2023. viewer.mainViewport.camera.updateProjectionMatrix()
  2024. viewer.dispatchEvent('content_changed')
  2025. }
  2026. }
  2027. function updateCamFar(){
  2028. let expand = 1.1 //for label
  2029. Potree.settings.cameraFar = THREE.Math.clamp((viewer.bound.boundingBox.distanceToPoint(viewer.mainViewport.camera.position)+viewer.bound.boundSize.length() ) * expand , 10000, 100000000000)
  2030. }
  2031. function updateMap(){
  2032. if (Potree.settings.showCesium && Potree.settings.displayMode == 'showPointCloud') {
  2033. let camera = MergeEditor.split ? viewer.viewports.find(e=>e.name == 'top').camera : viewer.mainViewport.camera
  2034. let pPos = new THREE.Vector3(0, 0, 0).applyMatrix4(camera.matrixWorld);
  2035. let orientation
  2036. /* let toCes = (pos) => {
  2037. let xy = [pos.x, pos.y];
  2038. let height = pos.z;
  2039. let deg = viewer.transform.lonlatToLocal.inverse(xy) // toMap.forward(xy);
  2040. let cPos = Cesium.Cartesian3.fromDegrees(...deg, height);
  2041. //console.log('toCes',cPos,height) //数字过大如e35会崩溃
  2042. return cPos;
  2043. }; */
  2044. let cPos = Potree.math.toCes(pPos);
  2045. if(MergeEditor.split){
  2046. orientation = {
  2047. heading: Cesium.Math.toRadians(0.0), // 方向角
  2048. pitch: Cesium.Math.toRadians(-90.0), // 俯仰角
  2049. roll: 0.0 // 翻滚角
  2050. }
  2051. if(!cesiumViewer.camera.perpFrustum_){
  2052. cesiumViewer.camera.perpFrustum_ = cesiumViewer.camera.frustum
  2053. cesiumViewer.camera.frustum = new Cesium.OrthographicOffCenterFrustum({//OrthographicFrustum OrthographicOffCenterFrustum
  2054. left: -10000, // 左边界
  2055. right: 10000, // 右边界
  2056. bottom: -10000, // 下边界
  2057. top: 10000, // 上边界
  2058. near: 1.0, // 近裁剪面距离
  2059. far: 100000000.0, // 远裁剪面距离
  2060. })
  2061. }
  2062. cesiumViewer.camera.frustum.left = camera.left / camera.zoom
  2063. cesiumViewer.camera.frustum.right = camera.right / camera.zoom
  2064. cesiumViewer.camera.frustum.top = camera.top / camera.zoom
  2065. cesiumViewer.camera.frustum.bottom = camera.bottom / camera.zoom
  2066. }else{
  2067. cesiumViewer.camera.perpFrustum_ && (cesiumViewer.camera.frustum = cesiumViewer.camera.perpFrustum_, cesiumViewer.camera.perpFrustum_ = null) //恢复
  2068. //let pRight = new THREE.Vector3(600, 0, 0).applyMatrix4(camera.matrixWorld);
  2069. let pUp = new THREE.Vector3(0, 600, 0).applyMatrix4(camera.matrixWorld);
  2070. let pTarget = viewer.scene.view.getPivot();
  2071. let cUpTarget = Potree.math.toCes(pUp);
  2072. let cTarget = Potree.math.toCes(pTarget);
  2073. let cDir = Cesium.Cartesian3.subtract(cTarget, cPos, new Cesium.Cartesian3());
  2074. let cUp = Cesium.Cartesian3.subtract(cUpTarget, cPos, new Cesium.Cartesian3());
  2075. cDir = Cesium.Cartesian3.normalize(cDir, new Cesium.Cartesian3());
  2076. cUp = Cesium.Cartesian3.normalize(cUp, new Cesium.Cartesian3());
  2077. //console.log('ces', 'cPos', cPos, 'cDir',cDir, 'cUp', cUp)
  2078. orientation = {
  2079. direction: cDir,
  2080. up: cUp
  2081. }
  2082. let aspect = cesAspect || camera.aspect;
  2083. //console.log('updateMap', aspect)
  2084. if (aspect < 1) {
  2085. let fovy = Math.PI * (viewer.scene.getActiveCamera().fov / 180);
  2086. cesiumViewer.camera.frustum.fov = fovy;
  2087. } else {
  2088. let fovy = Math.PI * (viewer.scene.getActiveCamera().fov / 180);
  2089. let fovx = Math.atan(Math.tan(0.5 * fovy) * aspect) * 2
  2090. cesiumViewer.camera.frustum.fov = fovx;
  2091. }
  2092. }
  2093. cesiumViewer.camera.setView({
  2094. destination: cPos,
  2095. orientation
  2096. });
  2097. cesiumViewer.scene.globe.show = camera.position.z > 0 //在地面之下地球会闪烁,故隐藏
  2098. cesiumViewer.render(); //立即render,否则会和点云render不同步而错位
  2099. }//cesium测试沙盒 https://sandcastle.cesium.com/
  2100. }
  2101. class AmapMercatorTilingScheme extends Cesium.WebMercatorTilingScheme {
  2102. constructor(options) {
  2103. super(options)
  2104. let projection = new Cesium.WebMercatorProjection()
  2105. this._projection.project = function(cartographic, result) {
  2106. //WGS84转GCJ02坐标
  2107. /* result = gcoord.transform([
  2108. Cesium.Math.toDegrees(cartographic.longitude),
  2109. Cesium.Math.toDegrees(cartographic.latitude)
  2110. ], gcoord.WGS84, gcoord.GCJ02) */
  2111. result = AMapWith84__.wgs84ToAMap({
  2112. x: Cesium.Math.toDegrees(cartographic.longitude),
  2113. y: Cesium.Math.toDegrees(cartographic.latitude)
  2114. })
  2115. result = projection.project(new Cesium.Cartographic(Cesium.Math.toRadians(result.x),Cesium.Math.toRadians(result.y)))
  2116. return new Cesium.Cartesian2(result.x,result.y)
  2117. }
  2118. this._projection.unproject = function(cartesian, result) {
  2119. let cartographic = projection.unproject(cartesian)
  2120. //GCJ02转WGS84坐标
  2121. /* result = gcoord.transform([
  2122. Cesium.Math.toDegrees(cartographic.longitude),
  2123. Cesium.Math.toDegrees(cartographic.latitude)
  2124. ], gcoord.GCJ02, gcoord.WGS84) */
  2125. result = AMapWith84__.aMapToWgs84({
  2126. x: Cesium.Math.toDegrees(cartographic.longitude),
  2127. y: Cesium.Math.toDegrees(cartographic.latitude)
  2128. })
  2129. return new Cesium.Cartographic(Cesium.Math.toRadians(result.x),Cesium.Math.toRadians(result.y))
  2130. }
  2131. }
  2132. }//see : https://blog.csdn.net/hongxianqiang/article/details/140541555 cesium加载高德地图并纠偏
  2133. return sdk
  2134. }
  2135. function load4dkkMedias(model){//加载四维看看的一些媒体物品
  2136. if(model.isPointcloud)return
  2137. let {sceneJsonPath,surveillancePath} = model.props.raw
  2138. console.log(sceneJsonPath,surveillancePath)
  2139. if(sceneJsonPath){//box图片视频 /oss/scene_view_data/SG-jm-Xwq0FwSkFy4/data/scene.json
  2140. Potree.loadFile(sceneJsonPath+ '?m='+Date.now(),null,(json)=>{
  2141. //console.log(model.name, 'sceneJson', json, json.surveillances)
  2142. if(json.surveillances){//监控
  2143. Potree.loadFile(surveillancePath + '?m='+Date.now(),null,(monitorJson)=>{
  2144. //console.log(model.name, 'surveillance', monitorJson)
  2145. monitorJson.forEach((e)=>{
  2146. e.showTitle = json.controls.showCameraTitle
  2147. e.showScope = json.controls.showSurveilScope
  2148. viewer.addMonitor(e,model)
  2149. })
  2150. })
  2151. }
  2152. if(json.boxPhotos || json.boxVideos){
  2153. let boxPhotos = json.boxPhotos ? JSON.parse(json.boxPhotos) : []
  2154. let boxVideos = json.boxVideos ? JSON.parse(json.boxVideos) : []
  2155. let medias = boxPhotos.concat(boxVideos)
  2156. medias.forEach(prop=>{
  2157. //Potree.addModel(prop, prop.done, prop.progressFun, prop.onError)
  2158. prop.type = 'media'
  2159. prop.model = model
  2160. prop.position = Potree.Utils.tran4dkkVecInModel(new THREE.Vector3().fromArray(prop.pos), model);
  2161. let qua = new THREE.Quaternion().fromArray(prop.qua).normalize()
  2162. if(model.props.baseRotation?.x == 0){ //该模型已经矫正,无需旋转90度,但是场景的数据需要,导致monitor和模型不匹配,需要再旋转
  2163. qua.copy(Potree.math.convertQuaternion.YupToZup(qua))
  2164. }
  2165. prop.rotation = new THREE.Euler().setFromQuaternion(qua)
  2166. //prop.url = location.origin + '/oss/scene_view_data/'+ model.props.raw.num + '/user/'+prop.url,
  2167. prop.url = sceneJsonPath.replace('data/scene.json','user/'+prop.url) //offline different
  2168. if(!prop.url.substr(0,5).includes('http')) prop.url = location.origin + prop.url
  2169. prop.id = prop.sid
  2170. //Potree.settings.urls.getPrefix(8,model)
  2171. //isNew:true, //是否新创建而非加载
  2172. Potree.addModel(prop,(overlay)=>{
  2173. //overlay.scale.set(10,10,10)
  2174. })
  2175. })
  2176. }
  2177. })
  2178. }
  2179. //cutModelPath裁剪模型路径
  2180. }
  2181. /*
  2182. 暂定不同场景间的漫游点不能互通。虽然它们可能是摆放正确的,如果是组成一整个场景的话还是要打通……
  2183. 不互通的方法是设置pano.enable
  2184. 现在需要互通了。但是还需要设置neibgbours, 有点麻烦,暂时没写。
  2185. */
  2186. export default enter