Android基于騰訊云實時音視頻仿微信視頻通話最小化懸浮
最近項目中有需要語音、視頻通話需求,看到這個像環(huán)信、融云等SDK都有具體Demo實現(xiàn),但咋的領(lǐng)導(dǎo)對騰訊情有獨鐘啊,IM要用騰訊云IM,不妙的是騰訊云IM并不包含有音視頻通話都要自己實現(xiàn),沒辦法深入了解騰訊云產(chǎn)品后,決定自己基于騰訊云實時音視頻做去語音、視頻通話功能。在這里把實現(xiàn)過程記錄下為以后用到便于查閱,另一方面也給有需要的人提供一個思路,讓大家少走彎路,有可能我的實現(xiàn)的方法不是最好,但是這或許是一個可行的方案,大家不喜勿噴。基于騰訊云實時音視頻SDK 6.5.7272版本,騰訊DEMO下載地址:鏈接: https://pan.baidu.com/s/1iJsVO3KBuhEiIUZcJPyv3g 提取碼: ueey
一、實現(xiàn)效果


二、實現(xiàn)思路
我把實現(xiàn)思路拆分為了兩步:1、視頻通話Activity的最小化。 2、視頻通話懸浮框的開啟
具體思路是這樣的:當用戶點擊左上角最小化按鈕的時候,最小化視頻通話Activity(這時Activity處于后臺狀態(tài)),于此同時開啟懸浮框,新建一個新的ViewGroup將全局Constents.mVideoViewLayout中用戶選中的最大View動態(tài)添加到懸浮框里面去,監(jiān)聽懸浮框的觸摸事件,讓懸浮框可以拖拽移動;自定義點擊事件,如果用戶點擊了懸浮框,則移除懸浮框然后重新調(diào)起我們在后臺的視頻通話Activity。
1.Activity是如何實現(xiàn)最小化的?
Activity本身自帶了一個moveTaskToBack(boolean nonRoot),我們要實現(xiàn)最小化只需要調(diào)用moveTaskToBack(true)傳入一個true值就可以了,但是這里有一個前提,就是需要設(shè)置Activity的啟動模式為singleInstance模式,兩步搞定。(注:activity最小化后重新從后臺回到前臺會回調(diào)onRestart()方法)
@Override
public boolean moveTaskToBack(boolean nonRoot) {
return super.moveTaskToBack(nonRoot);
}
2.懸浮框是如何開啟的?
懸浮框的實現(xiàn)方法最好寫在Service里面,將懸浮框的開啟關(guān)閉與服務(wù)Service的綁定解綁所關(guān)聯(lián)起來,開啟服務(wù)即相當于開啟我們的懸浮框,解綁服務(wù)則相當于關(guān)閉關(guān)閉的懸浮框,以此來達到更好的控制效果。
a. 首先我們聲明一個服務(wù)類,取名為FloatVideoWindowService:
public class FloatVideoWindowService extends Service {
@Nullable
@Override
public IBinder onBind(Intent intent) {
return new MyBinder();
}
public class MyBinder extends Binder {
public FloatVideoWindowService getService() {
return FloatVideoWindowService.this;
}
}
@Override
public void onCreate() {
super.onCreate();
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
return super.onStartCommand(intent, flags, startId);
}
@Override
public void onDestroy() {
super.onDestroy();
}
}
b. 為懸浮框建立一個布局文件float_video_window_layout,懸浮框大小我這里固定為長80dp,高120dp,id為small_size_preview的RelativeLayout主要是一個容器,可以動態(tài)的添加view到里面去
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/small_size_frame_layout" android:layout_width="wrap_content" android:layout_height="wrap_content" android:background="@color/colorComBg" android:orientation="vertical"> <com.tencent.rtmp.ui.TXCloudVideoView android:id="@+id/float_videoview" android:layout_width="80dp" android:layout_height="120dp" android:descendantFocusability="blocksDescendants" android:orientation="vertical" /> </LinearLayout>
c. 布局定義好后,接下來就要對懸浮框做一些初始化操作了,初始化操作這里我們放在服務(wù)的onCreate()生命周期里面執(zhí)行,因為只需要執(zhí)行一次就行了。這里的初始化主要包括對:懸浮框的基本參數(shù)(位置,寬高等),懸浮框的點擊事件以及懸浮框的觸摸事件(即可拖動范圍)等的設(shè)置,在onBind()中從Intent中取出了Activity中用戶選中最大View的id,以便在后面從 Constents.mVideoViewLayout中取出對應(yīng)View,然后加入懸浮窗布局中
/**
* 視頻懸浮窗服務(wù)
*/
public class FloatVideoWindowService extends Service {
private WindowManager mWindowManager;
private WindowManager.LayoutParams wmParams;
private LayoutInflater inflater;
private String currentBigUserId;
//浮動布局view
private View mFloatingLayout;
//容器父布局
private RelativeLayout smallSizePreviewLayout;
private TXCloudVideoView mLocalVideoView;
@Override
public void onCreate() {
super.onCreate();
initWindow();//設(shè)置懸浮窗基本參數(shù)(位置、寬高等)
}
@Nullable
@Override
public IBinder onBind(Intent intent) {
currentBigUserId = intent.getStringExtra("userId");
initFloating();//懸浮框點擊事件的處理
return new MyBinder();
}
public class MyBinder extends Binder {
public FloatVideoWindowService getService() {
return FloatVideoWindowService.this;
}
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
return super.onStartCommand(intent, flags, startId);
}
/**
* 設(shè)置懸浮框基本參數(shù)(位置、寬高等)
*/
private void initWindow() {
mWindowManager = (WindowManager) getApplicationContext().getSystemService(Context.WINDOW_SERVICE);
//設(shè)置好懸浮窗的參數(shù)
wmParams = getParams();
// 懸浮窗默認顯示以左上角為起始坐標
wmParams.gravity = Gravity.LEFT | Gravity.TOP;
//懸浮窗的開始位置,因為設(shè)置的是從左上角開始,所以屏幕左上角是x=0;y=0
wmParams.x = 70;
wmParams.y = 210;
//得到容器,通過這個inflater來獲得懸浮窗控件
inflater = LayoutInflater.from(getApplicationContext());
// 獲取浮動窗口視圖所在布局
mFloatingLayout = inflater.inflate(R.layout.alert_float_video_layout, null);
// 添加懸浮窗的視圖
mWindowManager.addView(mFloatingLayout, wmParams);
}
private WindowManager.LayoutParams getParams() {
wmParams = new WindowManager.LayoutParams();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
wmParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
} else {
wmParams.type = WindowManager.LayoutParams.TYPE_PHONE;
}
//設(shè)置可以顯示在狀態(tài)欄上
wmParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL |
WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN | WindowManager.LayoutParams.FLAG_LAYOUT_INSET_DECOR |
WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH;
//設(shè)置懸浮窗口長寬數(shù)據(jù)
wmParams.width = WindowManager.LayoutParams.WRAP_CONTENT;
wmParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
return wmParams;
}
private void initFloating() {
}
}
d. 在懸浮框成功被初始化以及相關(guān)參數(shù)被設(shè)置后,接下來就需要將Activity中用戶選中最大的View添加到懸浮框里面去了,這樣我們才能看到視頻畫面嘛,同樣我們是在Service的onCreate這個生命周期中initFloating()完成這個操作的,代碼如下所示:
TRTCVideoViewLayout mTRTCVideoViewLayout = Constents.mVideoViewLayout;
TXCloudVideoView mLocalVideoView = mTRTCVideoViewLayout.getCloudVideoViewByUseId(currentBigUserId);
if (mLocalVideoView == null) {
mLocalVideoView = mTRTCVideoViewLayout.getCloudVideoViewByIndex(0);
}
if (ConstData.userid.equals(currentBigUserId)) {
TXCGLSurfaceView mTXCGLSurfaceView = mLocalVideoView.getGLSurfaceView();
if (mTXCGLSurfaceView != null && mTXCGLSurfaceView.getParent() != null) {
((ViewGroup) mTXCGLSurfaceView.getParent()).removeView(mTXCGLSurfaceView);
mTXCloudVideoView.addVideoView(mTXCGLSurfaceView);
}
} else {
TextureView mTextureView = mLocalVideoView.getVideoView();
if (mTextureView != null && mTextureView.getParent() != null) {
((ViewGroup) mTextureView.getParent()).removeView(mTextureView);
mTXCloudVideoView.addVideoView(mTextureView);
}
}
e. 我們上面說到要將服務(wù)Service的綁定與解綁與懸浮框的開啟和關(guān)閉相結(jié)合,所以既然我們在服務(wù)的onCreate()方法中開啟了懸浮框,那么就應(yīng)該在其onDestroy()方法中對懸浮框進行關(guān)閉,關(guān)閉懸浮框的本質(zhì)是將相關(guān)View給移除掉,在服務(wù)的onDestroy()方法中執(zhí)行如下代碼:
@Override
public void onDestroy() {
super.onDestroy();
if (mFloatingLayout != null) {
// 移除懸浮窗口
mWindowManager.removeView(mFloatingLayout);
mFloatingLayout = null;
Constents.isShowFloatWindow = false;
}
}
f. 服務(wù)的綁定方式有bindService和startService兩種,使用不同的綁定方式其生命周期也會不一樣,已知我們需要讓懸浮框在視頻通話activity finish掉的時候也順便關(guān)掉,那么理所當然我們就應(yīng)該采用bind方式來啟動服務(wù),讓他的生命周期跟隨他的開啟者,也即是跟隨開啟它的activity生命周期。
intent = new Intent(this, FloatVideoWindowService.class);//開啟服務(wù)顯示懸浮框
bindService(intent, mVideoServiceConnection, Context.BIND_AUTO_CREATE);
ServiceConnection mVideoServiceConnection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// 獲取服務(wù)的操作對象
FloatVideoWindowService.MyBinder binder = (FloatVideoWindowService.MyBinder) service;
binder.getService();
}
@Override
public void onServiceDisconnected(ComponentName name) {
}
};
Service完整代碼如下:
/**
* 視頻懸浮窗服務(wù)
*/
public class FloatVideoWindowService extends Service {
private WindowManager mWindowManager;
private WindowManager.LayoutParams wmParams;
private LayoutInflater inflater;
private String currentBigUserId;
//浮動布局view
private View mFloatingLayout;
//容器父布局
private TXCloudVideoView mTXCloudVideoView;
@Override
public void onCreate() {
super.onCreate();
initWindow();//設(shè)置懸浮窗基本參數(shù)(位置、寬高等)
}
@Nullable
@Override
public IBinder onBind(Intent intent) {
currentBigUserId = intent.getStringExtra("userId");
initFloating();//懸浮框點擊事件的處理
return new MyBinder();
}
public class MyBinder extends Binder {
public FloatVideoWindowService getService() {
return FloatVideoWindowService.this;
}
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
return super.onStartCommand(intent, flags, startId);
}
@Override
public void onDestroy() {
super.onDestroy();
if (mFloatingLayout != null) {
// 移除懸浮窗口
mWindowManager.removeView(mFloatingLayout);
mFloatingLayout = null;
Constents.isShowFloatWindow = false;
}
}
/**
* 設(shè)置懸浮框基本參數(shù)(位置、寬高等)
*/
private void initWindow() {
mWindowManager = (WindowManager) getApplicationContext().getSystemService(Context.WINDOW_SERVICE);
//設(shè)置好懸浮窗的參數(shù)
wmParams = getParams();
// 懸浮窗默認顯示以左上角為起始坐標
wmParams.gravity = Gravity.LEFT | Gravity.TOP;
//懸浮窗的開始位置,因為設(shè)置的是從左上角開始,所以屏幕左上角是x=0;y=0
wmParams.x = 70;
wmParams.y = 210;
//得到容器,通過這個inflater來獲得懸浮窗控件
inflater = LayoutInflater.from(getApplicationContext());
// 獲取浮動窗口視圖所在布局
mFloatingLayout = inflater.inflate(R.layout.alert_float_video_layout, null);
// 添加懸浮窗的視圖
mWindowManager.addView(mFloatingLayout, wmParams);
}
private WindowManager.LayoutParams getParams() {
wmParams = new WindowManager.LayoutParams();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
wmParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
} else {
wmParams.type = WindowManager.LayoutParams.TYPE_PHONE;
}
//設(shè)置可以顯示在狀態(tài)欄上
wmParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL |
WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN | WindowManager.LayoutParams.FLAG_LAYOUT_INSET_DECOR |
WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH;
//設(shè)置懸浮窗口長寬數(shù)據(jù)
wmParams.width = WindowManager.LayoutParams.WRAP_CONTENT;
wmParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
return wmParams;
}
private void initFloating() {
mTXCloudVideoView = mFloatingLayout.findViewById(R.id.float_videoview);
TRTCVideoViewLayout mTRTCVideoViewLayout = Constents.mVideoViewLayout;
TXCloudVideoView mLocalVideoView = mTRTCVideoViewLayout.getCloudVideoViewByUseId(currentBigUserId);
if (mLocalVideoView == null) {
mLocalVideoView = mTRTCVideoViewLayout.getCloudVideoViewByIndex(0);
}
if (ConstData.userid.equals(currentBigUserId)) {
TXCGLSurfaceView mTXCGLSurfaceView = mLocalVideoView.getGLSurfaceView();
if (mTXCGLSurfaceView != null && mTXCGLSurfaceView.getParent() != null) {
((ViewGroup) mTXCGLSurfaceView.getParent()).removeView(mTXCGLSurfaceView);
mTXCloudVideoView.addVideoView(mTXCGLSurfaceView);
}
} else {
TextureView mTextureView = mLocalVideoView.getVideoView();
if (mTextureView != null && mTextureView.getParent() != null) {
((ViewGroup) mTextureView.getParent()).removeView(mTextureView);
mTXCloudVideoView.addVideoView(mTextureView);
}
}
Constents.isShowFloatWindow = true;
//懸浮框觸摸事件,設(shè)置懸浮框可拖動
mTXCloudVideoView.setOnTouchListener(new FloatingListener());
//懸浮框點擊事件
mTXCloudVideoView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//在這里實現(xiàn)點擊重新回到Activity
Intent intent = new Intent(FloatVideoWindowService.this, TRTCVideoCallActivity.class);
startActivity(intent);
}
});
}
//開始觸控的坐標,移動時的坐標(相對于屏幕左上角的坐標)
private int mTouchStartX, mTouchStartY, mTouchCurrentX, mTouchCurrentY;
//開始時的坐標和結(jié)束時的坐標(相對于自身控件的坐標)
private int mStartX, mStartY, mStopX, mStopY;
//判斷懸浮窗口是否移動,這里做個標記,防止移動后松手觸發(fā)了點擊事件
private boolean isMove;
private class FloatingListener implements View.OnTouchListener {
@Override
public boolean onTouch(View v, MotionEvent event) {
int action = event.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
isMove = false;
mTouchStartX = (int) event.getRawX();
mTouchStartY = (int) event.getRawY();
mStartX = (int) event.getX();
mStartY = (int) event.getY();
break;
case MotionEvent.ACTION_MOVE:
mTouchCurrentX = (int) event.getRawX();
mTouchCurrentY = (int) event.getRawY();
wmParams.x += mTouchCurrentX - mTouchStartX;
wmParams.y += mTouchCurrentY - mTouchStartY;
mWindowManager.updateViewLayout(mFloatingLayout, wmParams);
mTouchStartX = mTouchCurrentX;
mTouchStartY = mTouchCurrentY;
break;
case MotionEvent.ACTION_UP:
mStopX = (int) event.getX();
mStopY = (int) event.getY();
if (Math.abs(mStartX - mStopX) >= 1 || Math.abs(mStartY - mStopY) >= 1) {
isMove = true;
}
break;
default:
break;
}
//如果是移動事件不觸發(fā)OnClick事件,防止移動的時候一放手形成點擊事件
return isMove;
}
}
}
Activity中的操作
現(xiàn)在我們將思路了捋一下,假設(shè)現(xiàn)在我正在進行視頻通話,點擊視頻最小化按鈕,我們應(yīng)該按順序執(zhí)行如下步驟:應(yīng)該是會出現(xiàn)個懸浮框。我們用mServiceBound保存Service注冊狀態(tài),后面解綁時候用這個去判斷,不能有些從其他頁面過來調(diào)用OnRestart()方法的會報錯 說 Service not register之類的錯誤。
/*
* 開啟懸浮Video服務(wù)
*/
private void startVideoService() {
//最小化Activity
moveTaskToBack(true);
Constents.mVideoViewLayout = mVideoViewLayout;
//開啟服務(wù)顯示懸浮框
Intent floatVideoIntent = new Intent(this, FloatVideoWindowService.class);
floatVideoIntent.putExtra("userId", currentBigUserId);
mServiceBound=bindService(floatVideoIntent, mVideoCallServiceConnection, Context.BIND_AUTO_CREATE);
}
注意:這里用了一個全部變量 Constents.mVideoViewLayout 保存Activity中的mVideoViewLayout,以便在上面的Service中使用。
當我們點擊懸浮框的時候,可以使用startActivity(intent)來再次打開我們的activity,這時候視頻通話activity會回調(diào)onRestart()方法,我們在onRestart()生命周期里面unbind解綁掉懸浮框服務(wù),并且重新設(shè)置mVideoViewLayout展示
@Override
protected void onRestart() {
super.onRestart();
//不顯示懸浮框
if (mServiceBound) {
unbindService(mVideoCallServiceConnection);
mServiceBound = false;
}
TXCloudVideoView txCloudVideoView = mVideoViewLayout.getCloudVideoViewByUseId(currentBigUserId);
if (txCloudVideoView == null) {
txCloudVideoView = mVideoViewLayout.getCloudVideoViewByIndex(0);
}
if(ConstData.userid.equals(currentBigUserId)){
TXCGLSurfaceView mTXCGLSurfaceView=txCloudVideoView.getGLSurfaceView();
if (mTXCGLSurfaceView!=null && mTXCGLSurfaceView.getParent() != null) {
((ViewGroup) mTXCGLSurfaceView.getParent()).removeView(mTXCGLSurfaceView);
txCloudVideoView.addVideoView(mTXCGLSurfaceView);
}
}else{
TextureView mTextureView=txCloudVideoView.getVideoView();
if (mTextureView!=null && mTextureView.getParent() != null) {
((ViewGroup) mTextureView.getParent()).removeView(mTextureView);
txCloudVideoView.addVideoView(mTextureView);
}
}
}
視頻Activity是在Demo中TRTCMainActivity的基礎(chǔ)上修改完善的
視頻Activity全部代碼如下:
public class TRTCVideoCallActivity extends Activity implements View.OnClickListener,
TRTCSettingDialog.ISettingListener, TRTCMoreDialog.IMoreListener,
TRTCVideoViewLayout.ITRTCVideoViewLayoutListener, TRTCVideoViewLayout.OnVideoToChatClickListener,
TRTCCallMessageManager.TRTCVideoCallMessageCancelListener {
private final static String TAG = TRTCVideoCallActivity.class.getSimpleName();
private boolean bEnableVideo = true, bEnableAudio = true;
private boolean mCameraFront = true;
private TextView tvRoomId;
private ImageView ivCamera, ivVoice;
private TRTCVideoViewLayout mVideoViewLayout;
//通話計時
private Chronometer callTimeChronometer;
private TRTCCloudDef.TRTCParams trtcParams; /// TRTC SDK 視頻通話房間進入所必須的參數(shù)
private TRTCCloud trtcCloud; /// TRTC SDK 實例對象
private TRTCCloudListenerImpl trtcListener; /// TRTC SDK 回調(diào)監(jiān)聽
private HashSet<String> mRoomMembers = new HashSet<>();
private int mSdkAppId = -1;
private String trtcCallFrom;
private String trtcCallType;
private int roomId;
private String userSig;
private CountDownTimer countDownTimer;
private ImageView trtcSmallIv;
private String currentBigUserId = ConstData.userid;
private HomeWatcher mHomeWatcher;
private boolean mServiceBound = false;
/**
* 不包含自己的接收人列表(單聊情況)
*/
private List<SampleUser> receiveUsers = new ArrayList<>();
private static class VideoStream {
String userId;
int streamType;
public boolean equals(Object obj) {
if (obj == null || userId == null) return false;
VideoStream stream = (VideoStream) obj;
return (this.streamType == stream.streamType && this.userId.equals(stream.userId));
}
}
/**
* 定義服務(wù)綁定的回調(diào) 開啟視頻通話服務(wù)連接
*/
private ServiceConnection mVideoCallServiceConnection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// 獲取服務(wù)的操作對象
FloatVideoWindowService.MyBinder binder = (FloatVideoWindowService.MyBinder) service;
binder.getService();
}
@Override
public void onServiceDisconnected(ComponentName name) {
}
};
private ArrayList<VideoStream> mVideosInRoom = new ArrayList<>();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//應(yīng)用運行時,保持屏幕高亮,不鎖屏
getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
requestWindowFeature(Window.FEATURE_NO_TITLE);
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);
ActivityUtil.addDestoryActivityToMap(TRTCVideoCallActivity.this, TAG);
TRTCCallMessageManager.getInstance().setTRTCVideoCallMessageListener(this);
//獲取前一個頁面得到的進房參數(shù)
Intent intent = getIntent();
long mSdkAppIdTemp = intent.getLongExtra("sdkAppId", 0);
mSdkAppId = Integer.parseInt(String.valueOf(mSdkAppIdTemp));
roomId = intent.getIntExtra("roomId", 0);
trtcCallFrom = intent.getStringExtra("trtcCallFrom");
trtcCallType = intent.getStringExtra("trtcCallType");
ConstData.currentTrtcCallType = trtcCallType;
ConstData.currentRoomId = roomId + "";
receiveUsers = (List<SampleUser>) getIntent().getSerializableExtra("receiveUserList");
userSig = intent.getStringExtra("userSig");
trtcParams = new TRTCCloudDef.TRTCParams(mSdkAppId, ConstData.userid, userSig, roomId, "", "");
trtcParams.role = TRTCCloudDef.TRTCRoleAnchor;
//初始化 UI 控件
initView();
//創(chuàng)建 TRTC SDK 實例
trtcListener = new TRTCCloudListenerImpl(this);
trtcCloud = TRTCCloud.sharedInstance(this);
trtcCloud.setListener(trtcListener);
//開始進入視頻通話房間
enterRoom();
/** 倒計時30秒,一次1秒 */
countDownTimer = new CountDownTimer(30 * 1000, 1000) {
@Override
public void onTick(long millisUntilFinished) {
// TODO Auto-generated method stub
if (!TRTCVideoCallActivity.this.isFinishing() && ConstData.enterRoomUserIdSet.size() > 0) {
countDownTimer.cancel();
}
}
@Override
public void onFinish() {
//倒計時全部結(jié)束執(zhí)行操作
if (!TRTCVideoCallActivity.this.isFinishing() && ConstData.enterRoomUserIdSet.size() == 0) {
exitRoom();
}
}
};
countDownTimer.start();
/**
* home鍵監(jiān)聽相關(guān)
*/
mHomeWatcher = new HomeWatcher(this);
mHomeWatcher.setOnHomePressedListener(new HomeWatcher.OnHomePressedListener() {
@Override
public void onHomePressed() {
//按了HOME鍵
//如果懸浮窗沒有顯示 就開啟服務(wù)展示懸浮窗
if (!Constents.isShowFloatWindow) {
startVideoService();
}
}
@Override
public void onRecentAppsPressed() {
//最近app任務(wù)列表按鍵
if (!Constents.isShowFloatWindow) {
startVideoService();
}
}
});
mHomeWatcher.startWatch();
}
@Override
protected void onResume() {
super.onResume();
}
@Override
protected void onDestroy() {
super.onDestroy();
if (countDownTimer != null) {
countDownTimer.cancel();
}
trtcCloud.setListener(null);
TRTCCloud.destroySharedInstance();
ConstData.isEnterTRTCCALL = false;
//解綁 不顯示懸浮框
if (mServiceBound) {
unbindService(mVideoCallServiceConnection);
mServiceBound = false;
}
if (mHomeWatcher != null) {
mHomeWatcher.stopWatch();// 在銷毀時停止監(jiān)聽,不然會報錯的。
}
}
/**
* 重寫onBackPressed
* 屏蔽返回鍵
*/
@Override
public void onBackPressed() {
// super.onBackPressed();//要去掉這句
}
/**
* 初始化界面控件,包括主要的視頻顯示View,以及底部的一排功能按鈕
*/
private void initView() {
setContentView(R.layout.activity_trtc_video);
trtcSmallIv = (ImageView) findViewById(R.id.trtc_small_iv);
trtcSmallIv.setOnClickListener(this);
initClickableLayout(R.id.ll_camera);
initClickableLayout(R.id.ll_voice);
initClickableLayout(R.id.ll_change_camera);
mVideoViewLayout = (TRTCVideoViewLayout) findViewById(R.id.video_ll_mainview);
mVideoViewLayout.setUserId(trtcParams.userId);
mVideoViewLayout.setListener(this);
mVideoViewLayout.setOnVideoToChatListener(this);
callTimeChronometer = (Chronometer) findViewById(R.id.call_time_chronometer);
ivVoice = (ImageView) findViewById(R.id.iv_mic);
ivCamera = (ImageView) findViewById(R.id.iv_camera);
tvRoomId = (TextView) findViewById(R.id.tv_room_id);
tvRoomId.setText(ConstData.username + "(自己)");
findViewById(R.id.video_ring_off_btn).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
exitRoom();
/**
* 單人通話時
* 新增主叫方在接收方未接聽前掛斷時
* 發(fā)送消息給接收方 讓接收方取消響鈴頁面或者 來電彈框
*/
if ( trtcCallType.equals(Constents.ONE_TO_ONE_VIDEO_CALL)) {
//ConstData.enterRoomUserIdSet.size() == 0表示還沒有接收方加入房間
if (ConstData.enterRoomUserIdSet.size() == 0) {
sendDeclineMsg();
}
}
}
});
}
private LinearLayout initClickableLayout(int resId) {
LinearLayout layout = (LinearLayout) findViewById(resId);
layout.setOnClickListener(this);
return layout;
}
/**
* 設(shè)置視頻通話的視頻參數(shù):需要 TRTCSettingDialog 提供的分辨率、幀率和流暢模式等參數(shù)
*/
private void setTRTCCloudParam() {
// 大畫面的編碼器參數(shù)設(shè)置
// 設(shè)置視頻編碼參數(shù),包括分辨率、幀率、碼率等等,這些編碼參數(shù)來自于 TRTCSettingDialog 的設(shè)置
// 注意(1):不要在碼率很低的情況下設(shè)置很高的分辨率,會出現(xiàn)較大的馬賽克
// 注意(2):不要設(shè)置超過25FPS以上的幀率,因為電影才使用24FPS,我們一般推薦15FPS,這樣能將更多的碼率分配給畫質(zhì)
TRTCCloudDef.TRTCVideoEncParam encParam = new TRTCCloudDef.TRTCVideoEncParam();
encParam.videoResolution = TRTCCloudDef.TRTC_VIDEO_RESOLUTION_640_360;
encParam.videoFps = 15;
encParam.videoBitrate = 600;
encParam.videoResolutionMode = TRTCCloudDef.TRTC_VIDEO_RESOLUTION_MODE_PORTRAIT;
trtcCloud.setVideoEncoderParam(encParam);
TRTCCloudDef.TRTCNetworkQosParam qosParam = new TRTCCloudDef.TRTCNetworkQosParam();
qosParam.controlMode = TRTCCloudDef.VIDEO_QOS_CONTROL_SERVER;
qosParam.preference = TRTCCloudDef.TRTC_VIDEO_QOS_PREFERENCE_CLEAR;
trtcCloud.setNetworkQosParam(qosParam);
trtcCloud.setPriorRemoteVideoStreamType(TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_BIG);
}
/**
* 加入視頻房間:需要 TRTCNewViewActivity 提供的 TRTCParams 函數(shù)
*/
private void enterRoom() {
// 預(yù)覽前配置默認參數(shù)
setTRTCCloudParam();
// 開啟視頻采集預(yù)覽
if (trtcParams.role == TRTCCloudDef.TRTCRoleAnchor) {
startLocalVideo(true);
}
trtcCloud.setBeautyStyle(TRTCCloudDef.TRTC_BEAUTY_STYLE_SMOOTH, 5, 5, 5);
if (trtcParams.role == TRTCCloudDef.TRTCRoleAnchor) {
trtcCloud.startLocalAudio();
}
setVideoFillMode(true);
setVideoRotation(true);
enableAudioHandFree(true);
enableGSensor(true);
enableAudioVolumeEvaluation(false);
/**
* 2019/08/08
* 默認打開是前置攝像頭
* 前置攝像頭就設(shè)置鏡像 true
*/
enableVideoEncMirror(true);
setLocalViewMirrorMode(TRTCCloudDef.TRTC_VIDEO_MIRROR_TYPE_AUTO);
mVideosInRoom.clear();
mRoomMembers.clear();
trtcCloud.enterRoom(trtcParams, TRTCCloudDef.TRTC_APP_SCENE_VIDEOCALL);
}
/**
* 退出視頻房間
*/
private void exitRoom() {
if (trtcCloud != null) {
trtcCloud.exitRoom();
}
ToastUtil.toastShortMessage("通話已結(jié)束");
}
@Override
public void onClick(View v) {
if (v.getId() == R.id.trtc_small_iv) {
startVideoService();
} else if (v.getId() == R.id.ll_camera) {
onEnableVideo();
} else if (v.getId() == R.id.ll_voice) {
onEnableAudio();
} else if (v.getId() == R.id.ll_change_camera) {
onChangeCamera();
}
}
/**
* 發(fā)送掛斷/拒接電話消息
*/
private void sendDeclineMsg() {
TIMMessage timMessage = new TIMMessage();
TIMCustomElem ele = new TIMCustomElem();
/**
* 掛斷/拒接語音、視頻通話消息
* msgContent不放內(nèi)容
*/
String msgStr = null;
if (trtcCallType.equals(Constents.ONE_TO_ONE_AUDIO_CALL)
|| trtcCallType.equals(Constents.ONE_TO_MULTIPE_AUDIO_CALL)) {
msgStr = JsonUtil.toJson(Constents.AUDIO_CALL_MESSAGE_DECLINE_DESC, null);
} else if (trtcCallType.equals(Constents.ONE_TO_ONE_VIDEO_CALL)
|| trtcCallType.equals(Constents.ONE_TO_MULTIPE_VIDEO_CALL)) {
msgStr = JsonUtil.toJson(Constents.VIDEO_CALL_MESSAGE_DECLINE_DESC, null);
}
ele.setData(msgStr.getBytes());
timMessage.addElement(ele);
String receiveUserId = null;
if (!receiveUsers.isEmpty()) {
SampleUser sampleUser = receiveUsers.get(0);
receiveUserId = sampleUser.getUserid();
}
TIMConversation conversation = TIMManager.getInstance().getConversation(
TIMConversationType.C2C, receiveUserId);
//發(fā)送消息
conversation.sendOnlineMessage(timMessage, new TIMValueCallBack<TIMMessage>() {
@Override
public void onError(int code, String desc) {//發(fā)送消息失敗
//錯誤碼 code 和錯誤描述 desc,可用于定位請求失敗原因
//錯誤碼 code 含義請參見錯誤碼表
Log.d("NNN", "send message failed. code: " + code + " errmsg: " + desc);
}
@Override
public void onSuccess(TIMMessage msg) {//發(fā)送消息成功
Log.e("NNN", "SendMsg ok");
}
});
}
/**
* 開啟懸浮Video服務(wù)
*/
private void startVideoService() {
//最小化Activity
moveTaskToBack(true);
Constents.mVideoViewLayout = mVideoViewLayout;
//開啟服務(wù)顯示懸浮框
Intent floatVideoIntent = new Intent(this, FloatVideoWindowService.class);
floatVideoIntent.putExtra("userId", currentBigUserId);
mServiceBound = bindService(floatVideoIntent, mVideoCallServiceConnection, Context.BIND_AUTO_CREATE);
}
@Override
protected void onRestart() {
super.onRestart();
//不顯示懸浮框
if (mServiceBound) {
unbindService(mVideoCallServiceConnection);
mServiceBound = false;
}
TXCloudVideoView txCloudVideoView = mVideoViewLayout.getCloudVideoViewByUseId(currentBigUserId);
if (txCloudVideoView == null) {
txCloudVideoView = mVideoViewLayout.getCloudVideoViewByIndex(0);
}
if(ConstData.userid.equals(currentBigUserId)){
TXCGLSurfaceView mTXCGLSurfaceView=txCloudVideoView.getGLSurfaceView();
if (mTXCGLSurfaceView!=null && mTXCGLSurfaceView.getParent() != null) {
((ViewGroup) mTXCGLSurfaceView.getParent()).removeView(mTXCGLSurfaceView);
txCloudVideoView.addVideoView(mTXCGLSurfaceView);
}
}else{
TextureView mTextureView=txCloudVideoView.getVideoView();
if (mTextureView!=null && mTextureView.getParent() != null) {
((ViewGroup) mTextureView.getParent()).removeView(mTextureView);
txCloudVideoView.addVideoView(mTextureView);
}
}
}
/**
* 開啟/關(guān)閉視頻上行
*/
private void onEnableVideo() {
bEnableVideo = !bEnableVideo;
startLocalVideo(bEnableVideo);
mVideoViewLayout.updateVideoStatus(trtcParams.userId, bEnableVideo);
ivCamera.setImageResource(bEnableVideo ? R.mipmap.remote_video_enable : R.mipmap.remote_video_disable);
}
/**
* 開啟/關(guān)閉音頻上行
*/
private void onEnableAudio() {
bEnableAudio = !bEnableAudio;
trtcCloud.muteLocalAudio(!bEnableAudio);
ivVoice.setImageResource(bEnableAudio ? R.mipmap.mic_enable : R.mipmap.mic_disable);
}
/**
* 點擊切換攝像頭
*/
private void onChangeCamera() {
mCameraFront = !mCameraFront;
onSwitchCamera(mCameraFront);
}
@Override
public void onComplete() {
setTRTCCloudParam();
setVideoFillMode(true);
// moreDlg.updateVideoFillMode(true);
}
/**
* SDK內(nèi)部狀態(tài)回調(diào)
*/
static class TRTCCloudListenerImpl extends TRTCCloudListener implements TRTCCloudListener.TRTCVideoRenderListener {
private WeakReference<TRTCVideoCallActivity> mContext;
private HashMap<String, TestRenderVideoFrame> mCustomRender;
public TRTCCloudListenerImpl(TRTCVideoCallActivity activity) {
super();
mContext = new WeakReference<>(activity);
mCustomRender = new HashMap<>(10);
}
/**
* 加入房間
*/
@Override
public void onEnterRoom(long elapsed) {
final TRTCVideoCallActivity activity = mContext.get();
if (activity != null) {
activity.mVideoViewLayout.onRoomEnter();
activity.updateCloudMixtureParams();
activity.callTimeChronometer.setBase(SystemClock.elapsedRealtime());
activity.callTimeChronometer.start();
}
}
/**
* 離開房間
*/
@Override
public void onExitRoom(int reason) {
TRTCVideoCallActivity activity = mContext.get();
ConstData.enterRoomUserIdSet.clear();
ConstData.receiveUserSet.clear();
ConstData.isEnterTRTCCALL = false;
Log.e(TAG, "onExitRoom:11111111111111111111 ");
if (activity != null) {
activity.callTimeChronometer.stop();
activity.finish();
}
}
/**
* ERROR 大多是不可恢復(fù)的錯誤,需要通過 UI 提示用戶
*/
@Override
public void onError(int errCode, String errMsg, Bundle extraInfo) {
Log.d(TAG, "sdk callback onError");
TRTCVideoCallActivity activity = mContext.get();
if (activity == null) {
return;
}
if (errCode == TXLiteAVCode.ERR_ROOM_REQUEST_TOKEN_HTTPS_TIMEOUT ||
errCode == TXLiteAVCode.ERR_ROOM_REQUEST_IP_TIMEOUT ||
errCode == TXLiteAVCode.ERR_ROOM_REQUEST_ENTER_ROOM_TIMEOUT) {
Toast.makeText(activity, "進房超時,請檢查網(wǎng)絡(luò)或稍后重試:" + errCode + "[" + errMsg + "]", Toast.LENGTH_SHORT).show();
activity.exitRoom();
return;
}
if (errCode == TXLiteAVCode.ERR_ROOM_REQUEST_TOKEN_INVALID_PARAMETER ||
errCode == TXLiteAVCode.ERR_ENTER_ROOM_PARAM_NULL ||
errCode == TXLiteAVCode.ERR_SDK_APPID_INVALID ||
errCode == TXLiteAVCode.ERR_ROOM_ID_INVALID ||
errCode == TXLiteAVCode.ERR_USER_ID_INVALID ||
errCode == TXLiteAVCode.ERR_USER_SIG_INVALID) {
Toast.makeText(activity, "進房參數(shù)錯誤:" + errCode + "[" + errMsg + "]", Toast.LENGTH_SHORT).show();
activity.exitRoom();
return;
}
if (errCode == TXLiteAVCode.ERR_ACCIP_LIST_EMPTY ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_UNPACKING_ERROR ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_TOKEN_ERROR ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_ALLOCATE_ACCESS_FAILED ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_GENERATE_SIGN_FAILED ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_TOKEN_TIMEOUT ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_INVALID_COMMAND ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_GENERATE_KEN_ERROR ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_GENERATE_TOKEN_ERROR ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_DATABASE ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_BAD_ROOMID ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_BAD_SCENE_OR_ROLE ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_ROOMID_EXCHANGE_FAILED ||
errCode == TXLiteAVCode.ERR_SERVER_INFO_STRGROUP_HAS_INVALID_CHARS ||
errCode == TXLiteAVCode.ERR_SERVER_ACC_TOKEN_TIMEOUT ||
errCode == TXLiteAVCode.ERR_SERVER_ACC_SIGN_ERROR ||
errCode == TXLiteAVCode.ERR_SERVER_ACC_SIGN_TIMEOUT ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_INVALID_ROOMID ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_CREATE_ROOM_FAILED ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_SIGN_ERROR ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_SIGN_TIMEOUT ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_ADD_USER_FAILED ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_FIND_USER_FAILED ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_SWITCH_TERMINATION_FREQUENTLY ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_LOCATION_NOT_EXIST ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_ROUTE_TABLE_ERROR ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_INVALID_PARAMETER) {
Toast.makeText(activity, "進房失敗,請稍后重試:" + errCode + "[" + errMsg + "]", Toast.LENGTH_SHORT).show();
activity.exitRoom();
return;
}
if (errCode == TXLiteAVCode.ERR_SERVER_CENTER_ROOM_FULL ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_REACH_PROXY_MAX) {
Toast.makeText(activity, "進房失敗,房間滿了,請稍后重試:" + errCode + "[" + errMsg + "]", Toast.LENGTH_SHORT).show();
activity.exitRoom();
return;
}
if (errCode == TXLiteAVCode.ERR_SERVER_CENTER_ROOM_ID_TOO_LONG) {
Toast.makeText(activity, "進房失敗,roomID超出有效范圍:" + errCode + "[" + errMsg + "]", Toast.LENGTH_SHORT).show();
activity.exitRoom();
return;
}
if (errCode == TXLiteAVCode.ERR_SERVER_ACC_ROOM_NOT_EXIST ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_ROOM_NOT_EXIST) {
Toast.makeText(activity, "進房失敗,請確認房間號正確:" + errCode + "[" + errMsg + "]", Toast.LENGTH_SHORT).show();
activity.exitRoom();
return;
}
if (errCode == TXLiteAVCode.ERR_SERVER_INFO_SERVICE_SUSPENDED) {
Toast.makeText(activity, "進房失敗,請確認騰訊云實時音視頻賬號狀態(tài)是否欠費:" + errCode + "[" + errMsg + "]", Toast.LENGTH_SHORT).show();
activity.exitRoom();
return;
}
if (errCode == TXLiteAVCode.ERR_SERVER_INFO_PRIVILEGE_FLAG_ERROR ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_NO_PRIVILEDGE_CREATE_ROOM ||
errCode == TXLiteAVCode.ERR_SERVER_CENTER_NO_PRIVILEDGE_ENTER_ROOM) {
Toast.makeText(activity, "進房失敗,無權(quán)限進入房間:" + errCode + "[" + errMsg + "]", Toast.LENGTH_SHORT).show();
activity.exitRoom();
return;
}
if (errCode <= TXLiteAVCode.ERR_SERVER_SSO_SIG_EXPIRED &&
errCode >= TXLiteAVCode.ERR_SERVER_SSO_INTERNAL_ERROR) {
// 錯誤參考 https://cloud.tencent.com/document/product/269/1671#.E5.B8.90.E5.8F.B7.E7.B3.BB.E7.BB.9F
Toast.makeText(activity, "進房失敗,userSig錯誤:" + errCode + "[" + errMsg + "]", Toast.LENGTH_SHORT).show();
activity.exitRoom();
return;
}
Toast.makeText(activity, "onError: " + errMsg + "[" + errCode + "]", Toast.LENGTH_SHORT).show();
}
/**
* WARNING 大多是一些可以忽略的事件通知,SDK內(nèi)部會啟動一定的補救機制
*/
@Override
public void onWarning(int warningCode, String warningMsg, Bundle extraInfo) {
Log.d(TAG, "sdk callback onWarning");
}
/**
* 有新的用戶加入了當前視頻房間
*/
@Override
public void onUserEnter(String userId) {
TRTCVideoCallActivity activity = mContext.get();
ConstData.enterRoomUserIdSet.add(userId);
if (activity != null) {
// 創(chuàng)建一個View用來顯示新的一路畫面
// TXCloudVideoView renderView = activity.mVideoViewLayout.onMemberEnter(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_BIG);
TXCloudVideoView renderView = activity.mVideoViewLayout.onMemberEnter(userId);
if (renderView != null) {
// 設(shè)置儀表盤數(shù)據(jù)顯示
renderView.setVisibility(View.VISIBLE);
}
}
}
/**
* 有用戶離開了當前視頻房間
*/
@Override
public void onUserExit(String userId, int reason) {
TRTCVideoCallActivity activity = mContext.get();
ConstData.enterRoomUserIdSet.remove(userId);
if (activity != null) {
if (activity.trtcCallFrom.equals(userId)) {
activity.exitRoom();
} else {
if (ConstData.enterRoomUserIdSet.size() == 0) {
activity.exitRoom();
}
}
//停止觀看畫面
activity.trtcCloud.stopRemoteView(userId);
activity.trtcCloud.stopRemoteSubStreamView(userId);
//更新視頻UI
// activity.mVideoViewLayout.onMemberLeave(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_BIG);
// activity.mVideoViewLayout.onMemberLeave(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_SUB);
activity.mVideoViewLayout.onMemberLeave(userId );
activity.mVideoViewLayout.onMemberLeave(userId );
activity.mRoomMembers.remove(userId);
activity.updateCloudMixtureParams();
TestRenderVideoFrame customRender = mCustomRender.get(userId);
if (customRender != null) {
customRender.stop();
mCustomRender.remove(userId);
}
}
}
/**
* 有用戶屏蔽了畫面
*/
@Override
public void onUserVideoAvailable(final String userId, boolean available) {
TRTCVideoCallActivity activity = mContext.get();
if (activity != null) {
if (available) {
// final TXCloudVideoView renderView = activity.mVideoViewLayout.onMemberEnter(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_BIG);
final TXCloudVideoView renderView = activity.mVideoViewLayout.onMemberEnter(userId);
if (renderView != null) {
// 啟動遠程畫面的解碼和顯示邏輯,F(xiàn)illMode 可以設(shè)置是否顯示黑邊
activity.trtcCloud.setRemoteViewFillMode(userId, TRTCCloudDef.TRTC_VIDEO_RENDER_MODE_FIT);
activity.trtcCloud.startRemoteView(userId, renderView);
activity.runOnUiThread(new Runnable() {
@Override
public void run() {
// renderView.setUserId(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_BIG);
renderView.setUserId(userId );
}
});
}
activity.mRoomMembers.add(userId);
activity.updateCloudMixtureParams();
} else {
activity.trtcCloud.stopRemoteView(userId);
// activity.mVideoViewLayout.onMemberLeave(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_BIG);
activity.mVideoViewLayout.onMemberLeave(userId );
activity.mRoomMembers.remove(userId);
activity.updateCloudMixtureParams();
}
activity.mVideoViewLayout.updateVideoStatus(userId, available);
}
}
@Override
public void onUserSubStreamAvailable(final String userId, boolean available) {
TRTCVideoCallActivity activity = mContext.get();
if (activity != null) {
if (available) {
// final TXCloudVideoView renderView = activity.mVideoViewLayout.onMemberEnter(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_SUB);
final TXCloudVideoView renderView = activity.mVideoViewLayout.onMemberEnter(userId );
if (renderView != null) {
// 啟動遠程畫面的解碼和顯示邏輯,F(xiàn)illMode 可以設(shè)置是否顯示黑邊
activity.trtcCloud.setRemoteSubStreamViewFillMode(userId, TRTCCloudDef.TRTC_VIDEO_RENDER_MODE_FIT);
activity.trtcCloud.startRemoteSubStreamView(userId, renderView);
activity.runOnUiThread(new Runnable() {
@Override
public void run() {
// renderView.setUserId(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_SUB);
renderView.setUserId(userId );
}
});
}
} else {
activity.trtcCloud.stopRemoteSubStreamView(userId);
// activity.mVideoViewLayout.onMemberLeave(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_SUB);
activity.mVideoViewLayout.onMemberLeave(userId );
}
}
}
/**
* 有用戶屏蔽了聲音
*/
@Override
public void onUserAudioAvailable(String userId, boolean available) {
TRTCVideoCallActivity activity = mContext.get();
if (activity != null) {
if (available) {
// final TXCloudVideoView renderView = activity.mVideoViewLayout.onMemberEnter(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_BIG);
final TXCloudVideoView renderView = activity.mVideoViewLayout.onMemberEnter(userId );
if (renderView != null) {
renderView.setVisibility(View.VISIBLE);
}
}
}
}
/**
* 首幀渲染回調(diào)
*/
@Override
public void onFirstVideoFrame(String userId, int streamType, int width, int height) {
TRTCVideoCallActivity activity = mContext.get();
Log.e(TAG, "onFirstVideoFrame: 77777777777777777777777");
if (activity != null) {
// activity.mVideoViewLayout.freshToolbarLayoutOnMemberEnter(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_BIG);
activity.mVideoViewLayout.freshToolbarLayoutOnMemberEnter(userId );
}
}
@Override
public void onStartPublishCDNStream(int err, String errMsg) {
}
@Override
public void onStopPublishCDNStream(int err, String errMsg) {
}
@Override
public void onRenderVideoFrame(String userId, int streamType, TRTCCloudDef.TRTCVideoFrame frame) {
// Log.w(TAG, String.format("onRenderVideoFrame userId: %s, type: %d",userId, streamType));
}
@Override
public void onUserVoiceVolume(ArrayList<TRTCCloudDef.TRTCVolumeInfo> userVolumes, int totalVolume) {
// mContext.get().mVideoViewLayout.resetAudioVolume();
for (int i = 0; i < userVolumes.size(); ++i) {
mContext.get().mVideoViewLayout.updateAudioVolume(userVolumes.get(i).userId, userVolumes.get(i).volume);
}
}
@Override
public void onStatistics(TRTCStatistics statics) {
}
@Override
public void onConnectOtherRoom(final String userID, final int err, final String errMsg) {
TRTCVideoCallActivity activity = mContext.get();
if (activity != null) {
}
}
@Override
public void onDisConnectOtherRoom(final int err, final String errMsg) {
TRTCVideoCallActivity activity = mContext.get();
if (activity != null) {
}
}
@Override
public void onNetworkQuality(TRTCCloudDef.TRTCQuality localQuality, ArrayList<TRTCCloudDef.TRTCQuality> remoteQuality) {
TRTCVideoCallActivity activity = mContext.get();
if (activity != null) {
activity.mVideoViewLayout.updateNetworkQuality(localQuality.userId, localQuality.quality);
for (TRTCCloudDef.TRTCQuality qualityInfo : remoteQuality) {
activity.mVideoViewLayout.updateNetworkQuality(qualityInfo.userId, qualityInfo.quality);
}
}
}
}
@Override
public void onEnableRemoteVideo(final String userId, boolean enable) {
if (enable) {
// final TXCloudVideoView renderView = mVideoViewLayout.getCloudVideoViewByUseId(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_BIG);
final TXCloudVideoView renderView = mVideoViewLayout.getCloudVideoViewByUseId(userId );
if (renderView != null) {
trtcCloud.setRemoteViewFillMode(userId, TRTCCloudDef.TRTC_VIDEO_RENDER_MODE_FIT);
trtcCloud.startRemoteView(userId, renderView);
runOnUiThread(new Runnable() {
@Override
public void run() {
// renderView.setUserId(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_BIG);
renderView.setUserId(userId);
mVideoViewLayout.freshToolbarLayoutOnMemberEnter(userId);
}
});
}
} else {
trtcCloud.stopRemoteView(userId);
}
}
@Override
public void onEnableRemoteAudio(String userId, boolean enable) {
trtcCloud.muteRemoteAudio(userId, !enable);
}
@Override
public void onChangeVideoFillMode(String userId, boolean adjustMode) {
trtcCloud.setRemoteViewFillMode(userId, adjustMode ? TRTCCloudDef.TRTC_VIDEO_RENDER_MODE_FIT : TRTCCloudDef.TRTC_VIDEO_RENDER_MODE_FILL);
}
@Override
public void onChangeVideoShowFrame(String userId, String userName) {
currentBigUserId = userId;
tvRoomId.setText(userName);
}
@Override
public void onSwitchCamera(boolean bCameraFront) {
trtcCloud.switchCamera();
/**
* 2019/08/08
* 此處增加判斷
* 前置攝像頭就設(shè)置鏡像 true
* 后置攝像頭就不設(shè)置鏡像 false
*/
if (bCameraFront) {
enableVideoEncMirror(true);
} else {
enableVideoEncMirror(false);
}
}
/**
* 視頻里點擊進入和某人聊天
*
* @param userId
*/
@Override
public void onVideoToChatClick(String userId) {
Intent chatIntent = new Intent(TRTCVideoCallActivity.this, IMSingleActivity.class);
chatIntent.putExtra(IMKeys.INTENT_ID, userId);
startActivity(chatIntent);
if (!Constents.isShowFloatWindow) {
startVideoService();
}
}
/**
* 拒接視頻通話回調(diào)
*/
@Override
public void onTRTCVideoCallMessageCancel() {
exitRoom();
}
@Override
public void onFillModeChange(boolean bFillMode) {
setVideoFillMode(bFillMode);
}
@Override
public void onVideoRotationChange(boolean bVertical) {
setVideoRotation(bVertical);
}
@Override
public void onEnableAudioCapture(boolean bEnable) {
enableAudioCapture(bEnable);
}
@Override
public void onEnableAudioHandFree(boolean bEnable) {
enableAudioHandFree(bEnable);
}
@Override
public void onMirrorLocalVideo(int localViewMirror) {
setLocalViewMirrorMode(localViewMirror);
}
@Override
public void onMirrorRemoteVideo(boolean bMirror) {
enableVideoEncMirror(bMirror);
}
@Override
public void onEnableGSensor(boolean bEnable) {
enableGSensor(bEnable);
}
@Override
public void onEnableAudioVolumeEvaluation(boolean bEnable) {
enableAudioVolumeEvaluation(bEnable);
}
@Override
public void onEnableCloudMixture(boolean bEnable) {
updateCloudMixtureParams();
}
private void setVideoFillMode(boolean bFillMode) {
if (bFillMode) {
trtcCloud.setLocalViewFillMode(TRTCCloudDef.TRTC_VIDEO_RENDER_MODE_FILL);
} else {
trtcCloud.setLocalViewFillMode(TRTCCloudDef.TRTC_VIDEO_RENDER_MODE_FIT);
}
}
private void setVideoRotation(boolean bVertical) {
if (bVertical) {
trtcCloud.setLocalViewRotation(TRTCCloudDef.TRTC_VIDEO_ROTATION_0);
} else {
trtcCloud.setLocalViewRotation(TRTCCloudDef.TRTC_VIDEO_ROTATION_90);
}
}
private void enableAudioCapture(boolean bEnable) {
if (bEnable) {
trtcCloud.startLocalAudio();
} else {
trtcCloud.stopLocalAudio();
}
}
private void enableAudioHandFree(boolean bEnable) {
if (bEnable) {
trtcCloud.setAudioRoute(TRTCCloudDef.TRTC_AUDIO_ROUTE_SPEAKER);
} else {
trtcCloud.setAudioRoute(TRTCCloudDef.TRTC_AUDIO_ROUTE_EARPIECE);
}
}
private void enableVideoEncMirror(boolean bMirror) {
trtcCloud.setVideoEncoderMirror(bMirror);
}
private void setLocalViewMirrorMode(int mirrorMode) {
trtcCloud.setLocalViewMirror(mirrorMode);
}
private void enableGSensor(boolean bEnable) {
if (bEnable) {
trtcCloud.setGSensorMode(TRTCCloudDef.TRTC_GSENSOR_MODE_UIFIXLAYOUT);
} else {
trtcCloud.setGSensorMode(TRTCCloudDef.TRTC_GSENSOR_MODE_DISABLE);
}
}
private void enableAudioVolumeEvaluation(boolean bEnable) {
if (bEnable) {
trtcCloud.enableAudioVolumeEvaluation(300);
mVideoViewLayout.showAllAudioVolumeProgressBar();
} else {
trtcCloud.enableAudioVolumeEvaluation(0);
mVideoViewLayout.hideAllAudioVolumeProgressBar();
}
}
private void updateCloudMixtureParams() {
// 背景大畫面寬高
int videoWidth = 720;
int videoHeight = 1280;
// 小畫面寬高
int subWidth = 180;
int subHeight = 320;
int offsetX = 5;
int offsetY = 50;
int bitrate = 200;
int resolution = TRTCCloudDef.TRTC_VIDEO_RESOLUTION_640_360;
switch (resolution) {
case TRTCCloudDef.TRTC_VIDEO_RESOLUTION_160_160: {
videoWidth = 160;
videoHeight = 160;
subWidth = 27;
subHeight = 48;
offsetY = 20;
bitrate = 200;
break;
}
case TRTCCloudDef.TRTC_VIDEO_RESOLUTION_320_180: {
videoWidth = 192;
videoHeight = 336;
subWidth = 54;
subHeight = 96;
offsetY = 30;
bitrate = 400;
break;
}
case TRTCCloudDef.TRTC_VIDEO_RESOLUTION_320_240: {
videoWidth = 240;
videoHeight = 320;
subWidth = 54;
subHeight = 96;
bitrate = 400;
break;
}
case TRTCCloudDef.TRTC_VIDEO_RESOLUTION_480_480: {
videoWidth = 480;
videoHeight = 480;
subWidth = 72;
subHeight = 128;
bitrate = 600;
break;
}
case TRTCCloudDef.TRTC_VIDEO_RESOLUTION_640_360: {
videoWidth = 368;
videoHeight = 640;
subWidth = 90;
subHeight = 160;
bitrate = 800;
break;
}
case TRTCCloudDef.TRTC_VIDEO_RESOLUTION_640_480: {
videoWidth = 480;
videoHeight = 640;
subWidth = 90;
subHeight = 160;
bitrate = 800;
break;
}
case TRTCCloudDef.TRTC_VIDEO_RESOLUTION_960_540: {
videoWidth = 544;
videoHeight = 960;
subWidth = 171;
subHeight = 304;
bitrate = 1000;
break;
}
case TRTCCloudDef.TRTC_VIDEO_RESOLUTION_1280_720: {
videoWidth = 720;
videoHeight = 1280;
subWidth = 180;
subHeight = 320;
bitrate = 1500;
break;
}
default:
break;
}
TRTCCloudDef.TRTCTranscodingConfig config = new TRTCCloudDef.TRTCTranscodingConfig();
config.appId = -1; // 請從"實時音視頻"控制臺的帳號信息中獲取
config.bizId = -1; // 請進入 "實時音視頻"控制臺 https://console.cloud.tencent.com/rav,點擊對應(yīng)的應(yīng)用,然后進入“帳號信息”菜單中,復(fù)制“直播信息”模塊中的"bizid"
config.videoWidth = videoWidth;
config.videoHeight = videoHeight;
config.videoGOP = 1;
config.videoFramerate = 15;
config.videoBitrate = bitrate;
config.audioSampleRate = 48000;
config.audioBitrate = 64;
config.audioChannels = 1;
// 設(shè)置混流后主播的畫面位置
TRTCCloudDef.TRTCMixUser broadCaster = new TRTCCloudDef.TRTCMixUser();
broadCaster.userId = trtcParams.userId; // 以主播uid為broadcaster為例
broadCaster.zOrder = 0;
broadCaster.x = 0;
broadCaster.y = 0;
broadCaster.width = videoWidth;
broadCaster.height = videoHeight;
config.mixUsers = new ArrayList<>();
config.mixUsers.add(broadCaster);
// 設(shè)置混流后各個小畫面的位置
int index = 0;
for (String userId : mRoomMembers) {
TRTCCloudDef.TRTCMixUser audience = new TRTCCloudDef.TRTCMixUser();
audience.userId = userId;
audience.zOrder = 1 + index;
if (index < 3) {
// 前三個小畫面靠右從下往上鋪
audience.x = videoWidth - offsetX - subWidth;
audience.y = videoHeight - offsetY - index * subHeight - subHeight;
audience.width = subWidth;
audience.height = subHeight;
} else if (index < 6) {
// 后三個小畫面靠左從下往上鋪
audience.x = offsetX;
audience.y = videoHeight - offsetY - (index - 3) * subHeight - subHeight;
audience.width = subWidth;
audience.height = subHeight;
} else {
// 最多只疊加六個小畫面
}
config.mixUsers.add(audience);
++index;
}
trtcCloud.setMixTranscodingConfig(config);
}
protected String stringToMd5(String string) {
if (TextUtils.isEmpty(string)) {
return "";
}
MessageDigest md5 = null;
try {
md5 = MessageDigest.getInstance("MD5");
byte[] bytes = md5.digest(string.getBytes());
String result = "";
for (byte b : bytes) {
String temp = Integer.toHexString(b & 0xff);
if (temp.length() == 1) {
temp = "0" + temp;
}
result += temp;
}
return result;
} catch (NoSuchAlgorithmException e) {
e.printStackTrace();
}
return "";
}
private void startLocalVideo(boolean enable) {
TXCloudVideoView localVideoView = mVideoViewLayout.getCloudVideoViewByUseId(trtcParams.userId);
if (localVideoView == null) {
localVideoView = mVideoViewLayout.getFreeCloudVideoView();
}
localVideoView.setUserId(trtcParams.userId);
localVideoView.setVisibility(View.VISIBLE);
if (enable) {
// 設(shè)置 TRTC SDK 的狀態(tài)
trtcCloud.enableCustomVideoCapture(false);
//啟動SDK攝像頭采集和渲染
trtcCloud.startLocalPreview(mCameraFront, localVideoView);
} else {
trtcCloud.stopLocalPreview();
}
}
}
有評論區(qū)小伙伴要求曬出Constents.java,這里我也把這個類分享出來,Constents類主要是定義一些全局變量
Constents完整源碼如下:
public class Constents {
/**
* 1對1語音通話
*/
public final static String ONE_TO_ONE_AUDIO_CALL = "1";
/**
* 1對多語音通話
*/
public final static String ONE_TO_MULTIPE_AUDIO_CALL = "2";
/**
* 1對1視頻通話
*/
public final static String ONE_TO_ONE_VIDEO_CALL = "3";
/**
* 1對多視頻通話
*/
public final static String ONE_TO_MULTIPE_VIDEO_CALL = "4";
/**
* 實時語音通話消息描述內(nèi)容
*/
public final static String AUDIO_CALL_MESSAGE_DESC = "AUDIO_CALL_MESSAGE_DESC";
/**
* 實時視頻通話消息描述內(nèi)容
*/
public final static String VIDEO_CALL_MESSAGE_DESC = "VIDEO_CALL_MESSAGE_DESC";
/**
* 實時語音通話消息拒接
*/
public final static String AUDIO_CALL_MESSAGE_DECLINE_DESC = "AUDIO_CALL_MESSAGE_DECLINE_DESC";
/**
* 實時視頻通話消息拒接
*/
public final static String VIDEO_CALL_MESSAGE_DECLINE_DESC = "VIDEO_CALL_MESSAGE_DECLINE_DESC";
/**
* 懸浮窗與TRTCVideoActivity共享的視頻View
*/
public static TRTCVideoViewLayout mVideoViewLayout;
/**
* 懸浮窗是否開啟
*/
public static boolean isShowFloatWindow = false;
/**
* 語音通話開始計時時間(懸浮窗要顯示時間在這里記錄開始值)
*/
public static long audioCallStartTime;
}
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android開發(fā)Flutter?桌面應(yīng)用窗口化實戰(zhàn)示例
這篇文章主要為大家介紹了Android開發(fā)Flutter?桌面應(yīng)用窗口化實戰(zhàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-09-09
Android編程實現(xiàn)canvas繪制柱狀統(tǒng)計圖功能【自動計算寬高及分度值、可左右滑動】
這篇文章主要介紹了Android編程實現(xiàn)canvas繪制柱狀統(tǒng)計圖功能,具備自動計算寬高及分度值及左右滑動的功能,涉及Android canvas繪圖操作相關(guān)技巧,需要的朋友可以參考下2017-01-01
Android實現(xiàn)志愿者系統(tǒng)詳細步驟與代碼
這篇文章主要介紹了Android實現(xiàn)志愿者系統(tǒng),本系統(tǒng)采用MVC架構(gòu)設(shè)計,SQLite數(shù)據(jù)表有用戶表、成員表和活動表,有十多個Activity頁面。打開應(yīng)用,進入歡迎界面,3s后跳轉(zhuǎn)登錄界面,用戶先注冊賬號,登錄成功后進入主界面2023-02-02

