|
@@ -0,0 +1,174 @@
|
|
|
+import * as THREE from 'three';
|
|
|
+export const ThreeCubeDetailClass = {
|
|
|
+ constructor(options) {
|
|
|
+ this.webgl = options.webgl;
|
|
|
+ this.view = options.view;
|
|
|
+ this.points = options.points;
|
|
|
+ this.mesh = null;
|
|
|
+ this.edges = null;
|
|
|
+ this._camera = null;
|
|
|
+ },
|
|
|
+ initialize(){
|
|
|
+ this.renderer = new THREE.WebGLRenderer({
|
|
|
+ context:this.gl,
|
|
|
+ premultipliedAlpha:false
|
|
|
+ })
|
|
|
+ this.renderer.setPixelRatio(window.devicePixelRatio);
|
|
|
+ this.renderer.setViewport(0,0,this.view.width,this.view.height);
|
|
|
+
|
|
|
+ this.renderer.autoClear = false;
|
|
|
+ this.renderer.autoClearDepth = false;
|
|
|
+ this.renderer.autoClearColor = false;
|
|
|
+ let originalSetRenderTarget = this.renderer.setRenderTarget.bind(this.renderer);
|
|
|
+ this.renderer.setRenderTarget = function (target) {
|
|
|
+ originalSetRenderTarget(target);
|
|
|
+ if (target == null) {
|
|
|
+ context.bindRenderTarget();
|
|
|
+ }
|
|
|
+ };
|
|
|
+ // 场景和相机
|
|
|
+ this.scene = new THREE.Scene();
|
|
|
+ let cam = this.camera;
|
|
|
+ this._camera = new THREE.PerspectiveCamera(cam.fovY, cam.aspect, 0.1, 1000000000);
|
|
|
+ // 添加坐标轴辅助工具
|
|
|
+ const axesHelper = new THREE.AxesHelper(1);
|
|
|
+ axesHelper.position.copy(1000000, 100000, 100000);
|
|
|
+ this.scene.add(axesHelper);
|
|
|
+ let grid = new THREE.GridHelper(30, 10, 0xf0f0f0, 0xffffff);
|
|
|
+ this.scene.add(grid);
|
|
|
+
|
|
|
+ this.createInstancedMesh();
|
|
|
+ // 添加灯光(确保物体在场景中可以被看见)
|
|
|
+ const ambientLight = new THREE.AmbientLight(0xffffff, 0.5);
|
|
|
+ this.scene.add(ambientLight);
|
|
|
+
|
|
|
+ const directionalLight = new THREE.DirectionalLight(0xffffff, 0.5);
|
|
|
+ directionalLight.position.set(10, 10, 10);
|
|
|
+ this.scene.add(directionalLight);
|
|
|
+
|
|
|
+
|
|
|
+ },
|
|
|
+ createInstancedMesh() {
|
|
|
+ const boxGeometry = new THREE.BoxGeometry(
|
|
|
+ this.size.z,
|
|
|
+ this.size.x, this.size.y);
|
|
|
+ boxGeometry.computeVertexNormals(); // 计算并更新顶点法线
|
|
|
+
|
|
|
+ // 主体材质
|
|
|
+ const material = new THREE.MeshPhongMaterial({
|
|
|
+ transparent: true,
|
|
|
+ opacity: 0.1,
|
|
|
+ depthTest: true,
|
|
|
+ side: THREE.DoubleSide,
|
|
|
+ depthWrite: false, // 关闭深度写入
|
|
|
+ });
|
|
|
+
|
|
|
+ // 修改材质支持 instanceColor
|
|
|
+ material.onBeforeCompile = (shader) => {
|
|
|
+ shader.vertexShader = `
|
|
|
+ attribute vec3 instanceColor;
|
|
|
+ varying vec3 vColor;
|
|
|
+ ` + shader.vertexShader;
|
|
|
+
|
|
|
+ shader.vertexShader = shader.vertexShader.replace(
|
|
|
+ `#include <begin_vertex>`,
|
|
|
+ `
|
|
|
+ #include <begin_vertex>
|
|
|
+ vColor = instanceColor;
|
|
|
+ `
|
|
|
+ );
|
|
|
+ shader.fragmentShader = `
|
|
|
+ varying vec3 vColor;
|
|
|
+ ` + shader.fragmentShader;
|
|
|
+
|
|
|
+ shader.fragmentShader = shader.fragmentShader.replace(
|
|
|
+ `#include <dithering_fragment>`,
|
|
|
+ `
|
|
|
+ #include <dithering_fragment>
|
|
|
+ gl_FragColor.rgb = vColor;
|
|
|
+ `
|
|
|
+ );
|
|
|
+ };
|
|
|
+
|
|
|
+ // 创建主体 InstancedMesh
|
|
|
+ this.mesh = new THREE.InstancedMesh(boxGeometry, material, this.points.length);
|
|
|
+
|
|
|
+ // 添加发光边框
|
|
|
+ const edgesGeometry = new THREE.EdgesGeometry(boxGeometry); // 提取边框几何
|
|
|
+ const glowMaterial = new THREE.LineBasicMaterial({
|
|
|
+ color: 0xffffff, // 边框颜色
|
|
|
+ emissive: 0xff0000, // 发光颜色
|
|
|
+ linewidth: 2, // 线条宽度
|
|
|
+ transparent: true, // 透明度
|
|
|
+ });
|
|
|
+ const edgesGroup = new THREE.Group();
|
|
|
+
|
|
|
+ const dummy = new THREE.Object3D();
|
|
|
+ const instanceColors = new Float32Array(this.points.length * 3);
|
|
|
+ let index = 0;
|
|
|
+
|
|
|
+ for (let i = 0; i < this.points.length; i++) {
|
|
|
+ let worldX = this.points[i].x;
|
|
|
+ let worldY = this.points[i].y;
|
|
|
+ let worldZ = this.points[i].z;
|
|
|
+ let renderPos = [];
|
|
|
+ this.webgl.toRenderCoordinates(this.view, [worldX, worldY, worldZ], 0, this.view.spatialReference, renderPos, 0, 1);
|
|
|
+
|
|
|
+ // 设置实例的位置
|
|
|
+ dummy.position.set(renderPos[0], renderPos[1], renderPos[2]);
|
|
|
+ dummy.updateMatrix();
|
|
|
+
|
|
|
+ // 设置主网格实例
|
|
|
+ this.mesh.setMatrixAt(index, dummy.matrix);
|
|
|
+
|
|
|
+ // 创建并添加发光边框实例
|
|
|
+ const edgeLine = new THREE.LineSegments(edgesGeometry, glowMaterial);
|
|
|
+ edgeLine.position.copy(dummy.position);
|
|
|
+ edgeLine.rotation.copy(dummy.rotation);
|
|
|
+ edgeLine.scale.copy(dummy.scale);
|
|
|
+ edgesGroup.add(edgeLine);
|
|
|
+
|
|
|
+ // 设置颜色
|
|
|
+ // instanceColors[index * 3] = this.points[i].color[0]; // 红色
|
|
|
+ // instanceColors[index * 3 + 1] = this.points[i].color[1];
|
|
|
+ // instanceColors[index * 3 + 2] = this.points[i].color[2];
|
|
|
+
|
|
|
+ instanceColors[index * 3] = 1; // 红色
|
|
|
+ instanceColors[index * 3 + 1] = 0;
|
|
|
+ instanceColors[index * 3 + 2] = 0;
|
|
|
+
|
|
|
+
|
|
|
+ index++;
|
|
|
+ }
|
|
|
+
|
|
|
+ // 为实例化的网格设置颜色
|
|
|
+ boxGeometry.setAttribute('instanceColor', new THREE.InstancedBufferAttribute(instanceColors, 3));
|
|
|
+ boxGeometry.attributes.instanceColor.needsUpdate = true;
|
|
|
+ this.mesh.instanceMatrix.needsUpdate = true;
|
|
|
+
|
|
|
+ // 添加到场景
|
|
|
+ this.scene.add(this.mesh); // 添加主体
|
|
|
+ this.scene.add(edgesGroup); // 添加发光边框
|
|
|
+ },
|
|
|
+
|
|
|
+ render(){
|
|
|
+ let cam = this.camera;
|
|
|
+ //需要调整相机的视角
|
|
|
+ this._camera.position.set(cam.eye[0], cam.eye[1], cam.eye[2]);
|
|
|
+ this._camera.up.set(cam.up[0], cam.up[1], cam.up[2]);
|
|
|
+ this._camera.lookAt(new THREE.Vector3(cam.center[0], cam.center[1], cam.center[2]));
|
|
|
+ // Projection matrix can be copied directly
|
|
|
+ this._camera.projectionMatrix.fromArray(cam.projectionMatrix);
|
|
|
+
|
|
|
+ this.renderer.state.reset();
|
|
|
+
|
|
|
+ this.bindRenderTarget();
|
|
|
+ this.renderer.render(this.scene, this._camera);
|
|
|
+ // as we want to smoothly animate the ISS movement, immediately request a re-render
|
|
|
+ this.requestRender();
|
|
|
+
|
|
|
+ // cleanup
|
|
|
+ this.resetWebGLState();
|
|
|
+ }
|
|
|
+}
|
|
|
+
|