Android仿微信视屏悬浮窗效果

Android仿微信视屏悬浮窗效果,第1张

概述Android仿微信视屏悬浮效果 在项目中需要对接入的腾讯云音视频,可以悬浮窗显示,悬浮窗可拖拽,并且在悬浮窗不影响其他的activity的焦点. 这个大神的文章Android基于腾讯云实时音视频仿微信视频通话最小化悬浮,他讲的是视频通话时,将远端视频以悬浮窗形式展示,根据他的代码我进行了部分简化 1.悬浮窗效果:点击缩小按钮,将当前远端视屏加载进悬浮窗,且悬浮窗可拖拽,不影响其他界面焦点:点击悬浮窗可返回原来的Activity 2.实现悬浮窗需要: 在androidManifest中申请悬浮窗权限<uses-permis ...

在项目中需要对接入的腾讯云音视频,可以悬浮窗显示,悬浮窗可拖拽,并且在悬浮窗不影响其他的activity的焦点。

这个大神的文章AndroID基于腾讯云实时音视频仿微信视频通话最小化悬浮,他讲的是视频通话时,将远端视频以悬浮窗形式展示,根据他的代码我进行了部分简化

1.悬浮窗效果:点击缩小按钮,将当前远端视屏加载进悬浮窗,且悬浮窗可拖拽,不影响其他界面焦点;点击悬浮窗可返回原来的Activity

2.实现悬浮窗需要:

在androIDManifest中申请悬浮窗权限<uses-permission androID:name="androID.permission.SYstem_ALERT_WINDOW"/>

在androIDManifest中注册floatwindowservice

3.视屏activity实现:

-将activity置于后台关键代码:moveTaskToBack(true);//将activity置于后台
-开启悬浮窗

/**   * 定义服务绑定的回调 开启视频通话服务连接   */  private ServiceConnection mVIDeoCallServiceConnection = new ServiceConnection() {    @OverrIDe    public voID onServiceConnected(Componentname name,IBinder service) {      // 获取服务的 *** 作对象      floatwindowservice.mybinder binder = (floatwindowservice.mybinder) service;      binder.getService();    }    @OverrIDe    public voID onServicedisconnected(Componentname name) {    }  };/*   * 开启悬浮VIDeo服务   */  private voID startVIDeoService() {    //最小化Activity    moveTaskToBack(true);//将activity置于后台    //开启服务显示悬浮框    Intent serviceVIDeoIntent = new Intent(this,floatwindowservice.class);    mServiceBound = bindService(serviceVIDeoIntent,mVIDeoCallServiceConnection,Context.BIND_auto_CREATE);//绑定Service  }

-悬浮窗结束时

//在onDestroy()与onReStart()中解绑并销毁相关内容if (mServiceBound) {      unbindService(mVIDeoCallServiceConnection);//解绑      mServiceBound = false;    }

4.悬浮窗实现相关代码: 

/** * 视频悬浮窗服务 */public class floatwindowservice extends Service implements VIEw.OntouchListener {  private WindowManager mWindowManager;  private WindowManager.LayoutParams wmParams;  private LayoutInflater inflater;  //浮动布局vIEw  private VIEw mfloatingLayout;  //容器父布局  private VIEw mMainVIEw;  //开始触控的坐标,移动时的坐标(相对于屏幕左上角的坐标)  private int mtouchstartX,mtouchstartY,mtouchCurrentX,mtouchCurrentY;  //开始时的坐标和结束时的坐标(相对于自身控件的坐标)  private int mStartX,mStartY,mStopX,mStopY;  //判断悬浮窗口是否移动,这里做个标记,防止移动后松手触发了点击事件  private boolean isMove;  @OverrIDe  public voID onCreate() {    super.onCreate();    initwindow();//设置悬浮窗基本参数(位置、宽高等)  }  @Nullable  @OverrIDe  public IBinder onBind(Intent intent) {    currentBigUserID = intent.getStringExtra("localUserID");    remoteUserID = intent.getStringExtra("remoteUserID");    initfloating();//悬浮框点击事件的处理    return new mybinder();  }  public class mybinder extends Binder {    public floatwindowservice getService() {      return floatwindowservice.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;    }  }  /**   * 设置悬浮框基本参数(位置、宽高等)   */  private voID initwindow() {    mWindowManager = (WindowManager) getApplicationContext().getSystemService(Context.WINDOW_SERVICE);    //设置好悬浮窗的参数    wmParams = getParams();    // 悬浮窗默认显示以左上角为起始坐标    wmParams.gravity = Gravity.RIGHT | Gravity.top;    //悬浮窗的开始位置,因为设置的是从右上角开始,所以屏幕左上角是x=屏幕最大值;y=0    wmParams.x = 10;    wmParams.y = 120;    //得到容器,通过这个inflater来获得悬浮窗控件    inflater = LayoutInflater.from(getApplicationContext());    // 获取浮动窗口视图所在布局    mfloatingLayout = inflater.inflate(R.layout.dlg_floatvIEw,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;    }    //设置可以显示在状态栏上    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;    //设置悬浮窗口长宽数据    wmParams.wIDth = WindowManager.LayoutParams.WRAP_CONTENT;    wmParams.height = WindowManager.LayoutParams.WRAP_CONTENT;    return wmParams;  }  //加载远端视屏:在这对悬浮窗内内容做 *** 作  private voID initfloating() {    //将子VIEw加载进悬浮窗VIEw    mMainVIEw = mfloatingLayout.findVIEwByID(R.ID.trtc_vIDeo_vIEw_layout_float);//悬浮窗父布局    VIEw mChildVIEw = renderVIEw.getChildVIEw();//加载进悬浮窗的子VIEw,这个VIEw来自天转过来的那个Activity里面的那个需要加载的VIEw    mMainVIEw.addVIEw(mChildVIEw);//将需要悬浮显示的VIEwadd到mTXCloudVIDeoVIEw中    //悬浮框触摸事件,设置悬浮框可拖动    mTXCloudVIDeoVIEw.setontouchListener(this::ontouch);    //悬浮框点击事件    mTXCloudVIDeoVIEw.setonClickListener(new VIEw.OnClickListener() {      @OverrIDe      public voID onClick(VIEw v) {        //在这里实现点击重新回到Activity        Intent intent =        new Intent(floatwindowservice.this,RtcActivity.class);//从该service跳转至该activity会将该activity从后台唤醒,所以activity会走onReStart()        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);//从Service跳转至RTCActivity,需要Intent.FLAG_ACTIVITY_NEW_TASK,不然会崩溃        startActivity(intent);      }    });  }  //触摸事件  @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 += mtouchstartX - mtouchCurrentX;        wmParams.y += mtouchCurrentY - mtouchstartY;        ALog.dTag("floatingListener() ontouch",mtouchstartX,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;    }    //如果是移动事件不触发OnClick事件,防止移动的时候一放手形成点击事件    return isMove;  }}

ps:使用Service做悬浮窗的载体是为了,将悬浮框的开启关闭与服务Service的绑定解绑所关联起来,开启服务即相当于开启我们的悬浮框,解绑服务则相当于关闭悬浮框,以此来达到更好的控制效果。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

总结

以上是内存溢出为你收集整理的Android仿微信视屏悬浮窗效果全部内容,希望文章能够帮你解决Android仿微信视屏悬浮窗效果所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/web/1144299.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-31
下一篇 2022-05-31

发表评论

登录后才能评论

评论列表(0条)

保存