转载请注明本文出自大苞米的博客(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

3D语音天气球(源码分享)——完结篇的更多相关文章

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

    转载请注明本文出自大苞米的博客(http://blog.csdn.net/a396901990),谢谢支持! 开篇废话: 这个项目准备分四部分介绍: 一:创建可旋转的"3D球":3 ...

  2. 3D语音天气球(源码分享)——在Unity中使用Android语音服务

    转载请注明本文出自大苞米的博客(http://blog.csdn.net/a396901990),谢谢支持! 开篇废话: 这个项目准备分四部分介绍: 一:创建可旋转的"3D球":3 ...

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

    转载请注明本文出自大苞米的博客(http://blog.csdn.net/a396901990),谢谢支持! 开篇废话: 这个项目准备分四部分介绍: 一:创建可旋转的"3D球":3 ...

  4. 3D语音天气球(源码分享)——创建可旋转的3D球

    开篇废话: 在9月份时参加了一个网站的比赛,比赛的题目是需要使用第三方平台提供的服务做出创意的作品. 于是我选择使用语音服务,天气服务,Unity3D,Android来制作一个3D语音天气预报,我给它 ...

  5. 微信小程序——智能小秘“遥知之”源码分享(语义理解基于olami)

    微信小程序智能生活小秘书开发详解 >>>>>>>>>>>>>>>>>>>>> ...

  6. 支持语音识别、自然语言理解的微信小程序(“遥知之”智能小秘)完整源码分享

    记录自己搭建https的silk录音文件语音识别服务的调用过程,所有代码可在文中找链接打包下载 >>>>>>>>>>>>> ...

  7. 推荐!Html5精品效果源码分享

    一直在看别人的汇总,看到了一些不错的关于 HTML5内容的源码,我也汇总下分享出来,好东西需要共享!希望可以帮到需要的朋友. 1.劲爆分享:HTML5动感的火焰燃烧动画特效 这又是一款基于HTML5的 ...

  8. 全方位深度剖析PHP7底层源码(已完结)

    第1章 课程介绍本章主要介绍课程要讲的知识点,以及课程要求等. 第2章 PHP7的新特性本章主要介绍PHP7的新特性,做基准测试,与PHP5对比验证PHP7的性能提升程度,引出对PHP7源码学习的必要 ...

  9. BAT资深工程师 由浅入深分析 Tp5&Tp6底层源码 - 分享

    BAT资深工程师由浅入深分析Tp5&Tp6底层源码 第1章 课程简介 本章主要让大家知道本套课程的主线, 导学内容,如何学习源码等,看完本章要让小伙伴觉得这个是必须要掌握的,并且对加薪有很大的 ...

随机推荐

  1. 回退(pop&present)到根页面(根控制器)的方法,很不错~

    http://blog.csdn.net/assholeu/article/details/45897035

  2. 关于cocoa框架,你所要知道的一切(苹果官方文档,cocoa框架核心竞争力,必须收藏!)

    https://developer.apple.com/library/ios/documentation/General/Conceptual/DevPedia-CocoaCore/Accessib ...

  3. 【Java 基础篇】【第九课】继承

    继承就是为了提高代码的复用率. 利用继承,我们可以避免代码的重复.让Woman类继承自Human类,Woman类就自动拥有了Human类中所有public成员的功能.我们用extends关键字表示继承 ...

  4. Android笔记:java 中的数组

    在与嵌入式设备通讯的过程中使用的socket通讯 获取的字节流,通常转换为字节数组,需要根据协议将字节数组拆分.对于有规律的重复拆分可以使用,由于java中不能像c中直接进行内存操作例如使用struc ...

  5. JQuery 内容过滤选择器

    <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <m ...

  6. cocoapods没有自动补齐

    在install cocoapods成功之后,输入头文件名的时候,没有提示,这样,我还以为是我的cocoapods安装出错了.原来是配置没有配置好. 解决办法: 选择工程的 Target -> ...

  7. PySe-004-Se-WebDriver 启动浏览器之二 - Chrome

    上篇文章简略讲述了 WebDriver 启动 firefox 浏览器的示例脚本源码,具体请参阅: PySe-003-Se-WebDriver 启动浏览器之一 - Firefox 此文主要讲述在 Mac ...

  8. django 部署到 apache

    安装完django之后,每次都需要通过命令来启动启动开发服务器.虽然调试和测试方便,但只能在本地运行,并且不能承受许多用户同时使用的负载.所以需要将Django部署到生产级的服务器,这里选择apach ...

  9. 移动端a链接点击时取出背景色及边框

    a{blr:expression(this.onFocus=this.blur())} :focus{outline:0;} /*去掉a标签的虚线框,避免出现奇怪的选中区域*/*{-webkit-ta ...

  10. 跳到下个View

    nextWebView = [[ WEBViewController alloc ] initWithNibName : @"WEBViewController" bundle : ...