黑马程序员之手机卫士第五天

电话归属地在手机窗体上悬浮显示
1,悬浮在手机上(悬浮框不停留在手机卫士应用(界面)中,所以需要去开启一个服务,在服务中去管理toast代码逻辑)
1.点击是否开启归属地显示的自定义组合控件SettingItemView
(开启:开启服务,服务中管理吐司的显示
关闭:关闭服务,不需要显示吐司)
2.只有在来电的时候(响铃状态)显示吐司,挂断电话的时候,吐司移除掉

3.电话状态的监听(
服务开启的时候,需要去做监听,
mTM = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
mPhoneStateListener = new MyPhoneStateListener();
mTM.listen(mPhoneStateListener, PhoneStateListener.LISTEN_CALL_STATE);
关闭的时候电话状态就不需要监听)
if(mTM!=null && mPhoneStateListener!=null){
mTM.listen(mPhoneStateListener, PhoneStateListener.LISTEN_NONE);
}
4,是否开启来电归属地显示的判断条件(
服务正在运行==来电归属地开启
服务关闭==来电归属地关闭)


2,toast显示是不依赖于activity,但是现有展示效果需要自定义吐司才可实现
1.看吐司源码

final WindowManager.LayoutParams params = mParams;
params.height = WindowManager.LayoutParams.WRAP_CONTENT;
params.width = WindowManager.LayoutParams.WRAP_CONTENT;
params.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
                //| WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE 默认能够被触摸
| WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON;
params.format = PixelFormat.TRANSLUCENT;
//在响铃的时候显示吐司,和电话类型一致
params.type = WindowManager.LayoutParams.TYPE_PHONE;
params.setTitle("Toast");
//指定左上角
params.gravity = Gravity.LEFT+Gravity.TOP;
3,吐司背景颜色选择(透明,橙色,蓝色,灰色,绿色)
1.在弹出选择样式的对话框后,选择某种样式,并且将其所在数组中的索引值存储sp中
2.选中的样式,回显在SettingActivity
3.吐司在显示的时候,关联相同颜色的图片,作为背景

4,电话号码归属的方格,可以跟随手势去做移动
1.在设置界面添加一个可点击条目,点击此条目弹出activity(半透明)
2.双击居中的view,和描述文字处在不同的竖直(上下)区域
3.限制view的可拖拽范围
4.view双击居中

5,弹出单选框,修改toast展示背景色

6,指定吐司的所在位置,并且记录


7,双击事件
两次点击的时间戳做差<500
8,多击事件
最后一次点击的时间戳 - 第一次点击的时间戳<500

数组的拷贝多次点击事件时间戳存储


## Day06 ##
- 来电监听

创建后台服务 AddressService
 
public void onCreate() {
listener = new MyPhoneListener();
tm = (TelephonyManager) getSystemService(TELEPHONY_SERVICE);
tm.listen(listener, PhoneStateListener.LISTEN_CALL_STATE);
};

@Override
public void onDestroy() {
super.onDestroy();
tm.listen(listener, PhoneStateListener.LISTEN_NONE);
listener = null;
}

class MyPhoneListener extends PhoneStateListener {

@Override
public void onCallStateChanged(int state, String incomingNumber) {
switch (state) {
case TelephonyManager.CALL_STATE_RINGING:
String address = NumberAddressDao.getAddress(incomingNumber);
Toast.makeText(AddressService.this, address, Toast.LENGTH_LONG)
.show();
break;
default:
break;
}
super.onCallStateChanged(state, incomingNumber);
}
}

设置页面新增勾选框,点击后启动或停止service

- 判断服务是否在后台运行,更新checkbox

public static boolean isServiceRunning(String serviceName, Context ctx) {
ActivityManager am = (ActivityManager) ctx
.getSystemService(Context.ACTIVITY_SERVICE);
List runningServices = am.getRunningServices(100);//获取所有后台运行的服务
for (RunningServiceInfo runningServiceInfo : runningServices) {
String className = runningServiceInfo.service.getClassName();
if (className.equals(serviceName)) {
return true;
}
}
return false;
}

- 去电监听

- 静态注册广播

           
               
           
       

注意添加权限:  
问题: 当开关关闭时,仍然能显示去电地址信息

- 动态注册广播

当启动后台服务时,注册广播,服务停止后,注销广播,这样的话,来电和去电的位置显示都可以由一个开关来控制

- 自定义Toast
- Toast原理分析
查找transient_notification文件,查看布局样式, 在values/themes中搜索toastFrameBackground, 查看背景图片toast_frame.9.png

分析Toast源码, 创建自定义Toast

private void showToast(String address) {
view = new TextView(this);
view.setText(address);
view.setTextColor(Color.RED);
final WindowManager.LayoutParams params = new WindowManager.LayoutParams();
params.height = WindowManager.LayoutParams.WRAP_CONTENT;
params.width = WindowManager.LayoutParams.WRAP_CONTENT;
params.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
| WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE
| WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON;
params.format = PixelFormat.TRANSLUCENT;
params.type = WindowManager.LayoutParams.TYPE_TOAST;
params.setTitle("Toast");
wm.addView(view, params);
}

监听电话状态, 如果电话处于空闲状态,就从WindowManager中删除View
case TelephonyManager.CALL_STATE_IDLE:
if (wm != null && view != null) {
wm.removeView(view);
}
break;

- 金山手机卫士

演示金山手机卫士归属地样式, 模仿其样式进行开发. 解压金山手机卫士apk,获取相关资源文件. 注意: 相关图片在drawable目录下, 而非drawable-hdpi

- 自定义Toast样式

1. 布局文件
电话图标: @android:drawable/ic_menu_call

2. 自定义SettingClickView, 类似SettingItemView

去掉自定义属性,保留setDesc和setTitle两个方法

3. 初始化SettingClickView, 设置点击事件,弹出单选Dialog
// 选择归属地样式的弹窗
AlertDialog.Builder builder = new AlertDialog.Builder(
SettingActivity.this);
int style = sp.getInt("address_style", 0);
builder.setSingleChoiceItems(items, style,
new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog,
int which) {
sp.edit().putInt("address_style", which)
.commit();
scvStyle.setDesc(items[which]);
dialog.dismiss();
}
});

builder.setNegativeButton("取消", null);
builder.show();
4. 选择相应样式,保存在sp中
5. 从sp中读取样式,在AddressService中更改背景图片

SharedPreferences sp = getSharedPreferences("config", MODE_PRIVATE);
int style = sp.getInt("address_style", 0);
int[] bgs = new int[] { R.drawable.call_locate_white,
R.drawable.call_locate_orange, R.drawable.call_locate_blue,
R.drawable.call_locate_gray, R.drawable.call_locate_green };
view.setBackgroundResource(bgs[style]);

- 修改归属地显示位置

定义DragViewActivity

1. 布局文件:
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >
   
        android:id="@+id/tv_top"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentTop="true"
        android:layout_centerHorizontal="true"
        android:background="@drawable/call_locate_blue"
        android:gravity="center"
        android:text="按住提示框拖动到任意位置,按手机返回键立刻生效"
        android:textColor="#000"
        android:textSize="20sp" />
   
        android:id="@+id/tv_bottom"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentBottom="true"
        android:layout_centerHorizontal="true"
        android:background="@drawable/call_locate_blue"
        android:gravity="center"
        android:text="按住提示框拖动到任意位置,按手机返回键立刻生效"
        android:textColor="#000"
        android:textSize="20sp"
        android:visibility="invisible" />
   
        android:id="@+id/iv_drag"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="90dp"
        android:src="@drawable/drag" />

2. 拖拽事件监听

ivDrag.setOnTouchListener(new OnTouchListener() {

@Override
public boolean onTouch(View v, MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
//获取起始点坐标
startX = (int) event.getRawX();
startY = (int) event.getRawY();
break;
case MotionEvent.ACTION_MOVE:
int endX = (int) event.getRawX();
int endY = (int) event.getRawY();

int dx = endX - startX;
int dy = endY - startY;

System.out.println("位置偏移:(" + dx + "," + dy + ")");

//根据手指的移动偏移量,计算出图片相应的位置
int left = ivDrag.getLeft() + dx;
int top = ivDrag.getTop() + dy;
int right = ivDrag.getRight() + dx;
int bottom = ivDrag.getBottom() + dy;

//判断图片是否移出屏幕
if (left < 0 || right > windowWidth || top < 0
|| bottom > windowHeight - 20) {
break;
}

//判断图片位于屏幕上半部分还是下半部分
if (top > windowHeight / 2) {
tvBottom.setVisibility(View.INVISIBLE);
tvTop.setVisibility(View.VISIBLE);
} else {
tvBottom.setVisibility(View.VISIBLE);
tvTop.setVisibility(View.INVISIBLE);
}

//重新设定图片的位置
ivDrag.layout(left, top, right, bottom);

//重新获取起始点坐标
startX = (int) event.getRawX();
startY = (int) event.getRawY();
break;
case MotionEvent.ACTION_UP:
//记录拖拽结束后的坐标点
Editor edit = sp.edit();
edit.putInt("lastX", ivDrag.getLeft());
edit.putInt("lastY", ivDrag.getTop());
edit.commit();
break;

default:
break;
}

return true;
}
});

-----------------
获取屏幕宽高

WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
final int windowWidth = wm.getDefaultDisplay().getWidth();
final int windowHeight = wm.getDefaultDisplay().getHeight();

3. 初始化图片位置

LayoutParams params = (LayoutParams) ivDrag.getLayoutParams();
params.leftMargin = lastX;
params.topMargin = lastY;
ivDrag.setLayoutParams(params);
if (lastY > windowHeight / 2) {
tvBottom.setVisibility(View.INVISIBLE);
tvTop.setVisibility(View.VISIBLE);
} else {
tvBottom.setVisibility(View.VISIBLE);
tvTop.setVisibility(View.INVISIBLE);
}

注意:此处不能使用该方法: ivDrag.layout(lastX, lastY, lastX + ivDrag.getWidth(), lastY + ivDrag.getHeight());
因为当前还没有测量好, 所以不能直接调用layout. 顺序是measure,layout,ondraw

- 使用WindowManager设置归属地位置

int lastX = sp.getInt("lastX", 0);
int lastY = sp.getInt("lastY", 0);

params.gravity = Gravity.TOP + Gravity.LEFT; //注意要将重心设置在左上方,默认位于屏幕中央
params.x = lastX;
params.y = lastY;

- 半透明效果处理

1. 清单文件中增加样式, 将Activity设置为全透明
            android:name=".activity.DragViewActivity"
            android:theme="@android:style/Theme.Translucent.NoTitleBar" />

2. 将根布局的背景设置为半透明颜色
- 双击事件

public void onClick(View view) {
if (firstClickTime > 0) {
if (System.currentTimeMillis() - firstClickTime < 500) {
System.out.println("双击");
firstClickTime = 0;
return;
}
}
firstClickTime = System.currentTimeMillis();
}

- 多击事件

设置->关于手机->"Android 版本",多次点击后会跳转页面
查看系统源码Settings, 搜索"Android 版本"字符串,查找相关代码,拷贝到自己的项目中
long[] mHits = new long[3];//数组长度为点击次数

public void onClick(View view) {
// src 源数组
// srcPos 开始拷贝的位置
// dst 目标数组
// dstPos 目标数组的起始拷贝位置
// length 拷贝的数组长度
System.arraycopy(mHits, 1, mHits, 0, mHits.length - 1);//拷贝数组
mHits[mHits.length - 1] = SystemClock.uptimeMillis();
if (mHits[0] >= (SystemClock.uptimeMillis() - 500)) {
System.out.println("是男人!!!");
mHits = new long[3];
}
}

- 双击居中

//图片设置为屏幕居中
ivDrag.layout(windowWidth / 2 - ivDrag.getWidth() / 2,
ivDrag.getTop(),
windowWidth / 2 + ivDrag.getWidth() / 2,
ivDrag.getBottom());
//在sp中记录位置
Editor edit = sp.edit();
edit.putInt("lastX", ivDrag.getLeft());
edit.putInt("lastY", ivDrag.getTop());
edit.commit();

注意: 为了能响应点击事件,需要在onTouch中返回false,将事件传递给onClick
- 窗体触摸移动

1. 为了获取触摸事件,首先需要去掉WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE
2. 其次设置params.type = WindowManager.LayoutParams.TYPE_Phone;
3. 增加权限  
4. 移动逻辑处理

case MotionEvent.ACTION_MOVE:
int dx = (int) (event.getRawX() - startX);
int dy = (int) (event.getRawY() - startY);

params.x += dx;
params.y += dy;

//控制图片不要超出屏幕边界
if (params.x < 0) {
params.x = 0;
}

//控制图片不要超出屏幕边界
if (params.y < 0) {
params.y = 0;
}

//控制图片不要超出屏幕边界
if (params.x > wm.getDefaultDisplay().getWidth()
- view.getWidth()) {
params.x = wm.getDefaultDisplay().getWidth()
- view.getWidth();
}

//控制图片不要超出屏幕边界
if (params.y > wm.getDefaultDisplay().getHeight()
- view.getHeight()) {
params.y = wm.getDefaultDisplay().getHeight()
- view.getHeight();
}

System.out.println("当前位置:" + params.x + ";" + params.y);

wm.updateViewLayout(view, params);//更新图片的显示位置

startX = (int) event.getRawX();
startY = (int) event.getRawY();
break;
黑马程序员之手机卫士第五天

黑马程序员之手机卫士第五天

黑马程序员之手机卫士第五天

黑马程序员之手机卫士第五天

黑马程序员之手机卫士第五天

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值