在Cocos2dx中。对大图的处理已经封装好了一套自己的API,可是在Unity3D中貌似没有类似的API(好吧,实际上是有的,并且功能更强大),或者说我没找到。

只是这也在情理之中,毕竟Unity3D是做3D的。要分割图片的地方还是非常少的。

由于我用Unity3D主要是用于做2D游戏的(PS:非常蛋疼吧?我也认为),所以就不得不考虑切图和播放序列帧这两个在2D上常见的功能了,以下废话不多说。

我的任务是把以下这张图分割成16块。而且依照动画的序列播放出来。

查Unity3D的使用手冊的过程中,我发现了一个类:Texture2D,他是继承Texture的,主要是用于创建2D纹理的,很符合切图的须要。

首先,我们须要载入大图。载入大图有一个很easy的方法。就是创建一个public的Texture2D类成员变量,然后在编辑器中直接拖动到上去给他赋值就能够了。

当然也能够採用动态载入图片资源的方法,这样的方法比較麻烦。须要把图片先转换成二进制流,然后赋值给Texture2D

	//载入图片资源
void LoadTexture()
{
using (FileStream file = File.Open (Application.dataPath + "/Textures/Player.png", FileMode.Open))
{
using (BinaryReader reader = new BinaryReader(file))
{
m_texPlayer = new Texture2D (192, 256, TextureFormat.ARGB4444, false);
texture.LoadImage (reader.ReadBytes((int)file.Length));
}
}
}

载入完之后就要分割了。主要思路就是,两个for循环,一个表示行,一个表示列,然后再循环每一个像素点。把每一个像素点里面的颜色复制出来给分割的Texture2D,最后把Texture2D组合成一个4x4的矩阵数组。

以下是第一步:

        for (int i = 0; i < m_iMinPicColumnCount; ++i)
{
for (int j = 0; j < m_iMinPicRowCount; ++j)
DePackTexture(i, j);
}

上面的终于处理调用了一个DePackTexture,这个函数是用于实际上的分割的。

    //切图
void DePackTexture(int i, int j)
{
int cur_x = i * m_iMinPicWidth;
int cur_y = j * m_iMinPicHeight; Texture2D newTexture = new Texture2D(m_iMinPicWidth, m_iMinPicHeight); for (int m = cur_y; m < cur_y + m_iMinPicHeight; ++m)
{
for (int n = cur_x; n < cur_x + m_iMinPicWidth; ++n)
{
newTexture.SetPixel(n - cur_x, m - cur_y, m_texPlayer.GetPixel(n, m));
}
}
newTexture.Apply();
m_texPlayers[i, j] = newTexture;
}

切图值得注意的就是两点,一点就是找好位置,还有一点就是运行完SetPixel操作后一定要运行Apply,不然是没有效果的。

--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

以下是帧序列动画,帧序列动画实际上就是将图片依照一定的顺序载入上去,值得注意的是全部的GUI操作一定要放到OnGUI里面。

    void DrawAnimation(Texture[,] tex, Rect rect)
{
//绘制当前帧
GUI.DrawTexture(rect, tex[m_iCurFram, m_iCurAnimation], ScaleMode.StretchToFill, true, 0.0f);
//计算限制帧的时间
m_fTime += Time.deltaTime;
//超过限制帧切换贴图
if (m_fTime >= 1.0 / m_fFps && m_bStop == false)
{
//帧序列切换
m_iCurFram = ++m_iCurFram % m_iMinPicRowCount;
//限制帧清空
m_fTime = 0;
//超过帧动画总数从第0帧開始
if (m_iCurFram >= tex.Length)
{
m_iCurFram = 0;
}
}
}

然后没有什么了,代码还是非常easy的。以下附上所有的代码。这个我做成了一个小的demo,含有动画的開始和暂停功能。并且还有动画的帧速调整的功能。

(最后会附上demo的地址)

using UnityEngine;
using System.Collections;
using System; public class CTexture : MonoBehaviour
{
//大图的人
public Texture2D m_texPlayer;
//小图的人
private Texture2D[,] m_texPlayers;
//当前帧
private int m_iCurFram;
//当前动画
private int m_iCurAnimation;
//限制帧的时间
private float m_fTime = 0; //小图的宽和高
public int m_iMinPicWidth = 48;
public int m_iMinPicHeight = 64;
//一行有多少个小图
public int m_iMinPicRowCount = 4;
//一列有多少个小图
public int m_iMinPicColumnCount = 4; //动画控制
//暂停
private bool m_bStop = false;
//一秒多少帧
private float m_fFps = 4; private string m_sFps = ""; void Start()
{
m_texPlayers = new Texture2D[4, 4];
m_iCurAnimation = 0;
m_sFps = m_fFps.ToString();
//载入图片资源
LoadTexture(); for (int i = 0; i < m_iMinPicColumnCount; ++i)
{
for (int j = 0; j < m_iMinPicRowCount; ++j)
DePackTexture(i, j);
}
} void Update()
{
if(Input.GetKeyDown(KeyCode.A))
{
m_iCurAnimation = 2;
}
if (Input.GetKeyDown(KeyCode.S))
{
m_iCurAnimation = 3;
}
if (Input.GetKeyDown(KeyCode.W))
{
m_iCurAnimation = 0;
}
if (Input.GetKeyDown(KeyCode.D))
{
m_iCurAnimation = 1;
}
} void OnGUI()
{
DrawAnimation(m_texPlayers, new Rect(100, 100, m_iMinPicWidth, m_iMinPicHeight)); if(GUI.Button(new Rect(200,20,80,50),"開始/暂停"))
{
m_bStop = m_bStop == false ? true : false ;
} m_sFps = GUI.TextField(new Rect(200, 80, 80, 40), m_sFps);
if (GUI.Button(new Rect(200, 150, 50, 40), "应用"))
{
m_fFps = float.Parse(m_sFps);
}
} //载入图片资源
void LoadTexture()
{
using (FileStream file = File.Open (Application.dataPath + "/Textures/Player.png", FileMode.Open))
{
using (BinaryReader reader = new BinaryReader(file))
{
m_texPlayer = new Texture2D (192, 256, TextureFormat.ARGB4444, false);
texture.LoadImage (reader.ReadBytes((int)file.Length));
}
}
} //切图
void DePackTexture(int i, int j)
{
int cur_x = i * m_iMinPicWidth;
int cur_y = j * m_iMinPicHeight; Texture2D newTexture = new Texture2D(m_iMinPicWidth, m_iMinPicHeight); for (int m = cur_y; m < cur_y + m_iMinPicHeight; ++m)
{
for (int n = cur_x; n < cur_x + m_iMinPicWidth; ++n)
{
newTexture.SetPixel(n - cur_x, m - cur_y, m_texPlayer.GetPixel(n, m));
}
}
newTexture.Apply();
m_texPlayers[i, j] = newTexture;
} void DrawAnimation(Texture[,] tex, Rect rect)
{
//绘制当前帧
GUI.DrawTexture(rect, tex[m_iCurFram, m_iCurAnimation], ScaleMode.StretchToFill, true, 0.0f);
//计算限制帧的时间
m_fTime += Time.deltaTime;
//超过限制帧切换贴图
if (m_fTime >= 1.0 / m_fFps && m_bStop == false)
{
//帧序列切换
m_iCurFram = ++m_iCurFram % 4;
//限制帧清空
m_fTime = 0;
//超过帧动画总数从第0帧開始
if (m_iCurFram >= tex.Length)
{
m_iCurFram = 0;
}
}
}
}

demo地址:http://download.csdn.net/detail/baijiajie2012/8092625

【跟我一起学Unity3D】代码中分割图片而且载入帧序列动画的更多相关文章

  1. java如何从一段html代码中获取图片的src路径

    java如何从一段html代码中获取图片的src路径 package com.cellstrain.icell.Test; import java.util.ArrayList;import java ...

  2. 示例:WPF中自定义StoryBoarService在代码中封装StoryBoard、Animation用于简化动画编写

    原文:示例:WPF中自定义StoryBoarService在代码中封装StoryBoard.Animation用于简化动画编写 一.目的:通过对StoryBoard和Animation的封装来简化动画 ...

  3. JAVA-替换html中图片的路径-从html代码中提取图片路径并下载(完整版)

    transHtml方法实现提取网络图片中得图片路径,将其重新下载到本地,并替换html中原来得路径 package com.googosoft.until; import java.io.Buffer ...

  4. 用正则从html代码中提取图片路径

    $str = '<div align="center"> <img src="http://www.99tyg.com/public/images/e8 ...

  5. Unity3D GUI中的图片跟随鼠标旋转脚本

    var Mid : Texture2D; var mouse : Texture2D; //鼠标图片 var mousePs = Vector2.zero; //鼠标的位置 private var a ...

  6. Unity3D GUI中的图片尾随鼠标旋转脚本

    var Mid : Texture2D; var mouse : Texture2D; //鼠标图片 var mousePs = Vector2.zero; //鼠标的位置 private var a ...

  7. 代码中引用res里的颜色、图片

    1.imageButton userImgButton 在代码中设置图片,使用res/Drawable 里的图片 Resources res = getResources(); Bitmap inDr ...

  8. 代码: 两列图片瀑布流(一次后台取数据,图片懒加载。下拉后分批显示图片。图片高度未知,当图片onload后才显示容器)

    代码: 两列图片瀑布流(一次后台取数据,无ajax,图片懒加载.下拉后分批显示图片.图片高度未知,当图片onload后才显示容器) [思路]: 图片瀑布流,网上代码有多种实现方式,也有各类插件.没找到 ...

  9. imagesLoaded – 检测网页中的图片是否加载

    imagesLoaded 是一个用于来检测网页中的图片是否载入完成的 JavaScript 工具库.支持回调的获取图片加载的进度,还可以绑定自定义事件.可以结合 jQuery.RequireJS 使用 ...

随机推荐

  1. MyBatis学习总结(11)——MyBatis动态Sql语句

    MyBatis中对数据库的操作,有时要带一些条件,因此动态SQL语句非常有必要,下面就主要来讲讲几个常用的动态SQL语句的语法 MyBatis中用于实现动态SQL的元素主要有: if choose(w ...

  2. Java 学习(15):Java 数据结构

    Java 数据结构 Java工具包提供了强大的数据结构.在Java中的数据结构主要包括以下几种接口和类: 枚举(Enumeration) 位集合(BitSet) 向量(Vector) 栈(Stack) ...

  3. 【iOS与EV3混合机器人编程系列之中的一个】iOS要干嘛?EV3能够更酷!

    乐高Mindstorm EV3智能机器人(下面简称EV3)自从在2013年的CES(Consumer Electronics Show美国消费电子展)上展出之后,就吸引了全球广大机器人爱好者的眼球!E ...

  4. 从 QSplitter 中移除 QWidget(使用隐藏与显示,切换十分方便,不要真正销毁)

    Splitter 的函数中有addWidget,但是却没有removeWidget, 或者delete之类的功能,所以如果想删去或者暂时不显示其中的某些widget就要自己手动完成这个效果.方法一:取 ...

  5. 实战c++中的vector系列--copy set to vector(别混淆了reserve和resize)

    stl算法中有个copy函数.我们能够轻松的写出这种代码: #include <iostream> #include <algorithm> #include <vect ...

  6. java 三次样条插值 画光滑曲线 例子

    java 三次样条插值 画光滑曲线 例子 主要是做数值拟合,根据sin函数采点,取得数据后在java中插值并在swing中画出曲线,下面为截图  不光滑和光滑曲线前后对比:    代码: 执行类: p ...

  7. java DSA Signature Sign And Verify

    SignatureSignAndVerify import java.security.KeyPair; import java.security.KeyPairGenerator; import j ...

  8. MFC中对话框的各种消息触发时间

    小结:WM_CREATE是所有窗口都能响应的消息,表明本窗口已经创建完毕.可以安全的使用这个窗口了,例如在它上面画控件等等.这个状态肯定是在调用ShowWindows()显示窗口之前.WM_WM_IN ...

  9. Navigator对象关于语言的属性

    [摘要]在做国际化WEB项目的时候,遇到了一个根据用户浏览器所使用的自然语言切换默认语言版本的问题.于是,整理了这篇文章. 首先,W3Cschool关于Navigator的各个属性值说的很明确了,这里 ...

  10. TC快速搜索在win10下不可用

    今天突然发现TC的快速搜索在win10下突然不可用,按Ctrl + s 呼出快速搜索栏后半天不响应也无法输入文字.论坛里给出来的建议是将 QuickSearch 2.2.3 升级到 2.2.6,目前插 ...