Bootstrap

Unity 自定义批量打包工具

打包配置项

using UnityEngine;
using System.Collections.Generic;

namespace MYTOOL.Build
{
    [System.Flags]
    public enum VersionOptions
    {
        None = 0,
        Major = 1,
        Minor = 4,
        Build = 8,
        Revision = 0x10,
    }

    /// <summary>
    /// 批量打包配置文件
    /// </summary>
    [CreateAssetMenu]
    public class BatchBuildProfile : ScriptableObject
    {
        public VersionOptions versionOptions = VersionOptions.Revision;
        public List<BuildTask> tasks = new List<BuildTask>(0);
    }
}

打包功能

using UnityEditor;
using UnityEngine;
using System;
using System.IO;
using System.Collections.Generic;

namespace MYTOOL.Build
{
    public class LogMessage
    {
        public LogType type;
        public string message;

        public LogMessage(LogType type, string message)
        {
            this.type = type;
            this.message = message;
        }
    }

    [CustomEditor(typeof(BatchBuildProfile))]
    public class BatchBuildProfileInspector : Editor
    {
        //配置文件
        private BatchBuildProfile profile;
        //折叠栏
        private Dictionary<BuildTask, bool> foldoutMap;
        //记录日志
        private List<LogMessage> logsList;

        private void OnEnable()
        {
            profile = target as BatchBuildProfile;
            foldoutMap = new Dictionary<BuildTask, bool>();
            logsList = new List<LogMessage>();
        }

        public override void OnInspectorGUI()
        {
            OnMenuGUI();
            OnListGUI();
            serializedObject.ApplyModifiedProperties();
            if (GUI.changed)
                EditorUtility.SetDirty(profile);
        }

        /// <summary>
        /// 菜单项
        /// </summary>
        private void OnMenuGUI()
        {
            EditorGUILayout.HelpBox($"已有打包工作项:{profile.tasks.Count}个", MessageType.Info);
            EditorGUILayout.HelpBox($"打包时会先进行排序, 优先打包当前平台【{EditorUserBuildSettings.activeBuildTarget}】", MessageType.Info);
            //限制20个
            if (profile.tasks.Count < 20)
            {
                //新建工作项
                if (GUILayout.Button("新建工作项", GUILayout.Height(30)))
                {
                    Undo.RecordObject(profile, "Create");
                    string buildPath = Path.Combine(Directory.GetParent(Application.dataPath).FullName, ".Build");
                    if (Directory.Exists(buildPath) == false)
                    {
                        Directory.CreateDirectory(buildPath);
                        Debug.LogFormat("创建构建目录:{0}", buildPath);
                    }

                    var task = new BuildTask(PlayerSettings.productName, (MyBuildTarget)EditorUserBuildSettings.activeBuildTarget, buildPath);
                    profile.tasks.Add(task);
                }
            }
            else
            {
                EditorGUILayout.HelpBox($"无法新建打包工作项", MessageType.Warning);
            }

            if (profile.tasks.Count > 0)
            {
                //清空
                GUI.color = Color.yellow;
                if (GUILayout.Button("清空工作项", GUILayout.Height(30)))
                {
                    Undo.RecordObject(profile, "Clear");
                    if (EditorUtility.DisplayDialog("提醒", "是否确认清理所有打包工作项?", "确定", "取消"))
                    {
                        Debug.LogWarningFormat("清理{0}个打包工作项", profile.tasks.Count);
                        profile.tasks.Clear();
                        foldoutMap.Clear();
                    }
                }

                //开始打包
                GUI.color = Color.cyan;
                if (GUILayout.Button("开始打包", GUILayout.Height(30)))
                {
                    if (EditorUtility.DisplayDialog("确认操作", "即将开始打包过程,这可能需要一些时间。您希望继续吗?", "继续", "取消"))
                    {
                        logsList.Clear();
                        OnBuild(false);
                    }
                    return;
                }

                //清理并打包
                GUI.color = Color.yellow;
                if (GUILayout.Button("清理并打包", GUILayout.Height(30)))
                {
                    if (EditorUtility.DisplayDialog("确认操作", "即将进行清理并开始打包过程,这可能需要一些时间。您希望继续吗?", "继续", "取消"))
                    {
                        if (EditorUtility.DisplayDialog("重要提醒", "清理操作将移除当前构建平台的所有文件,请确保已备份重要数据。是否要继续?此操作不可逆。", "确定继续", "取消"))
                        {
                            logsList.Clear();
                            OnBuild(true);
                        }
                    }
                    return;
                }
            }

            GUI.color = Color.white;
            //排序
            if (profile.tasks.Count > 1)
            {
                if (GUILayout.Button("排序工作项", GUILayout.Height(30)))
                {
                    Debug.Log("排序打包工作项");
                    profile.tasks.Sort(new BuildTaskComparer());
                    return;
                }
            }
        }

        /// <summary>
        /// 任务项
        /// </summary>
        private void OnListGUI()
        {
            //新旧版本号
            if (profile.tasks.Count > 0)
            {
                GUILayout.Space(10);
                //版本选项
                GUILayout.BeginHorizontal();
                GUILayout.Label("版本选项:", GUILayout.Width(70));
                var newVO = (VersionOptions)EditorGUILayout.EnumFlagsField(profile.versionOptions);
                if (profile.versionOptions != newVO)
                {
                    Undo.RecordObject(profile, "Version Options");
                    profile.versionOptions = newVO;
                }
                GUILayout.EndHorizontal();

                GUILayout.BeginHorizontal();
                GUILayout.Label("旧版本号:", GUILayout.Width(70));
                GUILayout.Label(PlayerSettings.bundleVersion);
                GUILayout.EndHorizontal();

                GUILayout.BeginHorizontal();
                GUILayout.Label("新版本号:", GUILayout.Width(70));
                GUILayout.Label(GetNewVersion(profile.versionOptions));
                GUILayout.EndHorizontal();
            }

            for (int i = 0; i < profile.tasks.Count; i++)
            {
                var task = profile.tasks[i];
                if (foldoutMap.ContainsKey(task) == false)
                {
                    foldoutMap.Add(task, true);
                }

                GUI.contentColor = task.enableTask ? Color.green : Color.white;
                GUILayout.Space(10);
                GUILayout.BeginHorizontal("Badge");
                GUILayout.Space(20);
                foldoutMap[task] = EditorGUILayout.Foldout(foldoutMap[task], task.ToString(), true);
                if (GUILayout.Button(EditorGUIUtility.IconContent("TreeEditor.Trash"), "IconButton", GUILayout.Width(20)))
                {
                    Undo.RecordObject(profile, "Delete Task");
                    foldoutMap.Remove(task);
                    profile.tasks.Remove(task);
                    break;
                }
                GUILayout.EndHorizontal();

                //折叠栏
                if (foldoutMap[task])
                {
                    GUI.contentColor = Color.white;
                    GUILayout.BeginVertical("Box");

                    //是否激活
                    GUILayout.BeginHorizontal();
                    GUILayout.Label("是否激活:", GUILayout.Width(70));
                    task.enableTask = GUILayout.Toggle(task.enableTask, "");
                    GUILayout.EndHorizontal();

                    //打包场景
                    GUILayout.BeginHorizontal();
                    GUILayout.Label("打包场景:", GUILayout.Width(70));
                    if (GUILayout.Button("+", GUILayout.Width(20f)))
                    {
                        task.sceneAssets.Add(null);
                    }
                    GUILayout.EndHorizontal();

                    //场景列表
                    if (task.sceneAssets.Count > 0)
                    {
                        OnSceneAssetsList(task);
                    }

                    //产品名称
                    GUILayout.BeginHorizontal();
                    GUILayout.Label("产品名称:", GUILayout.Width(70));
                    var newPN = GUILayout.TextField(task.productName);
                    if (task.productName != newPN)
                    {
                        Undo.RecordObject(profile, "Product Name");
                        task.productName = newPN;
                    }
                    GUILayout.EndHorizontal();

                    //打包平台
                    GUILayout.BeginHorizontal();
                    GUILayout.Label("打包平台:", GUILayout.Width(70));
                    var newBT = (MyBuildTarget)EditorGUILayout.EnumPopup(task.buildTarget);
                    if (task.buildTarget != newBT)
                    {
                        Undo.RecordObject(profile, "Build Target");
                        task.buildTarget = newBT;
                        //这些平台只能使用IL2CPP
                        if (task.buildTarget == MyBuildTarget.iOS || task.buildTarget == MyBuildTarget.WebGL || task.buildTarget == MyBuildTarget.WeixinMiniGame)
                        {
                            task.scriptMode = ScriptingImplementation.IL2CPP;
                        }
                        //其它平台默认切换到Player
                        if (task.buildTarget != MyBuildTarget.StandaloneWindows64 && task.buildTarget != MyBuildTarget.StandaloneLinux64 && task.buildTarget != MyBuildTarget.NoTarget)
                        {
                            task.buildSubtarget = StandaloneBuildSubtarget.Player;
                        }
                    }
                    GUILayout.EndHorizontal();

                    //Windows Linux添加打包子平台
                    if (task.buildTarget == MyBuildTarget.StandaloneWindows64 || task.buildTarget == MyBuildTarget.StandaloneLinux64)
                    {
                        GUILayout.BeginHorizontal();
                        GUILayout.Label("打包子平台:", GUILayout.Width(70));
                        var newBS = (StandaloneBuildSubtarget)EditorGUILayout.EnumPopup(task.buildSubtarget);
                        if (task.buildSubtarget != newBS)
                        {
                            Undo.RecordObject(profile, "Build Subtarget");
                            task.buildSubtarget = newBS;
                        }
                        GUILayout.EndHorizontal();
                    }

                    //打包选项
                    GUILayout.BeginHorizontal();
                    GUILayout.Label("打包选项:", GUILayout.Width(70));
                    var newBO = (BuildOptions)EditorGUILayout.EnumFlagsField(task.buildOptions);
                    if (task.buildOptions != newBO)
                    {
                        Undo.RecordObject(profile, "Build Options");
                        task.buildOptions = newBO;
                    }
                    GUILayout.EndHorizontal();

                    //脚本模式
                    GUILayout.BeginHorizontal();
                    GUILayout.Label("脚本模式:", GUILayout.Width(70));
                    var newSM = (ScriptingImplementation)EditorGUILayout.EnumPopup(task.scriptMode);
                    if (task.scriptMode != newSM)
                    {
                        Undo.RecordObject(profile, "Script Mode");
                        task.scriptMode = newSM;
                    }
                    GUILayout.EndHorizontal();

                    //打包路径
                    GUILayout.BeginHorizontal();
                    GUILayout.Label("打包路径:", GUILayout.Width(70));
                    GUILayout.TextField(task.buildPath);
                    if (GUILayout.Button("浏览", GUILayout.Width(40f)))
                    {
                        string path = EditorUtility.SaveFolderPanel("Build Path", task.buildPath, "");
                        if (!string.IsNullOrWhiteSpace(path))
                        {
                            task.buildPath = path;
                        }

                        GUIUtility.ExitGUI();
                    }
                    GUILayout.EndHorizontal();

                    //安卓平台添加其它选项
                    if (task.buildTarget == MyBuildTarget.Android)
                    {
                        GUILayout.BeginHorizontal();
                        GUILayout.Label("keystore:", GUILayout.Width(70));
                        PlayerSettings.Android.keystorePass = EditorGUILayout.PasswordField(PlayerSettings.Android.keystorePass);
                        GUILayout.EndHorizontal();

                        GUILayout.BeginHorizontal();
                        GUILayout.Label("keyalias:", GUILayout.Width(70));
                        PlayerSettings.Android.keyaliasPass = EditorGUILayout.PasswordField(PlayerSettings.Android.keyaliasPass);
                        GUILayout.EndHorizontal();

                        //导出工程
                        GUILayout.BeginHorizontal();
                        GUILayout.Label("导出工程:", GUILayout.Width(70));
                        EditorUserBuildSettings.exportAsGoogleAndroidProject = GUILayout.Toggle(EditorUserBuildSettings.exportAsGoogleAndroidProject, "");
                        GUILayout.EndHorizontal();
                    }

                    GUILayout.EndVertical();
                }
            }
        }

        private void OnSceneAssetsList(BuildTask task)
        {
            GUILayout.BeginHorizontal();
            GUILayout.Space(75);
            GUILayout.BeginVertical("Badge");
            for (int j = 0; j < task.sceneAssets.Count; j++)
            {
                var sceneAsset = task.sceneAssets[j];
                GUILayout.BeginHorizontal();
                GUILayout.Label($"{j + 1}.", GUILayout.Width(20));
                task.sceneAssets[j] = EditorGUILayout.ObjectField(sceneAsset, typeof(SceneAsset), false) as SceneAsset;
                if (GUILayout.Button("↑", "MiniButtonLeft", GUILayout.Width(20)))
                {
                    if (j > 0)
                    {
                        Undo.RecordObject(profile, "Move Up Scene Assets");
                        var temp = task.sceneAssets[j - 1];
                        task.sceneAssets[j - 1] = sceneAsset;
                        task.sceneAssets[j] = temp;
                    }
                }
                if (GUILayout.Button("↓", "MiniButtonMid", GUILayout.Width(20)))
                {
                    if (j < task.sceneAssets.Count - 1)
                    {
                        Undo.RecordObject(profile, "Move Down Scene Assets");
                        var temp = task.sceneAssets[j + 1];
                        task.sceneAssets[j + 1] = sceneAsset;
                        task.sceneAssets[j] = temp;
                    }
                }
                if (GUILayout.Button("+", "MiniButtonMid", GUILayout.Width(20)))
                {
                    Undo.RecordObject(profile, "Add Scene Assets");
                    task.sceneAssets.Insert(j + 1, null);
                    break;
                }
                if (GUILayout.Button("-", "MiniButtonMid", GUILayout.Width(20)))
                {
                    Undo.RecordObject(profile, "Delete Scene Assets");
                    task.sceneAssets.RemoveAt(j);
                    break;
                }
                GUILayout.EndHorizontal();
            }
            GUILayout.EndVertical();
            GUILayout.EndHorizontal();
        }

        /// <summary>
        /// 开始打包
        /// </summary>
        /// <param name="clearBuild">清理旧的构建</param>
        private void OnBuild(bool clearBuild)
        {
            //排序,优先当前平台的任务
            profile.tasks.Sort(new BuildTaskComparer());
            //旧版本号
            string oldVersion = PlayerSettings.bundleVersion;
            //新版本号
            string newVersion = GetNewVersion(profile.versionOptions);
            //设置新版本号
            PlayerSettings.bundleVersion = newVersion;

            try
            {
                for (int i = 0; i < profile.tasks.Count; i++)
                {
                    var task = profile.tasks[i];
                    if (task.enableTask == false || task.buildTarget == MyBuildTarget.NoTarget)
                    {
                        logsList.Add(new LogMessage(LogType.Log, $"跳过: {task}"));
                        continue;
                    }

                    BuildTarget buildTarget = (BuildTarget)task.buildTarget;
                    BuildTargetGroup targetGroup = BuildPipeline.GetBuildTargetGroup(buildTarget);
                    BuildPlayerOptions buildPlayerOptions = SetBuildParams(targetGroup, task);
                    EditorUtility.DisplayProgressBar("正在打包", profile.tasks[i].ToString(), (float)i + 1 / profile.tasks.Count);

                    if (string.IsNullOrEmpty(buildPlayerOptions.locationPathName))
                    {
                        throw new Exception(($"无法打包 {task},产品名称可能为空"));
                    }
                    if (buildPlayerOptions.scenes.Length == 0)
                    {
                        throw new Exception($"无法打包 {task},打包场景为空");
                    }

                    //切换平台
                    if (buildTarget != EditorUserBuildSettings.activeBuildTarget)
                    {
                        EditorUserBuildSettings.SwitchActiveBuildTarget(targetGroup, buildTarget);
                    }

                    PlayerSettings.SetScriptingBackend(targetGroup, task.scriptMode);

                    string path = Path.GetDirectoryName(buildPlayerOptions.locationPathName);
                    if (clearBuild && Directory.Exists(path))
                    {
                        Directory.Delete(path, true);
                    }
                    if (Directory.Exists(path) == false)
                    {
                        Directory.CreateDirectory(path);
                    }

                    //开始打包
                    var report = BuildPipeline.BuildPlayer(buildPlayerOptions);
                    switch (report.summary.result)
                    {
                        case UnityEditor.Build.Reporting.BuildResult.Unknown:
                            logsList.Add(new LogMessage(LogType.Error, $"{task} 出现未知错误"));
                            break;

                        case UnityEditor.Build.Reporting.BuildResult.Succeeded:
                            logsList.Add(new LogMessage(LogType.Log, $"{task} 打包耗时: {(report.summary.buildEndedAt - report.summary.buildStartedAt).TotalSeconds}秒"));
                            break;

                        case UnityEditor.Build.Reporting.BuildResult.Failed:
                            string errorMsg = "\n";
                            foreach (var file in report.GetFiles())
                            {
                                errorMsg += file.path + "\n";
                            }
                            foreach (var step in report.steps)
                            {
                                foreach (var stepmsg in step.messages)
                                {
                                    errorMsg += "\n" + stepmsg.content;
                                }

                                errorMsg += "\n";
                            }
                            logsList.Add(new LogMessage(LogType.Error, $"{task} 打包失败: {errorMsg}"));
                            break;

                        case UnityEditor.Build.Reporting.BuildResult.Cancelled:
                            logsList.Add(new LogMessage(LogType.Log, $"{task} 取消打包"));
                            return;
                    }

                    //打包成功,打开目录并记录版本号
                    if (report.summary.result == UnityEditor.Build.Reporting.BuildResult.Succeeded)
                    {
                        File.WriteAllText(string.Format("{0}/__version.txt", path), newVersion);
                        Application.OpenURL(path);
                    }
                }
            }
            catch (Exception ex)
            {
                //异常情况下还原版本号
                PlayerSettings.bundleVersion = oldVersion;
                Debug.LogFormat("还原打包版本号:{0}", oldVersion);
                Debug.LogException(ex);
            }
            finally
            {
                EditorUtility.ClearProgressBar();
                Debug.LogFormat("当前打包版本号:{0}", newVersion);
                foreach (var log in logsList)
                {
                    Debug.unityLogger.Log(log.type, log.message);
                }

                logsList.Clear();
            }
        }

        /// <summary>
        /// 获取新版本号
        /// </summary>
        /// <returns></returns>
        private string GetNewVersion(VersionOptions options)
        {
            try
            {
                Version version = new Version(PlayerSettings.bundleVersion);
                int major = version.Major;              //主版本
                int minor = version.Minor;              //次版本
                int build = version.Build;              //构建版本
                int revision = version.Revision;        //修订版本
                //默认不处理
                if (options == VersionOptions.None)
                {
                    //revision += 1;
                }
                else
                {
                    major += options.HasFlag(VersionOptions.Major) ? 1 : 0;
                    minor += options.HasFlag(VersionOptions.Minor) ? 1 : 0;
                    build += options.HasFlag(VersionOptions.Build) ? 1 : 0;
                    revision += options.HasFlag(VersionOptions.Revision) ? 1 : 0;
                }

                if (revision >= 100)
                {
                    build += 1;
                    revision = 0;
                }
                if (build >= 100)
                {
                    minor += 1;
                    build = 0;
                }
                if (minor >= 100)
                {
                    major += 1;
                    minor = 0;
                }

                return $"{major}.{minor}.{build}.{revision}";
            }
            catch (Exception)
            {
                return "1.0.0.0";
            }
        }

        /// <summary>
        /// 设置构建参数
        /// </summary>
        /// <param name="targetGroup"></param>
        /// <param name="task"></param>
        /// <returns></returns>
        private BuildPlayerOptions SetBuildParams(BuildTargetGroup targetGroup, BuildTask task)
        {
            BuildPlayerOptions buildPlayerOptions = new BuildPlayerOptions();

            List<string> levels = new List<string>();
            string[] activeLevels = EditorBuildSettingsScene.GetActiveSceneList(EditorBuildSettings.scenes);
            if (activeLevels.Length > 0)
            {
                levels.AddRange(activeLevels);
            }
            for (int i = 0; i < task.sceneAssets.Count; i++)
            {
                var scenePath = AssetDatabase.GetAssetPath(task.sceneAssets[i]);
                if (!string.IsNullOrEmpty(scenePath) && !levels.Contains(scenePath))
                {
                    levels.Add(scenePath);
                }
            }

            buildPlayerOptions.scenes = levels.ToArray();
            buildPlayerOptions.target = (BuildTarget)task.buildTarget;
            buildPlayerOptions.subtarget = (int)task.buildSubtarget;
            buildPlayerOptions.targetGroup = targetGroup;
            buildPlayerOptions.options = task.buildOptions;
            buildPlayerOptions.locationPathName = GetBuildTargetPath(task.buildTarget, task.buildSubtarget, task.buildOptions, task.buildPath, task.productName);
            return buildPlayerOptions;
        }

        /// <summary>
        /// 获取构建路径
        /// </summary>
        /// <param name="buildTarget"></param>
        /// <param name="buildOptions"></param>
        /// <param name="buildPath"></param>
        /// <param name="productName"></param>
        /// <returns></returns>
        private string GetBuildTargetPath(MyBuildTarget buildTarget, StandaloneBuildSubtarget buildSubtarget, BuildOptions buildOptions, string buildPath, string productName)
        {
            if (string.IsNullOrEmpty(productName))
            {
                return string.Empty;
            }

            bool isDevelopment = buildOptions.HasFlag(BuildOptions.Development);
            string currentDate = DateTime.Now.ToString("yyMMdd");
            string locationPathName = Path.Combine(buildPath, buildTarget.ToString(), buildSubtarget.ToString(), currentDate, productName);
            switch (buildTarget)
            {
                case MyBuildTarget.StandaloneOSX:
                    {
                        if (isDevelopment)
                            locationPathName += "_dev.app";
                        else
                            locationPathName += ".app";
                    }
                    break;

                case MyBuildTarget.StandaloneWindows64:
                    {
                        if (isDevelopment)
                            locationPathName += "_dev.exe";
                        else
                            locationPathName += ".exe";
                    }
                    break;

                case MyBuildTarget.StandaloneLinux64:
                    {
                        if (isDevelopment)
                            locationPathName += "_dev.x86_64";
                        else
                            locationPathName += ".x86_64";
                    }
                    break;

                case MyBuildTarget.Android:
                    {
                        if (isDevelopment)
                            locationPathName += $"_{currentDate}_dev";
                        else
                            locationPathName += $"_{currentDate}";

                        if (EditorUserBuildSettings.exportAsGoogleAndroidProject == false)
                            locationPathName += ".APK";
                    }
                    break;

                case MyBuildTarget.iOS:
                    {
                        if (isDevelopment)
                            locationPathName += $"_{currentDate}_dev";
                        else
                            locationPathName += $"_{currentDate}";
                    }
                    break;
            }

            return locationPathName;
        }
    }
}

任务配置项

using System;
using System.Collections.Generic;
using UnityEditor;

namespace MYTOOL.Build
{
    /// <summary>
    /// 打包的目标平台
    /// </summary>
    public enum MyBuildTarget
    {
        NoTarget = -2,
        //
        // 摘要:
        //     Build a macOS standalone (Intel 64-bit).
        StandaloneOSX = 2,
        //
        // 摘要:
        //     Build a Windows standalone.
        StandaloneWindows = 5,
        //
        // 摘要:
        //     Build a Windows 64-bit standalone.
        StandaloneWindows64 = 19,
        //
        // 摘要:
        //     Build a Linux 64-bit standalone.
        StandaloneLinux64 = 24,
        //
        // 摘要:
        //     Build an iOS player.
        iOS = 9,
        //
        // 摘要:
        //     Build an Android .apk standalone app.
        Android = 13,
        //
        // 摘要:
        //     Build to WebGL platform.
        WebGL = 20,
        //
        // 摘要:
        //     Build to WeixinMiniGame platform.
        WeixinMiniGame = 47,
        //
        // 摘要:
        //     Build an OpenHarmony .hap standalone app.
        OpenHarmony = 48,
    }

    /// <summary>
    /// 打包工作项
    /// </summary>
    [Serializable]
    public class BuildTask
    {
        /// <summary>
        /// 是否激活
        /// </summary>
        public bool enableTask;
        /// <summary>
        /// 打包的产品名称
        /// </summary>
        public string productName;
        /// <summary>
        /// 打包的目标平台
        /// </summary>
        public MyBuildTarget buildTarget;
        /// <summary>
        /// 打包的目标子平台
        /// </summary>
        public StandaloneBuildSubtarget buildSubtarget;
        /// <summary>
        /// 打包的选项
        /// </summary>
        public BuildOptions buildOptions;
        /// <summary>
        /// 脚本模式
        /// </summary>
        public ScriptingImplementation scriptMode;
        /// <summary>
        /// 打包的保存路径
        /// </summary>
        public string buildPath;
        /// <summary>
        /// 打包的场景列表
        /// </summary>
        public List<SceneAsset> sceneAssets;

        /// <summary>
        /// 构造函数
        /// </summary>
        /// <param name="productName">产品名称</param>
        /// <param name="buildTarget">目标平台</param>
        /// <param name="buildPath">保存路径</param>
        public BuildTask(string productName, MyBuildTarget buildTarget, string buildPath)
        {
            this.productName = productName;
            this.buildTarget = buildTarget;
            this.buildPath = buildPath;

            enableTask = true;
            buildSubtarget = StandaloneBuildSubtarget.Player;
            buildOptions = BuildOptions.CleanBuildCache;
            scriptMode = ScriptingImplementation.IL2CPP;
            sceneAssets = new List<SceneAsset>();
        }

        public override string ToString()
        {
            return string.Format("{0}【{1}-{2}】", productName, buildTarget, buildSubtarget);
        }
    }

/// <summary>
/// BuildTask比较
/// </summary>
public class BuildTaskComparer : IComparer<BuildTask>
{
    public int Compare(BuildTask x, BuildTask y)
    {
        if (x.enableTask && (BuildTarget)x.buildTarget == EditorUserBuildSettings.activeBuildTarget && (BuildTarget)y.buildTarget != EditorUserBuildSettings.activeBuildTarget)
        {
            return -1; // x排在前
        }
        else if (y.enableTask && (BuildTarget)x.buildTarget != EditorUserBuildSettings.activeBuildTarget && (BuildTarget)y.buildTarget == EditorUserBuildSettings.activeBuildTarget)
        {
            return 1; // y排在前
        }
        else if (x.enableTask && y.enableTask && x.buildTarget == y.buildTarget)
        {
            return 0; //保持当前
        }
        else if (x.enableTask && x.buildTarget == y.buildTarget)
        {
            return -1; // x排在前
        }
        else if (y.enableTask && x.buildTarget == y.buildTarget)
        {
            return 1; // y排在前
        }
        else
        {
            return x.buildTarget.ToString().CompareTo(y.buildTarget.ToString());
        }
    }
}

效果图,可以将它锁定在这里,方便后面使用
使用也很简单,选择打包的平台,并设置一些参数。点击开始打包或清理并打包。
注意:打包场景字段是额外添加, 每次打包都会先获取Build Settings里激活的场景,并添加上打包场景中的设置
在这里插入图片描述

其它解释:
有些字段是直接使用Unity的,所以数据是共享的,比如安卓特有的选项,一个地方修改,其它相对应的位置也发生改变。
构建目录格式:打包路径+打包平台+打包子平台+日期(yyMMdd)
为什么添加打包子平台字段,因为我的项目中需要打包服务端(Dedicated Server)

;