高仿微信對話列表滑動刪除效果

轉載請註明出處:http://blog.csdn.net/singwhatiwanna/article/details/17515543html

前言

用過微信的都知道,微信對話列表滑動刪除效果是很不錯的,這個效果咱們也能夠有。思路其實很簡單,弄個ListView,而後裏面的每一個item作成一個能夠滑動的自定義控件便可。因爲ListView是上下滑動而item是左右滑動,所以會有滑動衝突,也許你須要瞭解下android中點擊事件的派發流程,請參考Android源碼分析-點擊事件派發機制。個人解決思路是這樣的:重寫ListView的onInterceptTouchEvent方法,在move的時候作判斷,若是是左右滑動就返回false,不然返回true;重寫SlideView(即自定義item控件)的onTouchEvent方法來處理滑動。整個思路沒有問題,滑動衝突也解決了,但是ListView沒法獲得焦點了,也就是ListView沒法處理點擊事件了。讓咱們回想下問題出在哪裏:個人理解是這樣的,上述處理滑動自己沒有問題,可是有一個反作用,就是會讓外層View失去焦點且沒法處理點擊事件。常見的滑動衝突場景,好比launcher內部嵌入ListView倒是沒有問題的,由於這個時候launcher不須要得到焦點,須要得到焦點的是內部的ListView。所以,上述處理方式對於外部須要得到焦點的狀況(好比外部是ListView)就不太適合了。因而我就和ttdevs探討,發現他採用了另一種思路,我歷來沒有想過還能夠這樣玩。下面介紹他的思路。java

新的思路

不考慮那麼複雜,不採用主流玩法,全部的事件均由外層的ListView作攔截,同時把事件傳遞給SlideView作滑動,這種實現的確能夠達到效果,並且代碼很簡單,根本不須要處理什麼複雜的滑動衝突。android

效果

下面分別爲微信和高仿效果緩存


代碼分析

先看SlideView是如何實現的微信

看layout xml:app

<?xml version="1.0" encoding="utf-8"?>
<merge xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >

    <LinearLayout
        android:id="@+id/view_content"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="horizontal" >
    </LinearLayout>

    <RelativeLayout
        android:id="@+id/holder"
        android:layout_width="120dp"
        android:layout_height="match_parent"
        android:clickable="true"
        android:background="@drawable/holder_bg">

        <TextView
            android:id="@+id/delete"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:drawableLeft="@drawable/del_icon_normal"
            android:layout_centerInParent="true"
            android:gravity="center"
            android:textColor="@color/floralwhite"
            android:text="刪除" />
    </RelativeLayout>

</merge>
上述xml文件中,全部的view都會被放在view_content中,而holder是放置諸如刪除按鈕之類的東西,咱們的SlideView會加載這個佈局。

再看SlideView.java:ide

/**
 * SlideView 繼承自LinearLayout
 */
public class SlideView extends LinearLayout {

    private static final String TAG = "SlideView";

    private Context mContext;

    // 用來放置全部view的容器
    private LinearLayout mViewContent;

    // 用來放置內置view的容器,好比刪除 按鈕
    private RelativeLayout mHolder;

    // 彈性滑動對象,提供彈性滑動效果
    private Scroller mScroller;

    // 滑動回調接口,用來向上層通知滑動事件
    private OnSlideListener mOnSlideListener;

    // 內置容器的寬度 單位:dp
    private int mHolderWidth = 120;

    // 分別記錄上次滑動的座標
    private int mLastX = 0;
    private int mLastY = 0;

    // 用來控制滑動角度,僅當角度a知足以下條件才進行滑動:tan a = deltaX / deltaY > 2
    private static final int TAN = 2;

    public interface OnSlideListener {
        // SlideView的三種狀態:開始滑動,打開,關閉
        public static final int SLIDE_STATUS_OFF = 0;
        public static final int SLIDE_STATUS_START_SCROLL = 1;
        public static final int SLIDE_STATUS_ON = 2;

        /**
         * @param view
         *            current SlideView
         * @param status
         *            SLIDE_STATUS_ON, SLIDE_STATUS_OFF or
         *            SLIDE_STATUS_START_SCROLL
         */
        public void onSlide(View view, int status);
    }

    public SlideView(Context context) {
        super(context);
        initView();
    }

    public SlideView(Context context, AttributeSet attrs) {
        super(context, attrs);
        initView();
    }

    private void initView() {
        mContext = getContext();
        // 初始化彈性滑動對象
        mScroller = new Scroller(mContext);
        // 設置其方向爲橫向
        setOrientation(LinearLayout.HORIZONTAL);
        // 將slide_view_merge加載進來
        View.inflate(mContext, R.layout.slide_view_merge, this);
        mViewContent = (LinearLayout) findViewById(R.id.view_content);
        mHolderWidth = Math.round(TypedValue.applyDimension(
                TypedValue.COMPLEX_UNIT_DIP, mHolderWidth, getResources()
                        .getDisplayMetrics()));
    }

    // 設置按鈕的內容,也能夠設置圖標啥的,我沒寫
    public void setButtonText(CharSequence text) {
        ((TextView) findViewById(R.id.delete)).setText(text);
    }

    // 將view加入到ViewContent中
    public void setContentView(View view) {
        mViewContent.addView(view);
    }

    // 設置滑動回調
    public void setOnSlideListener(OnSlideListener onSlideListener) {
        mOnSlideListener = onSlideListener;
    }

    // 將當前狀態置爲關閉
    public void shrink() {
        if (getScrollX() != 0) {
            this.smoothScrollTo(0, 0);
        }
    }

    // 根據MotionEvent來進行滑動,這個方法的做用至關於onTouchEvent
    // 若是你不須要處理滑動衝突,能夠直接重命名,照樣能正常工做
    public void onRequireTouchEvent(MotionEvent event) {
        int x = (int) event.getX();
        int y = (int) event.getY();
        int scrollX = getScrollX();
        Log.d(TAG, "x=" + x + "  y=" + y);

        switch (event.getAction()) {
        case MotionEvent.ACTION_DOWN: {
            if (!mScroller.isFinished()) {
                mScroller.abortAnimation();
            }
            if (mOnSlideListener != null) {
                mOnSlideListener.onSlide(this,
                        OnSlideListener.SLIDE_STATUS_START_SCROLL);
            }
            break;
        }
        case MotionEvent.ACTION_MOVE: {
            int deltaX = x - mLastX;
            int deltaY = y - mLastY;
            if (Math.abs(deltaX) < Math.abs(deltaY) * TAN) {
                // 滑動不知足條件,不作橫向滑動
                break;
            }

            // 計算滑動終點是否合法,防止滑動越界
            int newScrollX = scrollX - deltaX;
            if (deltaX != 0) {
                if (newScrollX < 0) {
                    newScrollX = 0;
                } else if (newScrollX > mHolderWidth) {
                    newScrollX = mHolderWidth;
                }
                this.scrollTo(newScrollX, 0);
            }
            break;
        }
        case MotionEvent.ACTION_UP: {
            int newScrollX = 0;
            // 這裏作了下判斷,當鬆開手的時候,會自動向兩邊滑動,具體向哪邊滑,要看當前所處的位置
            if (scrollX - mHolderWidth * 0.75 > 0) {
                newScrollX = mHolderWidth;
            }
            // 慢慢滑向終點
            this.smoothScrollTo(newScrollX, 0);
            // 通知上層滑動事件
            if (mOnSlideListener != null) {
                mOnSlideListener.onSlide(this,
                        newScrollX == 0 ? OnSlideListener.SLIDE_STATUS_OFF
                                : OnSlideListener.SLIDE_STATUS_ON);
            }
            break;
        }
        default:
            break;
        }

        mLastX = x;
        mLastY = y;
    }

    private void smoothScrollTo(int destX, int destY) {
        // 緩慢滾動到指定位置
        int scrollX = getScrollX();
        int delta = destX - scrollX;
        // 以三倍時長滑向destX,效果就是慢慢滑動
        mScroller.startScroll(scrollX, 0, delta, 0, Math.abs(delta) * 3);
        invalidate();
    }

    @Override
    public void computeScroll() {
        if (mScroller.computeScrollOffset()) {
            scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
            postInvalidate();
        }
    }

}
上述代碼作了很詳細的說明,這就是滑動控件的完整代碼,你們要明白的是:你所添加的view都是加在SlideView的子View : view_content中的,而不是直接加在SlideView中,只有這樣咱們才方便作滑動效果。

接着看ListView的代碼:核心就是下面這一個方法,將點擊事件發送給SlideView處理。源碼分析

@Override
    public boolean onTouchEvent(MotionEvent event) {
        switch (event.getAction()) {
        case MotionEvent.ACTION_DOWN: {
            int x = (int) event.getX();
            int y = (int) event.getY();
            //咱們想知道當前點擊了哪一行
            int position = pointToPosition(x, y);
            Log.e(TAG, "postion=" + position);
            if (position != INVALID_POSITION) {
                //獲得當前點擊行的數據從而取出當前行的item。
                //可能有人懷疑,爲何要這麼幹?爲何不用getChildAt(position)?
                //由於ListView會進行緩存,若是你不這麼幹,有些行的view你是得不到的。
                MessageItem data = (MessageItem) getItemAtPosition(position);
                mFocusedItemView = data.slideView;
                Log.e(TAG, "FocusedItemView=" + mFocusedItemView);
            }
        }
        default:
            break;
        }

        //向當前點擊的view發送滑動事件請求,其實就是向SlideView發請求
        if (mFocusedItemView != null) {
            mFocusedItemView.onRequireTouchEvent(event);
        }

        return super.onTouchEvent(event);
    }
最後看Activity的代碼:
public class MainActivity extends Activity implements OnItemClickListener,
        OnClickListener, OnSlideListener {

    private static final String TAG = "MainActivity";

    private ListViewCompat mListView;

    private List<MessageItem> mMessageItems = new ArrayList<MainActivity.MessageItem>();

    private SlideAdapter mSlideAdapter;

    // 上次處於打開狀態的SlideView
    private SlideView mLastSlideViewWithStatusOn;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        initView();
    }

    private void initView() {
        mListView = (ListViewCompat) findViewById(R.id.list);

        for (int i = 0; i < 20; i++) {
            MessageItem item = new MessageItem();
            if (i % 3 == 0) {
                item.iconRes = R.drawable.default_qq_avatar;
                item.title = "騰訊新聞";
                item.msg = "青島爆炸滿月:大量魚蝦死亡";
                item.time = "晚上18:18";
            } else {
                item.iconRes = R.drawable.wechat_icon;
                item.title = "微信團隊";
                item.msg = "歡迎你使用微信";
                item.time = "12月18日";
            }
            mMessageItems.add(item);
        }
        mSlideAdapter = new SlideAdapter();
        mListView.setAdapter(mSlideAdapter);
        mListView.setOnItemClickListener(this);
    }

    private class SlideAdapter extends BaseAdapter {

        private LayoutInflater mInflater;

        SlideAdapter() {
            super();
            mInflater = getLayoutInflater();
        }

        @Override
        public int getCount() {
            return mMessageItems.size();
        }

        @Override
        public Object getItem(int position) {
            return mMessageItems.get(position);
        }

        @Override
        public long getItemId(int position) {
            return position;
        }

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            ViewHolder holder;
            SlideView slideView = (SlideView) convertView;
            if (slideView == null) {
                // 這裏是咱們的item
                View itemView = mInflater.inflate(R.layout.list_item, null);

                slideView = new SlideView(MainActivity.this);
                // 這裏把item加入到slideView
                slideView.setContentView(itemView);
                // 下面是作一些數據緩存
                holder = new ViewHolder(slideView);
                slideView.setOnSlideListener(MainActivity.this);
                slideView.setTag(holder);
            } else {
                holder = (ViewHolder) slideView.getTag();
            }
            MessageItem item = mMessageItems.get(position);
            item.slideView = slideView;
            item.slideView.shrink();

            holder.icon.setImageResource(item.iconRes);
            holder.title.setText(item.title);
            holder.msg.setText(item.msg);
            holder.time.setText(item.time);
            holder.deleteHolder.setOnClickListener(MainActivity.this);

            return slideView;
        }

    }

    public class MessageItem {
        public int iconRes;
        public String title;
        public String msg;
        public String time;
        public SlideView slideView;
    }

    private static class ViewHolder {
        public ImageView icon;
        public TextView title;
        public TextView msg;
        public TextView time;
        public ViewGroup deleteHolder;

        ViewHolder(View view) {
            icon = (ImageView) view.findViewById(R.id.icon);
            title = (TextView) view.findViewById(R.id.title);
            msg = (TextView) view.findViewById(R.id.msg);
            time = (TextView) view.findViewById(R.id.time);
            deleteHolder = (ViewGroup) view.findViewById(R.id.holder);
        }
    }

    @Override
    public void onItemClick(AdapterView<?> parent, View view, int position,
            long id) {
        // 這裏處理ListItem的點擊事件
        Log.e(TAG, "onItemClick position=" + position);
    }

    @Override
    public void onSlide(View view, int status) {
        // 若是當前存在已經打開的SlideView,那麼將其關閉
        if (mLastSlideViewWithStatusOn != null
                && mLastSlideViewWithStatusOn != view) {
            mLastSlideViewWithStatusOn.shrink();
        }
        // 記錄本次處於打開狀態的view
        if (status == SLIDE_STATUS_ON) {
            mLastSlideViewWithStatusOn = (SlideView) view;
        }
    }

    @Override
    public void onClick(View v) {
        // 這裏處理刪除按鈕的點擊事件,能夠刪除對話
        if (v.getId() == R.id.holder) {
            int position = mListView.getPositionForView(v);
            if (position != ListView.INVALID_POSITION) {
                mMessageItems.remove(position);
                mSlideAdapter.notifyDataSetChanged();
            }
            Log.e(TAG, "onClick v=" + v);
        }
    }
}

代碼我都特地寫了註釋,就很少說了。佈局

代碼下載:http://download.csdn.net/detail/singwhatiwanna/6760085post

另外此博文采用了 ttdevs 所提供代碼的部分思想(他的博客是http://blog.csdn.net/ttdevs)

相關文章
相關標籤/搜索