图形化开发(九)01-Three.js之案例——王者荣耀demo制作

效果


demo目录结构

1. 场景搭建

scene = new THREE.Scene();
scene.background = new THREE.Color(0xa0a0a0);
scene.fog = new THREE.Fog(0xa0a0a0, 1000, 11000);

我们创建了场景,并设置了场景一个灰色的背景色。还设置了场景的雾化效果,这个雾的效果主要是针对于场景的相机的距离实现的,三个值分别是雾的颜色、雾的开始距离、完全雾化距离相机的位置。

2. 创建相机

我们创建了一个与地面呈45度角并朝向原点的相机:

camera = new THREE.PerspectiveCamera(45, window.innerWidth / window.innerHeight, 0.1, 20000);
camera.position.set(0, 800, -800);
camera.lookAt(new THREE.Vector3());

3. 创建灯光

我们创建了两个灯光:照射全局的环境光和可以产生阴影的平衡光。

scene.add(new THREE.AmbientLight(0x444444));light = new THREE.DirectionalLight(0xaaaaaa);
light.position.set(0, 200, 100);
light.lookAt(new THREE.Vector3());light.castShadow = true;
light.shadow.camera.top = 180;
light.shadow.camera.bottom = -180;
light.shadow.camera.left = -180;
light.shadow.camera.right = 180;//告诉平行光需要开启阴影投射
light.castShadow = true;scene.add(light);

4. 创建草地

我们使用平面几何体创建了一个贴有草皮贴图的材质的模型:

var groundTexture = new THREE.TextureLoader().load('../images/grasslight-big.jpg');
groundTexture.wrapS = groundTexture.wrapT = THREE.RepeatWrapping;
groundTexture.repeat.set(25, 25);
groundTexture.anisotropy = 16;
var groundMaterial = new THREE.MeshLambertMaterial({map: groundTexture});
var mesh = new THREE.Mesh(new THREE.PlaneBufferGeometry(20000, 20000), groundMaterial);
mesh.rotation.x = -Math.PI / 2;
mesh.receiveShadow = true;
scene.add(mesh);

到这里,场景、灯光、相机、舞台都已经备齐。接下来我们将请出我们主角naruto登场。

5. 添加人物模型

接下来主人公登场,首先我们将模型导入到场景内,注意,案例中的模型比较大,加载和处理需要一定的时间,请小伙伴们耐心等待即可:

var loader = new THREE.FBXLoader();loader.load("../js/models/fbx/Naruto.fbx", function (mesh) {scene.add(mesh);
});

我们不单单只是将模型添加到场景,还对模型的阴影和位置做了一下调整:

//设置模型的每个部位都可以投影
mesh.traverse(function (child) {if (child.isMesh) {child.castShadow = true;child.receiveShadow = true;}
});

调整模型的位置,站立在草地上面

mesh.position.y += 110;

设置灯光一直照射模型:

//设置光线焦点模型
light.target = mesh;

6. 添加动画

这个模型里面含有27个骨骼动画,我们可以通过设置不同的动画,来实现一整套的动作来实现相应的比如攻击效果,移动效果等。接下来我们通过模型的数据生成一下所需的动画:

actions = []; //所有的动画数组for (var i = 0; i < mesh.animations.length; i++) {createAction(i);
}function createAction(i) {actions[i] = mixer.clipAction(mesh.animations[i]);gui["action" + i] = function () {for (var j = 0; j < actions.length; j++) {if (j === i) {actions[j].play();}else {actions[j].stop();}}};
}//添加暂停所有动画的按键
gui.stop = function () {for (var i = 0; i < actions.length; i++) {actions[i].stop();}
};

模型加载成功后,我们需要让模型执行一个普通的站立效果:

//第24个动作是鸣人站立的动作
gui["action" + 24]();

7. 添加操作

在案例中,我们主要添加了两种操作:模型位置移动操作和攻击效果。
操作按钮为了方便,直接使用的dom标签模拟出来的。
模型位置移动操作中,我们需要模型的位置的变动和模型的朝向以及修改站立动画和奔跑动画的切换。
攻击效果则是实现攻击并且根据点击速度实现一整套的攻击动作切换。

实现位移

在实现位置移动效果中,我们为按钮绑定了三个事件:鼠标按下,鼠标移动,鼠标抬起。
在鼠标按下时,我们获取到了当前操作圆盘的中心点的位置,让模型进入跑步动画,绑定了鼠标的移动和抬起事件。重要的是更新模型的移动方向和移动速度。

dop.$(control).on("down", function (event) {event.preventDefault();//获取当前的按钮中心点center.x = window.innerWidth - parseFloat(dop.getFinalStyle(control, "right")) - parseFloat(dop.getFinalStyle(control, "width")) / 2;center.y = window.innerHeight - parseFloat(dop.getFinalStyle(control, "bottom")) - parseFloat(dop.getFinalStyle(control, "height")) / 2;getRadian(event);//鼠标按下切换跑步动作state.skills === 0 && gui["action" + 3]();//给document绑定拖拽和鼠标抬起事件doc.on("move", move);doc.on("up", up);
});

上面的dop类是一个兼容多端的事件库。
在鼠标移动回调事件中,我们更新模型的移动方向和移动速度。

function move(event) {getRadian(event);
}

最后在鼠标抬起事件中,我们解绑事件,将按键复原,并停止掉模型的移动状态,将模型动画恢复到站立状态。

function up() {doc.remove("move", move);doc.remove("up", up);//按钮复原bar.style.marginTop = 0;barWrap.style.transform = `translate(-50%, -50%) rotate(0deg)`;bar.style.transform = `translate(-50%, -50%) rotate(0deg)`;//设置移动距离为零characterMove(new THREE.Vector2(), 0);//鼠标抬起切换站立状态state.skills === 0 && gui["action" + 24]();
}

三个事件绑定完成后,我们需要将在回调中获得的值求出当前的偏转方向和移动速度:
首先我们获取一下当前鼠标的位置:

if (media === "pc") {mouse.x = event.clientX;mouse.y = event.clientY;
}
else {mouse.x = event.touches[0].clientX;mouse.y = event.touches[0].clientY;
}

根据位置求出距离操作圆盘中心的位置,并保证最大值也不会超出圆盘的半径:

let distance = center.distanceTo(mouse);
distance >= parseFloat(dop.getFinalStyle(control, "width")) / 2 && (distance = parseFloat(dop.getFinalStyle(control, "width")) / 2);

计算出来当前位置和中心的夹角,并修改dom的位置:

//计算两点之间的夹角
mouse.x = mouse.x - center.x;
mouse.y = mouse.y - center.y;//修改操作杆的css样式
bar.style.marginTop = `-${distance}px`;
bar.style.transform = `translate(-50%, -50%) rotate(-${(mouse.angle() / Math.PI * 180 + 90) % 360}deg)`;
barWrap.style.transform = `translate(-50%, -50%) rotate(${(mouse.angle() / Math.PI * 180 + 90) % 360}deg)`;

函数的最后,则调用的characterMove方法,将按钮数据转换成为模型实际需要移动的距离。

//修改当前的移动方向和移动速度
characterMove(mouse.normalize(), distance / (parseFloat(dop.getFinalStyle(control, "width")) / 2));

接下来我们查看一下characterMove方法,在这个方法中,我们计算出了模型每一帧需要移动的距离。这里有一个问题,我们所谓的操作杆向前让模型移动前方,其实是相机朝向的前方。所以我们需要先求出相机的前方矢量,再通过相机的前方矢量为基础,计算出来模型实际方向。
我们首先声明了两个变量,一个是旋转矩阵,另一个是移动矢量:

let direction = new THREE.Matrix4(); //当前移动的旋转矩阵
let move = new THREE.Vector3(); //当前位置移动的距离

characterMove函数内,我们根据相机的四元数获得了旋转矩阵:

//重置矩阵
direction.identity();//通过相机的四元数获取到相机的旋转矩阵
let quaternion = camera.quaternion;
direction.makeRotationFromQuaternion(quaternion);

然后通过旋转矩阵和当前的操作杆的方向通过相乘计算出来实际模型移动的方向:

//获取到操作杆的移动方向
move.x = vector.x;
move.y = 0;
move.z = vector.y;//通过相机方向和操作杆获得最终角色的移动方向
move.applyMatrix4(direction);
move.normalize();

最后,通过比例和方向得出当前模型每一帧移动的距离,因为我们不需要修改模型y轴,所以实际上也只是修改两个轴的位置:

move.x = move.x * ratio * 10;
move.z = move.z * ratio * 10;

我们获取到了模型的每一帧移动的距离,还需要在帧循环中调用:

//如果模型添加成功,则每帧都移动角色位置
if (naruto) {//获取当前位置position.x += move.x;position.z += move.z;//修改模型位置naruto.position.x = position.x;naruto.position.z = position.z;//修改平衡光的位置light.position.x = position.x;light.position.z = position.z + 100;//修改相机位置camera.position.x = position.x;camera.position.z = position.z - 800;
}

当前的模型,灯光,和相机都会跟随移动,实现了,我们上面动图中的模型移动的效果

实现攻击效果

在实现攻击效果时,我没有只是简单的实现一个普通的攻击,而是直接实现一套连招。
这一套连招是通过五个动作组成,在执行一个攻击动画时如果再次点击了攻击按钮,执行完这个攻击动画将不会切换到站立动画,而是直接切换到连招的下一个攻击动画中。
只要连续点按攻击按钮,模型将完成一整套的动作。实现这个效果,我们只是使用了一个简单的定时器即可实现,接下来我们通过代码了解一下实现过程。

在实现动画前,先设置一个连招的数组,将需要的动作添加到数组当中。我这里添加了五个手部攻击的效果:

let attackList = [12, 13, 14, 15, 16]; //连招的循序
let attackCombo = false; //是否连招,接下一个攻击

我们还设置了attackCombo设置当前是否可以连招的变量,这个变量state.skills值不为0时,将变为true。定时器每一次更新的时候,将判断attackCombo是否为true,在为true的状态下,将执行连招的下一个动作。否则,将停止连招。

//attackIndex 等于0,当前不处于攻击状态  不等于,当前处于攻击状态
if(state.skills === 0){state.skills++;gui["action" + attackList[state.skills-1]]();attackInterval = setInterval(function () {if(attackCombo){//如果设置了连招,上一个攻击动作完成后,进行下一个攻击动作state.skills++;//如果整套攻击动作已经执行完成,则清除定时器if(state.skills-1 >= attackList.length){closeAttack();return;}//进行下一个动作gui["action" + attackList[state.skills-1]]();attackCombo = false;}else{closeAttack();}}, naruto.animations[attackList[state.skills-1]].duration*1000);
}
else{attackCombo = true;
}

在关闭掉攻击动画的函数内,我们首先将state.skills设置为0,然后恢复到移动或者站立动画,最后清除掉定时器:

//关闭攻击状态
function closeAttack() {state.skills = 0;//根据状态设置是移动状态还是站立状态state.move ? gui["action" + 3]() :gui["action" + 24](); //回到站立状态clearInterval(attackInterval);
}

图形化开发(九)01-Three.js之案例——王者荣耀demo制作相关推荐

  1. 图形化开发(五)021-Three.js之材质——不受光影响-MeshBasicMaterial-同颜色MeshNormalMaterial-方向不同颜色LineBasicMaterial线条材质

    图形化开发(五)021-Three.js之材质--不受光影响-MeshBasicMaterial-整个物体颜色一样&MeshNormalMaterial-方向不同自动改变颜色 & Li ...

  2. 图形化开发(五)022-Three.js之材质——受光影响-MeshLambertMaterial 兰伯特材质 MeshPhongMaterial 高光材质

    图形化开发(五)022-Three.js之材质--受光影响-MeshLambertMaterial 兰伯特材质 & MeshPhongMaterial 高光材质 添加光 由于MeshBasic ...

  3. 图形化开发(一)——Three.js基本介绍-优缺点-在线编辑器 Babylon.JS是最好的JavaScript3D游戏引擎

    图形化开发(一)--Three.js基本介绍-优缺点-在线编辑器 & Babylon.JS是最好的JavaScript3D游戏引擎 课程主要学习目标 Threejs ( 3d ) D3 (做数 ...

  4. 图形化开发(六)01-Three.js之导入模型——3dmax和SketchUp-editor编辑器导出json文件,在创建模型initMesh中外部的JSON文件

    图形化开发(六)01-Three.js之导入模型--3dmax和SketchUp-editor编辑器导出json文件,在创建模型initMesh中外部的JSON文件 导入模型 官方推荐我们使用的3D模 ...

  5. 图形化开发(一)——D3.js的基本介绍、技术原理

    图形化开发(一)--D3.js的基本介绍.技术原理 D3.js 为什么学习D3 中文官网--https://www.d3js.org.cn/ github--https://github.com/d3 ...

  6. Arduino的图形化开发环境: ArduBlock

    转载自http://youngmakers.cn/groups/articles/54f9466b9835fed6656d4dd6 ArduBlock是一款为Arduino设计的开源图形化编程软件,由 ...

  7. 【MSP430G2553】图形化开发笔记(2) 系统时钟和低功耗模式

    目录 系统时钟概述 BCS+模块单元的基本构造 时钟-概览 介绍 Basic User 模式 Power User 模式 1. 数控振荡器 DCO 2. 出厂预校正频率 3. 低频振荡器 VLO 4. ...

  8. Scratch(图形化编程工具)绘制3D六面体,3D作品的制作过程!

    这个作品是在之前3D三面体的基础上的升级版. 优秀课程案例:使用Scratch绘制3D多面体! 适合想要提供自己编程能力,进阶学习的同学,不适合初级小朋友学习. 下边我们看一下代码实现部分: 这个截图 ...

  9. vue+node.js+moogodb仿王者荣耀移动端项目一

    注:本项目根据全栈之巅bilibili学习写的 学习链接https://www.bilibili.com/video/av51931842?from=search&seid=155804149 ...

  10. html+css+js写一个王者荣耀积分夺宝

    在游戏里面,积分夺宝是很抽中好东西的,于是想自己写一个页面,实现无限抽奖,而且必中荣耀水晶. 在线演示地址 先看效果(完整代码在后面): 开始要选择大区: 抽一次: 抽十次(一下中四个荣耀水晶,太爽了 ...

最新文章

  1. springboot 学习笔记(三)
  2. win7 docker java开发环境变量_java – Docker. Spring应用程序.设置和获取环境变量
  3. rodbc 连接oracle,在R中加密密码 – 使用RODBC连接到Oracle DB
  4. linux 全新编译安装,全新linux中通过编译方式安装nginx
  5. Fragment 与 Fragment 相互传值
  6. OpenJudge NOI 1.1 10:超级玛丽游戏
  7. mybatis insert返回主键_MyBatis官方文档XML 映射文件
  8. 独立ip 公司文件服务器,企业在外贸建站时,为什么要选择独立IP的服务器?
  9. SSM商城系统开发笔记-配置01-web.xml
  10. CentOS 7.2下Filebeat+Kafka+ELK生产部署(安全加固)
  11. Helix Streaming Server 简单配置
  12. 算法设计之数字三角形问题
  13. 移动终端浏览器初始设置apple-mobile-web-app-capable(转)
  14. 建站神器:使用Hexo+Kaze+Gitee 自建博客
  15. 教程丨GIS制图教程01
  16. Python淘宝App详情采集接口
  17. java中 enum什么意思_enum在java中是什么意思
  18. Unity3d中的走路动画
  19. 如何设置本电脑中的mysql让别人的电脑连接
  20. C语言编写万年历,解决1582年历史问题

热门文章

  1. ffmpeg批量合并ts文件为一个视频
  2. Latex输出大小写罗马数字
  3. 设计模式-word版
  4. 英语单词词性顺口溜_英语十大词性顺口溜
  5. 超级炫酷的3D旋转美女图——Python实现
  6. 电脑壁纸尺寸比例_怎么设置桌面壁纸尺寸比例
  7. 无法删除文件,无法读源文件或磁盘
  8. linux打开笔记本摄像头驱动程序,Linux下使用Opencv打开笔记本摄像头
  9. 什么是ETL?ETL是什么技术?
  10. Hyperledger Fabric Transaction Proposal过程