最新要闻
- 索尼被曝测试新版PS5:性能不变、增设可拆卸光驱
- 全球热消息:因担心投资者被吸引到他国:欧盟再次推迟将锂列为有害物质
- 世界今亮点!苹果担心的事儿发生了 男子用AirTag追踪妻子被逮捕
- 《巫师3》次时代版体积惊人:PS5升级包超50GB
- 【世界播资讯】国家邮政局官方回应“快递不快”:加快恢复 邮政顺丰京东开启夜派
- 锵锵三人行停播原因是什么?锵锵三人行女嘉宾名单大全
- 沙海吴邪的计划是什么?沙海吴邪的计划成功了吗?
- 什么边野草花什么口夕阳斜?什么边野草花什么口夕阳斜是什么诗?
- 九月青豆角产自什么村?九月青豆角怎么冻好吃?
- 天国的阶梯结局是什么?天国的阶梯演员表
- 天天看点:卡梅隆感染新冠 缺席洛杉矶首映式
- 想“白嫖”20万的私人飞机 推特被告了
- 【报资讯】短了5厘米照样强大 迷你SSD硬盘雄起 速度冲向5GB/s
- 全球聚焦:小米13、iPhone 14 Pro全角度对比:小米正面碾压式完胜 背面有争议
- 环球快消息!世界杯半决赛现场将播放两首中文歌:你肯定都听过
- 环球播报:硬刚小米13!moto X40核心配置官方全面揭晓
手机
iphone11大小尺寸是多少?苹果iPhone11和iPhone13的区别是什么?
警方通报辅警执法直播中被撞飞:犯罪嫌疑人已投案
- iphone11大小尺寸是多少?苹果iPhone11和iPhone13的区别是什么?
- 警方通报辅警执法直播中被撞飞:犯罪嫌疑人已投案
- 男子被关545天申国赔:获赔18万多 驳回精神抚慰金
- 3天内26名本土感染者,辽宁确诊人数已超安徽
- 广西柳州一男子因纠纷杀害三人后自首
- 洱海坠机4名机组人员被批准为烈士 数千干部群众悼念
家电
今日热议:零基础开启元宇宙|抖音快手虚拟形象直播【源码】
在上一篇文章零基础开启元宇宙——创建虚拟形象中,我们实现了创建虚拟形象,接下来我们可以利用虚拟形象“为所欲为”。今天我们利用虚拟形象在短视频平台如快手、抖音中直播,对于不希望露脸的主播们这是可是一大利器呀!话不多说,上绝活。
1 实现思路
通过即构免费提供的虚拟形象和实时RTC技术,结合抖音快手官方提供的直播伴侣,可以轻松实现虚拟形象在抖音快手平台直播,整个实现流程如下:
2 Android接入RTC推送实时预览画面
2.1 接入RTC SDK
前往https://doc-zh.zego.im/article/2969下载即构RTC SDK。将压缩包内容拷贝到app/libs
中,并修改app/build.gradle
添加如下内容:
(相关资料图)
// ...// 其他略// ...android { // ... // 其他略 // ... defaultConfig { // ... // 其他略 // ... ndk { abiFilters "armeabi-v7a", "arm64-v8a", "x86", "x86_64" } } sourceSets { main { jniLibs.srcDirs = ["libs"] } } }dependencies { implementation fileTree(dir: "libs", include: ["*.jar", "*.aar"]) //通配引入 // ... // 其他略 // ...}
在app/src/main/AndroidManifest.xml
文件中添加必要的权限信息:
2.1 虚拟形象实时推流
使用即构RTC SDK实现实时视频通话过程如下图:
根据我们目前的需求,只需实现在Android端推流,windows端拉流即可。因此我们接下来只介绍如何在android端推流,如果想实现更丰富的定制能力,参考官网https://doc-zh.zego.im/article/195即可。
出于篇幅考虑,我们这里只展示关键代码:
private ZegoExpressEngine createRTCEngine(Application app, IZegoEventHandler handler) { ZegoEngineProfile profile = new ZegoEngineProfile(); profile.appID = KeyCenter.APP_ID; profile.scenario = ZegoScenario.GENERAL; // 通用场景接入 profile.application = app; ZegoExpressEngine engine = ZegoExpressEngine.createEngine(profile, handler); return engine;}public void start(String userId, String userName, String roomId, RTCListener listener) { Log.e(TAG, "准备登陆房间"); loginRoom(userId, userName, roomId, listener);}public void stop() { loginOut();}public void setCustomVideo(int videoWidth, int videoHeight, RTCMngr.CaptureListener listener) { // 自定义视频采集 ZegoCustomVideoCaptureConfig videoCaptureConfig = new ZegoCustomVideoCaptureConfig(); // 选择 GL_TEXTURE_2D 类型视频帧数据 videoCaptureConfig.bufferType = ZegoVideoBufferType.GL_TEXTURE_2D; // 启动自定义视频采集 mRTCEngine.enableCustomVideoCapture(true, videoCaptureConfig, ZegoPublishChannel.MAIN); // 设置自定义视频采集回调 mRTCEngine.setCustomVideoCaptureHandler(new IZegoCustomVideoCaptureHandler() { @Override public void onStart(ZegoPublishChannel zegoPublishChannel) { if (listener != null) { listener.onStartCapture(); } } @Override public void onStop(ZegoPublishChannel zegoPublishChannel) { if (listener != null) { listener.onStopCapture(); } } }); // 设置视频配置, 要跟 avatar 的输出尺寸一致 ZegoVideoConfig videoConfig = new ZegoVideoConfig(ZegoVideoConfigPreset.PRESET_720P); // 输出纹理是正方形的, 要配置一下 videoConfig.setEncodeResolution(videoWidth, videoHeight); mRTCEngine.setVideoConfig(videoConfig);}//实时推流public void pushStream(String streamId, TextureView tv) { mRTCEngine.startPublishingStream(streamId); mRTCEngine.startPreview(new ZegoCanvas(tv));}public boolean loginRoom(String userId, String userName, String roomId, RTCListener listener) { mRoomId = roomId; mUserId = userId; ZegoUser user = new ZegoUser(userId, userName); ZegoRoomConfig config = new ZegoRoomConfig(); config.token = getToken(userId, roomId); // 请求开发者服务端获取 config.isUserStatusNotify = true; mRTCEngine.loginRoom(roomId, user, config, (int error, JSONObject extendedData) -> { if (listener != null) { listener.onLogin(error); } }); Log.e(TAG, "登录房间:" + roomId); return true;}public void loginOut() { mRTCEngine.stopPublishingStream(); mRTCEngine.logoutRoom(mRoomId);}@Overridepublic void onRoomTokenWillExpire(String roomID) { mRTCEngine.renewToken(roomID, getToken(mUserId, roomID));}/*** 此函数应该放在服务器端执行,以防止泄露ServerSecret*/public static String getToken(String userId, String roomId) { TokenEntity tokenEntity = new TokenEntity(KeyCenter.APP_ID, userId, roomId, 60 * 60, 1, 1); String token = TokenUtils.generateToken04(tokenEntity); return token;}
首先执行顺序如下:
- createRTCEngine, 获取RTC引擎对象:engine。
- setCustomVideo, 用于设置自定义推流采样视频帧数据相关属性。
- loginRoom,登录房间,登录房间函数会自动调用getToken获取token令牌做权鉴。
这里注意在setCustomVideo函数内执行了setCustomVideoCaptureHandler,这里我们将他间接转为了如下接口:
public interface CaptureListener { void onStartCapture(); void onStopCapture();}
上面对象用于监听开始抓取推流数据和停止抓取事件,在Avatar侧只需实现上面两个接口即可:
// 获取到 avatar 纹理后的处理public void onCaptureAvatar(int textureId, int width, int height) { if (mIsStop || mUser == null) { // rtc 的 onStop 是异步的, 可能activity已经运行到onStop了, rtc还没 return; } boolean useFBO = true; if (mBgRender == null) { mBgRender = new TextureBgRender(textureId, useFBO, width, height, Texture2dProgram.ProgramType.TEXTURE_2D_BG); } mBgRender.setInputTexture(textureId); float r = Color.red(mUser.bgColor) / 255f; float g = Color.green(mUser.bgColor) / 255f; float b = Color.blue(mUser.bgColor) / 255f; float a = Color.alpha(mUser.bgColor) / 255f; mBgRender.setBgColor(r, g, b, a); mBgRender.draw(useFBO); // 画到 fbo 上需要反向的 ZegoExpressEngine.getEngine().sendCustomVideoCaptureTextureData(mBgRender.getOutputTextureID(), width, height, System.currentTimeMillis());}@Overridepublic void onStartCapture() { if (mUser == null) return;// // 收到回调后,开发者需要执行启动视频采集相关的业务逻辑,例如开启摄像头等 AvatarCaptureConfig config = new AvatarCaptureConfig(mUser.width, mUser.height);// // 开始捕获纹理 mCharacterHelper.startCaptureAvatar(config, this::onCaptureAvatar);}@Overridepublic void onStopCapture() { Log.e(TAG, "结束推流"); mCharacterHelper.stopCaptureAvatar(); stopExpression();}
以上步骤实现了Android端将虚拟形象推流到服务器端,详细代码可以看附件。
3 PC端拉取实时虚拟形象并展示
前往https://doc-zh.zego.im/article/3209下载Web版RTC SDK。文件结构如下:
在keycenter.js
中定义APPID等属性值。
// 请从官网控制台获取对应的appIDconst APPID = 从官网控制台获取appid// 请从官网控制台获取对应的server地址,否则可能登录失败const SERVER = "wss://webliveroom510775561-api.imzego.com/ws" //下面这个密钥用于生成Token,最好不要客户端暴露,应当在私人服务器使用const SERVER_SECRET = 从官网控制台获取SERVER_SECRET
tokenUtils.js
文件用于创建token,这里跟android端的token是相同的改进,tokenUtils.js
的内容比较多,这里不展示了,只需将它作为创建token的工具即可。
zego.js
文件用于创建RTC引擎,登录房间以及监听推流事件,一旦有推流事件立马拉流。相关代码如下:
function newToken(userId) { const token = generateToken04(APPID, userId, SERVER_SECRET, 60 * 60 * 24, ""); console.log(">>>", generateToken04(APPID, "222", SERVER_SECRET, 60 * 60 * 24, "")) return token;}function createZegoExpressEngine() { var engine = new ZegoExpressEngine(APPID, SERVER); return engine;}// Step1 Check system requirementsfunction checkSystemRequirements(engine, cb) { console.log("sdk version is", engine.getVersion()); engine.checkSystemRequirements().then((result) => { if (!result.webRTC) { cb(false, "browser is not support webrtc!!"); } else if (!result.videoCodec.H264 && !result.videoCodec.VP8) { cb(false, "browser is not support H264 and VP8"); } else if (!result.camera && !result.microphone) { cb(false, "camera and microphones not allowed to use"); } else { if (result.videoCodec.VP8) { if (!result.screenSharing) console.warn("browser is not support screenSharing"); } else { console.log("不支持VP8,请前往混流转码测试"); } cb(true, null); } });}function initEvent(engine, onAddRemoteStream) { engine.on("roomUserUpdate", (roomID, updateType, userList) => { console.log(">>roomUserUpdate", roomId, state) }); engine.on("roomStateUpdate", (roomId, state) => { console.log(">>roomStateUpdate", roomId, state) }) engine.on("roomStreamUpdate", async (roomID, updateType, streamList, extendedData) => { console.log(">>update") // streams added if (updateType === "ADD") { const addStream = streamList[streamList.length - 1] if (addStream && addStream.streamID) { onAddRemoteStream(addStream.streamID) } } else if (updateType == "DELETE") { // del stream const delStream = streamList[streamList.length - 1] if (delStream && delStream.streamID) { if (delStream.streamID === remoteStreamID) { engine.stopPlayingStream(remoteStreamID) } } } });}// Step5 Start Play Streamfunction playingStream(engine, videoId, streamId, cb, options = { video: true, audio: true}) { engine.startPlayingStream(streamId, options).then((remoteStream) => { const remoteView = engine.createRemoteStreamView(remoteStream); remoteView.play(videoId, { objectFit: "cover", enableAutoplayDialog: true, }) cb(true, remoteStream); }).catch((err) => { cb(false, err) });}function stopPlaying(engine, stremId) { engine.stopPlayingStream(stremId)}// Login roomfunction loginRoom(engine, roomId, userId, userName, cb) { var token = newToken(userId); engine.loginRoom(roomId, token, { userID: userId, userName }).then((result) => { cb(true, result); }).catch((err) => { cb(false, err) });}// Logout roomfunction logoutRoom(engine, roomId) { engine.logoutRoom(roomId);}
在index.html中引用如上javascript文件,展示拉流内容:
<script src="./express_sdk/ZegoExpressWebRTC.js"></script> <script src="./js/tokenUtils.js"></script> <script src="./js/zego.js"></script> <script src="./js/keycenter.js"></script> <script src="./js/index.js"></script>
4 快手、抖音直播推送实时虚拟画面
接下来是振奋人心时刻,到了联调时刻。在android打开画面实时推理,并在浏览器中打开界面,可以看到如下画面:
接下来只需使用直播伴侣软件,将浏览器中的实时画面实时转发到快手或抖音。这里我们用快手直播伴侣实时截屏直播,可以看到如下画面
5 附件
源码: https://github.com/RTCWang/Virtual-Live
今日热议:零基础开启元宇宙|抖音快手虚拟形象直播【源码】
全球最资讯丨概率论之重要的随机变量
Zabbix技术分享——zabbix命令详解
索尼被曝测试新版PS5:性能不变、增设可拆卸光驱
全球热消息:因担心投资者被吸引到他国:欧盟再次推迟将锂列为有害物质
世界今亮点!苹果担心的事儿发生了 男子用AirTag追踪妻子被逮捕
《巫师3》次时代版体积惊人:PS5升级包超50GB
【世界播资讯】国家邮政局官方回应“快递不快”:加快恢复 邮政顺丰京东开启夜派
环球简讯:eval和ast.literal_eval区别
即时:MySQL索引必知必会
焦点速看:深入理解Whitelabel Error Page底层源码
短讯!字符串函数
【脚本项目源码】Python实现鲁迅名言查询系统
短讯!NGINX常见的变量说明
程序员也可以很浪漫,精选10个圣诞节特效及源码
天天资讯:使用Cpolar搭建一个图片网站2 (Piwigo网站环境准备及安装)
苹果笔记本是什么系统?苹果笔记本怎么安装win10系统?
关闭笔记本触摸板的方法有哪些?关闭笔记本触摸板的四种方法
柳叶刀是哪个国家的杂志?柳叶刀杂志是什么水平?
2023年元旦股市放假几天?2023年元旦股市休市时间表
消防车也要去加油站加油吗?消防车多少钱一台?
win10可以免费升级吗?免费升级win10的条件有哪些?
wlan和wifi哪个网速更快?wlan和wifi的区别是什么?
内存卡读不出来是什么原因?内存卡读不出来怎么修复?
一加8t和一加8pro哪个好?一加8t参数配置
环球微动态丨下载selenium及其适配谷歌浏览器插件chromedriver(含chrome各版本及下载地址)
19.13备库duplicate恢复新主库(二)
iPhone6多少钱?iphone6系统最高可以升级到多少?
锵锵三人行停播原因是什么?锵锵三人行女嘉宾名单大全
沙海吴邪的计划是什么?沙海吴邪的计划成功了吗?
什么边野草花什么口夕阳斜?什么边野草花什么口夕阳斜是什么诗?
九月青豆角产自什么村?九月青豆角怎么冻好吃?
天国的阶梯结局是什么?天国的阶梯演员表
天天热点!什么是Docker容器?(全面了解使用)
环球新资讯:【collection】4.java容器之LinkedList,Stack,CopyOnWriteArrayList
教你用JavaScript实现实时字符计数器
杭州联合银行 x 袋鼠云:打造智能标签体系,助力银行大零售业务转型
天天看点:卡梅隆感染新冠 缺席洛杉矶首映式
想“白嫖”20万的私人飞机 推特被告了
【报资讯】短了5厘米照样强大 迷你SSD硬盘雄起 速度冲向5GB/s
全球聚焦:小米13、iPhone 14 Pro全角度对比:小米正面碾压式完胜 背面有争议
环球快消息!世界杯半决赛现场将播放两首中文歌:你肯定都听过
当前热讯:CSS实现打字机动画效果
环球热消息:行为管理(锐捷交换篇)
环球播报:硬刚小米13!moto X40核心配置官方全面揭晓
环球热头条丨男子帮摔倒大爷报警反被讹引热议:已和解 对方赔偿男子2000元误工费
环球即时:旗舰猛兽!小米万兆路由器明天首销:1799元
天天时讯:太强了!卢伟冰称小米13 Pro应该叫“小米13 Pro Ultra”
【速看料】《三体》动画爆火!两部真人版剧集明年开播:网飞、腾讯对标
环球热推荐:又一合资倒下?家喻户晓的斯柯达或将退出中国市场
环球聚焦:历史首次!英国女孩通过基因编辑治好白血病对抗癌症:生物工程壮举
全球快资讯:框架第三课---作业讲解(数据增删改查),django请求生命周期流程图,django路由层,反向解析
天天热点!第一百一十二篇: JS数组Array(一)数组基本用法
焦点快播:SpringBoot+VUE
Dockerfile指令与Docker-compose容器编排-搭建docker私有仓库
【环球报资讯】伴娘穿露肩礼服肌肉健硕画面太美不敢看 网友:真金刚芭比
IGN给RX 7900 XTX显卡打7分:AMD性价比绝对YES
一加显示器X27发布:2K 165Hz屏、65W PD输出
观天下!小米13深度体验:补上最后一块短板!冲击高端真的要成了?
拒绝向SSD认输!机械硬盘每GB单价已暴跌87%:将越来越便宜
环球微动态丨python中的高阶函数
今日讯!雷军宣布小米13系列将很快登陆全球市场:国外用户激动坏了
【世界新要闻】《三体》动画播放量破1.3亿:豆瓣出现N多1星剧评
每日关注!奇葩!男子恋爱转账11万分手能向女方要回不 法院判决
动态焦点:AMD悄悄把RX 7900 XT的功耗提高了:游戏性能实测公布
世界热点评!FreeSWITCH学习笔记:模块
天天快播:服!敬业新郎一边结婚一边拉业务:39元套餐送1200分钟通话、90G流量和宽带
世界要闻:三亚游客乘观光直升机突遇意外迫降海面:发动机断油失效
视点!秒懂:JCTool 的 Mpsc 超高性能无锁队列 (史上最全+10W字长文)
全球要闻:免费可商用!荣耀HONOR Sans字体来了 附下载
0.89元/片:KN95口罩30片26.9元大促
全球热文:全球首款!大上科技墨水屏显示器上架:25.3寸超大屏幕
天天观速讯丨我们为什么使用Docker
天天滚动:win11上同时安装多个版本的python
当前信息:999元 小米Sound Pro太顶了:用户体验后感觉地板都在共振
天天看点:裸奔还是不行 Win10/11系统依然需要安全软件:免费的就行
每日速讯:顺丰突然又崩了!官方回应:系统异常 已恢复
系列最轻巧流畅系统!MIUI 14开发版首批推送来了
每日讯息!第一章作业
【全球报资讯】架构到底是指什么?
实验七-缓冲区溢出
全球看点:分析师称马斯克暴砍80%服务器订单:供应商Intel很受伤
新能源车换电池多少钱?比亚迪汉8万多 特斯拉13万
世界即时看!山东女生送闺蜜化妆品到内蒙被冻炸 当事人:心里很难过
全球首款真Hi-Fi无线耳机!vivo TWS 3 Pro图赏
2022最强电影!《阿凡达2》北京首映礼今晚举办:提前感受科幻巨作
PopClip使用教程图文详解 2022.12亲测有效
焦点快报!Html+CSS小案例项目:CSS开发小米商城电商产品展示效果
通过命令上传到GitHub
当前播报:对于async和await的使用方式、作用效果不怎么理解 ?没关系,初步看这篇就够了。
记录--记一次前端CSS升级
世界观焦点:中国生物最新研究!灭活疫苗对新冠康复者同样具有保护效力
环球热点评!女子住1楼质问物业为啥要交电梯费 网友力挺
OPPO Find N2 Flip亮相:副屏太惊艳
环球新资讯:日本公布2022年度热门汉字:“战”二次当选 理由奇葩
李诞入手小米13和13 Pro:大玩谐音梗
每日快报!【脚本项目源码】Python制作多功能音乐播放器,打造专属你的音乐播放器
每日关注!雷军分享小米13/MIUI 14内置壁纸原图:大批iPhone用户感谢
世界观天下!不用耳机也有立体听感!moto X40新增空间音频功能
电动自行车新国标充电插头曝光:三脚接口无了 不配套不上电