3D语音天气球(源码分享)——完结篇

转载请注明本文出自大苞米的博客(http://blog.csdn.net/a396901990),谢谢支持!


开篇废话:


由于这篇文章是本系列最后一篇,有必要进行简单的回顾和思路整理。


这个程序是由两部分组成,Android端和Unity端:

1.Unity端负责3D球的创建,显示和旋转:3D语音天气球(源码分享)——创建可旋转的3D球

2.通过天气服务动态创建3D球:3D语音天气球(源码分享)——通过天气服务动态创建3D球

3.Android端使用第三方的语音服务来进行语音识别:3D语音天气球(源码分享)——在Unity中使用Android语音服务

4.Unity中加入Android项目:Unity中加入Android项目的Build步骤

5.Android端和Unity端通信交互:ANDROID应用中嵌入Unity3D视图(展示3D模型)

6.将Android端4个按钮和语音指令传入Unity中进行相应操作。


本文将介绍最后的第6部分:从Android端传入“指令”,并在Unity中处理。



效果图:


    

左边是Unity做出后在电脑上运行效果图,仅支持鼠标拖动。

右边是Unity结合Android和语音控制之后在手机运行的效果图,我们可以通过这4个按钮和触屏滑动来操作这个程序。下面来简单介绍一下这4个按钮的实现方法:



4个按钮:


Android端的4个按钮其实就是一个非常普通的布局。布局有两层,里面一层用来显示Unity视图,外面一层用来显示4个按钮(4个按钮也可以做到Unity端):



下面是Android代码中4个按钮的点击触发事件:

	// 语音点击监听
	public class voiceListener implements OnClickListener {

		@Override
		public void onClick(View arg0) {
			voiceResult = "";
			// 设置参数
			setParam();
			mVoice.startListening(voiceListener);
		}
	}

	// 返回点击监听
	public class returnListener implements OnClickListener {

		@Override
		public void onClick(View arg0) {
			UnityPlayer.UnitySendMessage("Main Camera", "back", "");
		}
	}

	// 详细点击监听
	public class detailListener implements OnClickListener {

		@Override
		public void onClick(View arg0) {
			UnityPlayer.UnitySendMessage("Main Camera", "detail", "");
		}
	}

	// 退出点击监听
	public class quitListener implements OnClickListener {

		@Override
		public void onClick(View arg0) {
			System.exit(0);
		}
	}


下面以详细(detail)为例简单介绍下:

UnityPlayer.UnitySendMessage("Main Camera", "detail", "");
这段代码就是用来和Unity通讯的方法。这个函数中接收三个参数:接收的Unity对象,调用的方法,方法接收的参数。


本例中4个按键的接收对象都是Main Camera,就是我们程序的主摄像机。

再来看看用来接收detail方法的Main Camera脚本中的代码:

	// 详细
	void detail(string str) {

		string currentCity = mySphere.currentCity;
		string currentCityID = mySphere.currentCityID;
		bool isCity = mySphere.isCity;
		Hashtable allCityID = mySphere.allCityID;

		if (isCity)
		{
			mySphere.getDetail(currentCityID);
		}
		else
		{
			mySphere.setNewCity((string[])allCityID[currentCity],true,"");
		}

	}
这段代码的作用就是加载详细的省市信息。

先拿到当前小球的省市名称和编号,然后根据现在是省,还是城市来做不同操作。

如果当前是省,在点击“详细”按钮后将显示当前省下所有城市的信息。

如果当前是市,则显示详细天气信息。


mySphere就是主摄像机中对于“3D”大球的一个引用对象。因为摄像机只有一个,所以这里我将它当作一个控制器(controller,来控制小球(model),和UI的显示(view)。也算是一个简单的MVC模式的实现。

四个按钮中“退出”“后退”“详细”都很简单,下面讲解一下比较重要的功能——“语音”



语音:


Android端:

点击“语音”按钮后会触发相应监听,然后调用语音服务解析语音命令,将其转化为文字,根据这些文字(本例中就是省市名称)查询数据,之后传递到Unity端。

Android端代码我在上篇文章介绍过,就不多说了。


Unity端:

接收传递过来的信息(就是省市编号),根据省市编号在3D球中查找对应的小球。然后将小球转到正对着我们的主摄像机,这样就完成了语音查询。


实现很简单,其实只有两部分:

1.查找

2.旋转


如下图所示,为了方便截图我设置了一个按钮来模拟语音点击。

当点击它时会查找“北京”,然后自动将“北京”小球旋转到正中央(会有一个速度减缓效果slerp插值



查找:

查找很简单,就是循环所有的小球,然后找到名字相匹配的。

			foreach (Transform child in gameObject.transform)
			{
				if (child.GetComponent<TextMesh> ().text.Equals(findName)) {

					// rotation logic
				}
			}


旋转:

旋转是本文的重点,也是当初困扰我好久的一个问题,因为我对Unity不是很熟,数学知识也忘光光了。


Unity中提供很多关于旋转方面的函数,想详细了解的话官方文档是个好东西。

下面简单介绍一下我计算角度的方法,过几天等我把旋转角度的相关内容研究明白后,我计划写一篇关于旋转方面的总结。


先看下面的图,我用Debug.DrawLine方法将大球圆心到摄像机,和大球圆心到查找小球标记为两条红色的射线。

				Debug.DrawLine(gameObject.transform.position, cameraTarget.position, Color.red);
				Debug.DrawLine(child.transform.position, gameObject.transform.position, Color.red);
  
如图,假设大球圆心为O,圆心O到查找球的射线为OA,圆心O到摄像机的射线为OB。


第一种方法:

方法一的核心就是Quaternion.Euler(x : float, y : float, z : float)这个函数,简单理解就是该函数会返回一个旋转角度。3个参数为绕x轴旋转x度,绕y轴旋转y度,绕z轴旋转z度


所以第一种方法的思路就是求出目标球需要绕多少度可以转到屏幕正中间。因为两次旋转就可以确定,所以euler(x,y,z)这个方法中随便算出两个值就行。

计算绕Y轴的角度:取A点在XOZ平面的投影A'点绕Y轴方向所形成的夹角

计算绕Z轴的角度:取A点在XOY平面的投影A'点绕X轴负方向所形成的夹角。

				// 重置大球的角度
				gameObject.transform.rotation = new Quaternion(0,0,0,0);

				// 测试辅助射线
				Debug.DrawLine(gameObject.transform.position, cameraTarget.position, Color.red);
				Debug.DrawLine(child.transform.position, gameObject.transform.position, Color.red);

				// 小球到大球的向量
				Vector3 targetDir = child.position - transform.position;

				// X轴负方向的向量
				Vector3 dirR = -Vector3.right;
				// Z轴负方向的向量
				Vector3 dirF = -Vector3.forward;

				// 计算绕Y轴的旋转角度
				float angleY = Vector3.Angle(new Vector3(targetDir.x, 0, targetDir.z), dirF);
				// 计算绕Z轴的旋转角度
				float angleZ = Vector3.Angle(new Vector3(targetDir.x, targetDir.y, 0), dirR);
				
				// 旋转大球
				gameObject.transform.localRotation = Quaternion.Euler (0, angleY, angleZ);
Vector3.Angle (from : Vector3, to : Vector3)函数 的作用是返回from和to之间的夹角。


第二种方法:

第二种方法就简单太多了,核心是Quaternion.FromToRotation (fromDirection : Vector3, toDirection : Vector3)方法,它的作用就是返回从向量fromDirection到toDirection所需的旋转角度。


所以思路是通过求出圆心到小球的向量OA,和圆心到摄像机的向量OB,然后计算两个夹角的角度。在将大球旋转这个角度就ok了,代码如下:

					// 圆心到目标球的向量
					Vector3 targetDir = child.position - transform.position;
					// 圆心到摄像机的向量
					Vector3 cameraDir = -Vector3.forward;
					// 求出两个向量的旋转角度
					Quaternion rotation = Quaternion.FromToRotation(targetDir, cameraDir) * transform.rotation;
					// 大球旋转插值旋转这个角度
					transform.rotation = Quaternion.Slerp(transform.rotation, rotation, Time.deltaTime * 3f);

上面的代码中应用了Quaternion.Slerp (from : Quaternion, to : Quaternion, t : float)方法,它会将刚刚计算的Quaternion旋转值以插值进行旋转(也就是类似与球面曲线那种先快后慢的效果)。

Slerp翻译为球形插值,我个人觉得就像Android动画中设置Interpolator差不多的感觉。



写在最后:


3D语音天气球的所有内容终于介绍完了。说实话,当写完第二篇就不爱写了,因为这个demo写的很早,思路和Unity的相关知识点也忘差不多了。最近工作也忙,加上要过年了也越来越懒。。。不过最终还是坚持下来并给这个算不上项目的项目画上了一个句号。

当初要做它就是为了参加一个比赛,虽然天气查询这个功能有点鸡肋,不过创意还是有的,所以还是抱着要获奖的心态去的。结果很不幸,连个名次都没有,当时肯定有点小郁闷,毕竟熬夜熬了一个月。不过现在回头再来看这个东西确实比较屎,也就一个毕设的水平,现在也就坦然了。。。

不过在做它的过程中收获还是很大的,最直接的体现就是对于“驱动学习”的理解,比如当时对Unity的向量,角度旋转等等都不会时,一点点去看文档,看帖子论坛,查Google。这样学习比直接看书看视频来的深刻的多。

扯了好多废话。。。最后想说的就是谢谢大家的支持!


Android端和Unity端代码都放在Github上了:https://github.com/a396901990/3D_Sphere/tree/feature/Voice_Weather_3D_Sphere


原文地址:https://www.cnblogs.com/zhangyunlin/p/6168033.html