抖音快手如何轻松接入虚拟人直播

在上一篇文章零基础开启元宇宙——创建虚拟形象中,我们实现了创建虚拟形象,接下来我们可以利用虚拟形象“为所欲为”。今天我们利用虚拟形象在短视频平台如快手、抖音中直播,对于不希望露脸的主播们这是可是一大利器呀!话不多说,上绝活。

请添加图片描述在这里插入图片描述

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文件中添加必要的权限信息:


<uses-permission android:name="android.permission.RECORD_AUDIO" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.CAMERA" />
<uses-permission android:name="android.permission.BLUETOOTH" />
<uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /><uses-featureandroid:glEsVersion="0x00020000"android:required="true" />
<uses-feature android:name="android.hardware.camera" />
<uses-feature android:name="android.hardware.camera.autofocus" />

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() {@Overridepublic void onStart(ZegoPublishChannel zegoPublishChannel) {if (listener != null) {listener.onStartCapture();}}@Overridepublic 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);
}@Override
public 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;
}

首先执行顺序如下:

  1. createRTCEngine, 获取RTC引擎对象:engine。
  2. setCustomVideo, 用于设置自定义推流采样视频帧数据相关属性。
  3. 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());
}@Override
public void onStartCapture() {if (mUser == null) return;
//        // 收到回调后,开发者需要执行启动视频采集相关的业务逻辑,例如开启摄像头等AvatarCaptureConfig config = new AvatarCaptureConfig(mUser.width, mUser.height);
//        // 开始捕获纹理mCharacterHelper.startCaptureAvatar(config, this::onCaptureAvatar);
}@Override
public void onStopCapture() {Log.e(TAG, "结束推流");mCharacterHelper.stopCaptureAvatar();stopExpression();
}

以上步骤实现了Android端将虚拟形象推流到服务器端,详细代码可以看附件。

3 PC端拉取实时虚拟形象并展示

前往https://doc-zh.zego.im/article/3209下载Web版RTC SDK。文件结构如下:

在这里插入图片描述

keycenter.js中定义APPID等属性值。

// 请从官网控制台获取对应的appID
const 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 requirements
function 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 addedif (updateType === 'ADD') {const addStream = streamList[streamList.length - 1]if (addStream && addStream.streamID) {onAddRemoteStream(addStream.streamID)}} else if (updateType == 'DELETE') {//  del streamconst delStream = streamList[streamList.length - 1]if (delStream && delStream.streamID) {if (delStream.streamID === remoteStreamID) {engine.stopPlayingStream(remoteStreamID)}}}});
}// Step5 Start Play Stream
function 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 room
function 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 room
function logoutRoom(engine, roomId) {engine.logoutRoom(roomId);
}

在index.html中引用如上javascript文件,展示拉流内容:

<html> 
<head><link href="index.css" type="text/css" rel="stylesheet"/> <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></head><body>  <div class="toast_box"><p id="toast"></p></div><div class="loginPanel"><div class="formRow"><label id="loginErrorMsg"></label> </div><div class="formRow"><label>userId</label><input type="text" id="userId" value="S_0001"/></div><div class="formRow"><label>房间号</label><input type="text" id="roomId" value="R_0001"/></div><button id="loginBtn">登录</button></div><div id="playVideo"></div> <script src="./js/index.js"></script>
</body></html>

4 快手、抖音直播推送实时虚拟画面

接下来是振奋人心时刻,到了联调时刻。在android打开画面实时推理,并在浏览器中打开界面,可以看到如下画面:
请添加图片描述

接下来只需使用直播伴侣软件,将浏览器中的实时画面实时转发到快手或抖音。这里我们用快手直播伴侣实时截屏直播,可以看到如下画面

在这里插入图片描述

5 附件

源码: https://github.com/RTCWang/Virtual-Live

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/60782.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

开利成为独立上市公司,其股票开始在纽约证券交易所交易

Carrier Global Corporation成功从联合技术公司(United Technologies)分离出来&#xff0c;今天作为一家独立的上市公司首次亮相。Carrier是全球领先的建筑和制冷解决方案提供商&#xff0c;公司股票将在纽约证券交易所(NYSE)开始“常规”交易, 交易代码为“CARR”。 Carrier是…

海外问卷调查答题技巧,纯干货分享,新手小白看过来

海外问卷调查为什么别人赚得盆满钵满而我却连通过都不行&#xff1f;是不是经常有人发出这种疑问&#xff0c;东哥作为一个结交过很多做问卷调查行业的跨境人士&#xff0c;也了解到很多做这一行的去答题的时候都是掌握一定技巧的&#xff0c;而不是去乱答。 今天东哥就来说说国…

微信小程序做一个调查问卷(二)

即将不定期的开放代码 核心代码呈现源码下载在微信小程序如何展示富文本编辑器设置的内容多使用缓存技术&#xff0c;防止过多无效的访问题目分为单选题、多选题、简答题设置有其他选项&#xff0c;可进行手动输入选项多选题选择个数设置&#xff0c;只可最多选择三个 核心代码…

我想做国外问卷调查该怎么去入门?

目前2021年&#xff0c;很多的互联网创业项目横空出世。但是在这其中大家要有很好的眼光去识别并判断这个项目到底好不好呢&#xff1f;那我们该怎么样去判断这个呢&#xff0c;这是很多人存在疑惑的地方。 小编这里可以给大家说下咱们可以怎么样去判断一个创业项目的好与坏&am…

国外问卷调查该怎么做?入门须知!

现在跨境问卷调查的风越来越大&#xff0c;许多人想做但却苦于没有入门的方法&#xff0c;今天就来给大家讲讲做跨境调查问卷怎么做&#xff1f; 跨境问卷调查就是针对国外一些部门与各类组织机构等发布的问卷调查进行解答以获取报酬佣金。而且它操作简单&#xff0c;易上手&a…

国外调查问卷怎么做?

首先&#xff0c;我来说说什么是问卷调查&#xff1f;问卷调查是指通过制定详细的问卷&#xff0c;要求被调查者根据问卷进行回答来收集数据的方法。所谓问卷&#xff0c;就是与研究目标相关的一组问题&#xff0c;或者是为调查而编制的问题形式&#xff0c;也称调查表。 我们…

哪些软件可以做国外问卷调查

大家好&#xff0c;我是大成工作室大成&#xff0c;一个踩过很多坑的老韭菜&#xff0c;每天会给大家分享和拆解各种项目&#xff0c;引流思路。帮助每一个互联网的网漂人员&#xff0c;找到自己的方向&#xff0c;成为自己时间的主人&#xff0c;达到时间自由。 知道很多项目…

国外问卷调查这个项目可以做吗?国外问卷调查怎么赚钱?

大家好&#xff0c;我是小飞匠&#xff0c;今天来聊聊国外问卷调查这个项目可以做吗&#xff1f;国外问卷调查怎么赚钱&#xff1f; 我的答案是必须可以做&#xff01; 先来说我自己做国外问卷调查的一个工作状态&#xff0c;听着小说&#xff0c;点着鼠标然后就把美金挣了&a…

学习计算机基础有什么推荐的书和视频?

大家好&#xff0c;我是小林。 之前有读者问我&#xff0c;学习计算机基础有什么推荐的书&#xff1f; 这一个我就很有心得了&#xff0c;因为我大学的专业并不是计算机专业的&#xff0c;是电气自动化专业的&#xff0c;所以学校的课程并没有操作系统、计算机网络、计算机组…

文笔山记

上海的天空下着雨&#xff0c;我的心情也下着雨&#xff0c;最近因为工作的事情搞得我自己都感觉到心理上的扭曲&#xff0c;我很烦恼……按理说这个时候我什么也不会干&#xff0c;但这次我无论如何得找些事情&#xff0c;于是打算写篇游记&#xff0c;也算对 2006年“闷了一…

快冲!淘宝无货源副业,傻瓜式操作,日赚300-500元!!

如果有人问&#xff1a;有什么事&#xff0c;比穷更难受&#xff1f; 我的回答一定是&#xff1a;今年过半了&#xff0c;我还在喝西北风&#xff0c;我的朋友先暴富了&#xff01; 这段时间&#xff0c;我就被闺蜜美美狠狠扎了心。 以前她和我一样是个普通上班族&#xff0c;去…

Kotlin Flow响应式编程,操作符函数进阶

本文同步发表于我的微信公众号&#xff0c;扫一扫文章底部的二维码或在微信搜索 郭霖 即可关注&#xff0c;每个工作日都有文章更新。 大家好&#xff0c;今天原创。 在上一篇原创文章当中&#xff0c;我跟大家说了会开启一个新的系列&#xff0c;讲一讲Kotlin Flow响应式编程…

如果再来一次,你还会选择互联网么?

现在互联网的就业环境&#xff0c;大家都在感受着一股寒意。也不知道从什么时候开始&#xff0c;身边悲观的声音越来越多了。 如果再给你一次机会&#xff0c;你还会选择互联网吗&#xff1f; 回答这个问题之前&#xff0c;我想跟大家聊聊一个我朋友的故事。 他从学渣到大厂程…

奉劝各位学弟学妹们,该打造你的技术影响力了!

CSDN 的学弟学妹们&#xff0c;大家好呀&#xff0c;我是沉默王二。放在一年前&#xff0c;打死我也不相信&#xff0c;这四个平淡无奇的字组合在一起竟然充满了魔力&#xff01; 2019 年的时候&#xff0c;我看过一本书&#xff0c;名叫《影响力》&#xff0c;应该有不少学弟…

文笔如何迅速提升?国企老员工和你聊聊

很多人说在我们国企混&#xff0c;人际交往能力很重要&#xff0c;然而还有一项能力和人际交往能力一样重要&#xff0c;那就是——文笔。 文笔好有哪些好处&#xff1f; 1、工作总结汇报的时候&#xff0c;能让自己的工作看起来更有价值 2、在公司的一些活动比赛中&#xff…

刘宇凡:浅谈文笔、文采、文案的区别

声明:刘宇凡不是这方面的专家,我只是作为一个喜爱文字的搬运工谈谈自己的体会罢了。 此文的缘起,是一位微信好友的问题。本来当时就该回答的,但是一直拖到现在才以文章的形式作答,实在是对不住,也许你已经通过其它渠道了解到其本质与区别。但还是希望我的见解能够给你不…

5G与AI“点燃”数字应用,云原生重塑数据库未来

进入2023年&#xff0c;5G与AI技术逼近爆发的临界点&#xff0c;即将点燃数字应用的宇宙。以ChatGPT为代表的AI大模型&#xff0c;引发了机器智能的涌现&#xff0c;不仅能够与人类进行创造性对话&#xff0c;还能代替程序员编写代码&#xff01;而5G全面加速了物理世界的数字化…

下一代数据中心怎么建?数据中心能源十大趋势揭秘答案

国家发改委的信息显示&#xff0c;当前&#xff0c;“东数西算”工程的8个国家算力枢纽节点建设已全部开工&#xff0c;2023年新开工的数据中心项目近70个&#xff0c;“东数西算”工程正从系统布局进入全面建设阶段。可以预见&#xff0c;新一轮的数据中心建设大潮将蓬勃兴起。…

全网最详细中英文ChatGPT-GPT-4示例文档-从0到1快速入门自然语言指令创建支付API代码——官网推荐的48种最佳应用场景(附python/node.js/curl命令源代码,小白也能学)

从0到1快速入门自然语言指令创建支付API代码 Introduce 简介setting 设置Prompt 提示Sample response 回复样本API request 接口请求python接口请求示例node.js接口请求示例curl命令示例json格式示例 其它资料下载 ChatGPT是目前最先进的AI聊天机器人&#xff0c;它能够理解图片…

大模型与端到端会成为城市自动驾驶新范式吗?

摘要&#xff1a; 最近可以明显看到或者感受到第一梯队的城市自动驾驶量产已经进入快车道&#xff0c;他们背后所依靠的正是当下最热的大模型和端到端的技术。 近期&#xff0c;城市自动驾驶量产在产品和技术上都出现了新的变化。 在产品层面&#xff0c;出现了记性行车或者称…