Android實現(xiàn)類似3D Touch菜單功能
前言
在開發(fā)中,我們經(jīng)常遇到需要菜單功能的實現(xiàn),我們經(jīng)常會參考其他人的優(yōu)秀設計。比如3D Touch菜單,作為iphone6和iphone6s上引人注目的新功能。現(xiàn)在,我們希望盡力來模仿這種菜單設計,盡力,因為系統(tǒng)的差異,會導致很多東西實現(xiàn)起來有難度。
思路
想要盡力模仿這種菜單,經(jīng)過分析,我覺得主要實現(xiàn)以下幾個點:
1)菜單的出現(xiàn)方式,在ios上,方式是用戶用手指用力按下,然而在Android上,受限于硬件,我們無法捕捉用力按壓這種動作,所以,我改用另一種比較次的方式,長按彈出,捕捉手指長按動作。
2)菜單的界面上,需要處理背景模糊效果。
3)菜單的觸摸事件處理,我們看到,手指長按之后,菜單出現(xiàn),這時候手指不離開屏幕,滑動到菜單某個選項,再抬起,這時候這個選項會相應。
實現(xiàn)
背景模糊處理
經(jīng)過一番調(diào)研,除了調(diào)用github上面大神的各種繪圖效果庫,我們想要自己實現(xiàn)大概有兩個思路。
RenderScript方案
RenderScript是由Android3.0引入,用來在Android上編寫高性能代碼的一種語言。優(yōu)點:使用方便,Android官方API自帶,而且性能處理效果極好,缺點:需要API17以上。
使用非常簡單,我們只需要獲取RenderScript的實例,傳入模糊圖像需要的參數(shù)
@TargetApi(Build.VERSION_CODES.JELLY_BEAN_MR1) public Bitmap getRenderScriptBitmap(Context context, int radius, Bitmap bitmapOriginal) { RenderScript rs = RenderScript.create(context); final Allocation input = Allocation.createFromBitmap(rs, bitmapOriginal); final Allocation output = Allocation.createTyped(rs, input.getType()); final ScriptIntrinsicBlur script = ScriptIntrinsicBlur.create(rs, Element.U8_4(rs)); script.setRadius(radius); script.setInput(input); script.forEach(output); output.copyTo(bitmapOriginal); rs.destroy(); return bitmapOriginal; }
Java代碼層實現(xiàn)方案
通過java層代碼也可以實現(xiàn)圖像的模糊處理,github大神已經(jīng)為我們實現(xiàn)了這種圖像算法。
通過FastBlur算法實現(xiàn)圖片模糊,沒有版本兼容問題,但是如果我們需要模糊的圖像不小的時候,我們會發(fā)現(xiàn)模糊圖像需要的時間遠遠超過了我們能夠接受的范圍,如果加載大圖的話,那情況就更加糟糕了。一個比較好的處理方式是,在圖片進行模糊處理之前,先對圖像進行壓縮,在圖片模糊處理完畢之后,再按照原大小放大,這樣就能有效降低模糊處理的耗時。
這里我們做一個版本判斷
if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) { mIBlurry = BlurryFactory.createRenderScript(); } else { mIBlurry = BlurryFactory.createFastBlur(); }
觸摸事件的處理
先來說說模糊層如何出現(xiàn),肯定是要實現(xiàn)一個全屏效果,關(guān)于全屏效果,我們可以通過Dialog,懸浮窗,透明的Activity,或者在DectorView中插入覆蓋父布局的視圖,這四種方式都可以實現(xiàn)全屏效果,這里,我們選用在DectorView中插入視圖的方式來實現(xiàn)。
如何實現(xiàn)呢?
/** * 掛載到某個Activity的最頂層 * @param activity */ private void attachActivity(Activity activity) { ViewParent parent = getParent(); if(parent != null && parent instanceof ViewGroup) { ViewGroup parentView = (ViewGroup) parent; parentView.removeView(this); } FrameLayout decor = (FrameLayout)activity.getWindow().getDecorView(); FrameLayout.LayoutParams lp = new FrameLayout.LayoutParams( FrameLayout.LayoutParams.MATCH_PARENT, FrameLayout.LayoutParams.MATCH_PARENT); decor.addView(this, lp); }
前面提到,手指長按,在菜單彈出之后,我們在手指不抬起的情況下,可以選擇菜單選項。這樣聽起來可能有些難以理解,我們都看過很多大神寫的View的觸摸事件解析,也有自己去研讀源碼,都對觸摸事件傳遞有一定的了解。
一般情況下,當我們長按界面里的某個View,之后在我們手指抬起之前,所有的觸摸事件都會交由這個View來處理,也就是targetView(當某個View消費了觸摸事件,那么它就會被設置為targetView,并且接收接下來傳遞的觸摸事件)。那么我們?nèi)绾卧谑种覆惶鸬那闆r下,讓剛剛出現(xiàn)模糊層視圖來接管接下來的手指滑動,也就是ACTION_MOVE和ACTION_UP事件呢?
經(jīng)常一番思考,我想到了一種比較委婉的方式…
我們都知道手指抬起的時候,DecorView以及其子視圖都會接收到一個ACTION_UP的觸摸事件,這個事件會告訴DecorView,這個手指觸摸系列動作已經(jīng)結(jié)束,那么方法來了,我們并不需要手指真的抬起來,只需要模擬一個手指抬起的動作,也就是自己發(fā)送一個ACTION_UP事件,就可以讓DecorView接收到ACTION_UP事件,然后重新尋找targetView,也就是新出現(xiàn)的模糊層視圖,這時候我們再模擬發(fā)送一個ACTION_DOWN事件。
/** * 轉(zhuǎn)移觸摸事件 */ private void transferTouchEvent(final Activity activity) { postDelayed(new Runnable() { @Override public void run() { activity.dispatchTouchEvent(MotionEvent.obtain(SystemClock.uptimeMillis(), SystemClock.uptimeMillis(), MotionEvent.ACTION_UP, 0f, 0f, 0)); activity.getWindow().getDecorView().dispatchTouchEvent(MotionEvent.obtain(SystemClock .uptimeMillis(), SystemClock.uptimeMillis(), MotionEvent.ACTION_DOWN, 0f, 0f, 0)); } }, 200); }
效果圖
最后的效果圖如下,由于時間比較少,有很多地方?jīng)]有好好完善,菜單的出現(xiàn)角度以及菜單的樣式不夠美觀,同時沒有判斷菜單出現(xiàn)在ICON上面還是下面。
github項目地址:點擊打開地址。
效果思路仍不夠完善,待更好的方案。
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
- 安卓(Android)實現(xiàn)3DTouch效果
- Android左右滑出菜單實例分析
- android底部菜單欄實現(xiàn)原理與代碼
- android popwindow實現(xiàn)左側(cè)彈出菜單層及PopupWindow主要方法介紹
- 基于Android實現(xiàn)點擊某個按鈕讓菜單選項從按鈕周圍指定位置彈出
- Android ListView長按彈出菜單二種實現(xiàn)方式示例
- Android界面設計(APP設計趨勢 左側(cè)隱藏菜單右邊顯示content)
- Android開發(fā)技巧之我的菜單我做主(自定義菜單)
- Android仿QQ空間底部菜單示例代碼
- Android實現(xiàn)原生側(cè)滑菜單的超簡單方式
相關(guān)文章
Android中實現(xiàn)用命令行同步網(wǎng)絡時間
這篇文章主要介紹了Android中實現(xiàn)用命令行同步網(wǎng)絡時間,本文講解使用BusyBox實現(xiàn)同步網(wǎng)絡時間,并給出了詳細操作步驟,需要的朋友可以參考下2015-07-07android中AutoCompleteTextView的簡單用法(實現(xiàn)搜索歷史)
本篇文章主要介紹了android中AutoCompleteTextView的簡單用法(自動提示),有需要的可以了解一下。2016-11-11Android中Rxjava實現(xiàn)三級緩存的兩種方式
這篇文章主要介紹了Android中Rxjava實現(xiàn)三級緩存的兩種方式,具有一定的參考價值,感興趣的小伙伴們可以參考一下。2017-04-04Android UI設計系列之自定義SwitchButton開關(guān)實現(xiàn)類似IOS中UISwitch的動畫效果(2
這篇文章主要介紹了Android UI設計系列之自定義SwitchButton開關(guān)實現(xiàn)類似IOS中UISwitch的動畫效果,具有一定的實用性和參考價值,感興趣的小伙伴們可以參考一下2016-06-06Android?Framework原理Binder驅(qū)動源碼解析
這篇文章主要為大家介紹了Android?Framework原理Binder驅(qū)動源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-01-01Android開發(fā)之BottomSheetDialog組件的使用
BottomSheetDialog是底部操作控件,可在屏幕底部創(chuàng)建一個支持滑動關(guān)閉視圖。本文將通過示例詳細講解它的使用,感興趣的小伙伴可以了解一下2023-01-01