项目中用到了spine动画,使用Assetbundle打包后,在手机上运行会出现丢材质的情况。如果不进行打包,直接放到Resources目录下是可以正常加载的,但是,这样包就会很大,而且也不能进行热更新。进过测试,发现在代码中创建spine组件是可以解决这个问题,于是就有了下面的方案。


我们先说方案,再说问题。

方案:

spine动画制作人员提供的spine动画三个文件分别是.json,.atlas,.png,把这三个文件放到unity(我用的unity版本是5.3.4f1)中,会自动生成atlas.asset,.mat,skeletondata.asset三个文件,我们只使用.mat文件,因为mat文件上已经关联了.png文件,所以我们实际要打包的文件是三个源文件.json,.atlas,.png和.mat文件,而.png文件实际上是作为.mat的依赖打包的,下面在代码中创建spine组件如下:


GameObject addSpineComponent(UnityEngine.Object obj)

    {

        GameObject go = GameObject.Instantiate(obj) as GameObject;

        go.name = obj.name;

        GameObject.DontDestroyOnLoad(go);


        SkeletonAnimation sa;

        SkeletonDataAsset sda;


        sda = ScriptableObject.CreateInstance();

        sda.fromAnimation = new string[0];

        sda.toAnimation = new string[0];

        sda.duration = new float[0];

        sda.scale = 0.01f;

        sda.defaultMix = 0.15f;

      //SpineNameInfo中存着.json,.atlas,.mat的名称

        SpineNameInfo sni = dicMonsterSpineInfos[go.name];


        AtlasAsset[] arrAtlasData = new AtlasAsset[sni.AtlasFileNames.Length];

        for (int i = 0; i < arrAtlasData.Length; i++)

        {

            string atlasFileName = sni.AtlasFileNames[i];

            AtlasAsset atlasdata = ScriptableObject.CreateInstance();

            atlasdata.atlasFile = GetObj(atlasFileName) as TextAsset;

            string[] mNames = sni.GetMaterialNames(atlasFileName);

            atlasdata.materials = new Material[mNames.Length];

            for (int j = 0; j < mNames.Length; j++)

            {

                //Material m  = dicPreloadAssets[mNames[j]] as Material;

                atlasdata.materials[j] = GetObj(mNames[j]) as Material;

            }


            //atlasdata.materials = mats;

            atlasdata.Reset();

            arrAtlasData[i] = atlasdata;

        }


        sda.atlasAssets = arrAtlasData;

        sda.skeletonJSON = GetObj(sni.JsonFileName) as TextAsset;


        sa = go.AddComponent();

        sa.skeletonDataAsset = sda;

        sa.calculateNormals = true;

        sa.skeletonDataAsset.Reset();

        sa.AnimationName = "run";

        sa.loop = true;


        sa.Initialize(false);


        return go;

    }


    public class SpineNameInfo

    {

        private Dictionary dicAtlasFileNames = new Dictionary();

      //.json文件名

        public string JsonFileName = string.Empty;

      //.atlas文件名,可能有多个.atlas文件(虽然我的项目中只有一个)

        public string[] AtlasFileNames

        {

            get

            {

                string[] s = new string[dicAtlasFileNames.Count];

                dicAtlasFileNames.Keys.CopyTo(s, 0);

                return s;

            }

        }


      //.mat文件名,可能有多个mat文件

        public string[] GetMaterialNames(string atlasFileName)

        {

            if (dicAtlasFileNames.ContainsKey(atlasFileName))

                return dicAtlasFileNames[atlasFileName];

            return new string[0];

        }


        public void Add(string atlasFileName, string[] m)

        {

            if (!dicAtlasFileNames.ContainsKey(atlasFileName))

                dicAtlasFileNames.Add(atlasFileName, m);

        }

    }


   //这里是我项目中用到的对象池,根据资源名,获取资源对象

    public UnityEngine.Object GetObj(string name)

    {

        if (pool.ContainsKey(name))

            return pool[name];

        return null;

    }


可能您会注意到,怎么没有.png文件呢,我刚才在上面也提到,png文件会作为mat文件的依赖打包进assetbundle,所以在加载mat文件的时候png自然也已经被加载好了。


  以上是我的解决方案。可能有点乱,如果有什么不明白的,我再提供两个链接,也是网上的一些解决方案。


http://answers.unity3d.com/questions/710571/spine-creating-spine-skeletonanimation-using-cscri.html


http://zh.esotericsoftware.com/forum/Creating-Spine-SkeletonAnimation-using-c-3091


下面说一下问题:

使用上述方案以后,发现一个问题,就是创建spine动画很耗时。因为有大量的怪物要创建,这样就导致原先的场景预加载时间很长,长到不能接受。好在虽然怪物数量庞大,但种类不是很多,复用也很高,所以就把spine动画的创建和预加载放到了游戏数据初始化中,时间虽然会有点长,但因为游戏刚开始初始化的东西不多,就还能接受,后面再加载场景怪物的时候,直接获取预加载的种类的对象,然后实例化就OK了。


到此,问题解决。