本次学习:
1.第三人称视角研究
2.摇杆区域制作
3.视角限制设计
4.简单的战斗部分
1.第三人称视角研究
第三人称视角是什么大家玩玩游戏就懂了-0- 羽化不是专业制作人,只是一个玩家,下面是羽化玩家身份的总结出来的一些经验,很多词汇非专业,见谅。。。
1.人物移动 玩过魔兽世界的玩家可能都知道W A S D可以控制人物往8个方向移动,但人的面向不会改变,意味着有左移右移和后退等动作,羽化认为这样设计比较有真实感,而且效果很好,但有些游戏比如神鬼寓言就有所改变,W A S D不仅可以控制移动方向,而且还能改变人的面向,这种设计更加灵活,适合ACT,不适合ARPG,所以羽化这是做的一个ARPG,使用的是魔兽世界的移动标准,摇杆第一次点进区域就可自由移动,原来羽化就是这么想的,看到混乱与秩序的摇杆后觉得更加合理,所以就借鉴了下。
2.视角移动 ARPG也有很多种视角,有斜45°类似火炬之光这种,也有魔兽世界一样360°的,羽化原来做过斜45°的Demo,固定视角可以解决很多问题,但也伴随着更棘手的问题,既然公司要求,所以就做了这个360°的视角转换,包括手势放大缩小。视角改变很容易,但当视角更改以后问题出现了,玩过魔兽世界包括众多自由视角游戏的玩家都知道视角虽然可以随便移动,但存在着很多约束,比如你向上滑动时,视角不可能穿过地面而到地下,一定会沿着人的方向放大,这是前人给我们的经验,比如有一个物体挡在人物和玩家中间时,视角应该拉近,这样就不会产生视觉死角,诸如此类的情况很多,就运用到了射线Ray,进入我们今天的主题。
大家可以看到,羽化这个Demo用的东西也不多,这是前期的一个Demo,还不算完整,如果有机会给大家分享下羽化现在的Demo。
2.摇杆区域制作
触摸屏发展至今,羽化见过最好大的移动摇杆莫过于“混乱与秩序”的移动摇杆了,本身摇杆制作并不困难,主要是学习下这种为玩家带来方便的思维模式,先把代码送上:
Rocker.js:
- //移动速度
- varSpeed:float=0.05;
- //视角转动速度
- varSpeed_L:float=0.1;
- //角色
- varRole:Transform;
- //人称视角
- varPlayer_true:Transform;
- //攻击预设
- varAttackPrafab:Transform;
- //攻击方向
- varAttackRange:Transform;
- //第一点按下的初始点
- privatevarPO_X:int=0;
- privatevarPO_Y:int=0;
- //第二点按下的初始点
- privatevarPT_X:int=0;
- privatevarPT_Y:int=0;
- //判断人是否移动
- staticvarPR_M:boolean=false;
- //触摸点的记录
- privatevarPM_X:int=0;
- privatevarPM_Y:int=0;
- //第二点按下的初始点
- privatevarPS_X:int=0;
- privatevarPS_Y:int=0;
- //触摸点与中心点的距离
- privatevarM_X:int=0;
- privatevarM_Y:int=0;
- //中心的坐标
- privatevarPC_X:int=0;
- privatevarPC_Y:int=0;
- //旋转变量
- privatevartouchDeltaPosition:Vector2;
- //两点静态和动态距离
- privatevarDistance:int=0;
- privatevarDistance_D:int=0;
- //记录摄像机距离
- staticvarCamera_Record:float=-2.0;
- //两点的角度
- staticvarAngles:int=0;
- //战斗焦点
- staticvarFocus:boolean=false;
- //打击点
- privatevarhit:RaycastHit;
- //敌人方位
- privatevarEnemie:Vector3;
- //攻击模式范围
- privatevarRange:float=6.0;
- //记录时间
- privatevarCreationTime:double=-10.0;
- //处于攻击动作
- privatevarAttacking:boolean=false;
- //Test
- staticvarTest=0.0;
- functionUpdate()
- {
- //记录游戏时间
- //Test=Time.time;
- //战斗状态
- if(Focus)
- {
- if(Vector3.Distance(transform.position,Enemie)>Range)
- {
- Focus=false;
- Camera.main.depth=-1;
- return;
- }
- Camera.main.depth=-3;
- transform.LookAt(Enemie);
- Player_true.transform.LookAt(Enemie);
- }
- //攻击预设
- if(Time.time>(CreationTime+0.5)&&Role.animation.IsPlaying("Attack")&&Attacking)
- {
- Instantiate(AttackPrafab,AttackRange.transform.position,Quaternion.identity);
- Attacking=false;
- }
- //单点触摸时
- if(Input.touchCount==1)
- {
- Player_Move();
- Attack();
- //退出
- if(Input.GetTouch(0).position.y>Screen.height-50&&Input.GetTouch(0).position.x>Screen.width-50)
- {
- Application.Quit();
- }
- }
- //多点触摸时
- elseif(Input.touchCount==2)
- {
- Player_Move();
- Player_Look();
- Attack();
- }
- }
- //角色移动和静态视角
- functionPlayer_Move()
- {
- switch(Input.GetTouch(0).phase)
- {
- caseTouchPhase.Began:
- PO_X=Input.GetTouch(0).position.x;
- PO_Y=Input.GetTouch(0).position.y;
- PC_X=PO_X;
- PC_Y=PO_Y;
- varray=Camera.main.ScreenPointToRay(Input.GetTouch(0).position);
- if(Physics.Raycast(ray,hit))
- {
- if(hit.collider.gameObject.tag=="Enemies")
- {
- Focus=true;
- Enemie=hit.transform.position;
- }
- elseif((PO_X>200||PO_Y>200)&&(PO_X<Screen.width-100||PO_Y>100))
- {
- Focus=false;
- Camera.main.depth=-1;
- }
- }
- break;
- caseTouchPhase.Moved:
- PM_X=Input.GetTouch(0).position.x;
- PM_Y=Input.GetTouch(0).position.y;
- if(PO_X<200&&PO_Y<200&&PO_X>0&&PO_Y>0)
- {
- PR_M=true;
- }
- elseif(Input.touchCount==1)
- {
- touchDeltaPosition=Input.GetTouch(0).deltaPosition;
- //视角限制
- if(Player_true.transform.localEulerAngles.x<=60||Player_true.transform.localEulerAngles.x>=310)
- {
- Player_true.transform.Rotate(Vector3(touchDeltaPosition.y,touchDeltaPosition.x,0)*Time.deltaTime*100*Speed_L,Space.World);
- }
- elseif(Player_true.transform.localEulerAngles.x<310&&Player_true.transform.localEulerAngles.x>200)
- {
- Player_true.transform.localEulerAngles.x=311;
- }
- elseif(Player_true.transform.localEulerAngles.x>60&&Player_true.transform.localEulerAngles.x<200)
- {
- Player_true.transform.localEulerAngles.x=59;
- }
- Player_true.transform.localEulerAngles.z=0;
- }
- break;
- caseTouchPhase.Ended:
- PO_X=0;
- PO_Y=0;
- PM_X=0;
- PM_Y=0;
- PC_X=0;
- PC_Y=0;
- PR_M=false;
- Role.animation.Stop("Run");
- Role.animation.Stop("Back");
- Role.animation.Stop("Right");
- Role.animation.Stop("Left");
- Role.animation.PlayQueued("Wait",QueueMode.CompleteOthers);
- break;
- }
- if(PR_M)
- {
- M_X=PM_X-PC_X;
- M_Y=PM_Y-PC_Y;
- //求距离
- Distance=Mathf.Sqrt((M_X*M_X)+(M_Y*M_Y));
- //求角度
- Angles=Mathf.Atan2(M_X,M_Y)*Mathf.Rad2Deg;
- if(Distance>=50)
- {
- PC_X=PC_X*15/16+PM_X/16;
- PC_Y=PC_Y*15/16+PM_Y/16;
- }
- //最大速度限制
- if(M_X>40)
- {
- M_X=40;
- }
- elseif(M_X<-40)
- {
- M_X=-40;
- }
- if(M_Y>50)
- {
- M_Y=50;
- }
- elseif(M_Y<-30)
- {
- M_Y=-30;
- }
- //移动判断和优化
- if(Angles>=-45&&Angles<=45)
- {
- Role.animation.CrossFade("Run");
- if(Angles>=-20&&Angles<=20)
- {
- M_X=0;
- }
- }
- elseif(Angles>=135||Angles<=-135)
- {
- Role.animation.CrossFade("Back");
- if(Angles<=-160||Angles>=160)
- {
- M_X=0;
- }
- }
- elseif(Angles>45&&Angles<135)
- {
- Role.animation.CrossFade("Right");
- if(Angles>=70&&Angles<=110)
- {
- M_Y=0;
- }
- }
- elseif(Angles>-135&&Angles<-45)
- {
- Role.animation.CrossFade("Left");
- if(Angles>=-110&&Angles<=-70)
- {
- M_Y=0;
- }
- }
- //转身
- transform.localRotation=Quaternion.Euler(0,Player_true.transform.localEulerAngles.y,0);
- //vartarget=Quaternion.Euler(0,Player_true.transform.localEulerAngles.y,0);
- //transform.rotation=Quaternion.Slerp(transform.rotation,target,Time.deltaTime*2);
- //移动
- transform.Translate(M_X*Time.deltaTime*Speed,0,M_Y*Time.deltaTime*Speed);
- }
- }
- //游戏移动视角
- functionPlayer_Look()
- {
- switch(Input.GetTouch(1).phase)
- {
- caseTouchPhase.Began:
- PT_X=Input.GetTouch(1).position.x;
- PT_Y=Input.GetTouch(1).position.y;
- M_X=PO_X-PT_X;
- M_Y=PO_Y-PT_Y;
- Distance=Mathf.Sqrt((M_X*M_X)+(M_Y*M_Y));
- break;
- caseTouchPhase.Moved:
- PS_X=Input.GetTouch(1).position.x;
- PS_Y=Input.GetTouch(1).position.y;
- if(PR_M&&!Focus)
- {
- touchDeltaPosition=Input.GetTouch(1).deltaPosition;
- Player_true.transform.Rotate(Vector3(0,touchDeltaPosition.x,0)*Time.deltaTime*160*Speed_L,Space.World);
- Player_true.transform.Rotate(Vector3(touchDeltaPosition.y,0,0)*Time.deltaTime*90*Speed_L,Space.World);
- Player_true.transform.localEulerAngles.z=0;
- }
- elseif((PO_X>200||PO_Y>200)&&(PT_X>200||PT_Y>200))
- {
- M_X=PS_X-PM_X;
- M_Y=PS_Y-PM_Y;
- Distance_D=Mathf.Sqrt((M_X*M_X)+(M_Y*M_Y));
- if(Distance-Distance_D>20&&Camera.main.transform.localPosition.z<=0)
- {
- Camera.main.transform.localPosition.z+=0.1;
- Camera_Record=Camera.main.transform.localPosition.z;
- Distance=Distance_D;
- }
- elseif(Distance-Distance_D<-20&&Camera.main.transform.localPosition.z>=-4)
- {
- Camera.main.transform.localPosition.z-=0.1;
- Camera_Record=Camera.main.transform.localPosition.z;
- Distance=Distance_D;
- }
- }
- break;
- caseTouchPhase.Ended:
- PT_X=0;
- PT_Y=0;
- PS_X=0;
- PS_Y=0;
- Distance=0;
- Distance_D=0;
- break;
- }
- }
- //攻击判断
- functionAttack()
- {
- //普通攻击
- if(Input.GetTouch(0).position.y<100&&Input.GetTouch(0).position.x>Screen.width-100&&!(Role.animation.IsPlaying("Attack")))
- {
- Role.animation.Play("Attack");
- Role.animation.CrossFadeQueued("Wait",0.3,QueueMode.CompleteOthers);
- CreationTime=Time.time;
- Attacking=true;
- }
- //移动攻击
- elseif(PR_M&&Input.GetTouch(1).position.y<100&&Input.GetTouch(1).position.x>Screen.width-100&&!(Role.animation.IsPlaying("Attack")))
- {
- PR_M=false;
- Role.animation.CrossFadeQueued("Attack",0.3,QueueMode.PlayNow);
- Role.animation.CrossFadeQueued("Wait",0.3,QueueMode.CompleteOthers);
- CreationTime=Time.time;
- Attacking=true;
- }
- }
- functionOnGUI()
- {
- GUI.Box(Rect(0,Screen.height-200,200,200),"Rocker");
- GUI.Box(Rect(Screen.width-50,0,50,50),"Quit");
- GUI.Box(Rect(Screen.width-100,Screen.height-100,100,100),"Attack");
- }
啊,好长,其实也不多,就300行+,上面有羽化的一些注释,大家可以看看这个脚本,Rocker是绑在Player上的,大家可以从上一张图看到,羽化见了两个Player,一个叫Player_True的物体其实是一个空物体里面放置了一些基本东西包括光和摄像机,旋转视角的时候就是旋转的Player_True,其中的好处只有用过的人才能了解吧~ ~。羽化使用了预设制作攻击,这是个权益之计,以后说不定会改,这里最多判断了两个点的触控情况,羽化是分别判断的,这是吸取了前一个游戏的经验,这样不会乱,因为移动,攻击,转视角全部如果写在一个判断里面会造成很多冲突,羽化深有体会。
3.视角限制设计
前面部分羽化用不了半天就完成了,后面的视角限制花了将近2天时间,当然个人能力有限也是原因吧。这是我这次博客的重头,如何运用射线,我们主要说说Physics.Raycast这个脚本的应用,在Unity帮助文档里面提到了,这里羽化尝试了下,最好使用Raycast (origin :Vector3,direction :Vector3,out hitInfo :RaycastHit,distance : float =Mathf.Infinity,layerMask : int = kDefaultRaycastLayers),这个方法,别的不一定靠谱。。。有的地面会莫名其妙穿过去。
CameraZoom.js:
- varUp:Transform;
- privatevarForward:boolean=false;
- privatevarBack:boolean=false;
- functionUpdate()
- {
- //通过向下和前后激光判断视角
- varlayerMask=1<<2;
- layerMask=~layerMask;
- varhit:RaycastHit;
- //向前激光 判断中间阻挡的镜头靠近
- if(Physics.Raycast(transform.position,Camera.main.transform.TransformDirection(Vector3.forward),hit,(-Camera.main.transform.localPosition.z)))
- {
- Forward=true;
- vardistanceToForward=hit.distance;
- if(Camera.main.transform.localPosition.z<-0.5)
- {
- Camera.main.transform.localPosition.z+=0.05;
- }
- }
- else
- {
- Forward=false;
- }
- //向后激光 判断后退时靠墙的镜头拉近
- if(Physics.Raycast(transform.position,Camera.main.transform.TransformDirection(Vector3.forward*(-1)),hit,0.2,layerMask))
- {
- Back=true;
- vardistanceToBack=hit.distance;
- if(distanceToBack>0.1)
- {
- Back=false;
- }
- }
- //向下激光 用于判断下拉镜头的拉近
- if(Physics.Raycast(transform.position,-Vector3.up,hit,(-2)*Rocker.Camera_Record,layerMask))
- {
- vardistanceToGround=hit.distance;
- //print(distanceToGround);
- }
- if(distanceToGround<0.1)
- {
- Camera.main.transform.localPosition.z+=0.1;
- }
- elseif(Camera.main.transform.localPosition.z>Rocker.Camera_Record&&distanceToGround>0.2&&!Back&&!Forward)
- {
- Camera.main.transform.localPosition.z-=0.02;
- Camera.main.transform.localPosition.y=0.6;
- }
- if(!Rocker.Focus)
- {
- transform.LookAt(Up);
- }
- }
- functionOnTriggerStay(other:Collider)
- {
- if(/*other.attachedRigidbody&&*/Camera.main.transform.localPosition.z<-0.4)
- {
- Camera.main.transform.localPosition.z+=0.02;
- }
- Back=true;
- }
- functionOnTriggerExit(other:Collider)
- {
- Back=false;
- }
首先,羽化开始认为只有一个射线就可以搞定了,结果一个射线什么都判断不了,后来羽化仔细想了想就写成了三个射线,加上一个判断,终于效果出现了,话说羽化在摄像机设置Collider,是怕一个不小心,玩家把整个地图看透了-0- 如果你想视角更舒畅点可以把other.attachedRigidbody的注释去掉。这里有个函数layerMask是判断射线穿过那个层的,以后肯定用得上,1<<2代码射线只能穿过前两层。
4.简单的战斗部分
战斗部分很简单,做一个攻击预设,触发攻击动作时就产生预设,在极短时间给怪物造成伤害。绝大多数代码都在移动部分了,这里看看一个简单的AI。
AI.js:
- varrange=4;
- varPlayer:Transform;
- privatevarhealth=10;
- functionUpdate()
- {
- Discover();
- }
- functionOnTriggerEnter(other:Collider)
- {
- if(other.attachedRigidbody)
- {
- if(other.gameObject.tag=="Attack")
- {
- Rocker.Test=10.2;
- transform.Find("Small").animation.CrossFade("Hit");
- transform.Find("Small").animation.CrossFadeQueued("Wait",0.3,QueueMode.CompleteOthers);
- health--;
- if(health<=0)
- {
- transform.Find("Small").animation.CrossFade("Dead");
- Rocker.Focus=false;
- Camera.main.depth=-1;
- transform.tag="Dead";
- }
- }
- }
- }
- functionDiscover()
- {
- varlayerMask=1<<3;
- layerMask=~layerMask;
- if(Vector3.Distance(transform.position,Player.position)>range)
- {
- returnfalse;
- }
- varhit:RaycastHit;
- if(Physics.Linecast(transform.position,Player.position,hit,layerMask))
- {
- if(hit.collider.gameObject.tag=="Player"&&!(transform.Find("Small").animation.IsPlaying("Dead")))
- {
- transform.LookAt(Player);
- returntrue;
- }
- else
- {
- returnfalse;
- }
- }
- returntrue;
- }
怪物在一定范围内会发现你,面朝向敌人,简单的AI,可以添加怪物行走等,这就看大家对自己什么要求了,当玩家点击怪物后会进入战斗视角,这时的移动时围绕怪物的,是不是很像Fable~ ~羽化的Demo就初步完成了,最后来张截图。
自己认为视角做得很棒,有什么不足还望指出~ ~
本文转自羽化:http://blog.csdn.net/libeifs/article/details/6696424
APK下载地址: