热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Android中怎么实现微信视屏悬浮窗效果

今天就跟大家聊聊有关Android中怎么实现微信视屏悬浮窗效果,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根

今天就跟大家聊聊有关Android中怎么实现微信视屏悬浮窗效果,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。

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

2.实现悬浮窗需要:

在androidManifest中申请悬浮窗权限

在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",mTouchCurrentX,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中怎么实现微信视屏悬浮窗效果有进一步的了解吗?如果还想了解更多知识或者相关内容,请关注编程笔记行业资讯频道,感谢大家的支持。


推荐阅读
  • 本文介绍了pack布局管理器在Perl/Tk中的使用方法及注意事项。通过调用pack()方法,可以控制部件在显示窗口中的位置和大小。同时,本文还提到了在使用pack布局管理器时,应注意将部件分组以便在水平和垂直方向上进行堆放。此外,还介绍了使用Frame部件或Toplevel部件来组织部件在窗口内的方法。最后,本文强调了在使用pack布局管理器时,应避免在中间切换到grid布局管理器,以免造成混乱。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • Android开发实现的计时器功能示例
    本文分享了Android开发实现的计时器功能示例,包括效果图、布局和按钮的使用。通过使用Chronometer控件,可以实现计时器功能。该示例适用于Android平台,供开发者参考。 ... [详细]
  • Go GUIlxn/walk 学习3.菜单栏和工具栏的具体实现
    本文介绍了使用Go语言的GUI库lxn/walk实现菜单栏和工具栏的具体方法,包括消息窗口的产生、文件放置动作响应和提示框的应用。部分代码来自上一篇博客和lxn/walk官方示例。文章提供了学习GUI开发的实际案例和代码示例。 ... [详细]
  • 本文介绍了一款名为TimeSelector的Android日期时间选择器,采用了Material Design风格,可以在Android Studio中通过gradle添加依赖来使用,也可以在Eclipse中下载源码使用。文章详细介绍了TimeSelector的构造方法和参数说明,以及如何使用回调函数来处理选取时间后的操作。同时还提供了示例代码和可选的起始时间和结束时间设置。 ... [详细]
  • 本文详细介绍了Android中的坐标系以及与View相关的方法。首先介绍了Android坐标系和视图坐标系的概念,并通过图示进行了解释。接着提到了View的大小可以超过手机屏幕,并且只有在手机屏幕内才能看到。最后,作者表示将在后续文章中继续探讨与View相关的内容。 ... [详细]
  • 带添加按钮的GridView,item的删除事件
    先上图片效果;gridView无数据时显示添加按钮,有数据时,第一格显示添加按钮,后面显示数据:布局文件:addr_manage.xml<?xmlve ... [详细]
  • 拥抱Android Design Support Library新变化(导航视图、悬浮ActionBar)
    转载请注明明桑AndroidAndroid5.0Loollipop作为Android最重要的版本之一,为我们带来了全新的界面风格和设计语言。看起来很受欢迎࿰ ... [详细]
  • 深入理解CSS中的margin属性及其应用场景
    本文主要介绍了CSS中的margin属性及其应用场景,包括垂直外边距合并、padding的使用时机、行内替换元素与费替换元素的区别、margin的基线、盒子的物理大小、显示大小、逻辑大小等知识点。通过深入理解这些概念,读者可以更好地掌握margin的用法和原理。同时,文中提供了一些相关的文档和规范供读者参考。 ... [详细]
  • 在Xamarin XAML语言中如何在页面级别构建ControlTemplate控件模板
    本文介绍了在Xamarin XAML语言中如何在页面级别构建ControlTemplate控件模板的方法和步骤,包括将ResourceDictionary添加到页面中以及在ResourceDictionary中实现模板的构建。通过本文的阅读,读者可以了解到在Xamarin XAML语言中构建控件模板的具体操作步骤和语法形式。 ... [详细]
  • 本文介绍了在MFC下利用C++和MFC的特性动态创建窗口的方法,包括继承现有的MFC类并加以改造、插入工具栏和状态栏对象的声明等。同时还提到了窗口销毁的处理方法。本文详细介绍了实现方法并给出了相关注意事项。 ... [详细]
  • 如何在HTML中获取鼠标的当前位置
    本文介绍了在HTML中获取鼠标当前位置的三种方法,分别是相对于屏幕的位置、相对于窗口的位置以及考虑了页面滚动因素的位置。通过这些方法可以准确获取鼠标的坐标信息。 ... [详细]
  • JavaScript和HTML之间的交互是经由过程事宜完成的。事宜:文档或浏览器窗口中发作的一些特定的交互霎时。能够运用侦听器(或处置惩罚递次来预订事宜),以便事宜发作时实行相应的 ... [详细]
  • 用Vue实现的Demo商品管理效果图及实现代码
    本文介绍了一个使用Vue实现的Demo商品管理的效果图及实现代码。 ... [详细]
author-avatar
鱼鱼de眼泪2012
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有