Android之淘寶商品列表長(zhǎng)按遮罩效果的實(shí)現(xiàn)
先來(lái)看看淘寶、唯品會(huì)長(zhǎng)按商品的效果,以及簡(jiǎn)單Demo的效果:
首先分析一下場(chǎng)景:
長(zhǎng)按條目時(shí),彈出遮罩的效果遮擋在原來(lái)的條目布局上; 頁(yè)面滑動(dòng)或點(diǎn)擊其他的條目,上一個(gè)正在遮罩的條目遮罩消失。 長(zhǎng)按其他條目時(shí),上一個(gè)遮罩的條目撤銷遮罩,當(dāng)前長(zhǎng)按的顯示遮罩; 條目添加遮罩的時(shí)添加動(dòng)畫;
1. 遮罩的效果,我們會(huì)很容易的想到Android布局控件FrameLayout布局,是基于疊加在上方的布局。所以在列表?xiàng)l目布局的時(shí)候,可以使用FrameLayout布局,在長(zhǎng)按列表?xiàng)l目時(shí),用條目的根布局添加一個(gè)遮罩的布局,就達(dá)到我們想要的效果了。
2. 記錄當(dāng)前長(zhǎng)按的根布局,如果點(diǎn)擊或長(zhǎng)按其他的列表?xiàng)l目,亦或滑動(dòng)頁(yè)面(添加活動(dòng)監(jiān)聽(tīng))時(shí),就取消之前長(zhǎng)按的條目遮罩,從條目根布局中刪除遮罩布局就OK了;
3. 可以利用View動(dòng)畫或?qū)傩詣?dòng)畫,在添加遮罩布局時(shí)顯示動(dòng)畫;
接下來(lái), 來(lái)擼一下代碼吧:
1. 首先,先定義一下遮罩的布局,根據(jù)需求自定義View
/*** * 長(zhǎng)按條目遮罩界面 */ public class ItemMaskLayout extends LinearLayout { public ItemMaskLayout(Context context) { this(context, null); } public ItemMaskLayout(Context context, @Nullable AttributeSet attrs) { this(context, attrs, 0); } public ItemMaskLayout(Context context, @Nullable AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); LayoutInflater.from(context).inflate(R.layout.layout_product_list_item_mask, this, true); findViewById(R.id.tv_find_same).setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { if (mItemMaskClickListener != null) { mItemMaskClickListener.findTheSame(); } } }); findViewById(R.id.tv_collection).setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { if (mItemMaskClickListener != null) { mItemMaskClickListener.collection(); } } }); } public ItemMaskClickListener mItemMaskClickListener; public void setMaskItemClickListener(ItemMaskClickListener listener) { this.mItemMaskClickListener = listener; } //提供遮罩中按鈕點(diǎn)擊操作接口 自定義 public interface ItemMaskClickListener { void findTheSame(); void collection(); } }
2. 封裝一個(gè)幫助類,主要是根據(jù)該類的成員變量根據(jù)長(zhǎng)按的條目指向列表Item的布局,然后為條目添加遮罩的效果;
/** * 長(zhǎng)按條目添加遮罩操作幫助類 */ public class ItemLongClickMaskHelper { private FrameLayout mRootFrameLayout; private ItemMaskLayout mMaskItemLayout; private Context mContext; private ScaleAnimation anim; private String productId; public ItemLongClickMaskHelper(Context context){ this.mContext = context; mMaskItemLayout = new ItemMaskLayout(mContext); anim = new ScaleAnimation( 0f, 1.0f, 1.0f, 1.0f, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f ); anim.setDuration(300); mMaskItemLayout.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { dismissItemMaskLayout(); } }); mMaskItemLayout.setOnLongClickListener(new View.OnLongClickListener() { @Override public boolean onLongClick(View v) { dismissItemMaskLayout(); return true; } }); mMaskItemLayout.setMaskItemClickListener(new ItemMaskLayout.ItemMaskClickListener() { @Override public void findTheSame() { ToastUtil.showCustomToast("找相似 " + productId); } @Override public void collection() { ToastUtil.showCustomToast("收藏 " + productId); } }); } public ItemLongClickMaskHelper setRootFrameLayout(FrameLayout frameLayout, String fundId){ if (mRootFrameLayout != null){ mRootFrameLayout.removeView(mMaskItemLayout); } mRootFrameLayout = frameLayout; this.productId = fundId; mRootFrameLayout.addView(mMaskItemLayout); mMaskItemLayout.startAnimation(anim); return this; } public ItemLongClickMaskHelper setMaskItemListener(ItemMaskLayout.ItemMaskClickListener listener){ this.mMaskItemLayout.setMaskItemClickListener(listener); return this; } /** * 遮罩消失 */ public void dismissItemMaskLayout(){ if (mRootFrameLayout != null){ mRootFrameLayout.removeView(mMaskItemLayout); } } }
3.注意在滑動(dòng)RecyclerView列表的時(shí)候,監(jiān)聽(tīng)滑動(dòng),撤銷遮罩,直接定義RecyclerView的子類,添加滑動(dòng)監(jiān)聽(tīng)回調(diào);
public class TouchCallbackRecyclerView extends RecyclerView { public TouchCallbackRecyclerView(Context context) { super(context); } public TouchCallbackRecyclerView(Context context, @Nullable AttributeSet attrs) { super(context, attrs); } public TouchCallbackRecyclerView(Context context, @Nullable AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); } public interface ScrollCallback { /** * 滑動(dòng)手指抬起事件 * * @param diffY 抬起時(shí)相對(duì)于按下時(shí)的偏移量<br/>大于0:列表往下拉, 小于0: 列表往上拉 */ void onTouchUp(float diffY); } private ScrollCallback mScrollCallback; public void setScrollCallback(ScrollCallback callback) { this.mScrollCallback = callback; } private float mDownY, mMovingY, mUpY; private boolean isUp = false; @SuppressWarnings("deprecation") private static final float SLOP = ViewConfiguration.getTouchSlop(); @Override public boolean dispatchTouchEvent(MotionEvent ev) { switch (ev.getAction()) { case MotionEvent.ACTION_DOWN: mDownY = ev.getY(); isUp = false; break; case MotionEvent.ACTION_MOVE: mMovingY = ev.getY(); isUp = false; break; case MotionEvent.ACTION_UP: mUpY = ev.getY(); isUp = true; break; } if (isUp && mScrollCallback != null && Math.abs(mUpY - mDownY) > SLOP) { mScrollCallback.onTouchUp(mMovingY - mDownY); } return super.dispatchTouchEvent(ev); } }
以上就是主要的代碼實(shí)現(xiàn)部分,靈活地?cái)U(kuò)展應(yīng)用ItemLongClickMaskHelper基本就能實(shí)現(xiàn)類似淘寶長(zhǎng)按遮罩效果了。希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android自定義Dialog內(nèi)部透明、外部遮罩效果
- Android實(shí)現(xiàn)遮罩層(蒙板)效果
- Android PopupWindow實(shí)現(xiàn)遮罩層效果
- Android使用popUpWindow帶遮罩層的彈出框
- 360瀏覽器文本框獲得焦點(diǎn)后被android軟鍵盤遮罩該怎么辦
- Android頁(yè)面中引導(dǎo)蒙層的使用方法詳解
- Android實(shí)現(xiàn)新手引導(dǎo)半透明蒙層效果
- Android自定義ViewGroup實(shí)現(xiàn)豎向引導(dǎo)界面
- Android GuideView實(shí)現(xiàn)首次登陸引導(dǎo)
- 一分鐘實(shí)現(xiàn)Android遮罩引導(dǎo)視圖
相關(guān)文章
Android 使用Vitamio打造自己的萬(wàn)能播放器(7)——在線播放(下載視頻)
本文主要介紹Android Vitamio開(kāi)發(fā)播放器,這里提供在線播放和下載視頻實(shí)例代碼,有需要的小伙伴可以參考下2016-07-07Android編程實(shí)現(xiàn)播放視頻時(shí)切換全屏并隱藏狀態(tài)欄的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)播放視頻時(shí)切換全屏并隱藏狀態(tài)欄的方法,結(jié)合實(shí)例形式分析了Android視頻播放事件響應(yīng)及相關(guān)屬性設(shè)置操作技巧,需要的朋友可以參考下2017-08-08Android 微信搖一搖功能實(shí)現(xiàn)詳細(xì)介紹
這篇文章主要介紹了Android 微信搖一搖功能實(shí)現(xiàn)詳細(xì)介紹的相關(guān)資料,并附實(shí)例代碼及實(shí)現(xiàn)微信搖一搖的思路,需要的朋友可以參考下2016-11-11Android程序設(shè)計(jì)之AIDL實(shí)例詳解
這篇文章主要介紹了Android程序設(shè)計(jì)的AIDL,以一個(gè)完整實(shí)例的形式較為詳細(xì)的講述了AIDL的原理及實(shí)現(xiàn)方法,需要的朋友可以參考下2014-09-09Android?數(shù)據(jù)結(jié)構(gòu)全面總結(jié)分析
這篇文章主要為大家介紹了Android?數(shù)據(jù)結(jié)構(gòu)全面總結(jié)分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12Android編程實(shí)現(xiàn)攔截短信并屏蔽系統(tǒng)Notification的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)攔截短信并屏蔽系統(tǒng)Notification的方法,較為詳細(xì)的分析了Android短信與Notification的原理及對(duì)應(yīng)的設(shè)置取消技巧,需要的朋友可以參考下2015-12-12Android Studio之Debug運(yùn)行期代碼植入的方法
這篇文章主要介紹了Android Studio之Debug運(yùn)行期代碼植入的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-07-07Android 讀取文件內(nèi)容實(shí)現(xiàn)方法總結(jié)
這篇文章主要介紹了Android 讀取文件內(nèi)容實(shí)現(xiàn)方法的相關(guān)資料,這里提供了幾種方法,大家可以選擇使用,需要的朋友可以參考下2016-10-10