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線(xiàn)程中進(jìn)行工作。
//postInvalidate()可以在非UI線(xiàn)程調(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à)中畫(huà)模式的示例
本篇文章介紹了Android 8.0 中如何實(shí)現(xiàn)視頻通話(huà)的畫(huà)中畫(huà)模式的示例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11
快速解決Android適配底部返回鍵等虛擬鍵盤(pán)的問(wèn)題
今天小編就為大家分享一篇快速解決Android適配底部返回鍵等虛擬鍵盤(pán)的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-07-07
Android Loader詳細(xì)介紹及實(shí)例代碼
這篇文章主要介紹了Android Loader詳細(xì)介紹及實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2016-12-12
Android中監(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-06
Android折疊式Toolbar使用完全解析(CollapsingToolbarLayout)
這篇文章主要為大家詳細(xì)介紹了Android折疊式Toolbar的使用方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-02-02
Android編程實(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-04
Flutter使用?input?chip?標(biāo)簽組件示例詳解
這篇文章主要為大家介紹了Flutter使用?input?chip?標(biāo)簽組件示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10
Android 輕松實(shí)現(xiàn)語(yǔ)音識(shí)別詳解及實(shí)例代碼
這篇文章主要介紹了Android 輕松實(shí)現(xiàn)語(yǔ)音識(shí)別的相關(guān)資料,并附實(shí)例代碼,需要的朋友可以參考下2016-09-09
Android應(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

