国内比较好的互动直播sdk都有哪些?
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了国内比较好的互动直播sdk都有哪些?相关的知识,希望对你有一定的参考价值。
比较好的互动直播sdk有欢拓云直播、声网、腾讯云、网易云信等。更推荐欢拓云直播,功能丰富,解锁企业直播新玩法,涵盖各行业直播常用功能,支持特殊功能定制,轻松玩转直播。点击看看互动直播sdk系统如何选择直播SDK的选择要看这家平台的基础核心能力够不够强,不然很容易出现不稳定等现象,导致出现各种问题,影响到业务就不好了。SDK依托欢拓云直播API实现,对其进行包装和优化。解放B端用户的共性工作。将API调用逻辑和导常外理进行了封装优化 B端用户只需将请求参数封装后,交给欢拓云直播Java SDK处理即可,欢拓云直播SDK处理完成后返回结果,B端依据返回数据继续完成B端业务逻辑。
想要了解更多关于互动直播sdk的相关信息,推荐咨询欢拓云直播官网。欢拓十多年来一直专注于音频、视频的采样、编码、后处理及智能传输等直播核心技术的研究,其产品线涵盖教育、金融、电商、会议、培训、大会、医疗等全行业直播场景,更是获得国家高新技术企业和科技创新小巨人企业认证,满足客户的不同需求。 参考技术A 推荐ZEGO即构科技。ZEGO即构科技是一家全球云通讯服务商,提供优质的服务,为客户解决所需解决的问题。【点击免费试用,0成本启动】
采用第三方直播SDK,可以做的事正是从四个完整环节来进行直播产品基础设施搭建,即据业内朋友透露,一个完整的手机直播系统在产品层面包含四个环节:推流端(采集、前处理、编码、推流),服务端(转码、录制、截图、鉴黄),播放器(拉流、解码、渲染)、互动系统(聊天室、礼物系统、赞)。用户较为在意的美颜、鉴黄、互动以及云服务(PaaS),也可以获得一站式的解决。。
想要了解更多关于第三方直播sdk的信息,推荐咨询ZEGO即构科技。即构科技有限公司是为开发者提供高品质实时音视频、实时语音、互动直播和IM即时通讯等服务的公司。核心团队成员均来自QQ,有超过20年的互联网和音视频技术开发经验,以及亿量级海内外用户运营服务能力。专注于实时音视频领域,致力提供全世界范围内最清晰稳定实时的语音视频服务,已服务4000+企业,适用于游戏、娱乐社交、在线教育、在线医疗、远程会议、智能硬件等多种用户场景。官网欢拓云直播
2022-04-27·百度认证:广州欢拓网络科技有限公司官方帐号欢拓云直播欢拓云直播为企业深度打造最专业、稳定、快速、流畅的直播互动解决方案。只需简单对接,即可快速拥有完全自有品牌标识的直播平台。向TA提问关注比较好的互动直播sdk有欢拓云直播、声网、腾讯云、网易云信等。更推荐欢拓云直播,功能丰富,解锁企业直播新玩法,涵盖各行业直播常用功能,支持特殊功能定制,轻松玩转直播。点击看看互动直播sdk系统如何选择
直播SDK的选择要看这家平台的基础核心能力够不够强,不然很容易出现不稳定等现象,导致出现各种问题,影响到业务就不好了。SDK依托欢拓云直播API实现,对其进行包装和优化。解放B端用户的共性工作。将API调用逻辑和导常外理进行了封装优化 B端用户只需将请求参数封装后,交给欢拓云直播Java SDK处理即可,欢拓云直播SDK处理完成后返回结果,B端依据返回数据继续完成B端业务逻辑。
想要了解更多关于互动直播sdk的相关信息,推荐咨询欢拓云直播官网。欢拓十多年来一直专注于音频、视频的采样、编码、后处理及智能传输等直播核心技术的研究,其产品线涵盖教育、金融、电商、会议、培训、大会、医疗等全行业直播场景,更是获得国家高新技术企业和科技创新小巨人企业认证,满足客户的不同需求。详情已赞过已踩过<你对这个回答的评价是?评论收起ZEGO即构科技
2022-09-27·百度认证:深圳市即构科技有限公司官方帐号ZEGO即构科技深圳市即构科技有限公司致力于提供全球最清晰最稳定的实时语音视频云服务,深耕视频直播、视频社交、游戏语音、线上抓娃娃和在线教育等领域。向TA提问关注推荐ZEGO即构科技。ZEGO即构科技是一家全球云通讯服务商,提供优质的服务,为客户解决所需解决的问题。【点击免费试用,0成本启动】
采用第三方直播SDK,可以做的事正是从四个完整环节来进行直播产品基础设施搭建,即据业内朋友透露,一个完整的手机直播系统在产品层面包含四个环节:推流端(采集、前处理、编码、推流),服务端(转码、录制、截图、鉴黄),播放器(拉流、解码、渲染)、互动系统(聊天室、礼物系统、赞)。用户较为在意的美颜、鉴黄、互动以及云服务(PaaS),也可以获得一站式的解决。。
想要了解更多关于第三方直播sdk的信息,推荐咨询ZEGO即构科技。即构科技有限公司是为开发者提供高品质实时音视频、实时语音、互动直播和IM即时通讯等服务的公司。核心团队成员均来自QQ,有超过20年的互联网和音视频技术开发经验,以及亿量级海内外用户运营服务能力。专注于实时音视频领域,致力提供全世界范围内最清晰稳定实时的语音视频服务,已服务4000+企业,适用于游戏、娱乐社交、在线教育、在线医疗、远程会议、智能硬件等多种用户场景。 参考技术B 推荐微赞直播,微赞直播提供多种直播功能,支持API定制、SDK定制、账号打通、小程序组件。
微赞是建立于微信生态之上的专业企业直播平台,坐拥微信流量优势与社交优势,嵌入企业微信公众号, 微信端一键创建和分享直播。
专注于企业直播服务多年,高效助力企业营销,从解决企业营销痛点出发,内置多种互动营销方式, 数据分析与云端实时安全存储等功能,可实现高清直播且支持百万观看不卡顿。
累计企业直播超400万场,服务了包括腾讯、京东、百度、 中国银行、湖南广电等超100万企业用户,覆盖221个国家和地区的5亿用户。
基于anyrtc的sdk实现直播连麦互动
基于anyrtc的sdk实现直播连麦互动
前言
1.由于粘贴了较大的代码,造成内容比较长,可能会花费您较长的时间。
2.项目里面没有做权限判断,所以如果发现有页面发生崩溃可能是权限没有打开,请打开权限后再进行尝试。
3.录制的GIF画质好差,真实环境很好。
一.有图有真相
主播端 : http://d2.freep.cn/3tb_160831194207x8fa572812.gif
游客端 : http://d3.freep.cn/3tb_160831194126nt2e572812.gif
二.集成步骤,教你实现直播连麦互动
从github上https://github.com/AnyRTC/RTMPCHybirdEngine-Android下载下来,导入AS,结构如下图所示。
1.初始化RTMP引擎
MainActivity.java
RTMPCHybird.Inst().Init(getApplicationContext());
RTMPCHybird.Inst().InitEngineWithAnyrtcInfo("你的开发者ID", "liveDemo", "LslMfUyvcW8hv6NgY0n8FOWf5t1K+YWpoMzRsqwkBiE", "d6ff4b7d1b549f497a0b94cf500fd549");
以上的四个参数可以在https://www.anyrtc.io注册开发者账号并创建一个应用可得,如下图所示。你可以把它放在继承Application的onCreate()方法中。
2.获取直播列表,随时随地观看主播
MainActivity.java
RTMPCHttpSDK.GetLiveList(this, RTMPCHybird.Inst().GetHttpAddr(), 你的开发者ID", "liveDemo", "d6ff4b7d1b549f497a0b94cf500fd549", mRTMPCHttpCallback);
private RTMPCHttpSDK.RTMPCHttpCallback mRTMPCHttpCallback = new RTMPCHttpSDK.RTMPCHttpCallback() {
@Override
public void OnRTMPCHttpOK(String strContent) {
mRecyclerViewUtils.endRefreshing();
try {
listLive.clear();
JSONObject liveJson = new JSONObject(strContent);
JSONArray liveList = liveJson.getJSONArray("LiveList");
JSONArray memberList = liveJson.getJSONArray("LiveMembers");
for (int i = 0; i < liveList.length(); i++) {
LiveItemBean bean = new LiveItemBean();
JSONObject itemJson = new JSONObject(liveList.getString(i));
bean.setmHosterId(itemJson.getString("hosterId"));
bean.setmRtmpPullUrl(itemJson.getString("rtmp_url"));
bean.setmHlsUrl(itemJson.getString("hls_url"));
bean.setmLiveTopic(itemJson.getString("topic"));
bean.setmAnyrtcId(itemJson.getString("anyrtcId"));
bean.setmMemNumber(memberList.getInt(i));
listLive.add(bean);
}
mAdapter.setDatas(listLive);
} catch (JSONException e) {
e.printStackTrace();
}
}
@Override
public void OnRTMPCHttpFailed(int code) {
}
};
以上代码可以获取所有正在直播列表名单
3.发起直播,成就自己的直播梦
a.你可以自己设置推流地址和拉流地址,这里我直接用他提供的地址了
RTMPUrlHelper.java
public class RTMPUrlHelper {
/**
* rtmp 推流地址
*/
public static final String RTMP_PUSH_URL = "rtmp://192.168.7.207:1935/live/%s";
/**
* rtmp 拉流地址
*/
public static final String RTMP_PULL_URL = "rtmp://192.168.7.207:1935/live/%s";
/**
* hls 地址
*/
public static final String HLS_URL = "http://192.169.7.207/live/%s.m3u8";
/**
* 分享页面url地址
*/
public static final String SHARE_WEB_URL = "http://123.59.68.21/rtmpc-demo/?%s";
}
b.这里要使用到你创建应用的anyrtcid、推流拉流hls地址、直播名称。到这一步你已经离你的直播梦完成了一半了<( ̄︶ ̄)>。
PreStartLiveActivity.java
String anyrtcId = RTMPCHttpSDK.getRandomString(12);
String rtmpPushUrl = String.format(RTMPUrlHelper.RTMP_PUSH_URL, anyrtcId);
String rtmpPullUrl = String.format(RTMPUrlHelper.RTMP_PULL_URL, anyrtcId);
String hlsUrl = String.format(RTMPUrlHelper.HLS_URL, anyrtcId);
JSONObject item = new JSONObject();
try {
item.put("hosterId", "hostID");
item.put("rtmp_url", rtmpPullUrl);
item.put("hls_url", hlsUrl);
item.put("topic", topic);
item.put("anyrtcId", anyrtcId);
} catch (JSONException e) {
e.printStackTrace();
}
Bundle bundle = new Bundle();
bundle.putString("hosterId", "hostID");
bundle.putString("rtmp_url", rtmpPushUrl);
bundle.putString("hls_url", hlsUrl);
bundle.putString("topic", topic);
bundle.putString("andyrtcId", anyrtcId);
bundle.putString("userData", item.toString());
Intent intent = new Intent(this, HosterActivity.class);
intent.putExtras(bundle);
startActivity(intent);
4.你的直播梦就在此就要实现了,打起精神(⊙o⊙)哦
a.从上个Activity拿到传过来的数据
HosterActivity.java
mHosterId = getIntent().getExtras().getString("hosterId");
mRtmpPushUrl = getIntent().getExtras().getString("rtmp_url");
mAnyrtcId = getIntent().getExtras().getString("andyrtcId");
mUserData = getIntent().getExtras().getString("userData");
mHlsUrl = getIntent().getExtras().getString("hls_url");
mTopic = getIntent().getExtras().getString("topic");
这里的R.id.rl_rtmpc_videos表示打开手机摄像头捕获的像呈现的地方,如下图所示。
HosterActivity.java
mVideoView = new RTMPCVideoView((RelativeLayout) findViewById(R.id.rl_rtmpc_videos), RTMPCHybird.Inst().Egl(), true);
mVideoView.setBtnCloseEvent(mBtnVideoCloseEvent);
mHosterKit = new RTMPCHosterKit(this, mHosterListener);
{
VideoRenderer render = mVideoView.OnRtcOpenLocalRender();
mHosterKit.SetVideoCapturer(render.GetRenderPointer(), true);
}
b.这里是创建和音频管理器,将照顾音频路由,音频模式,音频设备枚举等、存储现有的音频设置更改音频模式达到最佳的VoIP性能。
{
// Create and audio manager that will take care of audio routing,
// audio modes, audio device enumeration etc.
mRtmpAudioManager = RTMPAudioManager.create(this, new Runnable() {
// This method will be called each time the audio state (number
// and
// type of devices) has been changed.
@Override
public void run() {
onAudioManagerChangedState();
}
});
// Store existing audio settings and change audio mode to
// MODE_IN_COMMUNICATION for best possible VoIP performance.
mRtmpAudioManager.init();
mRtmpAudioManager.setAudioDevice(RTMPAudioManager.AudioDevice.SPEAKER_PHONE);
}
mStartRtmp = true;
/**
* 开始推流
*/
mHosterKit.StartPushRtmpStream(mRtmpPushUrl);
/**
* 建立RTC连线连接
*/
mHosterKit.OpenRTCLine(mAnyrtcId, mHosterId, mUserData);
c.主播的直播回调接口
mHosterKit = new RTMPCHosterKit(this, mHosterListener);
这句代码是主播直播的回调接口,可以做连麦、取消连麦、接收消息和弹幕消息、实时观看人数和人员上下线等。实用的操作一般再此实行。
HosterActivity.java
/**
* 主播回调信息接口
*/
private RTMPCAbstractHoster mHosterListener = new RTMPCAbstractHoster() {
/**
* rtmp连接成功
*/
@Override
public void OnRtmpStreamOKCallback() {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
((TextView) findViewById(R.id.txt_rtmp_connection_status)).setText(R.string.str_rtmp_connect_success);
}
});
}
/**
* rtmp 重连次数
* @param times 重连次数
*/
@Override
public void OnRtmpStreamReconnectingCallback(final int times) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
((TextView) findViewById(R.id.txt_rtmp_connection_status)).setText(String.format(getString(R.string.str_reconnect_times), times));
}
});
}
/**
* rtmp 推流状态
* @param delayMs 推流延时
* @param netBand 推流码流
*/
@Override
public void OnRtmpStreamStatusCallback(final int delayMs, final int netBand) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
((TextView) findViewById(R.id.txt_rtmp_status)).setText(String.format(getString(R.string.str_rtmp_status), delayMs, netBand));
}
});
}
/**
* rtmp推流失败回调
* @param code
*/
@Override
public void OnRtmpStreamFailedCallback(int code) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
((TextView) findViewById(R.id.txt_rtmp_connection_status)).setTextColor(R.color.yellow);
((TextView) findViewById(R.id.txt_rtmp_connection_status)).setText(R.string.str_rtmp_connect_failed);
}
});
}
/**
* rtmp 推流关闭回调
*/
@Override
public void OnRtmpStreamClosedCallback() {
finish();
}
/**
* RTC 连接回调
* @param code 0: 连接成功
* @param strErr 原因
*/
@Override
public void OnRTCOpenLineResultCallback(final int code, String strErr) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
if (code == 0) {
((TextView) findViewById(R.id.txt_rtc_connection_status)).setText(R.string.str_rtc_connect_success);
} else {
((TextView) findViewById(R.id.txt_rtc_connection_status)).setTextColor(R.color.yellow);
((TextView) findViewById(R.id.txt_rtc_connection_status)).setText(R.string.str_rtmp_connect_failed);
}
}
});
}
/**
* 游客有申请连线回调
*
* @param strLivePeerID
* @param strCustomID
* @param strUserData
*/
@Override
public void OnRTCApplyToLineCallback(final String strLivePeerID, final String strCustomID, final String strUserData) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
ShowDialog(HosterActivity.this, strLivePeerID, strCustomID);
}
});
}
/**
* 视频连线超过4人时回调
* @param strLivePeerID
* @param strCustomID
* @param strUserData
*/
@Override
public void OnRTCLineFullCallback(final String strLivePeerID, String strCustomID, String strUserData) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
Toast.makeText(HosterActivity.this, getString(R.string.str_connect_full), Toast.LENGTH_LONG).show();
mHosterKit.RejectRTCLine(strLivePeerID, true);
}
});
}
/**
* 游客挂断连线回调
* @param strLivePeerID
*/
@Override
public void OnRTCCancelLineCallback(String strLivePeerID) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
Toast.makeText(HosterActivity.this, getString(R.string.str_line_disconnect), Toast.LENGTH_LONG).show();
}
});
}
/**
* RTC 连接关闭回调
* @param code 207:请去AnyRTC官网申请账号,如有疑问请联系客服!
* @param strReason
*/
@Override
public void OnRTCLineClosedCallback(final int code, String strReason) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
if (code == 207) {
Toast.makeText(HosterActivity.this, getString(R.string.str_apply_anyrtc_account), Toast.LENGTH_LONG).show();
finish();
}
}
});
}
/**
* 连线接通时的视频图像回调;
* @param strLivePeerID
*/
@Override
public void OnRTCOpenVideoRenderCallback(final String strLivePeerID) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
final VideoRenderer render = mVideoView.OnRtcOpenRemoteRender(strLivePeerID);
if (null != render) {
mHosterKit.SetRTCVideoRender(strLivePeerID, render.GetRenderPointer());
}
}
});
}
/**
* 连线关闭时的视频图像回调;
* @param strLivePeerID
*/
@Override
public void OnRTCCloseVideoRenderCallback(final String strLivePeerID) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
mHosterKit.SetRTCVideoRender(strLivePeerID, 0);
mVideoView.OnRtcRemoveRemoteRender(strLivePeerID);
}
});
}
/**
* 消息回调
* @param strCustomID 消息的发送者id
* @param strCustomName 消息的发送者昵称
* @param strCustomHeader 消息的发送者头像url
* @param strMessage 消息内容
*/
@Override
public void OnRTCUserMessageCallback(final String strCustomID, final String strCustomName, final String strCustomHeader, final String strMessage) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
addChatMessageList(new ChatMessageBean(strCustomID, strCustomName, "", strMessage));
}
});
}
/**
* 弹幕回调
* @param strCustomID 弹幕的发送者id
* @param strCustomName 弹幕的发送者昵称
* @param strCustomHeader 弹幕的发送者头像url
* @param strBarrage 弹幕的内容
*/
@Override
public void OnRTCUserBarrageCallback(final String strCustomID, final String strCustomName, final String strCustomHeader, final String strBarrage) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
addChatMessageList(new ChatMessageBean(strCustomID, strCustomName, "", strBarrage));
IDanmakuItem item = new DanmakuItem(HosterActivity.this, new SpannableString(strBarrage), mDanmakuView.getWidth(), 0, R.color.yellow_normol, 18, 1);
mDanmakuView.addItemToHead(item);
}
});
}
/**
* 直播观看总人数回调
* @param totalMembers 观看总人数
*/
@Override
public void OnRTCMemberListWillUpdateCallback(final int totalMembers) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
((TextView) findViewById(R.id.txt_watcher_number)).setText(String.format(getString(R.string.str_live_watcher_number), totalMembers));
}
});
}
/**
* 人员上下线回调
* @param strCustomID
* @param strUserData
*/
@Override
public void OnRTCMemberCallback(final String strCustomID, final String strUserData) {
HosterActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
try {
JSONObject userData = new JSONObject(strUserData);
addChatMessageList(new ChatMessageBean(userData.getString("nickName"), "", userData.getString("headUrl"), ""));
} catch (JSONException e) {
e.printStackTrace();
}
}
});
}
@Override
public void OnRTCMemberListUpdateDoneCallback() {
}
};
d.一些对应直播回调的方法
接下来就要说对应这些回调的方法了,这里可以做关闭直播、直播分享、切换摄像头和发送消息。
HosterActivity.java
public void OnBtnClicked(View btn) {
if (btn.getId() == R.id.btn_close) {
mStartRtmp = false;
mHosterKit.StopRtmpStream();
finish();
} else if (btn.getId() == R.id.btn_copy_hls) {
int index = mHlsUrl.lastIndexOf("/");
int lastIndex = mHlsUrl.lastIndexOf(".");
String shareUrl = String.format(RTMPUrlHelper.SHARE_WEB_URL, mHlsUrl.substring(index + 1, lastIndex));
mShareHelper.shareWeiXin(mTopic, shareUrl);
} else if (btn.getId() == R.id.btn_switch_camera) {
mHosterKit.SwitchCamera();
} else if (btn.getId() == R.id.btn_send_message) {
String message = editMessage.getText().toString();
editMessage.setText("");
if (message.equals("")) {
return;
}
if (mCheckBarrage.isChecked()) {
mHosterKit.SendBarrage(mNickname, "", message);
IDanmakuItem item = new DanmakuItem(HosterActivity.this, new SpannableString(message), mDanmakuView.getWidth(), 0, R.color.yellow_normol, 18, 1);
mDanmakuView.addItemToHead(item);
} else {
mHosterKit.SendUserMsg(mNickname, "", message);
}
addChatMessageList(new ChatMessageBean(mNickname, mNickname, "", message));
} else if (btn.getId() == R.id.iv_host_text) {
btnChat.clearFocus();
vaBottomBar.setDisplayedChild(1);
editMessage.requestFocus();
softKeyboardUtil.showKeyboard(HosterActivity.this, editMessage);
}
}
5.主播端的已经完成了,现在该轮到你的小伙伴们看你直播~(≧▽≦)/~啦啦啦
a.前面已经说了,我们可以实时获取开启直播的主播,点击列表就可以观看主播直播。
MainActivity.java
@Override
public void onItemChildClick(ViewGroup viewGroup, View view, int i) {
Intent it = new Intent(this, GuestActivity.class);
Bundle bundle = new Bundle();
bundle.putString("hls_url", listLive.get(i).getmHlsUrl());
bundle.putString("rtmp_url", listLive.get(i).getmRtmpPullUrl());
bundle.putString("anyrtcId", listLive.get(i).getmAnyrtcId());
bundle.putString("userData", new JSONObject().toString());
bundle.putString("topic", listLive.get(i).getmLiveTopic());
it.putExtras(bundle);
startActivity(it);
}
b.再实现游客端这个项目基本上就完成了,是不是有点小激动?
与主播端一样,拿上个Activity传过来的数据,其实这里代码与主播端的代码基本上可以说是一模一样,看完之后你应该会有我这样的体会。
GuestActivity.java
mNickname = ((HybirdApplication)HybirdApplication.app()).getmNickname();
mRtmpPullUrl = getIntent().getExtras().getString("rtmp_url");
mAnyrtcId = getIntent().getExtras().getString("anyrtcId");
mHlsUrl = getIntent().getExtras().getString("hls_url");
mGuestId = mNickname;//getIntent().getExtras().getString("guestId");
mTopic = getIntent().getExtras().getString("topic");
mVideoView = new RTMPCVideoView((RelativeLayout) findViewById(R.id.rl_rtmpc_videos), RTMPCHybird.Inst().Egl(), false);
mVideoView.setBtnCloseEvent(mBtnVideoCloseEvent);
/**
* 初始化rtmp播放器
*/
mGuestKit = new RTMPCGuestKit(this, mGuestListener);
VideoRenderer render = mVideoView.OnRtcOpenLocalRender();
{
// Create and audio manager that will take care of audio routing,
// audio modes, audio device enumeration etc.
mRtmpAudioManager = RTMPAudioManager.create(this, new Runnable() {
// This method will be called each time the audio state (number
// and
// type of devices) has been changed.
@Override
public void run() {
onAudioManagerChangedState();
}
});
// Store existing audio settings and change audio mode to
// MODE_IN_COMMUNICATION for best possible VoIP performance.
mRtmpAudioManager.init();
mRtmpAudioManager.setAudioDevice(RTMPAudioManager.AudioDevice.SPEAKER_PHONE);
}
mUserData = new JSONObject();
try {
mUserData.put("nickName", mNickname);
mUserData.put("headUrl", "");
} catch (JSONException e) {
e.printStackTrace();
}
/**
* 开始播放rtmp流
*/
mGuestKit.StartRtmpPlay(mRtmpPullUrl, render.GetRenderPointer());
/**
* 开启RTC连线连接
*/
mGuestKit.JoinRTCLine(mAnyrtcId, mGuestId, mUserData.toString());
c.观看直播的回调接口
mGuestKit = new RTMPCGuestKit(this, mGuestListener);
这句代码是观看直播的回调接口,可以做申请连麦、断开连麦、接收消息和弹幕消息、实时观看人数和人员上下线等。实用的操作一般再此实行。
GuestActivity.java
/**
* 观看直播回调信息接口
*/
private RTMPCAbstractGuest mGuestListener = new RTMPCAbstractGuest() {
/**
* rtmp 连接成功
*/
@Override
public void OnRtmplayerOKCallback() {
}
/**
* rtmp 当前播放状态
* @param cacheTime 当前缓存时间
* @param curBitrate 当前播放器码流
*/
@Override
public void OnRtmplayerStatusCallback(int cacheTime, int curBitrate) {
}
/**
* rtmp 播放缓冲区时长
* @param time 缓冲时间
*/
@Override
public void OnRtmplayerCacheCallback(int time) {
}
/**
* rtmp 播放器关闭
* @param errcode
*/
@Override
public void OnRtmplayerClosedCallback(int errcode) {
}
/**
* 游客RTC 状态回调
* @param code 回调响应码:0:正常;101:主播未开启直播;
* @param strReason 原因描述
*/
@Override
public void OnRTCJoinLineResultCallback(final int code, String strReason) {
GuestActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
if (code == 0) {
} else if (code == 101) {
Toast.makeText(GuestActivity.this, R.string.str_hoster_not_live, Toast.LENGTH_LONG).show();
mHandler.sendEmptyMessageDelayed(CLOSED, 2000);
}
}
});
}
/**
* 游客申请连线回调
* @param code 0:申请连线成功;-1:主播拒绝连线
*/
@Override
public void OnRTCApplyLineResultCallback(final int code) {
GuestActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
if (code == 0) {
VideoRenderer render = mVideoView.OnRtcOpenRemoteRender("LocalCameraRender");
mGuestKit.SetVideoCapturer(render.GetRenderPointer(), true);
} else if (code == -1) {
Toast.makeText(GuestActivity.this, R.string.str_hoster_refused, Toast.LENGTH_LONG).show();
mStartLine = false;
mBtnConnect.setText(R.string.str_connect_hoster);
}
}
});
}
/**
* 当与主播连线成功时,其他用户连线回调
* @param strLivePeerID
* @param strCustomID
* @param strUserData
*/
@Override
public void OnRTCOtherLineOpenCallback(String strLivePeerID, String strCustomID, String strUserData) {
}
/**
* 其他用户连线回调
* @param strLivePeerID
*/
@Override
public void OnRTCOtherLineCloseCallback(String strLivePeerID) {
}
/**
* 挂断连线回调
*/
@Override
public void OnRTCHangupLineCallback() {
//主播连线断开
GuestActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
mGuestKit.HangupRTCLine();
mVideoView.OnRtcRemoveRemoteRender("LocalCameraRender");
mStartLine = false;
mBtnConnect.setText(R.string.str_connect_hoster);
}
});
}
/**
* 主播已离开回调
* @param code
* @param strReason
*/
@Override
public void OnRTCLineLeaveCallback(int code, String strReason) {
//主播关闭直播
GuestActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
Toast.makeText(GuestActivity.this, R.string.str_hoster_leave, Toast.LENGTH_LONG).show();
mHandler.sendEmptyMessageDelayed(CLOSED, 2000);
}
});
}
/**
* 连线接通后回调
* @param strLivePeerID
*/
@Override
public void OnRTCOpenVideoRenderCallback(final String strLivePeerID) {
GuestActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
final VideoRenderer render = mVideoView.OnRtcOpenRemoteRender(strLivePeerID);
mGuestKit.SetRTCVideoRender(strLivePeerID, render.GetRenderPointer());
}
});
}
/**
* 连线关闭后图像回调
* @param strLivePeerID
*/
@Override
public void OnRTCCloseVideoRenderCallback(final String strLivePeerID) {
GuestActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
mGuestKit.SetRTCVideoRender(strLivePeerID, 0);
mVideoView.OnRtcRemoveRemoteRender(strLivePeerID);
}
});
}
/**
* 消息回调
* @param strCustomID 消息的发送者id
* @param strCustomName 消息的发送者昵称
* @param strCustomHeader 消息的发送者头像url
* @param strMessage 消息内容
*/
@Override
public void OnRTCUserMessageCallback(final String strCustomID, final String strCustomName, final String strCustomHeader, final String strMessage) {
GuestActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
addChatMessageList(new ChatMessageBean(strCustomID, strCustomName, "", strMessage));
}
});
}
/**
* 弹幕回调
* @param strCustomID 弹幕的发送者id
* @param strCustomName 弹幕的发送者昵称
* @param strCustomHeader 弹幕的发送者头像url
* @param strBarrage 弹幕的内容
*/
@Override
public void OnRTCUserBarrageCallback(final String strCustomID, final String strCustomName, final String strCustomHeader, final String strBarrage) {
GuestActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
addChatMessageList(new ChatMessageBean(strCustomID, strCustomName, "", strBarrage));
IDanmakuItem item = new DanmakuItem(GuestActivity.this, new SpannableString(strBarrage), mDanmakuView.getWidth(), 0, R.color.yellow_normol, 18, 1);
mDanmakuView.addItemToHead(item);
}
});
}
/**
* 观看直播的总人数回调
* @param totalMembers 观看直播的总人数
*/
@Override
public void OnRTCMemberListWillUpdateCallback(final int totalMembers) {
GuestActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
((TextView) findViewById(R.id.txt_watcher_number)).setText(String.format(getString(R.string.str_live_watcher_number), totalMembers));
}
});
}
/**
* 人员上下线回调
* @param strCustomID
* @param strUserData
*/
@Override
public void OnRTCMemberCallback(final String strCustomID, final String strUserData) {
GuestActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
try {
JSONObject userData = new JSONObject(strUserData);
addChatMessageList(new ChatMessageBean(userData.getString("nickName"), "", userData.getString("headUrl"), ""));
} catch (JSONException e) {
e.printStackTrace();
}
}
});
}
@Override
public void OnRTCMemberListUpdateDoneCallback() {
}
};
d.一些对应观看直播回调的方法
接下来就要说对应这些回调的方法了,这里可以做直播分享、申请连麦、断开连麦和发送消息。
GuestActivity.java
public void OnBtnClicked(View btn) {
if(btn.getId() == R.id.btn_copy_hls) {
int index = mHlsUrl.lastIndexOf("/");
int lastIndex = mHlsUrl.lastIndexOf(".");
String shareUrl = String.format(RTMPUrlHelper.SHARE_WEB_URL, mHlsUrl.substring(index + 1, lastIndex));
mShareHelper.shareWeiXin(mTopic, shareUrl);
} else if (btn.getId() == R.id.btn_line) {
if (!mStartLine) {
JSONObject json = new JSONObject();
try {
json.put("guestId", mNickname);
} catch (JSONException e) {
e.printStackTrace();
}
/**
* 向主播申请连线
*/
mGuestKit.ApplyRTCLine(json.toString());
mStartLine = true;
mBtnConnect.setText(R.string.str_hangup_connect);
} else {
/**
* 挂断连线
*/
mGuestKit.HangupRTCLine();
mVideoView.OnRtcRemoveRemoteRender("LocalCameraRender");
mStartLine = false;
mBtnConnect.setText(R.string.str_connect_hoster);
}
} else if (btn.getId() == R.id.btn_send_message) {
String message = editMessage.getText().toString();
editMessage.setText("");
if (message.equals("")) {
return;
}
if (mCheckBarrage.isChecked()) {
mGuestKit.SendBarrage(mNickname, "", message);
IDanmakuItem item = new DanmakuItem(GuestActivity.this, new SpannableString(message), mDanmakuView.getWidth(), 0, R.color.yellow_normol, 18, 1);
mDanmakuView.addItemToHead(item);
} else {
mGuestKit.SendUserMsg(mNickname, "", message);
}
addChatMessageList(new ChatMessageBean(mNickname, mNickname, "", message));
} else if (btn.getId() == R.id.iv_host_text) {
btnChat.clearFocus();
vaBottomBar.setDisplayedChild(1);
editMessage.requestFocus();
softKeyboardUtil.showKeyboard(GuestActivity.this, editMessage);
}
}
结束语
对于这个基于Anyrtc sdk的直播RTMPCHybirdEngine-Android核心的几个类基本上介绍完成,我在此仅仅是大概介绍了几点,可以让你大概了解这个SDK的直播流程,方便您快速上手。
转载请注明出处,谢谢!