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

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

元宇宙| 用虚拟人进行抖音/快手直播相关推荐

  1. 淘宝/天猫/淘特/京东/拼多多/唯品会/苏宁易购/考拉海购内部隐藏优惠券。饿了么/美团外卖/美团优选/美团酒店/大众点评/口碑红包。抖音/快手直播带货。优惠券APP源码怎么写?分享给大家。

    下载地址:喵惠应用宝https://a.app.qq.com/o/simple.jsp?pkgname=com.miaohui.xin 淘宝/天猫/淘特/京东/拼多多/唯品会/苏宁易购/考拉海购内部隐 ...

  2. 快手上热门的小技巧,抖音快手直播5个上热门技巧

    2020下半年,随着直播带货行业发展势头越来越猛,各大电商平台纷纷涉足直播电商,普通人也跃跃欲试想要加入直播带货,那么普通人该如何选择直播平台呢,这要根据每个人的自身条件,不管做抖音还是快手,初始最重 ...

  3. 2023 易语言 抖音快手直播源获取推流地址源码

    抖音快手直播源获取推流地址易语言源码,能够轻松获取抖音.快手等直播平台的推流地址,并且支持多种直播源格式,包括flv.m3u8等.同时它还能够获取某些直播平台的所有直播间直播源.此外,该源码还提供直播 ...

  4. 抖音带货什么产品都能带吗,抖音快手直播卖货技巧!

    直播带货被认为将是未来电商的趋势,随着抖音快手持续火热,基本上国内网民都已经接受了直播带货的现象,随着初始新鲜感的过期,很多人也看到了直播带货中的一些问题,比如造假严重,观看人数几十万,销量只有两位数 ...

  5. 多网聚合技术,让抖音快手直播效果更好

    目前直播行业发展迅猛,很多地方都建立了直播基地,配置了相关的硬件设备和技术支持人员.在以往直播过程时,大多数使用5GCPE或手机自身的网络来提供直播现场网络,但是在观看人数较多.晚上高峰期时会带来网络 ...

  6. 杨建允:2022年抖音快手电商直播带货选品的技巧和逻辑

    抖音快手电商直播间的流量推荐机制是什么样的? 我们首先来系统的分析一下抖音快手的核心指标: 抖音/快手电商GMV=平台总VV数x电商内容渗透率xGPM VV数=活跃用户数×人均时长 / 单条内容平均时 ...

  7. 零基础开启元宇宙|抖音快手虚拟形象直播【源码】

    在上一篇文章零基础开启元宇宙--创建虚拟形象中,我们实现了创建虚拟形象,接下来我们可以利用虚拟形象"为所欲为".今天我们利用虚拟形象在短视频平台如快手.抖音中直播,对于不希望露脸的 ...

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

    在上一篇文章零基础开启元宇宙--创建虚拟形象中,我们实现了创建虚拟形象,接下来我们可以利用虚拟形象"为所欲为".今天我们利用虚拟形象在短视频平台如快手.抖音中直播,对于不希望露脸的 ...

  9. 广告VS电商 抖音快手的变现之争

    文|曾响铃 来源|科技向令说(xiangling0815) 一向"佛系"的快手坐不住了. 继4月底低调试水"真会选"焕新季活动后,快手于近日(7月9日)举行&q ...

最新文章

  1. 删除SQL数据库中事务日志方法
  2. linux内核层功能 和核心,Linux内核研发工程师
  3. python环境搭建
  4. php5.23升级,2018年5月5号23:30分对php的学后感
  5. [云炬创业管理笔记]第四章把握创业机会测试3
  6. 限界分支法(实际上没有剪枝,介绍的是广度优先搜索):01背包问题,队列实现方式(FIFO)
  7. java自动gc_具有Java 7中自动资源管理功能的GC
  8. (找规律)Magic of David Copperfield
  9. IDEA、Eclipse的详细安装配置及Tomcat服务集成介绍
  10. idea 调试远程tomcat
  11. win7右键没有新建文件夹了
  12. vue封装websocket_Vue中使用websocket
  13. 【MySQL学习笔记】MySQL库与表
  14. pytorch线性模型 刘二
  15. 产品经理常用的分析模型方法
  16. ZEMAX | 照明设计中实用的光学模拟方法
  17. 学习笔记(1):FFmpeg打造Android万能音频播放器-实现变速变调功能(二)
  18. java document对象详解
  19. js监听只读文本框_js设置input文本框只读
  20. 笔记本制作WIFI热点经常掉线问题

热门文章

  1. 啥牌子的蓝牙耳机好用不贵?超帅的蓝牙耳机推荐
  2. wps怎么添加附录_wps用尾注生成参考文献后怎么插入致谢和附录等,来充电吧
  3. C++输入三角形三边判断三角形类型
  4. PLC如何实现循环定时开关(Cycle_time_switch)
  5. jquery事件委托off与on连用无效的问题
  6. Android 中的多线程简介
  7. unicloud中上传图片
  8. 码神之路博客部署教程【完整版】|基于Linux的Docker部署教程|非常详细
  9. MYSQL 最大连接数
  10. 下载微信公众号中的视频