index.js 114 KB

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