在vue-cli中使用threejs,并实现鼠标控制移动,以及点击交互效果

这里假设已经了解threejs中基本的三要素等基础知识

如题,前戏不多,直接提枪上阵
第一步: 创建一个vue-cli项目
按照vue-cli官网方式创建 , 这里话不多说,不需要过多设置,能运行起来就欧克(当然是vue2.x)。

第二步: 安装threejs插件
完全可以按照threejs官网的教程安装

注意:如果打开为英文状态,可以在左上角那个地方点击en,切换成中文

image.png

第三步: 初始项目展示页面

安装完成之后,运行项目, 当然没什么屌变化,还是初始界面。

所以现在以组件的形式继续实现标题内容。

这里只在vue-cli中运行threejs,所以不安装其他插件,干扰视线

image.png

上图是目录结构,下面的代码内容主要是在ThreeJs组件中实现。所以,目前主要是先把测试ThreeJs是否能正常引入并展示。(PS: 按照HelloWorld组件的方式,基本上不会出错)

App.vue中的代码:


image.png

ThreeJs中的代码:


image.png

到目前为止,基本上页面上会展示 hello-threejs


image.png

第四步:开始在ThreeJS组件中,实现threejs的舞台

  1. 初始化舞台,ThreeJs组件
<template>
    <div>
        <canvas class="c" ref="ThreeJS"></canvas>
    </div>
</template>

<script>
    import * as THREE from 'three'
    export default{
        data(){
            return{
                scene: null,
                camera: null,
                cameraPole: null,
                renderer: null,
                canvas: null,
                canvasW: 0,
                canvasH: 0,
                cameraParam: {
                    fov: 30,
                    aspect: 2,
                    near: .1,
                    far: 200
                },
            }
        },
        created() {
            this.canvasW = window.innerWidth;
            this.canvasH = window.innerHeight;
            // 初始化设置宽高比
            this.cameraParam.aspect = this.canvasW / this.canvasH
        },
        mounted(){
            this.start()
        },
        methods: {
            start(){
                // 初始化三要素
                this.initMain()
                // 启用渲染
                this.render()
            },
            initMain(){
                // 初始化三要素
                this.initScene()
                this.initCamera()
                this.initRenderer()
                // 添加环境光
                this.addLight()
            },
            initScene(){
                // 创建场景
                this.scene = new THREE.Scene();
                this.scene.background = new THREE.Color('white');
            },
            initCamera(){
                // 创建透视摄像头
                const cP = this.cameraParam;
                this.camera = new THREE.PerspectiveCamera(cP.fov, cP.aspect, cP.near, cP.far);
                this.camera.position.z = 30;
                this.scene.add(this.camera)
            },
            initRenderer(){
                // 渲染器
                this.canvas = this.$refs.ThreeJS;
                this.renderer = new THREE.WebGLRenderer({
                    canvas: this.canvas,
                    antialias: true,//是否开启反锯齿,设置为true开启反锯齿。
                    alpha: true,//是否可以设置背景色透明。
                    logarithmicDepthBuffer: true//模型的重叠部位便不停的闪烁起来。这便是Z-Fighting问题,为解决这个问题,我们可以采用该种方法
                })
            },
            addLight(){
                // 环境光
                const color = 0xFFFFFF;
                const intensity = 1;
                const light = new THREE.AmbientLight(color, intensity);
                this.scene.add(light)
            },
            render(){
                // 启动动画
                this.renderer.render(this.scene, this.camera);
                // 动态监听窗口尺寸变化
                if (this.resizeRendererToDisplaySize(this.renderer)) {
                    const canvas = this.renderer.domElement;
                    this.camera.aspect = canvas.clientWidth / canvas.clientHeight;
                    this.camera.updateProjectionMatrix();
                }
                requestAnimationFrame(this.render.bind(this))
            },
            resizeRendererToDisplaySize(renderer){
                const canvas = renderer.domElement;
                this.canvasW = window.innerWidth;
                this.canvasH = window.innerHeight;
                const needResize = canvas.width !== this.canvasW || canvas.height !== this.canvasH;
                if(needResize){
                    this.renderer.setSize(this.canvasW, this.canvasH, false);
                }
                return needResize;
            },
        }
    }
</script>

<style>
    html, body{
        height: 100%;
        margin: 0;
        background: #0033CC;
    }
    .c{
        width: 100%;
        height: 100%;
        display: block;
    }
</style>

到这里,基本上你会看到一个白色的地的空白页面,因为还没有往画布里面添加内容

  1. 添加形状以及文字
    data中添加
data(){
  return{
    ...,
    planArr: [
      { x: -3, y: 3, name: '第一块区域' },
      { x: 3, y: 3, name: '第二块区域'},
      { x: -3, y: -3, name: '第三块区域' },
      { x: 3, y: -3, name: '第四块区域' }
    ]
  }
}

methods方法中添加

methods: {
  ...,
  initMain(){
    // 这个是已经有的方法,
    // 只是需要在这里调用一下创建几何体的方法
    ..., 
    // 添加形状
    this.createCube()
  },
  geometry(width, height, depth){
    return new THREE.BoxGeometry(width, height, depth)
  },
  createCube(){
    for(let i = 0; i < this.planArr.length; i++){
        // 添加几何体
        const material = new THREE.MeshPhongMaterial({
            color: 0x8aff58
        })
                    
        const cube = new THREE.Mesh(this.geometry(5, 5, .001), material);
        this.scene.add(cube);
        // 添加名称
        cube.name = this.planArr[i].name
        // 设置几何模型形变
        cube.position.set(this.planArr[i].x, this.planArr[i].y, 0)
    }
  },
}

到这里,你会看到下面的效果图:


image.png
  1. 添加鼠标划过方块以及点击方块时的事件

同样首先在data中添加数据

data(){
  return{
    ...,
    events: {
      raycaster: new THREE.Raycaster(),
      pickedObject: null,
      pickedObjectSavedColor: 0,
      pickPosition: { x: 0, y: 0 }
    },
  }
}

methods中添加方法

methods: {
  ...,
  // 点击当前坐标
            clickPickPosition(){
                this.pickEvents(this.events.pickPosition, this.scene, this.camera, obj=>{
                    obj.userData.checked = !obj.userData.checked;
                    // alert(`您选中了--${obj.name}`)
                    if(!obj.userData.checked){
                        obj.material.color.setHex(0x8aff58)
                        alert(`您已经取消选中--${obj.name}`)
                    }else{
                        obj.material.color.setHex(0xFFFF00)
                        alert(`您选中了--${obj.name}`)
                    }
                })
            },
            // 获取当前焦点坐标
            setPickPosition(event){
                const pos = this.getCanvasRelativePosition(event);
                this.events.pickPosition.x = (pos.x / this.canvas.width) * 2 - 1;
                this.events.pickPosition.y = (pos.y / this.canvas.height) * -2 + 1;
                
                this.pickEvents(this.events.pickPosition, this.scene, this.camera)
            },
            // 获取当前事件焦点坐标所在位置
            getCanvasRelativePosition(event){
                const rect = this.canvas.getBoundingClientRect();
                return {
                    x: (event.clientX - rect.left) * this.canvas.width / rect.width,
                    y: (event.clientY - rect.top) * this.canvas.height / rect.height
                }
            },
            // 添加鼠标划过以及点击事件
            clickEvents(){
                window.addEventListener('click', this.clickPickPosition);   
                window.addEventListener('mousemove', this.setPickPosition);
            },
            // 创建点击事件(默认是离摄像头最近的相交)
            pickEvents(normalizedPosition, scene, camera, callback){
                // 如果存在拾取的对象,则恢复颜色
                if(this.events.pickedObject){
                    this.events.pickedObject.material.emissive.setHex(this.events.pickedObjectSavedColor);
                    this.events.pickedObject = undefined;
                }
                // 沿着摄像头的方向投射射线
                this.events.raycaster.setFromCamera(normalizedPosition, camera)
                // 获取与射线光线相交的对象列表
                const intersectedObjects = this.events.raycaster.intersectObjects(this.scene.children);
                if(intersectedObjects.length){
                    // 获取与射线光纤相交的第一个对象。也是最近的一个
                    this.events.pickedObject = intersectedObjects[0].object;
                    // 保存当前对象的颜色
                    this.events.pickedObjectSavedColor = this.events.pickedObject.material.emissive.getHex();
                    // 将其发射颜色设置为闪烁的红色/黄色
                    this.events.pickedObject.material.emissive.setHex(0xFFFF00)
                    
                    if(callback){
                        callback(this.events.pickedObject)
                    }
                }
            },
}

在methods的initMain方法中调用一下鼠标点击事件

methods: {
  initMain(){
    ...,
    this.clickEvents();
  }
}

至此,基本上已经完全实现效果: 鼠标滑过几何体变成黄色,点击几何体变成黄色
完整版代码中添加了文字

ThreeJS.vue完整版代码:

<template>
    <div>
        <canvas class="c" ref="ThreeJS"></canvas>
    </div>
</template>

<script>
    import * as THREE from 'three'
    export default{
        data(){
            return{
                scene: null,
                camera: null,
                cameraPole: null,
                renderer: null,
                canvas: null,
                canvasW: 0,
                canvasH: 0,
                cameraParam: {
                    fov: 30,
                    aspect: 2,
                    near: .1,
                    far: 200
                },
                planArr: [
                    { x: -3, y: 3, name: '第一块区域' },
                    { x: 3, y: 3, name: '第二块区域'},
                    { x: -3, y: -3, name: '第三块区域' },
                    { x: 3, y: -3, name: '第四块区域' }
                ],
                events: {
                    raycaster: new THREE.Raycaster(),
                    pickedObject: null,
                    pickedObjectSavedColor: 0,
                    pickPosition: { x: 0, y: 0 }
                }
            }
        },
        created() {
            this.canvasW = window.innerWidth;
            this.canvasH = window.innerHeight;
            // 初始化设置宽高比
            this.cameraParam.aspect = this.canvasW / this.canvasH
        },
        mounted(){
            this.start()
        },
        methods: {
            start(){
                // 初始化三要素
                this.initMain()
                // 启用渲染
                this.render()
            },
            initMain(){
                // 初始化三要素
                this.initScene()
                this.initCamera()
                this.initRenderer()
                
                // 添加环境光
                this.addLight()
                
                // 添加形状
                this.createCube()
                
                // 调用点击事件
                this.clickEvents()
            },
            initScene(){
                // 创建场景
                this.scene = new THREE.Scene();
                this.scene.background = new THREE.Color('white');
            },
            initCamera(){
                // 创建透视摄像头
                const cP = this.cameraParam;
                this.camera = new THREE.PerspectiveCamera(cP.fov, cP.aspect, cP.near, cP.far);
                this.camera.position.z = 30;
                this.scene.add(this.camera)
            },
            initRenderer(){
                // 渲染器
                this.canvas = this.$refs.ThreeJS;
                this.renderer = new THREE.WebGLRenderer({
                    canvas: this.canvas,
                    antialias: true,//是否开启反锯齿,设置为true开启反锯齿。
                    alpha: true,//是否可以设置背景色透明。
                    logarithmicDepthBuffer: true//模型的重叠部位便不停的闪烁起来。这便是Z-Fighting问题,为解决这个问题,我们可以采用该种方法
                })
            },
            addLight(){
                // 环境光
                const color = 0xFFFFFF;
                const intensity = 1;
                const light = new THREE.AmbientLight(color, intensity);
                this.scene.add(light)
            },
            render(){
                // 启动动画
                this.renderer.render(this.scene, this.camera);
                // 动态监听窗口尺寸变化
                if (this.resizeRendererToDisplaySize(this.renderer)) {
                    const canvas = this.renderer.domElement;
                    this.camera.aspect = canvas.clientWidth / canvas.clientHeight;
                    this.camera.updateProjectionMatrix();
                }
                requestAnimationFrame(this.render.bind(this))
            },
            resizeRendererToDisplaySize(renderer){
                const canvas = renderer.domElement;
                this.canvasW = window.innerWidth;
                this.canvasH = window.innerHeight;
                const needResize = canvas.width !== this.canvasW || canvas.height !== this.canvasH;
                if(needResize){
                    this.renderer.setSize(this.canvasW, this.canvasH, false);
                }
                return needResize;
            },
            
            // ====================================基础设置完成=======================
            // ====================================创建立方体========================
            geometry(width, height, depth){
                return new THREE.BoxGeometry(width, height, depth)
            },
            createCube(){
                for(let i = 0; i < this.planArr.length; i++){
                    // 添加几何体
                    const material = new THREE.MeshPhongMaterial({
                        color: 0x8aff58
                    })
                    
                    const cube = new THREE.Mesh(this.geometry(5, 5, .001), material);
                    this.scene.add(cube);
                    // 添加名称
                    cube.name = this.planArr[i].name
                    // 设置几何模型形变
                    cube.position.set(this.planArr[i].x, this.planArr[i].y, 0)
                    
                    // 添加文字
                    let texture = new THREE.Texture(this.getTextCanvas(this.planArr[i].name));
                    texture.needsUpdate = true;
                    let spriteMaterial = new THREE.PointsMaterial({
                            map: texture,
                            size: 12,
                            transparent: true,
                            opacity: 1,
                    });
                    //创建坐标点,并将材质给坐标
                    let geometry = new THREE.BufferGeometry();
                    let vertices = [0, 0, 0];
                    geometry.setAttribute('position', new THREE.Float32BufferAttribute(vertices, 3));
                    let sprite = new THREE.Points(geometry, spriteMaterial);
                    sprite.position.set(0, 0, .002);
                    
                    cube.add(sprite);
                }
            },
            // 创建文字canvas
            getTextCanvas(text){
                var width=100, height=100; 
                const canvas = document.createElement('canvas');
                canvas.width = width;
                canvas.height = height;
                const ctx = canvas.getContext('2d');
                ctx.fillStyle = 'transparent';
                ctx.fillRect(0, 0, width, height);
                ctx.font = '6px';
                ctx.fillStyle = '#2891FF';
                ctx.textAlign = 'center';
                ctx.textBaseline = 'middle';
                ctx.fillText(text, width/2,height/2); 
                return canvas;
            },
            // 点击当前坐标
            clickPickPosition(){
                this.pickEvents(this.events.pickPosition, this.scene, this.camera, obj=>{
                    obj.userData.checked = !obj.userData.checked;
                    // alert(`您选中了--${obj.name}`)
                    if(!obj.userData.checked){
                        obj.material.color.setHex(0x8aff58)
                        alert(`您已经取消选中--${obj.name}`)
                    }else{
                        obj.material.color.setHex(0xFFFF00)
                        alert(`您选中了--${obj.name}`)
                    }
                })
            },
            // 获取当前焦点坐标
            setPickPosition(event){
                const pos = this.getCanvasRelativePosition(event);
                this.events.pickPosition.x = (pos.x / this.canvas.width) * 2 - 1;
                this.events.pickPosition.y = (pos.y / this.canvas.height) * -2 + 1;
                
                this.pickEvents(this.events.pickPosition, this.scene, this.camera)
            },
            // 获取当前事件焦点坐标所在位置
            getCanvasRelativePosition(event){
                const rect = this.canvas.getBoundingClientRect();
                return {
                    x: (event.clientX - rect.left) * this.canvas.width / rect.width,
                    y: (event.clientY - rect.top) * this.canvas.height / rect.height
                }
            },
            // 添加鼠标划过以及点击事件
            clickEvents(){
                window.addEventListener('click', this.clickPickPosition);   
                window.addEventListener('mousemove', this.setPickPosition);
            },
            // 创建点击事件(默认是离摄像头最近的相交)
            pickEvents(normalizedPosition, scene, camera, callback){
                // 如果存在拾取的对象,则恢复颜色
                if(this.events.pickedObject){
                    this.events.pickedObject.material.emissive.setHex(this.events.pickedObjectSavedColor);
                    this.events.pickedObject = undefined;
                }
                // 沿着摄像头的方向投射射线
                this.events.raycaster.setFromCamera(normalizedPosition, camera)
                // 获取与射线光线相交的对象列表
                const intersectedObjects = this.events.raycaster.intersectObjects(this.scene.children);
                if(intersectedObjects.length){
                    // 获取与射线光纤相交的第一个对象。也是最近的一个
                    this.events.pickedObject = intersectedObjects[0].object;
                    // 保存当前对象的颜色
                    this.events.pickedObjectSavedColor = this.events.pickedObject.material.emissive.getHex();
                    // 将其发射颜色设置为闪烁的红色/黄色
                    this.events.pickedObject.material.emissive.setHex(0xFFFF00)
                    
                    if(callback){
                        callback(this.events.pickedObject)
                    }
                }
            },
        }
    }
</script>

<style>
    html, body{
        height: 100%;
        margin: 0;
        background: #0033CC;
    }
    .c{
        width: 100%;
        height: 100%;
        display: block;
    }
</style>

最终展示效果:


鼠标划过
第一次点击选中几何体
第二次点击取消选中几何体
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352

推荐阅读更多精彩内容