MeasuringTool.js 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646
  1. import * as THREE from "../../../libs/three.js/build/three.module.js";
  2. import {Measure} from "./Measure.js";
  3. import {Utils} from "../../utils.js";
  4. import math from "../../utils/math.js";
  5. import {CameraMode} from "../../defines.js";
  6. import {TextSprite} from '../TextSprite.js'
  7. import {ExtendEventDispatcher} from "../../custom/ExtendEventDispatcher.js";
  8. function updateAzimuth(viewer, measure){
  9. if(!measure.showAzimuth)return
  10. const azimuth = measure.azimuth;
  11. const isOkay = measure.points.length === 2;
  12. azimuth.node.visible = isOkay
  13. if(!azimuth.node.visible){
  14. return;
  15. }
  16. const camera = viewer.scene.getActiveCamera();
  17. const renderAreaSize = viewer.renderer.getSize(new THREE.Vector2());
  18. const width = renderAreaSize.width;
  19. const height = renderAreaSize.height;
  20. const [p0, p1] = measure.points;
  21. const r = p0.position.distanceTo(p1.position);
  22. const northVec = Utils.getNorthVec(p0.position, r, viewer.getProjection());
  23. const northPos = p0.position.clone().add(northVec);
  24. azimuth.center.position.copy(p0.position);
  25. azimuth.center.scale.set(2, 2, 2);
  26. azimuth.center.visible = false;
  27. // azimuth.target.visible = false;
  28. { // north
  29. azimuth.north.position.copy(northPos);
  30. azimuth.north.scale.set(2, 2, 2);
  31. let distance = azimuth.north.position.distanceTo(camera.position);
  32. let pr = Utils.projectedRadius(1, camera, distance, width, height);
  33. let scale = (5 / pr);
  34. azimuth.north.scale.set(scale, scale, scale);
  35. }
  36. { // target
  37. azimuth.target.position.copy(p1.position);
  38. azimuth.target.position.z = azimuth.north.position.z;
  39. let distance = azimuth.target.position.distanceTo(camera.position);
  40. let pr = Utils.projectedRadius(1, camera, distance, width, height);
  41. let scale = (5 / pr);
  42. azimuth.target.scale.set(scale, scale, scale);
  43. }
  44. azimuth.circle.position.copy(p0.position);
  45. azimuth.circle.scale.set(r, r, r);
  46. azimuth.circle.material.resolution.set(width, height);
  47. // to target
  48. azimuth.centerToTarget.geometry.setPositions([
  49. 0, 0, 0,
  50. ...p1.position.clone().sub(p0.position).toArray(),
  51. ]);
  52. azimuth.centerToTarget.position.copy(p0.position);
  53. azimuth.centerToTarget.geometry.verticesNeedUpdate = true;
  54. azimuth.centerToTarget.geometry.computeBoundingSphere();
  55. azimuth.centerToTarget.computeLineDistances();
  56. azimuth.centerToTarget.material.resolution.set(width, height);
  57. // to target ground
  58. azimuth.centerToTargetground.geometry.setPositions([
  59. 0, 0, 0,
  60. p1.position.x - p0.position.x,
  61. p1.position.y - p0.position.y,
  62. 0,
  63. ]);
  64. azimuth.centerToTargetground.position.copy(p0.position);
  65. azimuth.centerToTargetground.geometry.verticesNeedUpdate = true;
  66. azimuth.centerToTargetground.geometry.computeBoundingSphere();
  67. azimuth.centerToTargetground.computeLineDistances();
  68. azimuth.centerToTargetground.material.resolution.set(width, height);
  69. // to north
  70. azimuth.centerToNorth.geometry.setPositions([
  71. 0, 0, 0,
  72. northPos.x - p0.position.x,
  73. northPos.y - p0.position.y,
  74. 0,
  75. ]);
  76. azimuth.centerToNorth.position.copy(p0.position);
  77. azimuth.centerToNorth.geometry.verticesNeedUpdate = true;
  78. azimuth.centerToNorth.geometry.computeBoundingSphere();
  79. azimuth.centerToNorth.computeLineDistances();
  80. azimuth.centerToNorth.material.resolution.set(width, height);
  81. // label
  82. const radians = Utils.computeAzimuth(p0.position, p1.position, viewer.getProjection());
  83. let degrees = THREE.Math.radToDeg(radians);
  84. if(degrees < 0){
  85. degrees = 360 + degrees;
  86. }
  87. const txtDegrees = `${degrees.toFixed(2)}°`;
  88. const labelDir = northPos.clone().add(p1.position).multiplyScalar(0.5).sub(p0.position);
  89. if(labelDir.length() > 0){
  90. labelDir.z = 0;
  91. labelDir.normalize();
  92. const labelVec = labelDir.clone().multiplyScalar(r);
  93. const labelPos = p0.position.clone().add(labelVec);
  94. azimuth.label.position.copy(labelPos);
  95. }
  96. azimuth.label.setText(txtDegrees);
  97. let distance = azimuth.label.position.distanceTo(camera.position);
  98. let pr = Utils.projectedRadius(1, camera, distance, width, height);
  99. let scale = (70 / pr);
  100. azimuth.label.scale.set(scale, scale, scale);
  101. }
  102. export class MeasuringTool extends ExtendEventDispatcher{
  103. constructor (viewer) {
  104. super();
  105. this.viewer = viewer;
  106. this.renderer = viewer.renderer;
  107. this.viewer.addEventListener('start_inserting_measurement', e => {
  108. this.viewer.dispatchEvent({
  109. type: 'cancel_insertions'
  110. });
  111. });
  112. this.showLabels = true;
  113. this.scene = new THREE.Scene();
  114. this.scene.name = 'scene_measurement';
  115. //this.light = new THREE.PointLight(0xffffff, 1.0);
  116. //this.scene.add(this.light);
  117. this.viewer.inputHandler.registerInteractiveScene(this.scene);
  118. //this.scene = viewer.overlay//
  119. this.onRemove = (e) => { e.measurement.dispose()/* this.scene.remove(e.measurement); */};
  120. this.onAdd = e => {this.scene.add(e.measurement);};
  121. for(let measurement of viewer.scene.measurements){
  122. this.onAdd({measurement: measurement});
  123. }
  124. viewer.addEventListener('camera_changed',(e)=>{
  125. if(e.viewport == viewer.mainViewport ) this.update()
  126. })
  127. //viewer.addEventListener("update", this.update.bind(this));
  128. viewer.addEventListener("render.pass.perspective_overlay", this.render.bind(this));
  129. viewer.addEventListener("scene_changed", this.onSceneChange.bind(this));
  130. viewer.scene.addEventListener('measurement_added', this.onAdd);
  131. viewer.scene.addEventListener('measurement_removed', this.onRemove);
  132. viewer.addEventListener('resize',this.setSize.bind(this))
  133. }
  134. onSceneChange(e){
  135. if(e.oldScene){
  136. e.oldScene.removeEventListener('measurement_added', this.onAdd);
  137. e.oldScene.removeEventListener('measurement_removed', this.onRemove);
  138. }
  139. e.scene.addEventListener('measurement_added', this.onAdd);
  140. e.scene.addEventListener('measurement_removed', this.onRemove);
  141. }
  142. createMeasureFromData(data){//add
  143. const measure = new Measure(data);
  144. if(measure.failBuilded){
  145. return
  146. }
  147. viewer.scene.addMeasurement(measure);
  148. if(measure.guideLine)measure.guideLine.visible = false
  149. return measure
  150. }
  151. update(){
  152. return;
  153. let camera = this.viewer.scene.getActiveCamera();
  154. let domElement = this.renderer.domElement;
  155. let measurements = this.viewer.scene.measurements;
  156. // make size independant of distance
  157. let mainLabels = [], subLabels = [];
  158. for (let measure of measurements) {
  159. measure.lengthUnit = this.viewer.lengthUnit;
  160. measure.lengthUnitDisplay = this.viewer.lengthUnitDisplay;
  161. //measure.update();
  162. updateAzimuth(this.viewer, measure);
  163. /* [...measure.markers, ...measure.edgeLabels, measure.areaLabel].forEach(e=>{
  164. e && e.update()
  165. }); */
  166. // labels
  167. /* let labels = measure.edgeLabels.concat(measure.angleLabels);
  168. for(let label of labels){
  169. label.update()
  170. if(label.elem.hasClass('sub')){
  171. subLabels.push(label)
  172. }else{
  173. mainLabels.push(label)
  174. }
  175. }
  176. // coordinate labels
  177. for (let j = 0; j < measure.coordinateLabels.length; j++) {
  178. let label = measure.coordinateLabels[j];
  179. label.update()
  180. mainLabels.push(label)
  181. }
  182. if(measure.showArea){ // area label
  183. let label = measure.areaLabel;
  184. label.update()
  185. mainLabels.push(label)
  186. } */
  187. /* if(measure.showCircle){ // radius label
  188. let label = measure.circleRadiusLabel;
  189. let distance = label.position.distanceTo(camera.position);
  190. let pr = Utils.projectedRadius(1, camera, distance, clientWidth, clientHeight);
  191. let scale = (70 / pr);
  192. label.scale.set(scale, scale, scale);
  193. } */
  194. if(!this.showLabels){
  195. const labels = [
  196. ...measure.sphereLabels,
  197. ...measure.angleLabels,
  198. measure.circleRadiusLabel,
  199. ];
  200. for(const label of labels){
  201. label.visible = false;
  202. }
  203. }
  204. }
  205. //this.updateLabelZIndex([{labels:subLabels},{labels:mainLabels}])
  206. }
  207. setSize(e){ //e.resolution
  208. /* if(Measure.lineMats){
  209. for(var m in Measure.lineMats){
  210. Measure.lineMats[m].resolution.set(e.canvasWidth, e.canvasHeight);
  211. }
  212. }
  213. if(Measure.sphereMats){
  214. for(var s in Measure.sphereMats){
  215. Measure.sphereMats[s].uniforms.resolution.value.set(e.canvasWidth, e.canvasHeight);
  216. }
  217. }
  218. for (let measure of this.viewer.scene.measurements) {
  219. measure.edgeLabels.concat(measure.areaLabel).forEach(label=>{
  220. label.sprite.material.uniforms.resolution.value.set(e.canvasWidth, e.canvasHeight);
  221. })
  222. } */
  223. }
  224. updateLabelZIndex(group){//[{labels:[]},{}] 顺序按照z-index低到高
  225. group.forEach((e,i)=>{
  226. e.base = group[i-1] ? group[i-1].base + group[i-1].labels.length : 0
  227. var labels = e.labels.sort((a,b)=>{
  228. return b.pos2d.z - a.pos2d.z
  229. })
  230. labels.forEach((label,index)=>{
  231. $(label.elem).css('z-index', e.base+index)
  232. })
  233. })
  234. }
  235. editStateChange(e){
  236. //console.log("editStateChange" , e.state)
  237. let state = e.state
  238. if(!state){
  239. state = viewer.scene.measurements.some(e=>e.isEditing)
  240. }
  241. if(state){
  242. viewer.dispatchEvent({type:"measureMovePoint"})
  243. }else{
  244. viewer.dispatchEvent({type:"endMeasureMove"})
  245. }
  246. //this.editing =
  247. }
  248. startInsertion (args = {}, callback, cancelFun) {
  249. let domElement = this.viewer.renderer.domElement;
  250. const pick = (defaul, alternative) => {
  251. if(defaul != null){
  252. return defaul;
  253. }else{
  254. return alternative;
  255. }
  256. };
  257. args.showDistances = (args.showDistances === null) ? true : args.showDistances;
  258. args.showArea = pick(args.showArea, false);
  259. args.showAngles = pick(args.showAngles, false);
  260. args.showCoordinates = pick(args.showCoordinates, false);
  261. args.showHeight = pick(args.showHeight, false);
  262. args.showCircle = pick(args.showCircle, false);
  263. args.showAzimuth = pick(args.showAzimuth, false);
  264. args.showEdges = pick(args.showEdges, true);
  265. args.closed = pick(args.closed, false);
  266. args.maxMarkers = pick(args.maxMarkers, Infinity);
  267. args.direction = args.direction//add
  268. args.type = args.type /* || 'Measurement'; */
  269. args.showGuideLine = pick(args.showGuideLine, false);
  270. args.isRect = pick(args.isRect, false);
  271. let measure = new Measure(args);
  272. this.scene.add(measure);
  273. measure.isNew = true
  274. this.viewer.dispatchEvent({
  275. type: 'start_inserting_measurement',
  276. measure: measure
  277. });
  278. measure.addEventListener('editStateChange', this.editStateChange.bind(this))
  279. measure.editStateChange(true)
  280. let timer;
  281. let endDragFun = (e) => {
  282. let length = measure.points.length
  283. if (e.button == THREE.MOUSE.LEFT || e.isTouch) {
  284. if (length >= measure.maxMarkers) {
  285. end({finish:true});
  286. }else{
  287. var marker = measure.addMarker({point:measure.points[length - 1].clone()})
  288. if(args.isRect && measure.markers.length == 3){//marker全可见
  289. measure.addMarker({point:measure.points[0].clone()})
  290. }else{
  291. measure.markers[length].visible = false
  292. measure.edges[length].visible = false
  293. }
  294. measure.edges[length-1].visible = true
  295. measure.markers[length-1].visible = true;
  296. measure.editStateChange(true) //重新激活reticule状态
  297. marker.isDragging = true
  298. measure.continueDrag(marker, e)
  299. }
  300. } else if (e.button === THREE.MOUSE.RIGHT ) { //触屏怎么取消?
  301. if(e.pressDistance < Potree.config.clickMaxDragDis )end(e);//非拖拽的话
  302. else measure.continueDrag(null, e)
  303. }
  304. };
  305. let end = (e={}) => {//确定、结束
  306. if(!measure.isNew)return
  307. if(args.minMarkers != void 0){
  308. if(!e.finish && measure.markers.length<=args.minMarkers){//右键 当个数不够时取消
  309. //this.viewer.scene.removeMeasurement(measure)
  310. //cancelFun && cancelFun()
  311. //重新开始画
  312. measure.markers[0].removeEventListener('mousedown',end)
  313. measure.reDraw()
  314. this.viewer.addEventListener('global_click', click, 10)
  315. measure.editStateChange(true)
  316. return
  317. /* if(!Potree.settings.isOfficial) this.viewer.scene.removeMeasurement(measure)
  318. else if(e.drag){ //正式版本不允许右键退出, 继续
  319. continueDrag(e.drag.object)
  320. measure.editStateChange(true)
  321. return
  322. } */
  323. }
  324. }
  325. if (/* !e.finish && */measure.markers.length > 3) {
  326. measure.removeMarker(measure.points.length - 1);
  327. measure.markers[0].removeEventListener('mouseover', mouseover);
  328. measure.markers[0].removeEventListener('mouseleave', mouseleave);
  329. measure.markers[0].removeEventListener('click'/* 'mousedown' */,Exit)
  330. if(e.byClickMarker && measure.markers.length > 3){//通过点击第一个marker而结束的话,会多一个marker
  331. measure.removeMarker(measure.points.length - 1);
  332. }
  333. }
  334. measure.isNew = false
  335. let length = measure.points.length
  336. if(length){
  337. measure.markers[length-1].visible = true;
  338. measure.edges[length-1].visible = true
  339. measure.markers.forEach(marker=>{marker.dispatchEvent('addHoverEvent') })
  340. measure.edges.forEach(edge=>{edge.dispatchEvent('addHoverEvent') })
  341. }
  342. clearTimeout(timer)
  343. this.viewer.removeEventListener('cancel_insertions', Exit);
  344. //pressExit && this.viewer.inputHandler.removeEventListener('keydown', pressExit);
  345. this.viewer.removeEventListener('global_click', click)
  346. this.viewer.removeEventListener('global_mousemove', ifAtWrongPlace)
  347. viewer.dispatchEvent({
  348. type : "CursorChange", action : "remove", name:"polygon_AtWrongPlace"
  349. });
  350. viewer.inputHandler.dispatchEvent({type:'isMeasuring', v:false, cause:'stopInsertion'} )
  351. e.remove || callback && callback()
  352. /* this.viewer.dispatchEvent({
  353. type: 'finish_inserting_measurement',
  354. measure: measure
  355. }); */
  356. };
  357. let Exit = (e)=>{//强制退出
  358. if(e.measure && e.measure != measure){
  359. return;//若指定了退出的measure但和该measure不一致,就返回
  360. }
  361. console.log('Exit: ' + measure.id)
  362. if(e.remove){
  363. viewer.scene.removeMeasurement(measure)
  364. }
  365. measure.editStateChange(false)
  366. measure.cannotConfirmNormal = false //一些dropMarker中的句子
  367. measure.guideLine &&(measure.guideLine.visible = false)
  368. /*
  369. if(this.viewer.inputHandler.drag && !e.remove ){//还未触发drop的话
  370. this.viewer.inputHandler.drag.object.dispatchEvent({ //这句会导致又增一个marker
  371. type: 'drop',
  372. drag: this.viewer.inputHandler.drag,
  373. viewer: this.viewer,
  374. pressDistance:0,
  375. button : THREE.MOUSE.RIGHT
  376. });
  377. } else {*///未结束时添加新的measure时会触发
  378. end({finish:true, remove:e.remove, byClickMarker: e.type == 'click'})
  379. //}
  380. this.viewer.inputHandler.drag && (this.viewer.inputHandler.drag.object = null)
  381. }
  382. this.viewer.addEventListener('cancel_insertions', Exit);
  383. /*let pressExit
  384. if(!Potree.settings.isOfficial){
  385. pressExit = (e)=>{
  386. if(e.keyCode == 27){//Esc
  387. //Exit()
  388. //怎么模拟右键???//现由前端发出
  389. }
  390. }
  391. this.viewer.inputHandler.addEventListener('keydown', pressExit)
  392. } */
  393. let mouseover = (e) => {
  394. measure.setMarkerSelected(e.object, 'hover', 'single');
  395. };
  396. let mouseleave = (e) => {
  397. measure.setMarkerSelected(e.object, 'unhover', 'single');
  398. }
  399. let click = (e)=>{//一旦点击就立刻增加两marker
  400. if(ifAtWrongPlace(e))return
  401. if(e.clickElement)return //如点击label时focusOnObject
  402. if(e.button === THREE.MOUSE.RIGHT)return
  403. //console.log('measure clicked33', !!e.intersectPoint)
  404. //var I = e.intersectPoint && (e.intersectPoint.orthoIntersect || e.intersectPoint.location)
  405. var I = e.intersect && (e.intersect.orthoIntersect || e.intersect.location)
  406. if(!I){
  407. return measure.dispatchEvent('intersectNoPointcloud')
  408. }
  409. var atMap = e.drag.dragViewport.name == 'mapViewport'
  410. //在地图上测量的首个点按楼层高度(暂时先只按mainViewport相机高度吧,但navvis是按楼层,画在楼层的地面上,可能因为平面图显示的是楼层近地面),
  411. if(atMap){
  412. I = I.clone().setZ(viewer.mainViewport.camera.position.z )
  413. }
  414. var marker = measure.addMarker({point:I})
  415. marker.isDragging = true
  416. this.viewer.inputHandler.startDragging(marker , {endDragFun, notPressMouse:true} ); //notPressMouse代表不是通过按下鼠标来拖拽
  417. e.drag = this.viewer.inputHandler.drag
  418. e.drag.endDragFun = endDragFun
  419. e.drag.notPressMouse = true
  420. //if(!measure.dragMarker(e) || !measure.dropMarker(e))return
  421. measure.dragMarker(e)
  422. measure.dropMarker(e)
  423. if(measure.maxMarkers > 1 ){
  424. measure.markers[1].visible = false
  425. measure.edges[1].visible = false
  426. }
  427. if(measure.maxMarkers>2 && !measure.isRect){
  428. measure.markers[0].addEventListener('mouseover', mouseover);
  429. measure.markers[0].addEventListener('mouseleave', mouseleave);
  430. measure.markers[0].addEventListener('click'/* 'mousedown' */,Exit) //点击到第一个marker就结束
  431. }
  432. this.viewer.removeEventListener('global_click', click)///* global_drop */
  433. //console.log('measure clicked')
  434. e.consume && e.consume()
  435. return {stopContinue:true}//防止继续执行别的侦听,如flytopano
  436. }
  437. //点击第n下拥有n+1个marker, n>0
  438. viewer.inputHandler.dispatchEvent({type: 'isMeasuring', v: true, cause:'startInsertion'})
  439. this.viewer.addEventListener('global_click', click, 10)//add importance:10
  440. let ifAtWrongPlace = (e)=>{
  441. if(measure.unableDragAtMap && e.hoverViewport.name == 'mapViewport' ){
  442. if(e.isTouch){
  443. viewer.dispatchEvent({type:'reticule_forbit', v:true})
  444. }else{
  445. viewer.dispatchEvent({
  446. type : "CursorChange", action : "add", name:"polygon_AtWrongPlace"
  447. });
  448. }
  449. return true
  450. }else{
  451. if(e.isTouch){
  452. viewer.dispatchEvent({type:'reticule_forbit',v:false})
  453. }else{
  454. viewer.dispatchEvent({
  455. type : "CursorChange", action : "remove", name:"polygon_AtWrongPlace"
  456. });
  457. }
  458. }
  459. }
  460. if(measure.unableDragAtMap){
  461. this.viewer.addEventListener('global_mousemove', ifAtWrongPlace)
  462. }
  463. this.viewer.scene.addMeasurement(measure);
  464. return measure;
  465. }
  466. render(o={}){
  467. if(this.scene.children.length == 0)return
  468. viewer.setCameraLayers(o.camera, ['measure'])
  469. if(o.screenshot && this.viewer.ssaaRenderPass.enabled){ //抗锯齿
  470. this.viewer.ssaaRenderPass.sampleLevel = 4
  471. this.viewer.composer.render(this.scene, o.camera );
  472. /* viewer.scene.measurements.forEach(e=>{ //隐藏除了label以外的
  473. e.children.forEach((c)=>{
  474. if(!(c instanceof TextSprite)){
  475. c.visible = false
  476. }
  477. })
  478. }) */
  479. }else{
  480. this.viewer.renderer.render(this.scene, o.camera );
  481. }
  482. }
  483. };