黑马程序员之手机卫士第五天
生活随笔
收集整理的這篇文章主要介紹了
黑马程序员之手机卫士第五天
小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.
電話歸屬地在手機(jī)窗體上懸浮顯示 1,懸浮在手機(jī)上(懸浮框不停留在手機(jī)衛(wèi)士應(yīng)用(界面)中,所以需要去開啟一個服務(wù),在服務(wù)中去管理toast代碼邏輯) 1.點擊是否開啟歸屬地顯示的自定義組合控件SettingItemView (開啟:開啟服務(wù),服務(wù)中管理吐司的顯示 關(guān)閉:關(guān)閉服務(wù),不需要顯示吐司) 2.只有在來電的時候(響鈴狀態(tài))顯示吐司,掛斷電話的時候,吐司移除掉
3.電話狀態(tài)的監(jiān)聽( 服務(wù)開啟的時候,需要去做監(jiān)聽, mTM = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE); mPhoneStateListener = new MyPhoneStateListener(); mTM.listen(mPhoneStateListener, PhoneStateListener.LISTEN_CALL_STATE); 關(guān)閉的時候電話狀態(tài)就不需要監(jiān)聽) if(mTM!=null && mPhoneStateListener!=null){ mTM.listen(mPhoneStateListener, PhoneStateListener.LISTEN_NONE); } 4,是否開啟來電歸屬地顯示的判斷條件( 服務(wù)正在運(yùn)行==來電歸屬地開啟 服務(wù)關(guān)閉==來電歸屬地關(guān)閉)
2,toast顯示是不依賴于activity,但是現(xiàn)有展示效果需要自定義吐司才可實現(xiàn) 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 默認(rèn)能夠被觸摸 | 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,吐司背景顏色選擇(透明,橙色,藍(lán)色,灰色,綠色) 1.在彈出選擇樣式的對話框后,選擇某種樣式,并且將其所在數(shù)組中的索引值存儲sp中 2.選中的樣式,回顯在SettingActivity 3.吐司在顯示的時候,關(guān)聯(lián)相同顏色的圖片,作為背景
4,電話號碼歸屬的方格,可以跟隨手勢去做移動 1.在設(shè)置界面添加一個可點擊條目,點擊此條目彈出activity(半透明) 2.雙擊居中的view,和描述文字處在不同的豎直(上下)區(qū)域 3.限制view的可拖拽范圍 4.view雙擊居中
5,彈出單選框,修改toast展示背景色
6,指定吐司的所在位置,并且記錄
7,雙擊事件 兩次點擊的時間戳做差<500 8,多擊事件 最后一次點擊的時間戳 - 第一次點擊的時間戳<500
數(shù)組的拷貝多次點擊事件時間戳存儲
## Day06 ## - 來電監(jiān)聽
創(chuàng)建后臺服務(wù) 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); } }
設(shè)置頁面新增勾選框,點擊后啟動或停止service
- 判斷服務(wù)是否在后臺運(yùn)行,更新checkbox
public static boolean isServiceRunning(String serviceName, Context ctx) { ActivityManager am = (ActivityManager) ctx .getSystemService(Context.ACTIVITY_SERVICE); List runningServices = am.getRunningServices(100);//獲取所有后臺運(yùn)行的服務(wù) for (RunningServiceInfo runningServiceInfo : runningServices) { String className = runningServiceInfo.service.getClassName(); if (className.equals(serviceName)) { return true; } } return false; }
- 去電監(jiān)聽
- 靜態(tài)注冊廣播
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?
注意添加權(quán)限: ? 問題: 當(dāng)開關(guān)關(guān)閉時,仍然能顯示去電地址信息
- 動態(tài)注冊廣播
當(dāng)啟動后臺服務(wù)時,注冊廣播,服務(wù)停止后,注銷廣播,這樣的話,來電和去電的位置顯示都可以由一個開關(guān)來控制
- 自定義Toast - Toast原理分析 查找transient_notification文件,查看布局樣式, 在values/themes中搜索toastFrameBackground, 查看背景圖片toast_frame.9.png
分析Toast源碼, 創(chuàng)建自定義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); }
監(jiān)聽電話狀態(tài), 如果電話處于空閑狀態(tài),就從WindowManager中刪除View case TelephonyManager.CALL_STATE_IDLE: if (wm != null && view != null) { wm.removeView(view); } break;
- 金山手機(jī)衛(wèi)士
演示金山手機(jī)衛(wèi)士歸屬地樣式, 模仿其樣式進(jìn)行開發(fā). 解壓金山手機(jī)衛(wèi)士apk,獲取相關(guān)資源文件. 注意: 相關(guān)圖片在drawable目錄下, 而非drawable-hdpi
- 自定義Toast樣式
1. 布局文件 電話圖標(biāo): @android:drawable/ic_menu_call
2. 自定義SettingClickView, 類似SettingItemView
去掉自定義屬性,保留setDesc和setTitle兩個方法
3. 初始化SettingClickView, 設(shè)置點擊事件,彈出單選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. 選擇相應(yīng)樣式,保存在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="按住提示框拖動到任意位置,按手機(jī)返回鍵立刻生效" ? ? ?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="按住提示框拖動到任意位置,按手機(jī)返回鍵立刻生效" ? ? ?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. 拖拽事件監(jiān)聽
ivDrag.setOnTouchListener(new OnTouchListener() {
@Override public boolean onTouch(View v, MotionEvent event) { switch (event.getAction()) { case MotionEvent.ACTION_DOWN: //獲取起始點坐標(biāo) 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 + ")");
//根據(jù)手指的移動偏移量,計算出圖片相應(yīng)的位置 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); }
//重新設(shè)定圖片的位置 ivDrag.layout(left, top, right, bottom);
//重新獲取起始點坐標(biāo) startX = (int) event.getRawX(); startY = (int) event.getRawY(); break; case MotionEvent.ACTION_UP: //記錄拖拽結(jié)束后的坐標(biāo)點 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()); 因為當(dāng)前還沒有測量好, 所以不能直接調(diào)用layout. 順序是measure,layout,ondraw
- 使用WindowManager設(shè)置歸屬地位置
int lastX = sp.getInt("lastX", 0); int lastY = sp.getInt("lastY", 0);
params.gravity = Gravity.TOP + Gravity.LEFT; //注意要將重心設(shè)置在左上方,默認(rèn)位于屏幕中央 params.x = lastX; params.y = lastY;
- 半透明效果處理
1. 清單文件中增加樣式, 將Activity設(shè)置為全透明 ? ? ? ? ? android:name=".activity.DragViewActivity" ? ? ? ? ? android:theme="@android:style/Theme.Translucent.NoTitleBar" />
2. 將根布局的背景設(shè)置為半透明顏色 - 雙擊事件
public void onClick(View view) { if (firstClickTime > 0) { if (System.currentTimeMillis() - firstClickTime < 500) { System.out.println("雙擊"); firstClickTime = 0; return; } } firstClickTime = System.currentTimeMillis(); }
- 多擊事件
設(shè)置->關(guān)于手機(jī)->"Android 版本",多次點擊后會跳轉(zhuǎn)頁面 查看系統(tǒng)源碼Settings, 搜索"Android 版本"字符串,查找相關(guān)代碼,拷貝到自己的項目中 long[] mHits = new long[3];//數(shù)組長度為點擊次數(shù)
public void onClick(View view) { // src 源數(shù)組 // srcPos 開始拷貝的位置 // dst 目標(biāo)數(shù)組 // dstPos 目標(biāo)數(shù)組的起始拷貝位置 // length 拷貝的數(shù)組長度 System.arraycopy(mHits, 1, mHits, 0, mHits.length - 1);//拷貝數(shù)組 mHits[mHits.length - 1] = SystemClock.uptimeMillis(); if (mHits[0] >= (SystemClock.uptimeMillis() - 500)) { System.out.println("是男人!!!"); mHits = new long[3]; } }
- 雙擊居中
//圖片設(shè)置為屏幕居中 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();
注意: 為了能響應(yīng)點擊事件,需要在onTouch中返回false,將事件傳遞給onClick - 窗體觸摸移動
1. 為了獲取觸摸事件,首先需要去掉WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE 2. 其次設(shè)置params.type = WindowManager.LayoutParams.TYPE_Phone; 3. 增加權(quán)限 ? 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("當(dāng)前位置:" + params.x + ";" + params.y);
wm.updateViewLayout(view, params);//更新圖片的顯示位置
startX = (int) event.getRawX(); startY = (int) event.getRawY(); break;
3.電話狀態(tài)的監(jiān)聽( 服務(wù)開啟的時候,需要去做監(jiān)聽, mTM = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE); mPhoneStateListener = new MyPhoneStateListener(); mTM.listen(mPhoneStateListener, PhoneStateListener.LISTEN_CALL_STATE); 關(guān)閉的時候電話狀態(tài)就不需要監(jiān)聽) if(mTM!=null && mPhoneStateListener!=null){ mTM.listen(mPhoneStateListener, PhoneStateListener.LISTEN_NONE); } 4,是否開啟來電歸屬地顯示的判斷條件( 服務(wù)正在運(yùn)行==來電歸屬地開啟 服務(wù)關(guān)閉==來電歸屬地關(guān)閉)
2,toast顯示是不依賴于activity,但是現(xiàn)有展示效果需要自定義吐司才可實現(xiàn) 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 默認(rèn)能夠被觸摸 | 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,吐司背景顏色選擇(透明,橙色,藍(lán)色,灰色,綠色) 1.在彈出選擇樣式的對話框后,選擇某種樣式,并且將其所在數(shù)組中的索引值存儲sp中 2.選中的樣式,回顯在SettingActivity 3.吐司在顯示的時候,關(guān)聯(lián)相同顏色的圖片,作為背景
4,電話號碼歸屬的方格,可以跟隨手勢去做移動 1.在設(shè)置界面添加一個可點擊條目,點擊此條目彈出activity(半透明) 2.雙擊居中的view,和描述文字處在不同的豎直(上下)區(qū)域 3.限制view的可拖拽范圍 4.view雙擊居中
5,彈出單選框,修改toast展示背景色
6,指定吐司的所在位置,并且記錄
7,雙擊事件 兩次點擊的時間戳做差<500 8,多擊事件 最后一次點擊的時間戳 - 第一次點擊的時間戳<500
數(shù)組的拷貝多次點擊事件時間戳存儲
## Day06 ## - 來電監(jiān)聽
創(chuàng)建后臺服務(wù) 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); } }
設(shè)置頁面新增勾選框,點擊后啟動或停止service
- 判斷服務(wù)是否在后臺運(yùn)行,更新checkbox
public static boolean isServiceRunning(String serviceName, Context ctx) { ActivityManager am = (ActivityManager) ctx .getSystemService(Context.ACTIVITY_SERVICE); List runningServices = am.getRunningServices(100);//獲取所有后臺運(yùn)行的服務(wù) for (RunningServiceInfo runningServiceInfo : runningServices) { String className = runningServiceInfo.service.getClassName(); if (className.equals(serviceName)) { return true; } } return false; }
- 去電監(jiān)聽
- 靜態(tài)注冊廣播
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?
注意添加權(quán)限: ? 問題: 當(dāng)開關(guān)關(guān)閉時,仍然能顯示去電地址信息
- 動態(tài)注冊廣播
當(dāng)啟動后臺服務(wù)時,注冊廣播,服務(wù)停止后,注銷廣播,這樣的話,來電和去電的位置顯示都可以由一個開關(guān)來控制
- 自定義Toast - Toast原理分析 查找transient_notification文件,查看布局樣式, 在values/themes中搜索toastFrameBackground, 查看背景圖片toast_frame.9.png
分析Toast源碼, 創(chuàng)建自定義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); }
監(jiān)聽電話狀態(tài), 如果電話處于空閑狀態(tài),就從WindowManager中刪除View case TelephonyManager.CALL_STATE_IDLE: if (wm != null && view != null) { wm.removeView(view); } break;
- 金山手機(jī)衛(wèi)士
演示金山手機(jī)衛(wèi)士歸屬地樣式, 模仿其樣式進(jìn)行開發(fā). 解壓金山手機(jī)衛(wèi)士apk,獲取相關(guān)資源文件. 注意: 相關(guān)圖片在drawable目錄下, 而非drawable-hdpi
- 自定義Toast樣式
1. 布局文件 電話圖標(biāo): @android:drawable/ic_menu_call
2. 自定義SettingClickView, 類似SettingItemView
去掉自定義屬性,保留setDesc和setTitle兩個方法
3. 初始化SettingClickView, 設(shè)置點擊事件,彈出單選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. 選擇相應(yīng)樣式,保存在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="按住提示框拖動到任意位置,按手機(jī)返回鍵立刻生效" ? ? ?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="按住提示框拖動到任意位置,按手機(jī)返回鍵立刻生效" ? ? ?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. 拖拽事件監(jiān)聽
ivDrag.setOnTouchListener(new OnTouchListener() {
@Override public boolean onTouch(View v, MotionEvent event) { switch (event.getAction()) { case MotionEvent.ACTION_DOWN: //獲取起始點坐標(biāo) 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 + ")");
//根據(jù)手指的移動偏移量,計算出圖片相應(yīng)的位置 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); }
//重新設(shè)定圖片的位置 ivDrag.layout(left, top, right, bottom);
//重新獲取起始點坐標(biāo) startX = (int) event.getRawX(); startY = (int) event.getRawY(); break; case MotionEvent.ACTION_UP: //記錄拖拽結(jié)束后的坐標(biāo)點 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()); 因為當(dāng)前還沒有測量好, 所以不能直接調(diào)用layout. 順序是measure,layout,ondraw
- 使用WindowManager設(shè)置歸屬地位置
int lastX = sp.getInt("lastX", 0); int lastY = sp.getInt("lastY", 0);
params.gravity = Gravity.TOP + Gravity.LEFT; //注意要將重心設(shè)置在左上方,默認(rèn)位于屏幕中央 params.x = lastX; params.y = lastY;
- 半透明效果處理
1. 清單文件中增加樣式, 將Activity設(shè)置為全透明 ? ? ? ? ? android:name=".activity.DragViewActivity" ? ? ? ? ? android:theme="@android:style/Theme.Translucent.NoTitleBar" />
2. 將根布局的背景設(shè)置為半透明顏色 - 雙擊事件
public void onClick(View view) { if (firstClickTime > 0) { if (System.currentTimeMillis() - firstClickTime < 500) { System.out.println("雙擊"); firstClickTime = 0; return; } } firstClickTime = System.currentTimeMillis(); }
- 多擊事件
設(shè)置->關(guān)于手機(jī)->"Android 版本",多次點擊后會跳轉(zhuǎn)頁面 查看系統(tǒng)源碼Settings, 搜索"Android 版本"字符串,查找相關(guān)代碼,拷貝到自己的項目中 long[] mHits = new long[3];//數(shù)組長度為點擊次數(shù)
public void onClick(View view) { // src 源數(shù)組 // srcPos 開始拷貝的位置 // dst 目標(biāo)數(shù)組 // dstPos 目標(biāo)數(shù)組的起始拷貝位置 // length 拷貝的數(shù)組長度 System.arraycopy(mHits, 1, mHits, 0, mHits.length - 1);//拷貝數(shù)組 mHits[mHits.length - 1] = SystemClock.uptimeMillis(); if (mHits[0] >= (SystemClock.uptimeMillis() - 500)) { System.out.println("是男人!!!"); mHits = new long[3]; } }
- 雙擊居中
//圖片設(shè)置為屏幕居中 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();
注意: 為了能響應(yīng)點擊事件,需要在onTouch中返回false,將事件傳遞給onClick - 窗體觸摸移動
1. 為了獲取觸摸事件,首先需要去掉WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE 2. 其次設(shè)置params.type = WindowManager.LayoutParams.TYPE_Phone; 3. 增加權(quán)限 ? 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("當(dāng)前位置:" + params.x + ";" + params.y);
wm.updateViewLayout(view, params);//更新圖片的顯示位置
startX = (int) event.getRawX(); startY = (int) event.getRawY(); break;
總結(jié)
以上是生活随笔為你收集整理的黑马程序员之手机卫士第五天的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Worok:专门针对亚洲实体的网络间谍组
- 下一篇: ASP.NET-----Repeater