首页 > 程序开发 > 移动开发 > 其他 >

unity 一个拼图demo(七巧板)和一个切割demo

2016-08-04

好久没更新博客了,那么今天就来装一下b吧。还是先上一波图,看看我们到底要做什么。前面2张是切割的的demo,后面的这个是拼图,但是我们看到材质,这里的解释一下2个demo用的都是一个网& 26684;构建代码,

好久没更新博客了,那么今天就来装一下b吧。还是先上一波图,看看我们到底要做什么。

\
\
\

前面2张是切割的的demo,后面的这个是拼图,但是我们看到材质,这里的解释一下2个demo用的都是一个网格构建代码,所以才会出现这种情况,正确的做法应该是把添加uv的代码做个判断一下,把这个变量放到这个方法参数列表中,同时把它变成一个对外的方法。这里顺便给大家提一下封装的思想,厉害的程序员他们其实大部分的时间都在思考,思考如何把一个系统变得扩展性越高,这样软件的腐蚀就会越慢了。同样如果一个游戏能做成这样同样也便于我们维护和理解。这2个demo都用到了这个类,那么就将其定义为MeshBuilder。这个类他应该提供一个基本的方法就是我们给你一组点的数据,你给我们返回一个网格或者一个物体。但是如果我们需要给网格贴uv的话,我们的传进去一个网格的材质,网格的原始宽度和高度(便于我们计算点的uv信息),同样我们可以把材质,网格的原始宽度和高度放到构造函数或者用一个初始化方法,这样都是可以(我个人建议用后者,其实都一样),至于你这个meshbuilder类中怎么写,我其实都不怎么关心的,我只需要我想要的就是返回一个网格或者游戏物体。如果我们将这个方法约束成为一个行为的话,就是这个类它可以给我们提供生产网格而其他的类都没有这个功能。这个时候就是接口了。接口的表现更倾向于CanDo,能干什么行为而继承倾向于IS A,是什么东西这样的行为。言归正传吧。其实我是先做的后面那个demo,做的差不多的时候再去做切割的,所以在做切割的时候把前面的代码删掉了。先从第一个将吧,我们知道要做上面一个正方形的网格其实是很简单的。同时我们还需要在正方形的网格周围做一圈描边,这里还是画图从一般的情况进行分析这个border该怎么来添加吧。

\

我们就2中情况分析,一种是基本的凸多边形一种凹多边形。我们就凸多边形分析一波。如果我们想要做成上图的那种效果,首先我们通过b1点求得对应的b点,依次次类推求得剩下的2个点。根据基本的需求得出相应的规律,就是b1c1到bc之间的距离和a1c1到ac之间的距离及a1b1到ab之间的距离是相等的。所以b1到ab的距离等于b1到bc的距离,所以b这个点在a1b1和b1c1的平分线上,依次类推到c点和a点。我们可以轻松的求得b1a1向量和b1c1向量之间的和向量,然后通过一个反向量求得b的位置,当然没玩还得通过2条直线的距离求得这个斜线的距离就是bb1的距离,最后即可求得b点依次类推就可以求出所有的点,当然这只是仅限于我们的凸多边形,我们应该还需要讨论我们的凹多边形的情况。大家可以随便画个图就可以知道如果我们仍用凸的情况来讨论肯定是不对的。反正我们知道最后求得的那个点肯定在脚平分线上,只不过我们只是不知道是在角平分线的哪头罢了。如果我们把为负的所有情况都讨论出来了,然后我们取一下反不就为正的情况了么。那么我们用设定一个乘数,当在某种情况下该乘数为-1,我们最开始的时候为1,这样就可以轻松的解决这个问题了。那么问题来 怎么样的情况下我们需要将乘数是设为-1呢?当然这个结论同样也需要大家去思考。外人只能引导,自己理解了,将问题抽象为规律,那么你就可以轻松的解决这个问题了(这个问题我就不讲了,大家自己可以去总结一下他们之间的规律)。首先我们还是从最开始的网格生成类开始做起吧

public class MeshBuilder
{ 
}
定义一个功能类似于构造函数的方法。
    private Material _meshMaterial;
    private Material _borderMaterial;
    private Transform _parentTransform;
    public void Intilized(Material mat,Material bmat,Transform parent)
    {
        _meshMaterial = mat;
        _borderMaterial = bmat;

        _parentTransform = parent;
    }
初始化网格所需要的材质资源和基本的位置信息。添加一个对外的方法。
    public GameObject CreateMeshObj(List verticleList,float width,float height)
    {
        GameObject ob=new GameObject("sb");
        Mesh mesh=new Mesh();

        List templist = GetTriangulateByPoint.GetInsiderBorderList(verticleList,0.1f);


        if (GetTriangulateByPoint.Area(templist) > 0)
        {
            templist.Reverse();
            verticleList.Reverse();
        }

        mesh.vertices = templist.ToArray();
        mesh.triangles = GetTriangulateByPoint.GetTriByPoins(templist);

        var col = ob.AddComponent();
        col.sharedMesh = mesh;
        ob.AddComponent().mesh = mesh;

        var child = CreateBorderObj(templist, verticleList);
        child.transform.parent = ob.transform;
        return ob;
    }
这里就是根据点的信息来创建网格,这些点的信息我们通过我们去读json文件中存储点的基本信息,这些json我是通过写一个编辑器来编辑不同的图案。接着添加一个私有方法

CreateBorderObj。

 private GameObject CreateBorderObj(List inside,List outside)
 {
        int count = inside.Count;
        GameObject border=new GameObject("Border");
        Mesh mesh=new Mesh();

        List veritcles = new List(count*2);
        List indicats=new List();

        veritcles.AddRange(outside);
        veritcles.AddRange(inside);
        for (int i = 0; i < count; i++)
        {
            indicats.Add(i); indicats.Add((i + 1) % count); indicats.Add((i + 1) % count+count);
            indicats.Add(i); indicats.Add((i + 1) % count + count); indicats.Add(i + count);
        }
        mesh.vertices = veritcles.ToArray();
        mesh.triangles = indicats.ToArray();

        border.AddComponent().mesh = mesh;
        border.AddComponent().material = _borderMaterial;
        return border;
 }
因为我们操作中还需要一个画线操作,这里我们仍需要通过网格来动态控制大下和方向了,如果我们鼠标滑动的当前点和上一帧的点的距离判断及当前点和鼠标按下点与上一帧的点和鼠标按下的点的夹角大于一定角度的时候我们将其定义一次有效的移动。所以我们还需要通过2个点来创建的一个长方边形的网格。
    public GameObject CreateLineMeshObj(Vector3 start, Vector3 end, Material mat)
    {
        GameObject ob = new GameObject("sd");
        Mesh mesh = new Mesh();

        List verticles = new List();
        List indicats = new List();
        List uv = new List();

        Vector3 dir = (end - start).normalized;
        Vector3 normal = Quaternion.Euler(0, 0, 90) * dir;

        var start1 = start + normal.normalized * 0.1f;
        var end1 = end + normal.normalized * 0.1f;
        verticles.Add(start); verticles.Add(start1); verticles.Add(end); verticles.Add(end1);

        indicats.Add(0); indicats.Add(1); indicats.Add(3);
        indicats.Add(0); indicats.Add(3); indicats.Add(2);
        uv.Add(new Vector2(0, 0)); uv.Add(new Vector2(0, 1));
        uv.Add(new Vector2(1, 0)); uv.Add(new Vector2(1, 1));
        mesh.vertices = verticles.ToArray();
        mesh.triangles = indicats.ToArray();
        mesh.uv = uv.ToArray();
        ob.AddComponent().mesh = mesh;
        ob.AddComponent().material = mat;

        return ob;
    }
写完这些其实也跑不起来,因为查几个引用方法。今天就现讲到这里,下节再讲。
相关文章
最新文章
热点推荐