1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348 |
- import mitt from 'mitt'
- import axios from 'axios' //{ axios } from '@/api'
- let requestLoadCount = 0
- export const enter = ({ dom, mapDom, isLocal, lonlat, scenes }) => {
- console.warn('新的页面')
- Potree.settings.isOfficial = true //标记为正式、非测试版本
- //Potree.fileServer = axios
- Potree.settings.libsUrl = './lib/'
- //正式环境(本地调试会打不开)
- if (location.host === 'mix3d.4dkankan.com') {
- Potree.settings.urls.prefix = Potree.settings.urls.prefix6
- Potree.settings.webSite = 'datav1'
- } else if (location.host === 'xfhd.4dkankan.com') {
- Potree.settings.urls.prefix = Potree.settings.urls.prefix7
- Potree.settings.webSite = 'datav1'
- }
- const mapBus = mitt(), sceneBus = mitt()
- const tagLimitDis = 8;
- Potree.settings.showCompass = true
- Potree.settings.compassDom = dom.querySelector('#direction')
- Potree.settings.showObjectsOnMap = true
- Potree.settings.mergeType2 = true //标识新版
- Potree.settings.modelSkybox = true //是否将全景图贴在模型上(会导致卡顿)。若不显示模型将不显示Reticule
- Potree.settings.tiles3DMaxMemory = 300 //稍微增加点
- //Potree.settings.mergeTransCtlOnClick = true
- Potree.settings.canWalkThroughModel = true
-
- let { THREE } = Potree.mergeEditStart(dom, mapDom)
- let MergeEditor = viewer.modules.MergeEditor
- Potree.settings.unableNavigate = false
-
- Potree.settings.showCesium = true
- if (Potree.settings.showCesium) {
- viewer.backgroundOpacity = 0
- Cesium.Ion.defaultAccessToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJqdGkiOiI2ZGM2YzY0ZC1kNWE0LTRiYTgtYTkwNS1kYmJiODRjMWUwMmQiLCJpZCI6MjMzMTQ1LCJpYXQiOjE3MjI5OTUwNTB9.niqpkl6xOkQ2KeJjelyDDDydmSGqKXKb5cX2NyxSNAw'
- window.cesiumViewer = new Cesium.Viewer('app', {
- useDefaultRenderLoop: true,
- requestRenderMode: true, //add 只有需要render时才会render,如tile加载完后、镜头移动后
- animation: false,
- baseLayerPicker: false,
- fullscreenButton: false,
- geocoder: false,
- homeButton: false,
- infoBox: false,
- sceneModePicker: false,
- selectionIndicator: false,
- timeline: false,
- navigationHelpButton: false,
- //imageryProvider : Cesium.createOpenStreetMapImageryProvider({url : 'https://a.tile.openstreetmap.org/'}),
- imageryProvider: Cesium.UrlTemplateImageryProvider({ //直接用84坐标,不用转高德
- //"https://wprd04.is.autonavi.com/appmaptile?lang=zh_cn&style=7&yrs=m&x=${x}&y=${y}&z=${z}" //
- //url : 'https://webst0{0-7}.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=7&x={x}&y={y}&z={z}&token=YOUR_API_KEY',
- url: 'https://wprd04.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=7&x={x}&y={y}&z={z}&token=YOUR_API_KEY',
- minimumLevel: 0,
- maximumLevel: 19
- }),
- //高德秘钥版 imageryProvider: new Cesium.AmapImageryProvider({key, mapStyle: 'normal'})
- //报错 401 (Unauthorized) 的方法 https://blog.csdn.net/LBY_XK/article/details/121992641
- terrainShadows: Cesium.ShadowMode.DISABLED, //terrain地形
- });
-
- //lonlat = [113.595236803415,22.3665168584444]//[113.600356,22.364093]
- Potree.setLonlat(lonlat[0], lonlat[1])
- }
- if(Potree.settings.canWalkThroughModel){
- let lastPano
- viewer.images360.addEventListener('flyToPano',(e)=>{//漫游到另一个模型就要选中这个模型
- let model = e.toPano.pano.pointcloud
- if(lastPano?.pointcloud != model && model != MergeEditor.selected){
- MergeEditor.selectModel(model)
- model.result_.flyInPano(e.toPano.pano, {dontFly:true}) //切换模型显示,因为flyInPano有事件怕乱所以统一用这个函数
- }
- lastPano = e.toPano.pano
- })
-
- }
-
- viewer.addEventListener('camera_changed', e => {
- var camera = e.viewport.camera
- var pos = camera.position
- if (e.viewport.name == 'MainView') {
- sceneBus.emit('cameraChange', { x: pos.x, y: pos.y, z: pos.z, rotate: camera.rotation })
- if (Potree.settings.showCesium && Potree.settings.displayMode == 'showPointCloud') {
- let pPos = new THREE.Vector3(0, 0, 0).applyMatrix4(camera.matrixWorld);
- let pRight = new THREE.Vector3(600, 0, 0).applyMatrix4(camera.matrixWorld);
- let pUp = new THREE.Vector3(0, 600, 0).applyMatrix4(camera.matrixWorld);
- let pTarget = viewer.scene.view.getPivot();
- let toCes = (pos) => {
- let xy = [pos.x, pos.y];
- let height = pos.z;
- let deg = viewer.transform.lonlatToLocal.inverse(xy) // toMap.forward(xy);
- let cPos = Cesium.Cartesian3.fromDegrees(...deg, height);
- return cPos;
- };
- let cPos = toCes(pPos);
- let cUpTarget = toCes(pUp);
- let cTarget = toCes(pTarget);
- let cDir = Cesium.Cartesian3.subtract(cTarget, cPos, new Cesium.Cartesian3());
- let cUp = Cesium.Cartesian3.subtract(cUpTarget, cPos, new Cesium.Cartesian3());
- cDir = Cesium.Cartesian3.normalize(cDir, new Cesium.Cartesian3());
- cUp = Cesium.Cartesian3.normalize(cUp, new Cesium.Cartesian3());
- cesiumViewer.camera.setView({
- destination: cPos,
- orientation: {
- direction: cDir,
- up: cUp
- }
- });
- let aspect = viewer.scene.getActiveCamera().aspect;
- if (aspect < 1) {
- let fovy = Math.PI * (viewer.scene.getActiveCamera().fov / 180);
- cesiumViewer.camera.frustum.fov = fovy;
- } else {
- let fovy = Math.PI * (viewer.scene.getActiveCamera().fov / 180);
- let fovx = Math.atan(Math.tan(0.5 * fovy) * aspect) * 2
- cesiumViewer.camera.frustum.fov = fovx;
- }
- cesiumViewer.render(); //立即render,否则会和点云render不同步而错位
- }
- }
- })
- viewer.inputHandler.addEventListener('keydown', (e)=>{
- if(e.event.key == "e" ){
- MergeEditor.transformControls.mode = 'rotate'
- }else if(e.event.key == "w"){
- MergeEditor.transformControls.mode = 'translate'
- }else if(e.event.key == "s"){
- MergeEditor.transformControls.mode = 'scale'
- }
- })
- viewer.addEventListener('webglError', e => {
- console.error('viewer webglError: ' + e)
- sceneBus.emit('webglError', { msg: e.msg })
- })
- viewer.compass.setAutoDisplay(true)
- /* mapBus.on('visible', v => {
- //console.log('mapBus visible', v)
- viewer.mapViewer.visible = v
- if (v) {
- viewer.mapViewer.mapLayer.needUpdate = true
- }
- viewer.mapViewer.dispatchEvent({type:'forceVisible',visible:v})
- }) */
- {
- let index = 1;
- //let setDisplay()
- if (!Potree.isIframeChild) {
- /* viewer.addEventListener('createIframe',(e)=>{//创建了子页面
- }) */
- window.winIndex = 0;
- window.iframeCreated = function (iframe) {
- let child = iframe.contentWindow
- child.winIndex = index++
- //案件里视图提取页面子页面覆盖了父级页面,父级的模型可以隐藏以释放内存
- console.error('createdIframe', child.winIndex, child.location.href)
- viewer.setDisplay(false)
- child.beforeDestroy = function () { //注:在前端仍会找不到beforeDestroy,可能contentWindow变更??所以手动调用setDisplay
- console.warn('beforeDestroy', child.winIndex)
- child.viewer && child.viewer.setDisplay(false)
- //如果是四维看看的场景,先不管了,页面被销毁应该就没了吧
- viewer.setDisplay(true)//恢复主页的模型显示
- if (!child.viewer) {
- try {
- let player = child.__sdk.core.get('Player')
- /* let runtime = player.model._3dTilesRuntime
- let tileset = runtime.getTileset()
- tileset._cache.trim(); //使下一次update时dispose所有不可见的tiles
- let sceneRenderer = child.__sdk.core.get('SceneRenderer')
- player.model.visible = false
- runtime.update(16, sceneRenderer.renderer, sceneRenderer.camera, true) //没用,为何_trimTiles的while无法进入
- */
- player.model.traverse(e => {
- e.geometry && e.geometry.dispose()
- if (e.material) {
- e.material.map && e.material.map.dispose()
- if (e.material.uniforms && e.material.uniforms.map && e.material.uniforms.map.value) {
- e.material.uniforms.map.value.dispose()
- }
- }
- }) //效果甚微
- /* let sceneRenderer = child.__sdk.core.get('SceneRenderer')
- sceneRenderer.renderer.render(sceneRenderer.scene, sceneRenderer.camera)
- */
- } catch (e) {
- console.log(e)
- }
- }
- }
- }
- //不知道删除iframe时是否那些模型还在内存里,需要释放吗? 如果要需要加一个事件
- } else {
- }
- }
- window.THREE = THREE
- //isLocal = false
- let autoLoads = /* window.autoLoads = */ []
- let readyToAddModel
- let maxLoadingCount = /* isLocal ? 1 : */2; //正在加载模型的最大数目
- let mainBackground = viewer.background
- const units = { 1: 'metric', 2: 'imperial' }
- let getMeasureType = function (type, unit = 1) {
- let info
- switch (type) {
- case 'free':
- info = { measureType: 'Distance' }
- break
- case 'area':
- info = { measureType: 'Area' }
- break
- case 'vertical':
- info = { measureType: 'Ver Distance' }
- break
- default:
- console.error('无此 measure type')
- }
- info.unit = units[unit]
- return info
- }
- let getMeasureFunction = function (measure, bus) {
- measure.addEventListener('highlight', (e) => {
- //console.log('3d->2d highlight',e.state)
- bus.emit('highlight', e.state)
- })
- measure.addEventListener('marker_dropped', (e) => {//拖拽结束后发送changeCallBack
- if (measure.parent) {
- //未被删除
- bus.emit('update', [
- measure.dataset_points.map(p => p.clone()),
- measure.points_datasets
- ])
- }
- })
- return {
- /* quit: () => {
- Potree.Log('quit结束且删除: ' + measure.id, '#00c7b2')
- viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure })
- }, //触发结束。退出测量模式,清除之前操作 */
- destroy: () => {
- viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure })
- viewer.scene.removeMeasurement(measure)
- },
- /* getPoints: () => {
- return measure.points
- },
- getDatasetLocations: () => {
- return measure.dataset_points
- },
- getDatasets: () => {
- return measure.points_datasets
- },
- getDatasetId: () => {
- return measure.datasetId
- }, */
- getArea: () => {
- return measure.area //{value:area, string:..}
- },
- getDistance: () => {
- if (measure.points.length < 2) return 0
- var value = measure.points[0].distanceTo(measure.points[1])
- return {
- value, //米
- string: measure.getConvertString(value, 'distance')
- }
- },
- //手动开启或关闭:
- show: () => {
- Potree.Utils.updateVisible(measure, 'inListByUser', true)
- },
- hide: () => {
- Potree.Utils.updateVisible(measure, 'inListByUser', false)
- },
- fly() {
- let result = viewer.focusOnObject(measure, 'measure', 1200)
- return result.msg ? result.msg : result.promise
- //返回值 1 deferred 表示即将位移 2 'posNoChange' 表示已在最佳位置 3 'tooFar' 表示距离最佳位置太远
- },
- changeSelect(isHight) {
- console.log('2d->3d isHight ', isHight)
- measure.setSelected(isHight, 'byList')
- },
- }
- }
- let camera_changed = (e) => {
- if (e.viewport.name == 'MainView' && e.changeInfo.positionChanged) {
- //viewer.mainViewport.camera.position
- viewer.mainViewport.view.radius = 0.1 //使pivot在面前一丢丢距离
- viewer.setControls(viewer.orbitControls)
- viewer.removeEventListener('camera_changed', camera_changed)
- }
- }
- let requestInPano = false
- let sdk = {
- sceneBus, mapBus,
-
-
- canTurnToPanoMode(pos) {
- pos = pos ? new THREE.Vector3().copy(pos) : viewer.images360.position
- let pano = viewer.images360.findNearestPano(pos)
- if (pano && pano.position.distanceTo(pos) < Potree.config.panoFieldRadius) {
- return {model:pano.pointcloud.result_}
- }
- //poschange后会调用这个,如果返回false会变为点云模式,且不会自动变回原先的模式
- },
-
-
-
- getPositionByScreen(pos2d, hopeModelId) {//通过屏幕坐标获取真实坐标 . hopeModelId: 如果指定了模型,优先返回hopeModelId上的intersect
- //console.log('getPositionByScreen',hopeModelId)
- hopeModelId = null
- let worldPos, localPos, modelId, intersect
- let Handler = viewer.inputHandler
- let reGet = () => {//不使用当前鼠标所在位置的intersect,单独算
- pos2d.clientX = pos2d.x
- pos2d.clientY = pos2d.y
- pos2d.onlyGetIntersect = true
- pos2d.whichPointcloud = true
- if (hopeModelId != void 0) {//隐藏其他的模型
- let models = MergeEditor.getAllObjects()
- models.forEach(model => {
- Potree.Utils.updateVisible(model, 'forPick', model.dataset_id == hopeModelId)
- })
- }
- let intersect2 = Handler.onMouseMove(pos2d)
- if (hopeModelId != void 0) {//恢复
- let models = MergeEditor.getAllObjects()
- models.forEach(model => {
- Potree.Utils.updateVisible(model, 'forPick', true)
- })
- }
- if (intersect2 && intersect2.location) {
- intersect = intersect2
- }
- }
- if (pos2d && pos2d.inDrag) {
- reGet()
- } else {
- intersect = Handler.intersect
- if (intersect) {
- modelId = intersect.pointcloud ? intersect.pointcloud.dataset_id : intersect.object.dataset_id
- if (hopeModelId != void 0 && modelId != hopeModelId) {
- reGet()
- }
- }
- }
- if (intersect && intersect.location) {
- modelId = intersect.pointcloud ? intersect.pointcloud.dataset_id : intersect.object.dataset_id
- /* if(hopeModelId != void 0 && modelId != hopeModelId){
- return null
- } */
- worldPos = intersect.location.clone()
- localPos = Potree.Utils.datasetPosTransform({ toDataset: true, datasetId: modelId, position: worldPos })
- } else return null
- return { worldPos, modelId, localPos }
- },
- getScreenByPosition(pos3d, modelId, canShelter/* , disToCameraLimit */) {//通过模型局部坐标获取屏幕坐标
- //console.log('getScreenByPoint ')
- let isLocal = modelId != void 0
- pos3d = new THREE.Vector3().copy(pos3d)
- let worldPos = isLocal ? Potree.Utils.datasetPosTransform({ fromDataset: true, datasetId: modelId, position: pos3d }) : pos3d
- if (!worldPos) return
- if (canShelter) {
- if (viewer.inputHandler.ifBlockedByIntersect(worldPos, 0.1, true)) return { trueSide: false };
- }
- var viewport = viewer.mainViewport
- var camera = viewport.camera
- var dom = viewer.renderArea
- if (tagLimitDis != void 0) {
- if (camera.position.distanceToSquared(worldPos) > Math.pow(tagLimitDis, 2)) return false
- }
- //console.log('getScreenByPoint ' + pos3d.toArray())
- return Potree.Utils.getPos2d(worldPos, viewport, dom)
- },
- setCameraFov(fov) {
- viewer.setFOV(fov)
- },
- screenshot: (width, height, bgOpacity = 1) => {
- //截图
- var { getImagePromise, finishPromise } = viewer.startScreenshot({ type: 'default', /* useRenderTarget:true, */bgOpacity }, width, height)
- var deferred = $.Deferred();
- finishPromise.done(({ dataUrl }) => {
- deferred.resolve(dataUrl)
- })
- return deferred.promise()
- },
- getPose() {//获取当前点位和朝向
-
- const camera = viewer.scene.getActiveCamera()
- const target = viewer.scene.view.getPivot().clone()
- const position = viewer.scene.view.position.clone()
- const pose = { position, target, displayMode:Potree.settings.displayMode }
- if(Potree.settings.displayMode == 'showPanos'){
- pose.panoId = viewer.images360.currentPano.originID
- pose.model = viewer.images360.currentPano.pointcloud.result_
-
- pose.posInModel = Potree.Utils.datasetPosTransform({ toDataset: true, position: camera.position.clone(), object:pose.model })
- pose.rotInModel = Potree.Utils.datasetRotTransform({ toDataset: true, quaternion: camera.position.clone(), getQuaternion: true, object:pose.model }) //拿第一个数据集
-
- }
-
- //console.log('getPose',position, target)
- return pose
- },
- comeTo(o = {}) {
- //console.log('comeTo',o.position, o.target)
- //飞到某个点
- /* if (o.modelId) {
- ['position', 'target'].forEach(e => {
- if (o[e]) {
- o[e] = Potree.Utils.datasetPosTransform({ fromDataset: true, datasetId: o.modelId, position: o[e] })
- }
- })
- } */
- let deferred = $.Deferred()
- if(o.displayMode == 'showPanos'){
- let pano = o.model.panos.find(a=>a.originID == e.panoId)
- if(pano){
- let quaternion = Potree.Utils.datasetRotTransform({ fromDataset: true, quaternion: o.rotInModel, getQuaternion: true, object:o.model })
- o.model.result_.flyInPano(pano, {quaternion, callback(){
- o.callback && o.callback()
- deferred.resolve(true)
- }})
- return deferred.promise()
-
- }else{
- console.warn('没有找到漫游点',o)
- }
- }
-
- if (o.distance) {
- let position = o.target || o.position
- return viewer.focusOnObject({ position }, 'tag', null, { distance: o.distance }).promise
- }
-
- viewer.scene.view.setView($.extend({}, o, {
- duration: o.dur,
- callback: () => {
- o.callback && o.callback()
- deferred.resolve(true)
- }
- }))
- return deferred.promise()
- },
- setBackdrop(sky, type, { scale, rotate }) {//天空盒背景
- //console.log('天空盒背景', sky,type)
- let setGroundAndText = (color) => {
- MergeEditor.secondCompass.dom.find(".dirText").css({ 'color': color })
- viewer.compass.dom.find(".dirText").css({ 'color': color })
- MergeEditor.ground.material.uniforms.uColor.value.set(color)
- //MergeEditor.ground.children[0].material.color.set(color)
- }
- if (type == 'bimg') {//地面图
- MergeEditor.setGroundPlaneImg(sky, scale, rotate)
- setGroundAndText('#e0e0e0')
- viewer.setBackground(mainBackground)
- } else {
- MergeEditor.setGroundPlaneImg(null)
- if (sky == 'none') {
- viewer.setBackground(mainBackground)
- setGroundAndText('#eee')
- } else if (sky[0] == '#') {
- viewer.setBackground(new THREE.Color(sky))
- let color = sky == '#fff' ? '#666' : sky == '#333' ? '#eee' : '#bbb' //反相
- setGroundAndText(color)
- } else if (type == 'image-map' || type == 'vector-map') {//影像|矢量 地图
- } else {//环境
- viewer.setBackground('skybox', sky)
- setGroundAndText('#e0e0e0')
- }
- }
- viewer.dispatchEvent('content_changed')
- },
- switchMapType(type) {
- let map = viewer.mapViewer.mapLayer.maps.find(e => e.name == 'map')
- map.switchStyle(type/* map.style == 'satellite' ? 'standard' : 'satellite' */)
- },
- enableMap(mapArea, latlng) {
- if (!viewer.mapViewer) {
- //--------------------------------
- viewer.mapViewer = new Potree.MapViewer(mapArea)
- viewer.mapViewer.initProjection()
- //focus
- let boundSize = new THREE.Vector3(200, 150, 1).max(viewer.bound.boundSize)
- viewer.mapViewer.addEventListener('viewerResize', () => {
- viewer.mapViewer.moveTo(viewer.bound.center, boundSize, 0)
- }, { once: true })
- }
- },
- enterSceneGuide(pathArr) {//导览 (不需要修改参数)
- let editor = viewer.modules.CamAniEditor
- console.log('pathArr', pathArr)
-
- //console.log('enterSceneGuide',pathArr)
- let data = {
- //duration: pathArr.slice(0, pathArr.length - 1).reduce(function (total, currentValue) { return total + currentValue.time }, 0), //总时长(要去掉最后一个,因为已到终点,该点time无意义)
- points: pathArr,
- useDurSlice: true
- }
-
-
- data.pathArr.forEach(e=>{
- if(e.inPano){
- e.model = e.model.model
- }
- })
- let ani = editor.createMulAnimation(data)
-
-
- //注:最多只存在一条导览
- let bus = mitt()
- //播放完成
- ani.event_.addEventListener('playDone', () => {
- bus.emit('playComplete')
- })
- //切换点
- ani.event_.addEventListener('updateCurrentIndex', e => {
- bus.emit('changePoint', e.currentIndex + 1)
- })
- return {
- bus,
- play() {
- MergeEditor.selectModel(null)
- ani.play()
- },
- pause() {
- ani.pause()
- },
- clear() {
- ani.remove()
- },
- }
- },
- //[path1, paht2], { time, speed }
- calcPathInfo(paths, info) { //传入的time, speed仅有一个。返回完整的 time, speed
- //这一版的control似乎无法在某个位置上改变角度,位置和角度一般都是一起变的,所以先不增加单位更换功能。
- let pos1 = new THREE.Vector3().copy(paths[0].position)
- let pos2 = new THREE.Vector3().copy(paths[1].position)
- let dis = pos1.distanceTo(pos2)
- if (info.time != void 0) {
- info.speed = dis / info.time
- } else {
- info.time = dis / info.speed
- }
- return info
- },
- addModel(props) {
- let bus = props.bus = mitt()
- //console.log('addModel',props)
- props.isFirstLoad = isLocal ? props.bottom == void 0 : (props.isDynamicAdded || props.mode == 'single') // 在编辑时用户添加的 或 展示单个模型 (props.mode='single'模型展示页, props.mode='many'融合页)
- if (props.opacity == void 0) props.opacity = 1
- if (props.type == 'obj') props.type = 'glb'
- props.scale /= 100
- if (props.rotation) {
- if (props.rotation._x == void 0 && props.rotation.x != void 0) {
- props.rotation = new THREE.Euler().setFromVector3(props.rotation)
- }
- }
- let getDefaultRotation = () => {
- //0看看,1看见,2深时,3用户上传三维模型,4深时mesh,5深光点云,6深光mesh
- if (props.fromType == 1 || props.fromType == 6 || props.fromType == 4 && props.type != 'glb') {//来自4dkk的3dtiles初始需要转90度
- return new THREE.Euler(Math.PI / 2, 0, 0)
- } else return new THREE.Euler(0, 0, 0)
- }
- if (!props.isFirstLoad) {
- if (autoLoads.length == 0) { //首次加载
- setTimeout(() => {
- let sizes = autoLoads.map(e => e.size || 0)
- console.log('需要请求加载的模型大小为', sizes, '总大小', sizes.reduce(function (total, currentValue) {
- let current = parseFloat(currentValue)
- return total + ((typeof currentValue == 'number' || currentValue.includes('M')) ? current : current / 1024)
- }, 0))
- readyToAddModel = true //准备开始加载
- loadNext()//startLoad(autoLoads[0])
- }, 30)
- }
- autoLoads.push(props)
- readyToAddModel = false
- } else {
- readyToAddModel = true
- props.rotation = getDefaultRotation()
- }
-
-
-
- let model
- let done = (model_) => {
- model = model_
- model.result_ = result
- if (!props.isFirstLoad) {
- model.visible = false//先不显示,防止卡顿
- }
- model.showInPano = props.raw.showInPano
- props.opacity < 100 && result.changeOpacity(props.opacity)
- model.addEventListener('changeSelect', (e) => {
- bus.emit('changeSelect', e.selected)
- })
- let lastState = {}
- model.addEventListener('transformChanged', (e) => {
- let msg = {}
- if (!lastState.position || !model.position.equals(lastState.position)) {
- lastState.position = msg.position = model.position.clone()
- }
- if (!lastState.rotation || !model.rotation.equals(lastState.rotation)) {
- lastState.rotation = msg.rotation = model.rotation.clone()
- }
- if (lastState.scale == void 0 || model.scale.x * 100 != lastState.scale) {
- lastState.scale = msg.scale = model.scale.x * 100
- }
- msg = Potree.Common.CloneObject(msg)
- //console.log(msg)
- bus.emit('transformChanged', msg)
- })
- spliceFromArr(model, props, true)
-
- /* model.addEventListener('changeSelect', (e) => {
- e.selected ? MergeEditor.transformControls.attach(model, e.clickPos) : MergeEditor.transformControls.detach()
- }) */
- if (props.mode == 'single') {//模型查看页
- MergeEditor.noNeedSelection = true
- setTimeout(() => {
- MergeEditor.focusOn([model], 1000, true, true)
- }, 1)
- }
- if (props.fromType == 6 || props.fromType == 4 || props.fromType == 0 || props.fromType == 1) {
- Potree.load4dkkPanos(props.raw.num, model, () => {
- bus.emit('loadDone')
- })
- } else {
- bus.emit('loadDone')
- }
- //console.log('loadDone' )
- }
- let progressFun = (progress) => {
- bus.emit('loadProgress', progress)
- }
- let onError = function (xhr) {
- bus.emit('loadError', xhr)
- console.log('loadError!!!!!!!!!', Potree.Common.getNameFromURL(props.url), props.size, xhr)
- spliceFromArr(model, props, false)
- }
- try {
- props.url = JSON.parse(props.url) //去掉 '\'
- } catch (e) { }
- props.done = done; props.progressFun = progressFun; props.onError = onError
- if (readyToAddModel) {
- if (autoLoads.filter(e => e.loading).length < maxLoadingCount) {
- startLoad(props)
- }
- }
- let scaleMeasure
- let result = {
- bus,
- model,
- getDefaultRotation,
- supportPano() { //是否支持全景图
- return model.panos && model.panos.length > 0
- },
- flyInPano(pano, {dontFly, quaternion}={}) {// 飞入全景图
- requestInPano = model
- pano = pano || viewer.images360.findNearestPano(null, model.panos)
- if (pano) {
- viewer.removeEventListener('camera_changed', camera_changed)
- Potree.settings.canWalkThroughModel || viewer.images360.panos.forEach(pano => {
- pano.setEnable(pano.pointcloud == model)
- })
- viewer.setControls(viewer.fpControls)
- //MergeEditor.transformControls.detach(model)
- Potree.Utils.updateVisible(MergeEditor.transformControls, 'showPanos', false)
- Potree.Utils.updateVisible(MergeEditor.boxHelper, 'showPanos', false)
- let changeVisi = (object) => {
- Potree.Utils.updateVisible(object, 'showPanos', object == model && Potree.settings.modelSkybox || object.showInPano)
- }
-
- let eventName = Potree.settings.displayMode != 'showPanos' ? 'endChangeMode' : 'flyToPanoDone'
- viewer.images360.addEventListener(eventName, (e) => {
- if (Potree.settings.displayMode == 'showPanos' && requestInPano == model) {
- //Potree.Utils.updateVisible(viewer.objs, 'showPanos', false)
- viewer.objs.children.forEach(changeVisi)
- }
- }, { once: true })
- dontFly || viewer.images360.flyToPano({ pano, canCancelLast: true, quaternion })
- Potree.settings.displayMode = 'showPanos'
-
- }
- },
- flyOutPano() {// 飞出全景图(就是切换到正常融合视角)
- requestInPano = false
- let panoPos = viewer.images360.position.clone()
- //Potree.Utils.updateVisible(viewer.objs, 'showPanos', true)
- let changeVisi = (object) => {
- Potree.Utils.updateVisible(object, 'showPanos', true)
- }
- viewer.objs.children.forEach(changeVisi)
- Potree.settings.canWalkThroughModel || viewer.images360.panos.forEach(pano => {
- pano.setEnable(true)
- })
- Potree.Utils.updateVisible(MergeEditor.transformControls, 'showPanos', true)
- setTimeout(() => {//在下一帧再变,因为3dtiles需要更新一下才会显示tiles
- if (!requestInPano) {
- Potree.settings.displayMode = 'showPointCloud'
- Potree.Utils.updateVisible(MergeEditor.boxHelper, 'showPanos', true)
- }
- }, 50)
- viewer.addEventListener('camera_changed', camera_changed)
- },
- changeShow(show) {
- props.show = show //for autoLoads show model
- if (model) {
- Potree.Utils.updateVisible(model, 'datasetSelection', show)
- if (model.panos) {
- model.panos.forEach(e => e.setEnable(show))
- }
- viewer.dispatchEvent('content_changed')
- }
- },
- changeSelect(state) {
- console.error('select', state)
- if (model) {
- let fly = viewer.images360.latestRequestMode != 'showPanos'
- MergeEditor.selectModel(model, state, fly, true)
- if (state && viewer.inputHandler.selection[0]) {
- MergeEditor.transformControls.attach(model) //viewer.transformObject(model); //交换
- }
- //console.log('changeSelect', props.id, state)
- }
- },
- changeScale(s) {
- if (model) {
- s /= 100
- if (model.scale.x == s) return
- //MergeEditor.history.beforeChange(model)//但不知道什么时候结束拖拽
- model.scale.set(s, s, s)
- model.isPointcloud && model.changePointSize(Potree.config.material.realPointSize * s)
- model.dispatchEvent("scale_changed")
- }
- },
- changeOpacity(opacity) { //见笔记:透明物体的材质设置
- if (opacity == void 0) opacity = 100
- opacity /= 100
- MergeEditor.changeOpacity(model, opacity)
- },
- changeBottom(z) {
- /* model && MergeEditor.setModelBtmHeight(model,z)
- model.dispatchEvent('transformChanged') //改了position */
- },
- changePosition(pos) {//校准取消时执行
- console.log('changePosition', pos.x, pos.y, pos.z)
- model && model.position.copy(pos)
- model.dispatchEvent({ type: 'position_changed' })
- },
- changeRotation(rot) {//校准取消时执行
- console.log('changeRotation', rot.x, rot.y, rot.z)
- model && model.rotation.setFromVector3(rot)
- model.dispatchEvent({ type: 'rotation_changed' })
- },
- enterRotateMode() {
- if (model) {
- if (MergeEditor.split) {//分屏校准
- MergeEditor.setTransformState('rotate')
- MergeEditor.transformControls2.attach(model)
- MergeEditor.transformControls2.mode = 'rotate'
- }
- MergeEditor.transformControls.attach(model)
- MergeEditor.transformControls.mode = 'rotate'
- }
- },
- enterMoveMode() {
- console.log('enterMoveMode')
- if (model) {
- if (MergeEditor.split) {//分屏校准
- MergeEditor.setTransformState('translate')
- MergeEditor.transformControls2.attach(model)
- MergeEditor.transformControls2.mode = 'translate'
- }
- MergeEditor.transformControls.attach(model)
- MergeEditor.transformControls.mode = 'translate'
- }
- },
- leaveTransform() {
- console.log('leaveTransform')
- if (MergeEditor.split) {//分屏校准
- MergeEditor.setTransformState(null)
- } else {
- MergeEditor.transformControls.detach()
- MergeEditor.transformControls2.detach()
- }
- MergeEditor.history.clear()
- },
- enterAlignment() {//开始校准
- result.leaveTransform()
- MergeEditor.enterSplit()
- //console.log('enterAlignment',model.position, model.rotation)
- let bus = new mitt()
- /* MergeEditor.transformControls.attach(model)
- MergeEditor.transformControls.mode = 'translate' */
- return {
- bus
- }
- },
- leaveAlignment() {
- //console.log('leaveAlignment',model.position, model.rotation)
- MergeEditor.leaveSplit()
- MergeEditor.transformControls.detach()
- MergeEditor.transformControls2.detach()
- },
- enterScaleSet() {//设置比例
- let bus = new mitt()
- let length, measureBuilded;
- viewer.outlinePass.selectedObjects = []
- if (!Potree.Utils.isInsideFrustum(model.boundingBox.clone().applyMatrix4(model.matrixWorld), viewer.scene.getActiveCamera())) {
- MergeEditor.focusOn(model, 600)
- }
- MergeEditor.getAllObjects().forEach(m => {//隐藏其他的模型
- if (m != model) Potree.Utils.updateVisible(m, 'enterScaleSet', false)
- })
- let setScale = () => {
- if (length == void 0 || !measureBuilded) return
- let vec = new THREE.Vector3().subVectors(viewer.mainViewport.camera.position, scaleMeasure.points[1])
- let s = length / (scaleMeasure.points[0].distanceTo(scaleMeasure.points[1]))
- result.changeScale(model.scale.x * s * 100)
- /* setTimeout(()=>{
- viewer.focusOnObject(scaleMeasure , 'measure', 500)
- },1) */
- let newCamPos = new THREE.Vector3().addVectors(scaleMeasure.points[1], vec.multiplyScalar(s))
- viewer.scene.view.setView({
- position: newCamPos, target: scaleMeasure.getCenter(), duration: 0, callback: () => {
- //更改target到measure中心的好处就是可以让相机绕measure中心转,坏处是每次更改都会变一下画面
- }
- })
- }
- return {
- bus,
- setLength(v) {
- if (!v) return
- length = v
- setScale()
- },
- startMeasure() {
- if (scaleMeasure) {
- viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure: scaleMeasure })
- viewer.scene.removeMeasurement(scaleMeasure)
- }
- measureBuilded = false
- scaleMeasure = viewer.measuringTool.startInsertion(
- { measureType: "Distance", unit: "metric" },
- () => {
- //done:
- //bus.emit('end' ) //完成
- measureBuilded = true
- setScale()
- },
- () => {
- //cancel
- //bus.emit('quit') //删除
- }
- )
- scaleMeasure.addEventListener('marker_dropped', (e) => {//拖拽结束后发送changeCallBack
- if (scaleMeasure.parent) {
- //未被删除
- measureBuilded && setScale()
- }
- })
- }
- }
- },
- leaveScaleSet() {
- if (scaleMeasure) {
- viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure: scaleMeasure })
- viewer.scene.removeMeasurement(scaleMeasure)
- scaleMeasure = null
- }
- viewer.outlinePass.selectedObjects = [model];
- MergeEditor.getAllObjects().forEach(m => {//恢复其他的模型
- if (m != model) Potree.Utils.updateVisible(m, 'enterScaleSet', true)
- })
- },
- destroy() {
- model && MergeEditor.removeModel(model)
- viewer.dispatchEvent('content_changed')
- }
- }
- return result
- },
- //测量线的点都附着于各个模型,当模型变化时,点跟着变化。
- // 新的测量创建方法,传入type 返回新测量对象
- startMeasure(type) {
- // 寻创建的测量对象有上面绘画测量对象的所有方法
- const bus = mitt()
- let info = getMeasureType(type)
- let measure = viewer.measuringTool.startInsertion(
- info,
- () => {
- //done:
- /* bus.emit('submit', {
- dataset_points: measure.dataset_points.map(p=>p.clone()) ,
- points_datasets: measure.points_datasets
- } ) //完成 */
- bus.emit('submit')
- bus.emit('update', [
- measure.dataset_points.map(p => p.clone()),
- measure.points_datasets
- ])
- },
- () => {
- //cancel
- bus.emit('cancel'/* , ret */) //删除
- }
- )
- Potree.Log('startMeasure: ' + measure.id, '#00c7b2')
- /* let cancel = ()=>{
- Potree.Log('clear删除: ' + measure.id, '#00c7b2')
- viewer.dispatchEvent({ type: 'cancel_insertions', remove: true, measure })
- viewer.scene.removeMeasurement(measure)
- } */
- let result = {
- bus,
- ...getMeasureFunction(measure, bus),
- }
- /* StartMeasure = Measure & {
- // 多了cancel 取消测量的事件,没有参数
- // 多了invalidPoint 当用户测量了无效点时的事件,抛出无效原因
- bus: Emitter<{ cancel: void; invalidPoint: string }>
- } */
- return result
- },
- // 绘画测量线(非新增使用)
- // type = 'free' (自由) || 'vertical' (垂直) || 'area' (面积)
- // positions 点数组 构成如下 [{ point: {x,y,z}, modelId: 1 }]
- drawMeasure(type, dataset_points, points_datasets) {
- // 返回测量对象有如下
- const bus = mitt()
- let info = getMeasureType(type /* , unit */)
- //info.points = positions
- info.dataset_points = dataset_points
- info.points_datasets = points_datasets
- //info.sid = sid
- info.bus = bus
- let measure = viewer.measuringTool.createMeasureFromData(info)
- if (!measure) return { bus }
- Potree.Log('drawMeasure由数据新建: ' + measure.id, '#00c7b2')
- let result = {
- bus,
- setPositions(dataset_points, points_datasets) {//用于恢复measure的点,不会修改点的个数
- measure.dataset_points = dataset_points.map(e => {
- return e && new THREE.Vector3().copy(e)
- })
- measure.points_datasets = points_datasets
- measure.points = measure.dataset_points.map((p, i) => {
- return Potree.Utils.datasetPosTransform({ fromDataset: true, datasetId: measure.points_datasets[i], position: p })
- })
- measure.getPoint2dInfo(measure.points)
- measure.update({ ifUpdateMarkers: true })
- measure.setSelected(false)//隐藏edgelabel
- },
- ...getMeasureFunction(measure, bus),
- }
- return result
- },
- addTag(info) {//加热点
- let bus = mitt()
- let tag
- let done = () => {
- bus.emit('added')
- bus.emit('update', { position: tag.position.clone(), normal: o.normal.clone(), modelId: tag.root.dataset_id })
- tag = tag_
- tag.spot.addEventListener('mouseover', () => {
- bus.emit('hoverState', true)
- })
- tag.spot.addEventListener('mouseout', () => {
- bus.emit('hoverState', false)
- })
- }
- if (!info.position) {
- viewer.tagTool.startInsertion().done(tag_ => {
- done()
- })
- } else {
- info.root = MergeEditor.getAllObjects().find(e => e.dataset_id == info.modelId)
- if (!info.root) {
- console.error('没有找到该modelId')
- }
- tag = viewer.tagTool.createTagFromData(info)
- done()
- }
- let result = {
- bus,
- getScreenPos() {
- let pos3d = new THREE.Vector3().setFromMatrixPosition(tag.matrixWorld)
- return sdk.getScreenByPosition(pos3d)
- },
- show() {
- Potree.Utils.updateVisible(tag, 'byList', true)
- },
- hide() {
- Potree.Utils.updateVisible(tag, 'byList', false)
- },
- destroy() {
- if (tag) {
- tag.dispose()
- }
- viewer.dispatchEvent({ type: 'cancel_insertions', remove: true })
- },
- changeTitle(title) {
- tag.changeTitle(title)
- }
- }
- return result
- },
- showGrid() {
- Potree.Utils.updateVisible(viewer.modules.MergeEditor.ground, 'hideGrid', true)
- viewer.dispatchEvent('content_changed')
- },
- hideGrid() {
- Potree.Utils.updateVisible(viewer.modules.MergeEditor.ground, 'hideGrid', false)
- viewer.dispatchEvent('content_changed')
- }
- }
-
- function spliceFromArr(model, props, loaded){
- //let autoLoads.find()
- props.loadFinish = true
- props.loading = false
- if (loaded) {
- props.loaded = true
- props.model = model
- } else {
- props.error = true
- }
- /* let haventLoad = autoLoads.filter(e=>!e.loading && !e.loadFinish);
- if( haventLoad[0]){
- startLoad(haventLoad[0])
- */
- if (!loadNext()) {
- if (autoLoads.filter(e => !e.loadFinish).length == 0 && autoLoads.filter(e => e.loaded).length > 0 && !props.isFirstLoad) {//设置相机位置:当自动开始加载第一个模型时(其余的也跟着自动加载),等这批加载完后;
- let autoLoadsDone = autoLoads.filter(e => e.loaded).map(e => e.model)
- console.log('所有模型加载完毕')
- autoLoads.filter(e => e.loaded && e.show).forEach(e => e.model.visible = true)
- MergeEditor.focusOn(autoLoadsDone, 1000, true, true)
- autoLoads.length = 0
- }
- }
- }
-
-
- function loadNext(){
- let haventLoad = autoLoads.filter(e => !e.loading && !e.loadFinish);
- let loading = autoLoads.filter(e => e.loading);
- let needLoad = haventLoad.slice(0, maxLoadingCount - loading.length)
- needLoad.forEach(e => startLoad(e))
- return haventLoad.length > 0
- }
-
- function startLoad(prop){
- /* if(prop.raw.visible !== 1){//用于临时隐藏
- setTimeout(()=>{
- spliceFromArr(null, prop, false)
- prop.bus.emit('loadError' )
- },1)
- return
- } */
-
- if(prop.loading || prop.loadFinish)return
-
- Potree.Log(`--开始加载--`, { font: { color: '#f68' } });
- console.log('id:', prop.id, ', title:', prop.title, ', filename:', Potree.Common.getNameFromURL(prop.url), ', type:', prop.type, prop)
- prop.unlit = prop.renderType != 'normal'
- prop.maximumScreenSpaceError = 70
- prop.prefix = prop.raw.prefix
- Potree.addModel(prop, prop.done, prop.progressFun, prop.onError)
- prop.loading = true
- }
- return sdk
- }
-
-
- /*
- 暂定不同场景间的漫游点不能互通。虽然它们可能是摆放正确的,如果是组成一整个场景的话还是要打通……
- 不互通的方法是设置pano.enable
- */
- export default enter
|