自定義控件之-----Listview

puto.gif

繼承Listview,增強Listview的功能

實現(xiàn)下拉刷新的功能

1捎琐、繼承Listview


        public class RefreshListView extends ListView {

            private int downY;
            private View header;
            private int headerHeight;
            public RefreshListView(Context context, AttributeSet attrs) {
                super(context, attrs);
            }
        }

2蓄髓、添加頭布局妹懒,關鍵方法addHeaderView



        public RefreshListView(Context context, AttributeSet attrs) {
            super(context, attrs);
            initHeader();
        }

        private void initHeader() {
            header = View.inflate(getContext(), R.layout.refresh_header, null);
            // 把布局添加到Listview的頭上
            this.addHeaderView(header);
        }

3、隱藏頭布局双吆,關鍵方法setPadding


        private void initHeader() {
            header = View.inflate(getContext(), R.layout.refresh_header, null);
            // 隱藏頭布局
            // 主動測量控件眨唬,獲取測量的寬高
            header.measure(0, 0);// 把布局中的寬高給測量出來
            // 獲取測量的寬高
            headerHeight = header.getMeasuredHeight();
            header.setPadding(0, -headerHeight, 0, 0);
            // 把布局添加到Listview的頭上
            this.addHeaderView(header);
        }

4、處理事件好乐,讓頭布局隨手指移動匾竿,關鍵方法setPadding,計算手指移動的距離蔚万,再計算出頭布局要設置的頂部padding值岭妖,通過setPadding方法達到移動頭布局的效果



        // 處理觸摸事件
        @Override
        public boolean onTouchEvent(MotionEvent ev) {
            switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                downY = (int) ev.getY();
                break;
            case MotionEvent.ACTION_MOVE:
                int moveY = (int) ev.getY();
                // 計算手指移動的距離
                int diffY = moveY - downY;
                // 只處理從上往下的事件
                if(diffY>0){
                    // 計算頭布局距離頂部的padding值
                    int topPadding = diffY - headerHeight;
                    header.setPadding(0, topPadding, 0, 0);
                    return true;// 自己處理的從上往下的觸摸事件,需要消費掉
                }
                break;
    
            default:
                break;
            }
            return super.onTouchEvent(ev);
        }

5反璃、當Listview第一個條目沒有完全展示時昵慌,給頭布局設置padding沒有效果,需要判斷當?shù)谝粋€條目完全展示時淮蜈,才處理下拉刷新



        // 處理觸摸事件
        @Override
        public boolean onTouchEvent(MotionEvent ev) {
            switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                downY = (int) ev.getY();
                break;
            case MotionEvent.ACTION_MOVE:
                int moveY = (int) ev.getY();
                // 計算手指移動的距離
                int diffY = moveY - downY;
                // 只有當Listview中的第一個條目完全展示時斋攀,header.setPadding才有效果,才能自己處理事件
                if (getFirstVisiblePosition() != 0) {
                    // 如果自己不處理事件梧田,每次移動需要給downY重新賦值
                    downY = (int) ev.getY();
                    break;
                }
                // 只處理從上往下的事件
                if(diffY>0){
                    // 計算頭布局距離頂部的padding值
                    int topPadding = diffY - headerHeight;
                    header.setPadding(0, topPadding, 0, 0);
                    return true;// 自己處理的從上往下的觸摸事件淳蔼,需要消費掉
                }
                break;
    
            default:
                break;
            }
            return super.onTouchEvent(ev);
        }

6、根據(jù)手指移動的距離裁眯,設置刷新狀態(tài)



        private static final int PULLREFRESH_STATE = 1;// 下拉刷新狀態(tài)
        private static final int RELEASE_STATE = 2;// 松開刷新狀態(tài)
        private static final int REFRESHING_STATE = 3;// 正在刷新狀態(tài)
        private int current_state = PULLREFRESH_STATE;// 當前刷新狀態(tài)

        case MotionEvent.ACTION_MOVE:
            int moveY = (int) ev.getY();
            // 計算手指移動的距離
            int diffY = moveY - downY;
            // 只有當Listview中的第一個條目完全展示時鹉梨,header.setPadding才有效果,才能自己處理事件
            if (getFirstVisiblePosition() != 0) {
                // 如果自己不處理事件穿稳,每次移動需要給downY重新賦值
                downY = (int) ev.getY();
                break;
            }
            // 只處理從上往下的事件
            if (diffY > 0) {
                // 計算頭布局距離頂部的padding值
                int topPadding = diffY - headerHeight;
                // 根據(jù)toppadding值是否大于0 頭布局是否完全展示存皂,判斷狀態(tài)的切換
                if (topPadding >= 0 && current_state != RELEASE_STATE) {// 頭布局完全展示,切換到松開刷新
                                                                        // 逢艘,如果已經(jīng)是松開刷新狀態(tài)旦袋,就不用再切換
                    current_state = RELEASE_STATE;
                    System.out.println("切換到松開刷新");
                    switchState();
                } else if (topPadding < 0 && current_state != PULLREFRESH_STATE) {// 頭布局沒有完全展示,切換到下拉刷新
                    current_state = PULLREFRESH_STATE;
                    System.out.println("切換到下拉刷新");
                    switchState();
                }

                header.setPadding(0, topPadding, 0, 0);
                return true;// 自己處理的從上往下的觸摸事件埋虹,需要消費掉
            }
            break;
        case MotionEvent.ACTION_UP:
            // 手指抬起時猜憎,根據(jù)當前的狀態(tài)判斷是否切換到正在刷新
            if (current_state == PULLREFRESH_STATE) {// 抬起時,是下拉刷新搔课,頭布局沒有完全展示胰柑,不切換到正在刷新
                // 隱藏頭布局
                header.setPadding(0, -headerHeight, 0, 0);
            } else if (current_state == RELEASE_STATE) {// 抬起時,是松開刷新爬泥,切換到正在刷新
                current_state = REFRESHING_STATE;
                // 讓頭布局正好完全展示
                header.setPadding(0, 0, 0, 0);
                System.out.println("切換到正在刷新");
                switchState();
            }
            break;

        * 箭頭動畫
        
        public RefreshListView(Context context, AttributeSet attrs) {
            super(context, attrs);
            initHeader();
            initAnimation();
        }

        private void initAnimation() {
            up = new RotateAnimation(0, -180, Animation.RELATIVE_TO_SELF, 0.5f,
                    Animation.RELATIVE_TO_SELF, 0.5f);
            up.setDuration(200);
            up.setFillAfter(true);
            down = new RotateAnimation(-180, -360, Animation.RELATIVE_TO_SELF,
                    0.5f, Animation.RELATIVE_TO_SELF, 0.5f);
            down.setDuration(200);
            down.setFillAfter(true);
        }

        * 根據(jù)狀態(tài)更新控件

        // 切換狀態(tài)時柬讨,更新界面
        private void switchState() {
            switch (current_state) {
            case PULLREFRESH_STATE:
                state.setText("下拉刷新");
                progress.setVisibility(View.INVISIBLE);
                arrow.setVisibility(View.VISIBLE);
                arrow.startAnimation(down);
                break;
            case RELEASE_STATE:
                state.setText("松開刷新");
                arrow.startAnimation(up);
                break;
            case REFRESHING_STATE:
                // 由于動畫設置了setFillAfter 控件就停留在結(jié)束時的效果
                arrow.clearAnimation();
                state.setText("正在刷新");
                progress.setVisibility(View.VISIBLE);
                arrow.setVisibility(View.INVISIBLE);
                break;
    
            default:
                break;
            }
        }

7、暴露接口袍啡,讓外界實現(xiàn)業(yè)務


        // 對外暴露接口
        public interface OnRefreshListener {
            // 正在刷新時踩官,回調(diào)
            void onRefreshing();
        }
    
        // 提供傳遞監(jiān)聽器的方法
        public void setOnRefreshListener(OnRefreshListener listener) {
            this.mListener = listener;
        }

        case MotionEvent.ACTION_UP:
            // 手指抬起時,根據(jù)當前的狀態(tài)判斷是否切換到正在刷新
            if (current_state == PULLREFRESH_STATE) {// 抬起時境输,是下拉刷新蔗牡,頭布局沒有完全展示颖系,不切換到正在刷新
                // 隱藏頭布局
                header.setPadding(0, -headerHeight, 0, 0);
            } else if (current_state == RELEASE_STATE) {// 抬起時,是松開刷新辩越,切換到正在刷新
                current_state = REFRESHING_STATE;
                // 讓頭布局正好完全展示
                header.setPadding(0, 0, 0, 0);
                System.out.println("切換到正在刷新");
                switchState();
                // 當處于正在刷新狀態(tài)時嘁扼,回調(diào)監(jiān)聽器的onRefreshing
                if (mListener != null) {
                    mListener.onRefreshing();
                }
            }
            break;

        * 外界監(jiān)聽刷新狀態(tài),處理業(yè)務

        listview.setOnRefreshListener(new MyListener());

        class MyListener implements OnRefreshListener{

            @Override
            public void onRefreshing() {
                // 處理業(yè)務
                new Thread(){
                    public void run() {
                        try {
                            Thread.sleep(3000);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                        runOnUiThread(new Runnable() {
                            
                            @Override
                            public void run() {
                                arrayList.add(0, "我是拉出來的");
                                adapter.notifyDataSetChanged();
                                // 刷新完成后黔攒,調(diào)用恢復下拉刷新控件的方法
                                listview.refreshFinished();
                            }
                        });
                    };
                }.start();
            }
        }

8趁啸、刷新業(yè)務完成后,恢復下拉刷新狀態(tài)



        // 下拉刷新完成后督惰,恢復狀態(tài)不傅,隱藏頭布局
        public void refreshFinished() {
            header.setPadding(0, -headerHeight, 0, 0);
            state.setText("下拉刷新");
            progress.setVisibility(View.INVISIBLE);
            arrow.setVisibility(View.VISIBLE);
            current_state = PULLREFRESH_STATE;
        }

實現(xiàn)上拉加載更多功能

流程:

    1、添加腳布局赏胚,addFooterView
    2访娶、隱藏腳布局,setPadding
    3栅哀、監(jiān)聽Listview的滾動狀態(tài)震肮,當處于停止或慣性停止狀態(tài)時,而且Listview最后一個條目完全展示留拾,才加載更多
    4戳晌、對外暴露接口,讓外界處理加載更多的業(yè)務
    5痴柔、加載更多業(yè)務完成后沦偎,恢復加載更多狀態(tài)

2.1、添加腳布局



        public RefreshListView(Context context, AttributeSet attrs) {
            super(context, attrs);
            initHeader();
            initAnimation();
            initFooter();
        }

        private void initFooter() {
            footer = View.inflate(getContext(), R.layout.refresh_footer, null);
            // 添加腳布局
            addFooterView(footer);
        }

        * 腳布局

        <?xml version="1.0" encoding="utf-8"?>
        <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:gravity="center"
            android:orientation="horizontal" >
        
            <ProgressBar
                android:layout_width="wrap_content"
                android:layout_height="wrap_content" />
        
            <TextView
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="加載更多中咳蔚。豪嚎。。"
                android:textColor="#f00"
                android:textSize="25sp" />
        
        </LinearLayout>

2.2谈火、隱藏腳布局



        private void initFooter() {
            footer = View.inflate(getContext(), R.layout.refresh_footer, null);
            footer.measure(0, 0);
            footerHeight = footer.getMeasuredHeight();
            footer.setPadding(0, 0, 0, -footerHeight);
            // 添加腳布局
            addFooterView(footer);
        }


2.3侈询、監(jiān)聽Listview滾動狀態(tài),根據(jù)狀態(tài)判斷是否顯示加載更多腳布局


        private void initFooter() {
            footer = View.inflate(getContext(), R.layout.refresh_footer, null);
            footer.measure(0, 0);
            footerHeight = footer.getMeasuredHeight();
            footer.setPadding(0, 0, 0, -footerHeight);
            // 添加腳布局
            addFooterView(footer);
            // 監(jiān)聽Listview的滾動狀態(tài)
            this.setOnScrollListener(new MyOnScrollListener());
        }

        class MyOnScrollListener implements OnScrollListener {
            // 狀態(tài)發(fā)生變化時調(diào)用
            @Override
            public void onScrollStateChanged(AbsListView view, int scrollState) {
                // 當處于停止或慣性停止狀態(tài)時
                if (scrollState == OnScrollListener.SCROLL_STATE_IDLE
                        || scrollState == OnScrollListener.SCROLL_STATE_FLING) {
                    // 而且Listview最后一個條目完全展示
                    if(getLastVisiblePosition()==getCount()-1&&!isLoadMore){
                        isLoadMore = true;
                        // 顯示加載更多布局
                        footer.setPadding(0, 0, 0, 0);
                        System.out.println("加載更多了");
                        // 自動顯示加載更多布局
                        setSelection(getCount());
                    }
                }
            }
        }


2.4糯耍、暴露接口扔字,讓外界處理加載更多業(yè)務


        // 對外暴露接口
        public interface OnRefreshListener {
            // 正在刷新時,回調(diào)
            void onRefreshing();
            // 加載更多時温技,回調(diào)
            void onLoadingMore();
        }

        @Override
        public void onScrollStateChanged(AbsListView view, int scrollState) {
            // 當處于停止或慣性停止狀態(tài)時
            if (scrollState == OnScrollListener.SCROLL_STATE_IDLE
                    || scrollState == OnScrollListener.SCROLL_STATE_FLING) {
                // 而且Listview最后一個條目完全展示
                if(getLastVisiblePosition()==getCount()-1&&!isLoadMore){
                    isLoadMore = true;
                    // 顯示加載更多布局
                    footer.setPadding(0, 0, 0, 0);
                    System.out.println("加載更多了");
                    // 自動顯示加載更多布局
                    setSelection(getCount());
                    // 當處于加載更多時革为,調(diào)用監(jiān)聽器的onLoadingMore方法
                    if(mListener!=null){
                        mListener.onLoadingMore();
                    }
                }
            }
        }
        
        

2.5外界處理加載更多業(yè)務



        @Override
        public void onLoadingMore() {
            // 處理業(yè)務
                        new Thread(){
                            public void run() {
                                try {
                                    Thread.sleep(3000);
                                } catch (InterruptedException e) {
                                    e.printStackTrace();
                                }
                                runOnUiThread(new Runnable() {
                                    
                                    @Override
                                    public void run() {
                                        arrayList.add("我是加載出來的");
                                        arrayList.add("我是加載出來的");
                                        adapter.notifyDataSetChanged();
                                        // 加載更多完成后,調(diào)用控件恢復狀態(tài)的方法
                                        listview.loadMoreFinished();
                                    }
                                });
                            };
                        }.start();
        }



2. 6舵鳞、外界處理完業(yè)務震檩,恢復加載更多狀態(tài)



        // 加載更多完成后,恢復狀態(tài)
        public void loadMoreFinished(){
            isLoadMore = false;
            footer.setPadding(0, 0, 0, -footerHeight);
        }

github地址

https://github.com/zssAndroid/RefreshListView/tree/master

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末蜓堕,一起剝皮案震驚了整個濱河市抛虏,隨后出現(xiàn)的幾起案子博其,更是在濱河造成了極大的恐慌,老刑警劉巖嘉蕾,帶你破解...
    沈念sama閱讀 206,378評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件贺奠,死亡現(xiàn)場離奇詭異,居然都是意外死亡错忱,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評論 2 382
  • 文/潘曉璐 我一進店門挂据,熙熙樓的掌柜王于貴愁眉苦臉地迎上來以清,“玉大人,你說我怎么就攤上這事崎逃≈谰螅” “怎么了?”我有些...
    開封第一講書人閱讀 152,702評論 0 342
  • 文/不壞的土叔 我叫張陵个绍,是天一觀的道長勒葱。 經(jīng)常有香客問我,道長巴柿,這世上最難降的妖魔是什么凛虽? 我笑而不...
    開封第一講書人閱讀 55,259評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮广恢,結(jié)果婚禮上凯旋,老公的妹妹穿的比我還像新娘。我一直安慰自己钉迷,他們只是感情好至非,可當我...
    茶點故事閱讀 64,263評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著糠聪,像睡著了一般荒椭。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上舰蟆,一...
    開封第一講書人閱讀 49,036評論 1 285
  • 那天趣惠,我揣著相機與錄音,去河邊找鬼夭苗。 笑死信卡,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的题造。 我是一名探鬼主播傍菇,決...
    沈念sama閱讀 38,349評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼界赔!你這毒婦竟也來了丢习?” 一聲冷哼從身側(cè)響起牵触,我...
    開封第一講書人閱讀 36,979評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎咐低,沒想到半個月后揽思,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,469評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡见擦,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,938評論 2 323
  • 正文 我和宋清朗相戀三年钉汗,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片鲤屡。...
    茶點故事閱讀 38,059評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡损痰,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出酒来,到底是詐尸還是另有隱情卢未,我是刑警寧澤,帶...
    沈念sama閱讀 33,703評論 4 323
  • 正文 年R本政府宣布堰汉,位于F島的核電站辽社,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏翘鸭。R本人自食惡果不足惜滴铅,卻給世界環(huán)境...
    茶點故事閱讀 39,257評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望矮固。 院中可真熱鬧失息,春花似錦、人聲如沸档址。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽守伸。三九已至绎秒,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間尼摹,已是汗流浹背见芹。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留蠢涝,地道東北人玄呛。 一個月前我還...
    沈念sama閱讀 45,501評論 2 354
  • 正文 我出身青樓,卻偏偏與公主長得像和二,于是被迫代替她去往敵國和親徘铝。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 42,792評論 2 345

推薦閱讀更多精彩內(nèi)容