Android自定義控件ScrollView實(shí)現(xiàn)上下滑動(dòng)功能
本文實(shí)例為大家分享了Android ScrollView實(shí)現(xiàn)上下滑動(dòng)功能的具體代碼,供大家參考,具體內(nèi)容如下
package com.example.zhuang; import android.content.Context; import android.util.AttributeSet; import android.util.DisplayMetrics; import android.view.MotionEvent; import android.view.View; import android.view.ViewGroup; import android.view.WindowManager; import android.widget.Scroller; public class MyScrollView extends ViewGroup { private int mScreeHeight;//屏幕高度 private Scroller mScroller; private int mLastY; private int mStart; private int mEnd; private Context context; public MyScrollView(Context context) { super(context); initView(context); } public MyScrollView(Context context, AttributeSet attrs) { super(context, attrs); initView(context); } public MyScrollView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); initView(context); } private void initView(Context context) { WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE); //DisplayMetrics 類(lèi)提供了一種關(guān)于顯示的通用信息,如顯示大小,分辨率和字體。 DisplayMetrics dm = new DisplayMetrics(); wm.getDefaultDisplay().getMetrics(dm); mScreeHeight = dm.heightPixels;//高度(像素) mScroller = new Scroller(context); } //繼承ViewGroup必須要實(shí)現(xiàn)的方法 @Override protected void onLayout(boolean changed, int l, int t, int r, int b) { int childCount = getChildCount();//獲取子view的個(gè)數(shù) //設(shè)置ViewGroup的高度 MarginLayoutParams mlp = (MarginLayoutParams) getLayoutParams(); mlp.height = mScreeHeight * childCount; setLayoutParams(mlp); for (int i = 0; i < childCount; i++) { View child = getChildAt(i); if (child.getVisibility() != View.GONE) { //參數(shù)為相對(duì)父容器的左上右下位置,第三個(gè)參數(shù)必須為r child.layout(0, i * mScreeHeight, r, (i + 1) * mScreeHeight); } } } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(widthMeasureSpec, heightMeasureSpec); int count = getChildCount(); for (int i = 0; i < count; i++) { View child = getChildAt(i); measureChild(child, widthMeasureSpec, heightMeasureSpec); } } @Override public boolean onTouchEvent(MotionEvent event) { int y = (int) event.getY();//相對(duì)于view的y值,getRawY()是相對(duì)屏幕 switch (event.getAction()) { case MotionEvent.ACTION_DOWN: mLastY = y;//上一次的y值 mStart = getScrollY();//記錄觸摸起點(diǎn) break; case MotionEvent.ACTION_MOVE: if(!mScroller.isFinished()) { mScroller.abortAnimation();//放棄移到最終位置 } int dy = mLastY - y;//偏移距離 //如果滑動(dòng)距離小于0或大于屏幕高度,不偏移 if(getScrollY()<0){ dy = 0; } if(getScrollY() > getHeight()-mScreeHeight){ dy = 0; } scrollBy(0,dy);//移動(dòng) mLastY = y; break; case MotionEvent.ACTION_UP: int dScrollY = checkAlignment();//整體移動(dòng)的距離 if(dScrollY > 0){ if(dScrollY < mScreeHeight / 3){ mScroller.startScroll(0,getScrollY(),0,-dScrollY); }else{ mScroller.startScroll(0,getScrollY(),0,mScreeHeight-dScrollY); } }else{ if(-dScrollY < mScreeHeight / 3){ mScroller.startScroll(0,getScrollY(),0,-dScrollY); }else{ mScroller.startScroll(0,getScrollY(),0,-mScreeHeight-dScrollY); } } break; } postInvalidate(); return true; } private int checkAlignment(){ mEnd = getScrollY();//記錄觸摸終點(diǎn) boolean isUp = ((mEnd - mStart)>0) ? true : false; int lastPrev = mEnd % mScreeHeight; int lastNext = mScreeHeight - lastPrev; if(isUp){ return lastPrev;//向上 }else return -lastNext; } @Override public void computeScroll() { super.computeScroll(); if(mScroller.computeScrollOffset()){//返回true,表示還未移動(dòng)完 scrollTo(0,mScroller.getCurrY());//移到當(dāng)前位置 postInvalidate(); //invalidate()是用來(lái)刷新View的,必須是在UI線程中進(jìn)行工作。 //postInvalidate()可以在非UI線程調(diào)用 } } }
知識(shí)點(diǎn):
1、獲取屏幕參數(shù)代碼:
DisplayMetrics metric = new DisplayMetrics(); //API 17之后使用,獲取的像素寬高包含虛擬鍵所占空間,在API 17之前通過(guò)反射獲取 context.getWindowManager().getDefaultDisplay().getRealMetrics(metric); //獲取的像素寬高不包含虛擬鍵所占空間 //context.getWindowManager().getDefaultDisplay().getMetrics(metric); int width = metric.widthPixels; // 寬度(像素) int height = metric.heightPixels; // 高度(像素) float density = metric.density; // dp縮放因子 int densityDpi = metric.densityDpi; // 廣義密度 float xdpi = metric.xdpi;//x軸方向的真實(shí)密度 float ydpi = metric.ydpi;//y軸方向的真實(shí)密度
屏幕高度值包含了狀態(tài)欄的像素,非沉浸模式下真實(shí)的Activity高度需要減去狀態(tài)欄的高度。獲取狀態(tài)欄高度代碼:
private int getStatusBarHeight() { Rect rect = new Rect(); getWindow().getDecorView().getWindowVisibleDisplayFrame(rect); return rect.top; }
屏幕參數(shù)Width和Height的值和屏幕方向有關(guān),另外4個(gè)值和屏幕方向無(wú)關(guān)。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android自定義控件實(shí)現(xiàn)可左右滑動(dòng)的導(dǎo)航條
- Android控件之SlidingDrawer(滑動(dòng)式抽屜)詳解與實(shí)例分享
- Android開(kāi)源堆疊滑動(dòng)控件仿探探效果
- Android實(shí)現(xiàn)可滑動(dòng)的自定義日歷控件
- Android控件SeekBar仿淘寶滑動(dòng)驗(yàn)證效果
- Android自定義View實(shí)現(xiàn)隨手勢(shì)滑動(dòng)控件
- Android仿微信列表滑動(dòng)刪除之可滑動(dòng)控件(一)
- Android自定義滑動(dòng)解鎖控件使用詳解
- Android自定義控件實(shí)現(xiàn)滑動(dòng)開(kāi)關(guān)效果
- Android自定義雙向滑動(dòng)控件
相關(guān)文章
Android 8.0 中如何實(shí)現(xiàn)視頻通話的畫(huà)中畫(huà)模式的示例
本篇文章介紹了Android 8.0 中如何實(shí)現(xiàn)視頻通話的畫(huà)中畫(huà)模式的示例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11快速解決Android適配底部返回鍵等虛擬鍵盤(pán)的問(wèn)題
今天小編就為大家分享一篇快速解決Android適配底部返回鍵等虛擬鍵盤(pán)的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-07-07Android Loader詳細(xì)介紹及實(shí)例代碼
這篇文章主要介紹了Android Loader詳細(xì)介紹及實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2016-12-12Android中監(jiān)聽(tīng)判斷網(wǎng)絡(luò)連接狀態(tài)的方法
這篇文章主要介紹了Android中監(jiān)聽(tīng)判斷網(wǎng)絡(luò)連接狀態(tài)的方法,介紹了是否有網(wǎng)絡(luò)連接判斷、連接的類(lèi)型和監(jiān)聽(tīng)網(wǎng)絡(luò)狀態(tài)的方法,需要的朋友可以參考下2014-06-06Android折疊式Toolbar使用完全解析(CollapsingToolbarLayout)
這篇文章主要為大家詳細(xì)介紹了Android折疊式Toolbar的使用方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-02-02Android編程實(shí)現(xiàn)向SD卡寫(xiě)入數(shù)據(jù)的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)向SD卡寫(xiě)入數(shù)據(jù)的方法,涉及Android針對(duì)SD卡狀態(tài)判斷,文件及權(quán)限操作等相關(guān)技巧,需要的朋友可以參考下2016-04-04Flutter使用?input?chip?標(biāo)簽組件示例詳解
這篇文章主要為大家介紹了Flutter使用?input?chip?標(biāo)簽組件示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10Android 輕松實(shí)現(xiàn)語(yǔ)音識(shí)別詳解及實(shí)例代碼
這篇文章主要介紹了Android 輕松實(shí)現(xiàn)語(yǔ)音識(shí)別的相關(guān)資料,并附實(shí)例代碼,需要的朋友可以參考下2016-09-09Android應(yīng)用開(kāi)發(fā)中Fragment的靜態(tài)加載與動(dòng)態(tài)加載實(shí)例
這篇文章主要介紹了Android應(yīng)用開(kāi)發(fā)中Fragment的靜態(tài)加載與動(dòng)態(tài)加載實(shí)例,例子中包括動(dòng)態(tài)的添加更新以及刪除Fragment等操作,很有借鑒意義,需要的朋友可以參考下2016-02-02