jjzjj

【Unity】填坑,Unity接入Epic Online Service上架Epic游戏商城

TopGames 2024-04-27 原文

EOS SDK For Unity地址:https://github.com/PlayEveryWare/eos_plugin_for_unity_upm

可能是Epic官方SDK写得太烂了吧,知名游戏糖豆人(FALL GUYS)开发公司PlayEveryWare基于官方SDK维护了一套EOS Unity插件。

Epic是虚幻游戏引擎开发商,2018年12月Epic宣布推出Epic游戏商城至今刚好三年,Epic将平台分成定为12%(远低于当时Steam的30%),并且频繁推出各种让利、免费活动。

天下苦G胖久矣,Epic靠着这些骚操作迅速崛起。身边很多朋友买游戏先去Epic看价格,纷纷入坑。作为旁观者的我,一度以为Epic形势大好,取代Steam只是一丢丢时间的问题了。直到我接到接入EOS SDK的需求。

想当年咱可是接过无数各种SDK, Android、iOS双端信手拈来,小众SDK也没少见,再不专业的sdk也都轻松拿捏。直到遇到EOS SDK,大名鼎鼎虚幻家的sdk。瞬间让我觉得,我还太嫩,任重道远。。。

感情Epic把钱都拿去砸营销、抛诱饵钓用户去了,SDK和开发者后台管理严重经费不足! 甚至还是个半成品。官方开发者文档和最新的eos sdk demo,可以说是驴唇不对马嘴。看demo是一辆车,加上油就能开。看文档是一堆零件,想用可以,自己组装造车。我$^%^&*^&*^*&^!

Epic游戏商城发展至今三年了,国内/外几乎搜不到一篇正经接入文档,只有官方文档和官方论坛能找到线索。。

记录踩坑过程,栽树于此,以便后人乘凉:

0, 首先抛弃官方文档,一个字都不用看,毫无参考意义;

1, 下载eos sdk, https://github.com/PlayEveryWare/eos_plugin_for_unity_upm.git

①打开Unity Package Manager以git方式添加eos插件。

②把添加到本地的eos插件从Library目录移动到项目的Packages目录下(不要改插件文件夹名!)

为什么要脱裤子放屁?

因为Epic插件在Eidtor模式会根据插件路径动态加载dll文件,dll文件目录是按规则写死的,路径改变就会报错!

其次,插件依赖的dll默认是Any Platform, 然而插件dll实际只支持win32/64, 打包Switch、PS4/5等其它平台必然报错,打包失败。因此要把插件从Library下的缓存目录移动到Packages下,作为本地插件,这样才能修改dll的平台;

③ 把eos的所有dll平台都改为Editor、Windows 32-bit、 Windows 64-bit

2,导入eos sdk的示例,在Unity Package Manager中选中eos插件左侧信息栏有导入示例按钮。

示例代码中有很多xxManager文件,示例把乱七八槽的汽车零件(半成品)通过各种Manager组装成了接近成品的汽车,直接复用这些Manager接入eos就变得可以接受了。

3,把EOSManager和EOSHostManager两个脚本挂到启动场景,并勾选Awake时初始化;

4,接入模块功能:

demo把各个模块功能封装成了EOSXXXManager(其中XXX为模块名),sdk初始化后可通过代码var manager = EOSManager.Instance.GetOrCreateManager<EOSXXXManager>()动态添加需要接入的功能模块,对应的EOSXXXManager已经封装好了模块功能。

5,登录,登录为必接模块。可直接抄示例中的登录实现代码。

Epic提供了4种登录模式,首先排除每次打开游戏都弹出登录窗口,只考虑持久化登录方式:

首先如果是上架到Epic游戏商城,官方回强烈建议你使用LoginCredentialType.ExchangeCode方式登录,即登录时向Epic Game Launcher获取用户信息,直接作为参数静默登录游戏,此方式不会弹出登录窗口。

为了登录流程逻辑严谨,若ExchangeCode方式登录失败,再使用LoginCredentialType.AccountPortal方式登录, AccountPortal登录会弹出Epic登录窗口,登录成功后会在本地存储一个长期的刷新令牌;登录流程为:

1.使用EOSManager.Instance.StartPersistentLogin()先从本地令牌登录,若本地有令牌则登录成功;

2.如果本地没有令牌StartPersistentLogin()会登录失败,在登录失败后再通过

EOSManager.Instance.StartLoginWithLoginTypeAndToken(LoginCredentialType.AccountPortal,ExternalCredentialType.Epic, null, null, loginResult=>{}); 弹出登录授权窗口方式登录,此方式登录成功后会在本地保存令牌,以供下载StartPersistentLogin()静默登录。

public void Login()
        {
#if UNITY_EDITOR
            LoginWithPersistentMode();//编辑器模式Persistent模式登录
#else
                    var token = string.Empty;
                    string[] commandArgs = Environment.GetCommandLineArgs();
                    foreach (var commandArg in commandArgs)
                    {
                        if (commandArg.Contains("AUTH_PASSWORD"))
                        {
                            var args = commandArg.Split('=');
                            if (args.Length >= 2)
                            {
                                token = args[1];
                            }
                        }
                    }

                    EOSManager.Instance.StartLoginWithLoginTypeAndToken(LoginCredentialType.ExchangeCode, null, token, callbackInfo =>
                    {
                        if (callbackInfo.ResultCode != Epic.OnlineServices.Result.Success)
                        {
                            LoginWithPersistentMode();
                        }
                        else
                        {
                            StartLoginWithLoginTypeAndTokenCallback(callbackInfo);
                        }
                    });
#endif
        }


public void LoginWithPersistentMode()
        {
        EOSManager.Instance.StartPersistentLogin((Epic.OnlineServices.Auth.LoginCallbackInfo callbackInfo) =>
            {
                if (callbackInfo.ResultCode != Epic.OnlineServices.Result.Success)
                {
                    LoginWithLoginTypeAndToken();
                }
                else
                {
                    StartLoginWithLoginTypeAndTokenCallback(callbackInfo);
                }
            });
        }

private void LoginWithLoginTypeAndToken()
        {
            EOSManager.Instance.StartLoginWithLoginTypeAndToken(
                Epic.OnlineServices.Auth.LoginCredentialType.AccountPortal, ExternalCredentialType.Epic, null, null,
                loginResult =>
                {
                    EOSManager.Instance.StartConnectLoginWithEpicAccount(loginResult.LocalUserId, (Epic.OnlineServices.Connect.LoginCallbackInfo connectLoginCallbackInfo) =>
                    {
                        if (connectLoginCallbackInfo.ResultCode == Result.Success)
                        {
                            _productUserId = connectLoginCallbackInfo.LocalUserId;
                        }
                        else if (connectLoginCallbackInfo.ResultCode == Result.InvalidUser)
                        {
                            // ask user if they want to connect; sample assumes they do
                            EOSManager.Instance.CreateConnectUserWithContinuanceToken(connectLoginCallbackInfo.ContinuanceToken, (Epic.OnlineServices.Connect.CreateUserCallbackInfo createUserCallbackInfo) =>
                            {
                                EOSManager.Instance.StartConnectLoginWithEpicAccount(loginResult.LocalUserId, (Epic.OnlineServices.Connect.LoginCallbackInfo retryConnectLoginCallbackInfo) =>
                                {
                                    if (retryConnectLoginCallbackInfo.ResultCode == Result.Success)
                                    {
                                        _productUserId = retryConnectLoginCallbackInfo.LocalUserId;
                                    }

                                });
                            });
                        }
                        else
                        {

                        }
                    });

                });
        }


private void StartConnectLoginWithLoginCallbackInfo(LoginCallbackInfo loginCallbackInfo)
        {
            EOSManager.Instance.StartConnectLoginWithEpicAccount(loginCallbackInfo.LocalUserId, (Epic.OnlineServices.Connect.LoginCallbackInfo connectLoginCallbackInfo) =>
            {
                if (connectLoginCallbackInfo.ResultCode == Result.Success)
                {
                    _productUserId = connectLoginCallbackInfo.LocalUserId;

                }
                else if (connectLoginCallbackInfo.ResultCode == Result.InvalidUser)
                {
                    EOSManager.Instance.CreateConnectUserWithContinuanceToken(connectLoginCallbackInfo.ContinuanceToken, (Epic.OnlineServices.Connect.CreateUserCallbackInfo createUserCallbackInfo) =>
                    {
                        EOSManager.Instance.StartConnectLoginWithEpicAccount(loginCallbackInfo.LocalUserId, (Epic.OnlineServices.Connect.LoginCallbackInfo retryConnectLoginCallbackInfo) =>
                        {
                            if (retryConnectLoginCallbackInfo.ResultCode == Result.Success)
                            {
                                _productUserId = retryConnectLoginCallbackInfo.LocalUserId;
                            }
                        });
                    });
                }
            });
        }

6,Unity菜单栏Tool下有eos参数配置工具,其中产品名(游戏名),版本号,产品id,沙盒id, 部署id, 客户端id, 客户端密钥必须配置,密钥下方的一串字符可点击生成自动产生。参数在开发者后台->产品设置;

7,获取DLC状态,Epic开发者后台创建DLC商品,然后通过下面API可以获取用户是否购买了DLC:

public void CheckOwnedDLC(string dlcId, Action<bool> callback)
        {
            var localUserId = EOSManager.Instance.GetLocalUserId();
            var ecomInterface = EOSManager.Instance.GetEOSEcomInterface();
            var options = new QueryOwnershipOptions()
            {
                CatalogItemIds = new[] { new Utf8String(dlcId) },
                LocalUserId = localUserId,

            };
            ecomInterface.QueryOwnership(ref options, null, (ref QueryOwnershipCallbackInfo data) =>
            {
                if (data.ResultCode == Result.Success)
                {
                    bool ownedDlc = false;
                    foreach (var item in data.ItemOwnership)
                    {
                        var itemId = item.Id.ToString();
                        if (itemId.CompareTo(dlcId) == 0)
                        {
                            ownedDlc = item.OwnershipStatus == OwnershipStatus.Owned;
                            break;
                        }
                    }
                    callback.Invoke(ownedDlc);
                }
                else
                {
                    callback.Invoke(false);
                }

            });
        }

至此,你非常Happy的以为你接完了,可以测试了,当你Happy地点下运行按钮,不出意外的情况下必然eos sdk必然登录失败!然而当你把参数换上官方demo里的参数后一下就登录成功了。。。由此可以推断,一定不是接入代码的锅,一定是开发者后台配置的锅。嗯,你可以怀疑人生了。

恭喜,踩到了第一个巨坑。无任何征兆和提示,到处找不到线索。

最后发现需要把开发者后台的获取位置权限关闭才能成功登录,设置入口如下图:

 7. 发布上传:

此时官方文档终于派上了用场,Epic发布游戏像Steam一样也需要专门的发布工具,需要下载BuildPatch Tool:BuildPatch Tool Instructions (1.5.0) | Epic Online Services Developer

 这个工具是通过命令行工作!使用方法如图:最重要是把必须参数填对,否则上传失败。

    .\BuildPatchTool.exe
    -OrganizationId="<组织id,谁有开发者账号最高权限找谁要>"
    -ProductId="<产品id>"
    -ArtifactId="<应用id>"
    -ClientId="<BPT客户端id>"
    -ClientSecretEnvVar="<BPT客户端密钥>"
    -mode=UploadBinary
    -BuildRoot="<打包目录>"
    -CloudDir="<选个本地目录>"
    -BuildVersion="<版本号>"
    -AppLaunch="<游戏客户端.exe>"
    -AppArgs=""

 BuildPatch Tool很鸡肋,不支持忽略上传文件夹,需要把每一个要忽略上传的文件列出来。所以写了个可视化上传工具,支持配置文件夹和文件:https://github.com/sunsvip/EpicGameUploader :

ArtifactId是程序包id,  藏得比蛟龙号都深。寻宝路线如下图,千万别迷路:

注意:Build Patch Tool需要的ClientId和ClientSecretEnvVar不是eos接入所需的客户端id,客户端密钥,而是BPT ClientId 和 BPT ClientSecret,(Epic的人是用脚思考问题吗?这么制造歧义误导开发者合适吗?), 寻宝路径为 开发者后台=>产品设置=>通用,寻宝图如下:

有关【Unity】填坑,Unity接入Epic Online Service上架Epic游戏商城的更多相关文章

  1. Unity 热更新技术 | (三) Lua语言基本介绍及下载安装 - 2

    ?博客主页:https://xiaoy.blog.csdn.net?本文由呆呆敲代码的小Y原创,首发于CSDN??学习专栏推荐:Unity系统学习专栏?游戏制作专栏推荐:游戏制作?Unity实战100例专栏推荐:Unity实战100例教程?欢迎点赞?收藏⭐留言?如有错误敬请指正!?未来很长,值得我们全力奔赴更美好的生活✨------------------❤️分割线❤️-------------------------

  2. FOHEART H1数据手套驱动Optitrack光学动捕双手运动(Unity3D) - 2

    本教程将在Unity3D中混合Optitrack与数据手套的数据流,在人体运动的基础上,添加双手手指部分的运动。双手手背的角度仍由Optitrack提供,数据手套提供双手手指的角度。 01  客户端软件分别安装MotiveBody与MotionVenus并校准人体与数据手套。MotiveBodyMotionVenus数据手套使用、校准流程参照:https://gitee.com/foheart_1/foheart-h1-data-summary.git02  数据转发打开MotiveBody软件的Streaming,开始向Unity3D广播数据;MotionVenus中设置->选项选择Unit

  3. unity---接入Admob - 2

    目录1.AdmobSDK下载地址2.将下载好的unityPackagesdk导入到unity里​编辑 3.解析依赖到项目中

  4. Unity 3D 制作开关门动画,旋转门制作,推拉门制作,门把手动画制作 - 2

    Unity自动旋转动画1.开门需要门把手先动,门再动2.关门需要门先动,门把手再动3.中途播放过程中不可以再次进行操作觉得太复杂?查看我的文章开关门简易进阶版效果:如果这个门可以直接打开的话,就不需要放置"门把手"如果门把手还有钥匙需要旋转,那就可以把钥匙放在门把手的"门把手",理论上是可以无限套娃的可调整参数有:角度,反向,轴向,速度运行时点击Test进行测试自己写的代码比较垃圾,命名与结构比较拉,高手轻点喷,新手有类似的需求可以拿去做参考上代码usingSystem.Collections;usingSystem.Collections.Generic;usingUnityEngine;u

  5. ruby - 我需要从 facebook 游戏中抓取数据——使用 ruby - 2

    修改(澄清问题)我已经花了几天时间试图弄清楚如何从Facebook游戏中抓取特定信息;但是,我遇到了一堵又一堵砖墙。据我所知,主要问题如下。我可以使用Chrome的检查元素工具手动查找我需要的html-它似乎位于iframe中。但是,当我尝试抓取该iframe时,它​​是空的(属性除外):如果我使用浏览器的“查看页面源代码”工具,这与我看到的输出相同。我不明白为什么我看不到iframe中的数据。答案不是它是由AJAX之后添加的。(我知道这既是因为“查看页面源代码”可以读取Ajax添加的数据,也是因为我有b/c我一直等到我可以看到数据页面之后才抓取它,但它仍然不存在)。发生这种情况是因为

  6. python - Ruby 或 Python 的 3d 游戏引擎? - 2

    关闭。这个问题不符合StackOverflowguidelines.它目前不接受答案。要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于StackOverflow来说是偏离主题的,因为它们往往会吸引自以为是的答案和垃圾邮件。相反,describetheproblem以及迄今为止为解决该问题所做的工作。关闭9年前。Improvethisquestion是否有适用于这些的3d游戏引擎?

  7. Unity Shader 学习笔记(5)Shader变体、Shader属性定义技巧、自定义材质面板 - 2

    写在之前Shader变体、Shader属性定义技巧、自定义材质面板,这三个知识点任何一个单拿出来都是一套知识体系,不能一概而论,本文章目的在于将学习和实际工作中遇见的问题进行总结,类似于网络笔记之用,方便后续回顾查看,如有以偏概全、不祥不尽之处,还望海涵。1、Shader变体先看一段代码......Properties{ [KeywordEnum(on,off)]USL_USE_COL("IsUseColorMixTex?",int)=0 [Toggle(IS_RED_ON)]_IsRed("IsRed?",int)=0}......//中间省略,后续会有完整代码 #pragmamulti_c

  8. 三分钟集成 TapTap 防沉迷 SDK(Unity 版) - 2

    三分钟集成Tap防沉迷SDK(Unity版)一、SDK介绍基于国家对上线所有游戏必须增加防沉迷功能的政策下,TapTap推出防沉迷SDK,供游戏开发者进行接入;允许未成年用户在周五、六、日以及法定节假日晚上8:00-9:00进行游戏,防沉谜时间段进入游戏会弹窗进行提示!开发环境要求:Unity2019.4或更高版本iOS10或更高版本Android5.0(APIlevel21)或更高版本🔗Unity集成Demo参考链接🔗UnityTapSDK功能体验APK下载链接二、集成前准备1.创建应用进入开发者后台,按照提示开始创建应用;2.开通服务在使用TDS实名认证和防沉迷服务之前,需要在上面创建的应

  9. 【Unity大气散射】GAMES104:3A中如何实现大气散射 - 2

    写在前面前两天学习并整理的大气散射基础知识:【Unity大气渲染】关于单次大气散射的理论知识,收获了很多,但不得不承认的是,这其实已经是最早的、90年代的非常古老的方法了,后来也出现了一些优化性的计算思路和方法。因此,我打算先不急着跟各种教程在Unity中实现大气散射,而是再花时间来看看最近的游戏是如何去实现大气渲染的:06.游戏中地形大气和云的渲染(下)|GAMES104-现代游戏引擎:从入门到实践接下来就跟着GAMES104讲地形大气和云渲染的部分学习并做简单的记录,涉及到之前没提到的Mie散射也只选择直接截图PPT的方式记录啦!毕竟对于做作品来说,之后实现出来才是重要的~当然,May佬的

  10. ruby - 使用 Ruby 编写 Unity 游戏 - 2

    所以我看到unity支持c#、JS和Boo。我可以学习其中一个,但我想制作一个“编译器”或类似的东西,让我可以编写ruby​​代码并输出JS代码或制作一个可以被Unity编译器读取的层。这有可能吗?我愿意在这方面投入很多时间并且有相当多的经验。 最佳答案 如果您的问题实际上是“我如何将Ruby编译为JavaScript”,那么这更容易回答:Opal:RubytoJavaScriptcompiler但是,学习其中一种受支持的语言会更好。当运行的是用另一种语言解释的代码时,很难调试“您的”代码。

随机推荐