欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android基于騰訊云實(shí)時(shí)音視頻仿微信視頻通話最小化懸浮

 更新時(shí)間:2019年11月16日 10:48:12   作者:雨打芭蕉  
這篇文章主要為大家詳細(xì)介紹了Android基于騰訊云實(shí)時(shí)音視頻實(shí)現(xiàn)類(lèi)似微信視頻通話最小化懸浮,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下

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

一、實(shí)現(xiàn)效果

二、實(shí)現(xiàn)思路

我把實(shí)現(xiàn)思路拆分為了兩步:1、視頻通話Activity的最小化。 2、視頻通話懸浮框的開(kāi)啟

具體思路是這樣的:當(dāng)用戶點(diǎn)擊左上角最小化按鈕的時(shí)候,最小化視頻通話Activity(這時(shí)Activity處于后臺(tái)狀態(tài)),于此同時(shí)開(kāi)啟懸浮框,新建一個(gè)新的ViewGroup將全局Constents.mVideoViewLayout中用戶選中的最大View動(dòng)態(tài)添加到懸浮框里面去,監(jiān)聽(tīng)?wèi)腋】虻挠|摸事件,讓?xiě)腋】蚩梢酝献б苿?dòng);自定義點(diǎn)擊事件,如果用戶點(diǎn)擊了懸浮框,則移除懸浮框然后重新調(diào)起我們?cè)诤笈_(tái)的視頻通話Activity。

1.Activity是如何實(shí)現(xiàn)最小化的?

Activity本身自帶了一個(gè)moveTaskToBack(boolean nonRoot),我們要實(shí)現(xiàn)最小化只需要調(diào)用moveTaskToBack(true)傳入一個(gè)true值就可以了,但是這里有一個(gè)前提,就是需要設(shè)置Activity的啟動(dòng)模式為singleInstance模式,兩步搞定。(注:activity最小化后重新從后臺(tái)回到前臺(tái)會(huì)回調(diào)onRestart()方法)

@Override
 public boolean moveTaskToBack(boolean nonRoot) {
 return super.moveTaskToBack(nonRoot);
 }

2.懸浮框是如何開(kāi)啟的?

懸浮框的實(shí)現(xiàn)方法最好寫(xiě)在Service里面,將懸浮框的開(kāi)啟關(guān)閉與服務(wù)Service的綁定解綁所關(guān)聯(lián)起來(lái),開(kāi)啟服務(wù)即相當(dāng)于開(kāi)啟我們的懸浮框,解綁服務(wù)則相當(dāng)于關(guān)閉關(guān)閉的懸浮框,以此來(lái)達(dá)到更好的控制效果。

a. 首先我們聲明一個(gè)服務(wù)類(lèi),取名為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. 為懸浮框建立一個(gè)布局文件float_video_window_layout,懸浮框大小我這里固定為長(zhǎng)80dp,高120dp,id為small_size_preview的RelativeLayout主要是一個(gè)容器,可以動(dòng)態(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. 布局定義好后,接下來(lái)就要對(duì)懸浮框做一些初始化操作了,初始化操作這里我們放在服務(wù)的onCreate()生命周期里面執(zhí)行,因?yàn)橹恍枰獔?zhí)行一次就行了。這里的初始化主要包括對(duì):懸浮框的基本參數(shù)(位置,寬高等),懸浮框的點(diǎn)擊事件以及懸浮框的觸摸事件(即可拖動(dòng)范圍)等的設(shè)置,在onBind()中從Intent中取出了Activity中用戶選中最大View的id,以便在后面從 Constents.mVideoViewLayout中取出對(duì)應(yīng)View,然后加入懸浮窗布局中

/**
 * 視頻懸浮窗服務(wù)
 */
public class FloatVideoWindowService extends Service {
 private WindowManager mWindowManager;
 private WindowManager.LayoutParams wmParams;
 private LayoutInflater inflater;
 private String currentBigUserId;
 //浮動(dòng)布局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();//懸浮框點(diǎn)擊事件的處理
 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();
 // 懸浮窗默認(rèn)顯示以左上角為起始坐標(biāo)
 wmParams.gravity = Gravity.LEFT | Gravity.TOP;
 //懸浮窗的開(kāi)始位置,因?yàn)樵O(shè)置的是從左上角開(kāi)始,所以屏幕左上角是x=0;y=0
 wmParams.x = 70;
 wmParams.y = 210;
 //得到容器,通過(guò)這個(gè)inflater來(lái)獲得懸浮窗控件
 inflater = LayoutInflater.from(getApplicationContext());
 // 獲取浮動(dòng)窗口視圖所在布局
 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è)置懸浮窗口長(zhǎng)寬數(shù)據(jù)
 wmParams.width = WindowManager.LayoutParams.WRAP_CONTENT;
 wmParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
 return wmParams;
 }
 
 private void initFloating() {
 
 
 }
}

d. 在懸浮框成功被初始化以及相關(guān)參數(shù)被設(shè)置后,接下來(lái)就需要將Activity中用戶選中最大的View添加到懸浮框里面去了,這樣我們才能看到視頻畫(huà)面嘛,同樣我們是在Service的onCreate這個(gè)生命周期中initFloating()完成這個(gè)操作的,代碼如下所示:

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. 我們上面說(shuō)到要將服務(wù)Service的綁定與解綁與懸浮框的開(kāi)啟和關(guān)閉相結(jié)合,所以既然我們?cè)诜?wù)的onCreate()方法中開(kāi)啟了懸浮框,那么就應(yīng)該在其onDestroy()方法中對(duì)懸浮框進(jìn)行關(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兩種,使用不同的綁定方式其生命周期也會(huì)不一樣,已知我們需要讓?xiě)腋】蛟谝曨l通話activity finish掉的時(shí)候也順便關(guān)掉,那么理所當(dāng)然我們就應(yīng)該采用bind方式來(lái)啟動(dòng)服務(wù),讓他的生命周期跟隨他的開(kāi)啟者,也即是跟隨開(kāi)啟它的activity生命周期。

intent = new Intent(this, FloatVideoWindowService.class);//開(kāi)啟服務(wù)顯示懸浮框
bindService(intent, mVideoServiceConnection, Context.BIND_AUTO_CREATE);
 
ServiceConnection mVideoServiceConnection = new ServiceConnection() {
 
 @Override
 public void onServiceConnected(ComponentName name, IBinder service) {
  // 獲取服務(wù)的操作對(duì)象
  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;
 //浮動(dòng)布局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();//懸浮框點(diǎn)擊事件的處理
 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();
 // 懸浮窗默認(rèn)顯示以左上角為起始坐標(biāo)
 wmParams.gravity = Gravity.LEFT | Gravity.TOP;
 //懸浮窗的開(kāi)始位置,因?yàn)樵O(shè)置的是從左上角開(kāi)始,所以屏幕左上角是x=0;y=0
 wmParams.x = 70;
 wmParams.y = 210;
 //得到容器,通過(guò)這個(gè)inflater來(lái)獲得懸浮窗控件
 inflater = LayoutInflater.from(getApplicationContext());
 // 獲取浮動(dòng)窗口視圖所在布局
 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è)置懸浮窗口長(zhǎng)寬數(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è)置懸浮框可拖動(dòng)
 mTXCloudVideoView.setOnTouchListener(new FloatingListener());
 //懸浮框點(diǎn)擊事件
 mTXCloudVideoView.setOnClickListener(new View.OnClickListener() {
  @Override
  public void onClick(View v) {
  //在這里實(shí)現(xiàn)點(diǎn)擊重新回到Activity
  Intent intent = new Intent(FloatVideoWindowService.this, TRTCVideoCallActivity.class);
  startActivity(intent);
  }
 });
 
 }
 
 //開(kāi)始觸控的坐標(biāo),移動(dòng)時(shí)的坐標(biāo)(相對(duì)于屏幕左上角的坐標(biāo))
 private int mTouchStartX, mTouchStartY, mTouchCurrentX, mTouchCurrentY;
 //開(kāi)始時(shí)的坐標(biāo)和結(jié)束時(shí)的坐標(biāo)(相對(duì)于自身控件的坐標(biāo))
 private int mStartX, mStartY, mStopX, mStopY;
 //判斷懸浮窗口是否移動(dòng),這里做個(gè)標(biāo)記,防止移動(dòng)后松手觸發(fā)了點(diǎn)擊事件
 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;
  }
  //如果是移動(dòng)事件不觸發(fā)OnClick事件,防止移動(dòng)的時(shí)候一放手形成點(diǎn)擊事件
  return isMove;
 }
 }
 
}

Activity中的操作

現(xiàn)在我們將思路了捋一下,假設(shè)現(xiàn)在我正在進(jìn)行視頻通話,點(diǎn)擊視頻最小化按鈕,我們應(yīng)該按順序執(zhí)行如下步驟:應(yīng)該是會(huì)出現(xiàn)個(gè)懸浮框。我們用mServiceBound保存Service注冊(cè)狀態(tài),后面解綁時(shí)候用這個(gè)去判斷,不能有些從其他頁(yè)面過(guò)來(lái)調(diào)用OnRestart()方法的會(huì)報(bào)錯(cuò) 說(shuō) Service not register之類(lèi)的錯(cuò)誤。

/*
 * 開(kāi)啟懸浮Video服務(wù)
 */
private void startVideoService() {
 //最小化Activity
 moveTaskToBack(true);
 Constents.mVideoViewLayout = mVideoViewLayout;
 //開(kāi)啟服務(wù)顯示懸浮框
 Intent floatVideoIntent = new Intent(this, FloatVideoWindowService.class);
 floatVideoIntent.putExtra("userId", currentBigUserId);
 mServiceBound=bindService(floatVideoIntent, mVideoCallServiceConnection, Context.BIND_AUTO_CREATE);
}

注意:這里用了一個(gè)全部變量 Constents.mVideoViewLayout 保存Activity中的mVideoViewLayout,以便在上面的Service中使用。

當(dāng)我們點(diǎn)擊懸浮框的時(shí)候,可以使用startActivity(intent)來(lái)再次打開(kāi)我們的activity,這時(shí)候視頻通話activity會(huì)回調(diào)onRestart()方法,我們?cè)趏nRestart()生命周期里面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;
 //通話計(jì)時(shí)
 private Chronometer callTimeChronometer;
 
 private TRTCCloudDef.TRTCParams trtcParams; /// TRTC SDK 視頻通話房間進(jìn)入所必須的參數(shù)
 private TRTCCloud trtcCloud;  /// TRTC SDK 實(shí)例對(duì)象
 private TRTCCloudListenerImpl trtcListener; /// TRTC SDK 回調(diào)監(jiān)聽(tīng)
 
 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) 開(kāi)啟視頻通話服務(wù)連接
 */
 private ServiceConnection mVideoCallServiceConnection = new ServiceConnection() {
 
 @Override
 public void onServiceConnected(ComponentName name, IBinder service) {
  // 獲取服務(wù)的操作對(duì)象
  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)用運(yùn)行時(shí),保持屏幕高亮,不鎖屏
 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);
 
 //獲取前一個(gè)頁(yè)面得到的進(jìn)房參數(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 實(shí)例
 trtcListener = new TRTCCloudListenerImpl(this);
 trtcCloud = TRTCCloud.sharedInstance(this);
 trtcCloud.setListener(trtcListener);
 
 //開(kāi)始進(jìn)入視頻通話房間
 enterRoom();
 
 /** 倒計(jì)時(shí)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() {
  //倒計(jì)時(shí)全部結(jié)束執(zhí)行操作
  if (!TRTCVideoCallActivity.this.isFinishing() && ConstData.enterRoomUserIdSet.size() == 0) {
   exitRoom();
  }
  }
 };
 countDownTimer.start();
 /**
  * home鍵監(jiān)聽(tīng)相關(guān)
  */
 mHomeWatcher = new HomeWatcher(this);
 mHomeWatcher.setOnHomePressedListener(new HomeWatcher.OnHomePressedListener() {
  @Override
  public void onHomePressed() {
  //按了HOME鍵
  //如果懸浮窗沒(méi)有顯示 就開(kāi)啟服務(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();// 在銷(xiāo)毀時(shí)停止監(jiān)聽(tīng),不然會(huì)報(bào)錯(cuò)的。
 }
 }
 
 /**
 * 重寫(xiě)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();
  /**
   * 單人通話時(shí)
   * 新增主叫方在接收方未接聽(tīng)前掛斷時(shí)
   * 發(fā)送消息給接收方 讓接收方取消響鈴頁(yè)面或者 來(lái)電彈框
   */
  if ( trtcCallType.equals(Constents.ONE_TO_ONE_VIDEO_CALL)) {
   //ConstData.enterRoomUserIdSet.size() == 0表示還沒(méi)有接收方加入房間
   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() {
 // 大畫(huà)面的編碼器參數(shù)設(shè)置
 // 設(shè)置視頻編碼參數(shù),包括分辨率、幀率、碼率等等,這些編碼參數(shù)來(lái)自于 TRTCSettingDialog 的設(shè)置
 // 注意(1):不要在碼率很低的情況下設(shè)置很高的分辨率,會(huì)出現(xiàn)較大的馬賽克
 // 注意(2):不要設(shè)置超過(guò)25FPS以上的幀率,因?yàn)殡娪安攀褂?4FPS,我們一般推薦15FPS,這樣能將更多的碼率分配給畫(huà)質(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ù)覽前配置默認(rèn)參數(shù)
 setTRTCCloudParam();
 // 開(kāi)啟視頻采集預(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
  * 默認(rèn)打開(kāi)是前置攝像頭
  * 前置攝像頭就設(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();
 /**
  * 掛斷/拒接語(yǔ)音、視頻通話消息
  * 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ā)送消息失敗
  //錯(cuò)誤碼 code 和錯(cuò)誤描述 desc,可用于定位請(qǐng)求失敗原因
  //錯(cuò)誤碼 code 含義請(qǐng)參見(jiàn)錯(cuò)誤碼表
  Log.d("NNN", "send message failed. code: " + code + " errmsg: " + desc);
  }
 
  @Override
  public void onSuccess(TIMMessage msg) {//發(fā)送消息成功
  Log.e("NNN", "SendMsg ok");
  }
 });
 }
 
 /**
 * 開(kāi)啟懸浮Video服務(wù)
 */
 private void startVideoService() {
 //最小化Activity
 moveTaskToBack(true);
 Constents.mVideoViewLayout = mVideoViewLayout;
 //開(kāi)啟服務(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);
  }
 }
 }
 
 
 /**
 * 開(kāi)啟/關(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);
 }
 
 /**
 * 開(kāi)啟/關(guān)閉音頻上行
 */
 private void onEnableAudio() {
 bEnableAudio = !bEnableAudio;
 trtcCloud.muteLocalAudio(!bEnableAudio);
 ivVoice.setImageResource(bEnableAudio ? R.mipmap.mic_enable : R.mipmap.mic_disable);
 }
 
 /**
 * 點(diǎn)擊切換攝像頭
 */
 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();
  }
 }
 
 /**
  * 離開(kāi)房間
  */
 @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ù)的錯(cuò)誤,需要通過(guò) 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, "進(jìn)房超時(shí),請(qǐng)檢查網(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, "進(jìn)房參數(shù)錯(cuò)誤:" + 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, "進(jìn)房失敗,請(qǐng)稍后重試:" + 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, "進(jìn)房失敗,房間滿了,請(qǐng)稍后重試:" + errCode + "[" + errMsg + "]", Toast.LENGTH_SHORT).show();
  activity.exitRoom();
  return;
  }
 
  if (errCode == TXLiteAVCode.ERR_SERVER_CENTER_ROOM_ID_TOO_LONG) {
  Toast.makeText(activity, "進(jìn)房失敗,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, "進(jìn)房失敗,請(qǐng)確認(rèn)房間號(hào)正確:" + errCode + "[" + errMsg + "]", Toast.LENGTH_SHORT).show();
  activity.exitRoom();
  return;
  }
 
  if (errCode == TXLiteAVCode.ERR_SERVER_INFO_SERVICE_SUSPENDED) {
  Toast.makeText(activity, "進(jìn)房失敗,請(qǐng)確認(rèn)騰訊云實(shí)時(shí)音視頻賬號(hào)狀態(tài)是否欠費(fè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, "進(jìn)房失敗,無(wú)權(quán)限進(jì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) {
  // 錯(cuò)誤參考 https://cloud.tencent.com/document/product/269/1671#.E5.B8.90.E5.8F.B7.E7.B3.BB.E7.BB.9F
  Toast.makeText(activity, "進(jìn)房失敗,userSig錯(cuò)誤:" + errCode + "[" + errMsg + "]", Toast.LENGTH_SHORT).show();
  activity.exitRoom();
  return;
  }
  Toast.makeText(activity, "onError: " + errMsg + "[" + errCode + "]", Toast.LENGTH_SHORT).show();
 }
 
 /**
  * WARNING 大多是一些可以忽略的事件通知,SDK內(nèi)部會(huì)啟動(dòng)一定的補(bǔ)救機(jī)制
  */
 @Override
 public void onWarning(int warningCode, String warningMsg, Bundle extraInfo) {
  Log.d(TAG, "sdk callback onWarning");
 }
 
 /**
  * 有新的用戶加入了當(dāng)前視頻房間
  */
 @Override
 public void onUserEnter(String userId) {
  TRTCVideoCallActivity activity = mContext.get();
  ConstData.enterRoomUserIdSet.add(userId);
  if (activity != null) {
  // 創(chuàng)建一個(gè)View用來(lái)顯示新的一路畫(huà)面
//  TXCloudVideoView renderView = activity.mVideoViewLayout.onMemberEnter(userId + TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_BIG);
  TXCloudVideoView renderView = activity.mVideoViewLayout.onMemberEnter(userId);
  if (renderView != null) {
   // 設(shè)置儀表盤(pán)數(shù)據(jù)顯示
   renderView.setVisibility(View.VISIBLE);
  }
  }
 }
 
 /**
  * 有用戶離開(kāi)了當(dāng)前視頻房間
  */
 @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();
   }
  }
  //停止觀看畫(huà)面
  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);
  }
  }
 }
 
 /**
  * 有用戶屏蔽了畫(huà)面
  */
 @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) {
   // 啟動(dòng)遠(yuǎn)程畫(huà)面的解碼和顯示邏輯,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) {
   // 啟動(dòng)遠(yuǎn)程畫(huà)面的解碼和顯示邏輯,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);
 }
 }
 
 /**
 * 視頻里點(diǎn)擊進(jìn)入和某人聊天
 *
 * @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() {
 // 背景大畫(huà)面寬高
 int videoWidth = 720;
 int videoHeight = 1280;
 
 // 小畫(huà)面寬高
 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; // 請(qǐng)從"實(shí)時(shí)音視頻"控制臺(tái)的帳號(hào)信息中獲取
 config.bizId = -1; // 請(qǐng)進(jìn)入 "實(shí)時(shí)音視頻"控制臺(tái) https://console.cloud.tencent.com/rav,點(diǎn)擊對(duì)應(yīng)的應(yīng)用,然后進(jìn)入“帳號(hào)信息”菜單中,復(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è)置混流后主播的畫(huà)面位置
 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è)置混流后各個(gè)小畫(huà)面的位置
 int index = 0;
 for (String userId : mRoomMembers) {
  TRTCCloudDef.TRTCMixUser audience = new TRTCCloudDef.TRTCMixUser();
  audience.userId = userId;
  audience.zOrder = 1 + index;
  if (index < 3) {
  // 前三個(gè)小畫(huà)面靠右從下往上鋪
  audience.x = videoWidth - offsetX - subWidth;
  audience.y = videoHeight - offsetY - index * subHeight - subHeight;
  audience.width = subWidth;
  audience.height = subHeight;
  } else if (index < 6) {
  // 后三個(gè)小畫(huà)面靠左從下往上鋪
  audience.x = offsetX;
  audience.y = videoHeight - offsetY - (index - 3) * subHeight - subHeight;
  audience.width = subWidth;
  audience.height = subHeight;
  } else {
  // 最多只疊加六個(gè)小畫(huà)面
  }
 
  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);
  //啟動(dòng)SDK攝像頭采集和渲染
  trtcCloud.startLocalPreview(mCameraFront, localVideoView);
 } else {
  trtcCloud.stopLocalPreview();
 }
 }
}

有評(píng)論區(qū)小伙伴要求曬出Constents.java,這里我也把這個(gè)類(lèi)分享出來(lái),Constents類(lèi)主要是定義一些全局變量

Constents完整源碼如下:

public class Constents {
 
 /**
 * 1對(duì)1語(yǔ)音通話
 */
 public final static String ONE_TO_ONE_AUDIO_CALL = "1";
 /**
 * 1對(duì)多語(yǔ)音通話
 */
 public final static String ONE_TO_MULTIPE_AUDIO_CALL = "2";
 /**
 * 1對(duì)1視頻通話
 */
 public final static String ONE_TO_ONE_VIDEO_CALL = "3";
 
 /**
 * 1對(duì)多視頻通話
 */
 public final static String ONE_TO_MULTIPE_VIDEO_CALL = "4";
 
 /**
 * 實(shí)時(shí)語(yǔ)音通話消息描述內(nèi)容
 */
 public final static String AUDIO_CALL_MESSAGE_DESC = "AUDIO_CALL_MESSAGE_DESC";
 /**
 * 實(shí)時(shí)視頻通話消息描述內(nèi)容
 */
 public final static String VIDEO_CALL_MESSAGE_DESC = "VIDEO_CALL_MESSAGE_DESC";
 
 /**
 * 實(shí)時(shí)語(yǔ)音通話消息拒接
 */
 public final static String AUDIO_CALL_MESSAGE_DECLINE_DESC = "AUDIO_CALL_MESSAGE_DECLINE_DESC";
 /**
 * 實(shí)時(shí)視頻通話消息拒接
 */
 public final static String VIDEO_CALL_MESSAGE_DECLINE_DESC = "VIDEO_CALL_MESSAGE_DECLINE_DESC";
 
 /**
 * 懸浮窗與TRTCVideoActivity共享的視頻View
 */
 public static TRTCVideoViewLayout mVideoViewLayout;
 
 /**
 * 懸浮窗是否開(kāi)啟
 */
 public static boolean isShowFloatWindow = false;
 
 /**
 * 語(yǔ)音通話開(kāi)始計(jì)時(shí)時(shí)間(懸浮窗要顯示時(shí)間在這里記錄開(kāi)始值)
 */
 public static long audioCallStartTime;

}

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Android Fragment 基本了解(圖文介紹)

    Android Fragment 基本了解(圖文介紹)

    Android是在Android 3.0 (API level 11)開(kāi)始引入Fragment的可以把Fragment想成Activity中的模塊,這個(gè)模塊有自己的布局,有自己的生命周期,單獨(dú)處理自己的輸入,在Activity運(yùn)行的時(shí)候可以加載或者移除Fragment模塊
    2013-01-01
  • RecyclerView的萬(wàn)能分割線

    RecyclerView的萬(wàn)能分割線

    RecyclerView 是Android L版本中新添加的一個(gè)用來(lái)取代ListView的SDK,它的靈活性與可替代性比listview更好。本文重點(diǎn)給大家介紹RecyclerView的萬(wàn)能分割線的知識(shí),非常不錯(cuò),感興趣的朋友一起看下吧
    2016-07-07
  • Android開(kāi)發(fā)Flutter?桌面應(yīng)用窗口化實(shí)戰(zhàn)示例

    Android開(kāi)發(fā)Flutter?桌面應(yīng)用窗口化實(shí)戰(zhàn)示例

    這篇文章主要為大家介紹了Android開(kāi)發(fā)Flutter?桌面應(yīng)用窗口化實(shí)戰(zhàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-09-09
  • Android編程實(shí)現(xiàn)canvas繪制柱狀統(tǒng)計(jì)圖功能【自動(dòng)計(jì)算寬高及分度值、可左右滑動(dòng)】

    Android編程實(shí)現(xiàn)canvas繪制柱狀統(tǒng)計(jì)圖功能【自動(dòng)計(jì)算寬高及分度值、可左右滑動(dòng)】

    這篇文章主要介紹了Android編程實(shí)現(xiàn)canvas繪制柱狀統(tǒng)計(jì)圖功能,具備自動(dòng)計(jì)算寬高及分度值及左右滑動(dòng)的功能,涉及Android canvas繪圖操作相關(guān)技巧,需要的朋友可以參考下
    2017-01-01
  • 詳解Android中Drawable方法

    詳解Android中Drawable方法

    這篇文章主要為大家詳細(xì)介紹了Android中Drawable方法,感興趣的朋友可以參考一下
    2016-05-05
  • Android簡(jiǎn)單自定義音樂(lè)波動(dòng)特效圖

    Android簡(jiǎn)單自定義音樂(lè)波動(dòng)特效圖

    這篇文章主要為大家詳細(xì)介紹了Android簡(jiǎn)單自定義音樂(lè)波動(dòng)特效圖,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-04-04
  • Android文件操作工具類(lèi)詳解

    Android文件操作工具類(lèi)詳解

    這篇文章主要為大家詳細(xì)介紹了Android文件操作工具類(lèi),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-07-07
  • Android自定義View彈性滑動(dòng)Scroller詳解

    Android自定義View彈性滑動(dòng)Scroller詳解

    這篇文章主要為大家詳細(xì)介紹了Android自定義View彈性滑動(dòng)Scroller,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-12-12
  • android實(shí)現(xiàn)簡(jiǎn)易計(jì)算器

    android實(shí)現(xiàn)簡(jiǎn)易計(jì)算器

    這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)簡(jiǎn)易計(jì)算器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-07-07
  • Android實(shí)現(xiàn)志愿者系統(tǒng)詳細(xì)步驟與代碼

    Android實(shí)現(xiàn)志愿者系統(tǒng)詳細(xì)步驟與代碼

    這篇文章主要介紹了Android實(shí)現(xiàn)志愿者系統(tǒng),本系統(tǒng)采用MVC架構(gòu)設(shè)計(jì),SQLite數(shù)據(jù)表有用戶表、成員表和活動(dòng)表,有十多個(gè)Activity頁(yè)面。打開(kāi)應(yīng)用,進(jìn)入歡迎界面,3s后跳轉(zhuǎn)登錄界面,用戶先注冊(cè)賬號(hào),登錄成功后進(jìn)入主界面
    2023-02-02

最新評(píng)論