Android之淘寶商品列表長按遮罩效果的實(shí)現(xiàn)
先來看看淘寶、唯品會(huì)長按商品的效果,以及簡單Demo的效果:

首先分析一下場(chǎng)景:
長按條目時(shí),彈出遮罩的效果遮擋在原來的條目布局上; 頁面滑動(dòng)或點(diǎn)擊其他的條目,上一個(gè)正在遮罩的條目遮罩消失。 長按其他條目時(shí),上一個(gè)遮罩的條目撤銷遮罩,當(dāng)前長按的顯示遮罩; 條目添加遮罩的時(shí)添加動(dòng)畫;
1. 遮罩的效果,我們會(huì)很容易的想到Android布局控件FrameLayout布局,是基于疊加在上方的布局。所以在列表?xiàng)l目布局的時(shí)候,可以使用FrameLayout布局,在長按列表?xiàng)l目時(shí),用條目的根布局添加一個(gè)遮罩的布局,就達(dá)到我們想要的效果了。

2. 記錄當(dāng)前長按的根布局,如果點(diǎn)擊或長按其他的列表?xiàng)l目,亦或滑動(dòng)頁面(添加活動(dòng)監(jiān)聽)時(shí),就取消之前長按的條目遮罩,從條目根布局中刪除遮罩布局就OK了;
3. 可以利用View動(dòng)畫或?qū)傩詣?dòng)畫,在添加遮罩布局時(shí)顯示動(dòng)畫;
接下來, 來擼一下代碼吧:
1. 首先,先定義一下遮罩的布局,根據(jù)需求自定義View
/***
* 長按條目遮罩界面
*/
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ù)長按的條目指向列表Item的布局,然后為條目添加遮罩的效果;
/**
* 長按條目添加遮罩操作幫助類
*/
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)聽滑動(dòng),撤銷遮罩,直接定義RecyclerView的子類,添加滑動(dòng)監(jiān)聽回調(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)類似淘寶長按遮罩效果了。希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android自定義Dialog內(nèi)部透明、外部遮罩效果
- Android實(shí)現(xiàn)遮罩層(蒙板)效果
- Android PopupWindow實(shí)現(xiàn)遮罩層效果
- Android使用popUpWindow帶遮罩層的彈出框
- 360瀏覽器文本框獲得焦點(diǎn)后被android軟鍵盤遮罩該怎么辦
- Android頁面中引導(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關(guān)于打開虛擬機(jī)時(shí)右側(cè)工具欄不顯示的問題
下面小編就為大家分享一篇解決android關(guān)于打開虛擬機(jī)時(shí)右側(cè)工具欄不顯示的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-01-01
android編程之下拉刷新實(shí)現(xiàn)方法分析
這篇文章主要介紹了android編程之下拉刷新實(shí)現(xiàn)方法,以實(shí)例形式詳細(xì)分析了Android編程中針對(duì)ListView下拉刷新的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-11-11
解決Android SELinux權(quán)限問題記錄分析
這篇文章主要為大家介紹了解決Android SELinux權(quán)限問題記錄分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11
Android調(diào)用系統(tǒng)自帶瀏覽器打開網(wǎng)頁的實(shí)現(xiàn)方法
在Android中可以調(diào)用自帶的瀏覽器,或者指定一個(gè)瀏覽器來打開一個(gè)鏈接。只需要傳入一個(gè)uri,可以是鏈接地址。接下來通過本文給大家分享android 自帶瀏覽器打開網(wǎng)頁的實(shí)現(xiàn)方法,需要的朋友參考下吧2017-09-09
Android5.1系統(tǒng)通過包名給應(yīng)用開放系統(tǒng)權(quán)限的方法
這篇文章主要介紹了Android5.1系統(tǒng)通過包名給應(yīng)用開放系統(tǒng)權(quán)限的方法,此文介紹一種通過修改Android平臺(tái)系統(tǒng)層代碼,根據(jù)指定的應(yīng)用包名給對(duì)應(yīng)的應(yīng)用在該平臺(tái)上開放系統(tǒng)權(quán)限,需要的朋友可以參考下2017-11-11
Android Handler 機(jī)制實(shí)現(xiàn)原理分析
本文主要介紹 Android Handle機(jī)制實(shí)現(xiàn)的原理,這里整理了詳細(xì)的關(guān)于Handler的資料以及工作流程和實(shí)際應(yīng)用,有興趣的小伙伴可以參考下2016-08-08
AndroidQ(10)黑暗模式適配的實(shí)現(xiàn)
這篇文章主要介紹了AndroidQ(10)黑暗模式適配的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06
Android簡單實(shí)用的可拖拽GridView組件分享
在我們?nèi)粘i_發(fā)中,使用?GridView?這種網(wǎng)格視圖的場(chǎng)合還是不少的,本篇我們來介紹一個(gè)支持拖拽的?GridView?組件,可以輕松搞定網(wǎng)格視圖的拖拽排序,需要的可以參考一下2023-06-06
Android Studio實(shí)現(xiàn)音樂播放器
這篇文章主要為大家詳細(xì)介紹了Android Studio實(shí)現(xiàn)音樂播放器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-11-11

