Android开发模仿qq视频通话悬浮按钮(实例代码)

模仿qq视频通话的悬浮按钮的实例代码,如下所示;

public class FloatingWindowService extends Service{
  private static final String TAG="OnTouchListener";
  private static View mView = null;
  private static WindowManager mWindowManager = null;
  private static Context mContext = null;
  public static Boolean isShown = false;
  public WindowManager.LayoutParams params = null;
  private int pixel;
  private int TheOffset;
  @Override
  public void onCreate() {
    super.onCreate();
  }
  @Override
  public int onStartCommand(Intent intent, int flags, int startId) {
    pixel = intent.getIntExtra("pixel",1);
    showPopupWindow(this);
    return super.onStartCommand(intent, flags, startId);
  }
  /**
   * 显示弹出框
   *
   * @param context
   *
   */
  private void showPopupWindow(final Context context) {
    if (isShown) {
      return;
    }
    isShown = true;
    // 获取应用的Context
    mContext = context.getApplicationContext();
    // 获取WindowManager
    mWindowManager = (WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE);
    params = new WindowManager.LayoutParams();
    mView = setUpView(context);
    // 类型
    params.type = WindowManager.LayoutParams.TYPE_SYSTEM_ALERT;
    int flags=WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
    params.flags = flags;
    params.format = PixelFormat.TRANSLUCENT;
    params.width = WindowManager.LayoutParams.WRAP_CONTENT;
    params.height = WindowManager.LayoutParams.WRAP_CONTENT;
    params.gravity = Gravity.CENTER;
    mWindowManager.addView(mView, params);
  }
  /**
   * 隐藏弹出框
   */
  private static void hidePopupWindow() {
    if (isShown && null != mView) {
      mWindowManager.removeView(mView);
      isShown = false;
    }
  }
  private  int x=0;
  private  int y=0;
  private  int startX=0;
  private  int startY=0;
  private View setUpView(final Context context) {
    View view = LayoutInflater.from(context).inflate(R.layout.popupwindow,
        null);
     TextView tv= (TextView) view.findViewById(R.id.title);
    int w = View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED);
    int h = View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED);
    tv.measure(w, h);
    TheOffset=(pixel-tv.getMeasuredWidth())/2-50;
    tv.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View v) {
        Intent intent =new Intent(context,MainActivity.class);
        context.startActivity(intent);
        // Toast.makeText(context,"点击事件",Toast.LENGTH_LONG).show();
      }
    });
    tv.setOnTouchListener(new View.OnTouchListener() {
      @Override
      public boolean onTouch(View v, MotionEvent event) {
        switch (event.getAction()){
          case MotionEvent.ACTION_MOVE:
            int newX= (int) (event.getRawX()-x);
            int newY= (int) (event.getRawY()-y);
            params.x=newX+startX;
            params.y=newY+startY;
              mWindowManager.updateViewLayout(mView,params);
            break;
          case MotionEvent.ACTION_DOWN:
            x= (int) event.getRawX();
            y= (int) event.getRawY();
            break;
          case MotionEvent.ACTION_UP:
            if(params.x>=0){
              params.x=TheOffset;
              mWindowManager.updateViewLayout(mView,params);
            }
            if(params.x<=-0){
              params.x=-TheOffset;
              mWindowManager.updateViewLayout(mView,params);
            }
            Log.i(TAG,params.x+"");
            Log.i(TAG,params.y+"");
            //判断 从按住到抬起时候的移动距离, 如果如果移动距离大于20 那么就拦截事件,否则就不拦截事件,主要是处理点击事件的冲突
            if(Math.abs(startX-params.x)>20 ||Math.abs(startY-params.y)>20 ){
              //记录上一次的偏移量
              startX=params.x;
              startY=params.y;
              return true;
            }else {
              startX=params.x;
              startY=params.y;
              return false;
            }
        }
        return false;
      }
    });
    return view;
  }
  @Nullable
  @Override
  public IBinder onBind(Intent intent) {
    return null;
  }
  @Override
  public void onDestroy() {
    super.onDestroy();
    if (mView != null) {
      isShown=false;
      mWindowManager.removeView(mView);
    }
  }
  }

Main  

@Override
    protected void onCreate(Bundle savedInstanceState) {
      super.onCreate(savedInstanceState);
      setContentView(R.layout.activity_main);
      findViewById(R.id.open).setOnClickListener(this);
      findViewById(R.id.close).setOnClickListener(this);
    }

-点击开启 关闭悬浮按钮 

@Override
    public void onClick(View v) {
      switch (v.getId()){
        case R.id.open:
          //判断是否拥有悬浮权限
          //op 的值是 0 ~ 47,其中0代表粗略定位权限,1代表精确定位权限,24代表悬浮窗权限。(具体可以看看Android源码在android.app下就有个AppOpsManager类)
          if(utils.checkOp(this,24)==0) {
            Intent intent=new Intent(MainActivity.this, FloatingWindowService.class);
            intent.putExtra("pixel",utils.pixel(this)[0]);
            startService(intent);
          }else {
            //引导用户进入悬浮权限设置界面
            Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,
                Uri.parse("package:" + getPackageName()));
            startActivityForResult(intent, 200);
          }
          break;
        case R.id.close:
          stopService(new Intent(MainActivity.this,FloatingWindowService.class));
          break;
      }
    }

判断权限 -获取屏幕的宽高 

public class utils {
      public static int checkOp(Context context, int op){
        final int version = Build.VERSION.SDK_INT;
        if (version >= 19){
          Object object = context.getSystemService("appops");
          Class c = object.getClass();
          try {
            Class[] cArg = new Class[3];
            cArg[0] = int.class;
            cArg[1] = int.class;
            cArg[2] = String.class;
            Method lMethod = c.getDeclaredMethod("checkOp", cArg);
            return (Integer) lMethod.invoke(object, op, Binder.getCallingUid(), context.getPackageName());
          } catch(NoSuchMethodException e) {
            e.printStackTrace();
          } catch (IllegalAccessException e) {
            e.printStackTrace();
          } catch (IllegalArgumentException e) {
            e.printStackTrace();
          } catch (InvocationTargetException e) {
            e.printStackTrace();
          }
        }
        return -1;
      }
      /**
       * 获取屏幕的宽高
       * @param context
       * @return
       */
      public static int[] pixel(Activity context){
        DisplayMetrics dm = new DisplayMetrics();
        context.getWindowManager().getDefaultDisplay().getMetrics(dm);
        return new int[]{dm.widthPixels,dm.heightPixels};
      }
    }

--popupwindow填充布局文件 

 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
         android:layout_width="match_parent"
         android:layout_height="match_parent"
         android:orientation="vertical">
    <LinearLayout
      android:id="@+id/popup_window"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:background="@android:color/white"
      android:orientation="vertical" >
      <TextView
        android:background="@mipmap/ic_launcher"
        android:id="@+id/title"
        android:layout_width="50dp"
        android:layout_height="50dp"/>
      </LinearLayout>
  </LinearLayout>

以上所述是小编给大家介绍的Android开发模仿qq视频通话悬浮按钮(实例代码),希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

(0)

相关推荐

  • Android开发实现删除联系人通话记录的方法

    本文实例讲述了Android开发实现删除联系人通话记录的方法.分享给大家供大家参考,具体如下: 1. 负责存放呼叫记录的内容提供者源码在 ContactsProvider 项目下: 源码路径: com/Android/providers/contacts/CallLogProvider.Java 使用到的数据库在: /data/data/com.android.providers.contacts/databases/contacts2.db 表名:calls 呼叫记录有三种类型: 来电:Cal

  • Android获取手机通话记录的方法

    Android如何获取手机通话记录,本文为大家揭晓. 获取手机通话记录流程: 1. 获取ContentResolver; ContentResolver resolver = getContentResolver(); 2.resolver.query(*); 需要传入通话记录的URI:CallLog.Calls.CONTENT_URI 3.对查询得到的Cursor进行数据获取. 主要代码如下: MainActivity.java package com.noonecode.contentres

  • Android开发四大组件之实现电话拦截和电话录音

    一.问题描述 使用BordercastReceiver和Service组件实现下述功能: 1.当手机处于来电状态,启动监听服务,对来电进行监听录音. 2.设置电话黑名单,当来电是黑名单电话,则直接挂断. 当拨打电话或电话状态发生改变时,系统就会发出有序广播,因此我们可以使用BordercastReceiver接受广播,因BordercastReceiver执行时间短不能执行耗时任务也不能使用子线程,因此我们应启动一个Service来监听电话并进行处理 二.加入AIDL文件 Android没有对外

  • Android录音应用实例教程

    本文以实例形式较为详细的展示了Android录音的实现方法,分享给大家供大家参考之用.具体方法如下: 首先是xml布局文件: <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="fill_parent" andr

  • android 添加按(power键)电源键结束通话(挂断电话)

    首先我们发现现在我们所用的android智能手机大部分都有当你在打电话时按power键来挂断电话,一般都是在设置中. 我主要是在原生源码中添加这一功能,主要用于学习....先看一张图:  看到那个按电源键挂断电话吧,那就是我所添加的,本来原生源码中是没有这一栏的..... 大概思路: 首先我先添加这一个checkboxPreference,然后将是否选择这一功能的值(0和1)存到data/data/com.android.providers.settings /databases/setting

  • Android应用开发:电话监听和录音代码示例

    在oncreate 中执行: 复制代码 代码如下: public void onCreate() {  super.onCreate();  Log.i("TAG", "服务启动了"); // 对电话的来电状态进行监听  TelephonyManager telManager = (TelephonyManager) this    .getSystemService(Context.TELEPHONY_SERVICE);  // 注册一个监听器对电话状态进行监听 

  • Android获取通话时间实例分析

    本文章总结了一段Android获取通话时间程序代码,有需要的朋友可参考一下. 我们知道安卓系统中通话时长应该是归Callog管,所以建议去查查ContactProvider,或者是TelephonyProvider Service测试 可以的通话开始的时候启动Service 记录当前时间A, 然后stopSelf(); 另外在通话结束的时候再次启动一下Service,再次获得当前时间B, 然后把时间A和B进行比较处理 String time = Long.toString(比较后处理的时间) 然

  • Android编程实现通话录音功能的方法

    本文实例讲述了Android编程实现通话录音功能的方法.分享给大家供大家参考,具体如下: 因受系统限制,只能录自已麦的声音,录不到对方的声音,可能需要改内核才能实现双向录音: 接通电话和挂断电话时,震动一下: 使用广播接收者实现自启动: 服务代码: package com.eboy.phoneListener; import java.io.File; import android.app.Service; import android.content.Context; import andro

  • Android通话记录备份实现代码

    (一) 前言 Android默认提供了联系人备份到sd卡的功能(代码在com.android.vcard包里面),我们可以把联系人导出成.vcf文件存在sd卡中:如果换手机了,我们又可以把联系人从sd卡文件中导入进来.那么,通话记录我们也能不能做出类似的功能呢?答案是肯定的! (二) 导出通话记录 既然是备份通话记录,那就肯定包括导出和导入的功能,这里我们先讲导出通话记录. 1. 根据通话记录导出的规范,导出的文件一般以.vcl后缀结尾,中间的内容是 复制代码 代码如下: BEGIN:VCALL

  • Android简单的利用MediaRecorder进行录音的实例代码

    复制代码 代码如下: package com.ppmeet; import java.io.IOException; import android.app.Activity;  import android.graphics.PixelFormat;  import android.media.MediaRecorder;  import android.os.Bundle;  import android.view.View;  import android.view.View.OnClick

  • Android6.0编程实现双向通话自动录音功能的方法详解

    本文实例讲述了Android6.0编程实现双向通话自动录音功能的方法.分享给大家供大家参考,具体如下: 项目中需要实现基于Android 6.0 的双向通话自动录音功能,在查阅相关android电话状态监听文章以及Git上的开源录音项目后,整理出此文 首先,介绍一下android 电话状态的监听(来电和去电): http://www.jb51.net/article/32433.htm 实现手机电话状态的监听,主要依靠两个类: TelephoneManger和PhoneStateListener

  • Android 实现电话来去自动录音的功能

    我们在使用Android手机打电话时,有时可能会需要对来去电通话自动录音,本文就详细讲解实现Android来去电通话自动录音的方法,大家按照文中的方法编写程序就可以完成此功能. 来去电自动录音的关键在于如何监听手机电话状态的转变: 1)来电的状态的转换如下(红色标记是我们要用到的状态) 空闲(IDEL)--> 响铃(RINGING)--> 接听(ACTIVE)--> 挂断(经历DISCONNECTING--DISCONNECTED)--> 空闲(IDEL) 或者  空闲(IDEL)

随机推荐