SOURCE

console 命令行工具 X clear

                    
>
console
// 第三方FPS显示窗口
javascript: (function () {
    var script = document.createElement('script');
    script.onload = function () {
        var stats = new Stats();
        document.body.appendChild(stats.dom);
        requestAnimationFrame(function loop() {
        stats.update();
        requestAnimationFrame(loop);
        });
    };
    script.src = 'https://mrdoob.github.io/stats.js/build/stats.min.js';
    document.head.appendChild(script);
})();

var map = new AMap.Map('container', {
    center: [116.54, 39.79],
    zooms: [2, 20],
    zoom: 14,
    viewMode: '3D',
    pitch: 50,
});

var camera;
var renderer;
var scene;
var meshes = [];
let texture;
// 数据转换工具
var customCoords = map.customCoords;
// 数据使用转换工具进行转换,这个操作必须要提前执行(在获取镜头参数 函数之前执行),否则将会获得一个错误信息。
var data = customCoords.lngLatsToCoords([
    [116.52, 39.79],
    [116.54, 39.78],
    [116.56, 39.79],
]);

// 闭环数据
const testData = customCoords.lngLatsToCoords([
    [
        116.18657032,
        39.82981112
    ],
    [
        116.46982646,
        39.7899015
    ],
    [
        116.56519504,
        39.92211609
    ],
    [
        116.44064767,
        40.05080839
    ],
    [
        116.20151513,
        39.99412627
    ],
    [
        116.18657032,
        39.82981112
    ]
])

function calcDis([x1, y1], [x2, y2]) {
    return Math.sqrt(Math.pow(x2-x1, 2) + Math.pow(y2-y1, 2))
} 
// 创建 GL 图层
var gllayer = new AMap.GLCustomLayer({
    // 图层的层级
    zIndex: 8,
    // 初始化的操作,创建图层过程中执行一次。
    init: (gl) => {
        // 这里我们的地图模式是 3D,所以创建一个透视相机,相机的参数初始化可以随意设置,因为在 render 函数中,每一帧都需要同步相机参数,因此这里变得不那么重要。
        // 如果你需要 2D 地图(viewMode: '2D'),那么你需要创建一个正交相机
        camera = new THREE.PerspectiveCamera(
            60,
            window.innerWidth / window.innerHeight,
            100,
            1 << 30
        );

        renderer = new THREE.WebGLRenderer({
            context: gl, // 地图的 gl 上下文
            // alpha: true,
            // antialias: true,
            // canvas: gl.canvas,
        });

        // 自动清空画布这里必须设置为 false,否则地图底图将无法显示
        renderer.autoClear = false;
        scene = new THREE.Scene();

        // 环境光照和平行光
        var aLight = new THREE.AmbientLight(0xffffff, 1);
        // var dLight = new THREE.DirectionalLight(0xffffff, 1);
        // dLight.position.set(1000, -100, 900);
        // scene.add(dLight);
        scene.add(aLight);

        const base_64 = ''
        // texture = new THREE.TextureLoader().load(
        //     'https://a.amap.com/jsapi_demos/static/demo-center-v2/three.jpeg'
        // );

        texture = new THREE.TextureLoader().load(base_64)
        // 纹理重复
        texture.wrapS = THREE.RepeatWrapping;
        texture.wrapT = THREE.RepeatWrapping;
        // texture.repeat.set(4, 1)

        // 
        texture.minFilter = THREE.LinearFilter;
        //  这里可以使用 three 的各种材质
        var mat = new THREE.MeshPhongMaterial({
            color: 0xfff0f0,
            depthTest: true,
            transparent: true,
            map: texture,
            // wireframe: true,
            side: THREE.DoubleSide, 
        });
        // var geo = new THREE.BoxBufferGeometry(1000, 1000, 1000);
        // for (let i = 0; i < data.length; i++) {
        //     const d = data[i];
        //     var mesh = new THREE.Mesh(geo, mat);
        //     mesh.position.set(d[0], d[1], 500);
        //     meshes.push({
        //         mesh,
        //         count: i,
        //     });
        //     scene.add(mesh);
        // }

        // 创建geometry
        const geometry = new THREE.BufferGeometry();
        let indices = [];
        let vetices = [];
        let normals = [];
        let colors = [];
        let uvs = [];

        const vec3List = []; 
        console.log(testData)
        for(let i =0; i< testData.length; i++) {
            const [x1, y1] = testData[i]
            vec3List.push([x1, y1, 100])
            vec3List.push([x1, y1, 2500])
        }

        // 竖(u)轴uv重复次数 
        // const repeatCount = 4;
        const t0 = [0, 0]
        const t1 = [1, 0]
        const t2 = [1, 1]
        const t3 = [0, 1] 

        for(let i =0; i< vec3List.length - 2; i++) {
            // 长宽比,用于uv拉伸
            const repeatCount  = Math.round(calcDis(vec3List[i], vec3List[i+2])/ 2500)
            if (i % 2 === 0) {
                // 下面片
                vetices = [...vetices, ...vec3List[i], ...vec3List[i+2], ...vec3List[i+1] ]
                uvs = [
                    ...uvs, 
                    t0[0] * repeatCount, t0[1],
                    t1[0] * repeatCount, t1[1],
                    t3[0] * repeatCount, t3[1],
                ]

                // 计算法线 (简单方法: 使用叉乘)
                const vA = new THREE.Vector3().fromArray(vec3List[i]);
                const vB = new THREE.Vector3().fromArray(vec3List[i+2]);
                const vC = new THREE.Vector3().fromArray(vec3List[i+1]);
                
                const ab = new THREE.Vector3().subVectors(vB, vA);
                const ac = new THREE.Vector3().subVectors(vC, vA);
                const normal = new THREE.Vector3().crossVectors(ab, ac).normalize();
                
                normals = [...normals, normal.x, normal.y, normal.z]; // 顶点A的法线
                normals = [...normals, normal.x, normal.y, normal.z]; // 顶点B的法线
                normals = [...normals, normal.x, normal.y, normal.z];
            } else {
                // 上面片
                vetices = [...vetices, ...vec3List[i], ...vec3List[i+1], ...vec3List[i+2] ]
                uvs = [
                    ...uvs, 
                    t3[0] * repeatCount, t3[1], 
                    t1[0] * repeatCount, t1[1],
                    t2[0] * repeatCount, t2[1],
                ]

                // 计算法线
                const vA = new THREE.Vector3().fromArray(vec3List[i]);
                const vB = new THREE.Vector3().fromArray(vec3List[i+1]);
                const vC = new THREE.Vector3().fromArray(vec3List[i+2]);
                
                const ab = new THREE.Vector3().subVectors(vB, vA);
                const ac = new THREE.Vector3().subVectors(vC, vA);
                const normal = new THREE.Vector3().crossVectors(ab, ac).normalize();
                
                normals = [...normals, normal.x, normal.y, normal.z]; // 顶点A的法线
                normals = [...normals, normal.x, normal.y, normal.z]; // 顶点B的法线
                normals = [...normals, normal.x, normal.y, normal.z]; // 顶点C的法线
            }
        }
        // geometry.setIndex( indices );
        geometry.setAttribute( 'position', new THREE.Float32BufferAttribute( vetices, 3 ) );
        geometry.setAttribute( 'normal', new THREE.Float32BufferAttribute( normals, 3 ) );
        // geometry.setAttribute( 'color', new THREE.Float32BufferAttribute( colors, 3 ) );
        geometry.setAttribute( 'uv', new THREE.Float32BufferAttribute( uvs, 2 ) );

        testMesh = new THREE.Mesh( geometry, mat );
        scene.add( testMesh );

    },
    render: () => {
        // 这里必须执行!!重新设置 three 的 gl 上下文状态。
        renderer.resetState();
        // 重新设置图层的渲染中心点,将模型等物体的渲染中心点重置
        // 否则和 LOCA 可视化等多个图层能力使用的时候会出现物体位置偏移的问题
        customCoords.setCenter([116.52, 39.79]);
        var { near, far, fov, up, lookAt, position } =
        customCoords.getCameraParams();

        // 2D 地图下使用的正交相机
        // var { near, far, top, bottom, left, right, position, rotation } = customCoords.getCameraParams();

        // 这里的顺序不能颠倒,否则可能会出现绘制卡顿的效果。
        camera.near = near;
        camera.far = far;
        camera.fov = fov;
        camera.position.set(...position);
        camera.up.set(...up);
        camera.lookAt(...lookAt);
        camera.updateProjectionMatrix();

        // 2D 地图使用的正交相机参数赋值
        // camera.top = top;
        // camera.bottom = bottom;
        // camera.left = left;
        // camera.right = right;
        // camera.position.set(...position);
        // camera.updateProjectionMatrix();

        renderer.render(scene, camera);

        // 这里必须执行!!重新设置 three 的 gl 上下文状态。
        renderer.resetState();
    },
});
map.add(gllayer);

// 动画
let offset = 0
const speed = 0.01
function animate() {
    // for (let i = 0; i < meshes.length; i++) {
    //     let { mesh, count } = meshes[i];
    //     count += 1;
    //     mesh.rotateZ((count / 180) * Math.PI);
    // }
    
    offset += speed;
    if(offset >= 1) {
        offset = 0
    }
    console.log('texture: ', texture)
    texture.offset.set(0, offset)

    map.render();
    requestAnimationFrame(animate);
}

animate();

function onWindowResize() {
    camera.aspect = window.innerWidth / window.innerHeight;
    camera.updateProjectionMatrix();
    renderer.setSize(window.innerWidth, window.innerHeight);
}
window.addEventListener('resize', onWindowResize);
<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta
      name="viewport"
      content="initial-scale=1.0, user-scalable=no, width=device-width"
    />
    <title>自定义三维图层</title>
    <link
      rel="stylesheet"
      href="https://a.amap.com/jsapi_demos/static/demo-center/css/demo-center.css"
    />
    <style>
      html,
      body,
      #container {
        width: 100%;
        height: 100%;
      }
    </style>
  </head>
  <body>
    <div id="container"></div>
    <script src="https://cdn.jsdelivr.net/npm/three@0.142/build/three.js"></script>
    <script src="//webapi.amap.com/maps?v=2.0&key=4ee1bad27d5a07c45f128e904022f5f2"></script>
  </body>
</html>