Android自定義ViewGroup實現(xiàn)彈性滑動效果
自定義View實現(xiàn)一個彈性滑動的效果,供大家參考,具體內(nèi)容如下


實現(xiàn)原理
onMeasure()中測量所有子View
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
// 測量所有子View
int count = getChildCount();
for (int i = 0; i < count; i++) {
View childView = getChildAt(i);
measureChild(childView, widthMeasureSpec, heightMeasureSpec);
}
setMeasuredDimension(widthMeasureSpec, heightMeasureSpec);
}
onLayout()中,將所有的子View按照位置依次往下排列
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
// 設(shè)置ViewGroup的高度,對所有子View進(jìn)行排列
int childCount = getChildCount();
MarginLayoutParams params = (MarginLayoutParams) getLayoutParams();
params.height = mScreenHeight * childCount;
for (int i = 0; i < childCount; i++) {
View childView = getChildAt(i);
if (childView.getVisibility() != View.GONE) {
// 給每個ChildView放置在指定位置
childView.layout(l, i * mScreenHeight, r, (i + 1) * mScreenHeight);
}
}
}
onTouchEvent()中處理滑動
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
mLastY = (int) event.getY();
mStart = getScrollY();
return true;
case MotionEvent.ACTION_MOVE:
if (!mScroller.isFinished()) {
// 終止滑動
mScroller.abortAnimation();
}
int offsetY = (int) (mLastY - event.getY());
Log.d(TAG, "onTouchEvent: getScrollY: " + getScrollY());
Log.d(TAG, "onTouchEvent: offsetY " + offsetY);
// 到達(dá)頂部,使用offset判斷方向
if (getScrollY() + offsetY < 0) { // 當(dāng)前已經(jīng)滑動的 Y 位置
offsetY = 0;
}
// 到達(dá)底部
if (getScrollY() > getHeight() - mScreenHeight && offsetY > 0) {
offsetY = 0;
}
scrollBy(0, offsetY);
// 滑動完成后,重新設(shè)置LastY位置
mLastY = (int) event.getY();
break;
case MotionEvent.ACTION_UP:
mEnd = getScrollY();
int distance = mEnd - mStart;
if (distance > 0) { // 向上滑動
if (distance < mScreenHeight / 3) {
Log.d(TAG, "onTouchEvent: distance < screen/3");
// 回到原來位置
mScroller.startScroll(0, getScrollY(), 0, -distance);
} else {
// 滾到屏幕的剩余位置
mScroller.startScroll(0, getScrollY(), 0, mScreenHeight - distance);
}
} else { // 向下滑動
if (-distance < mScreenHeight / 3) {
mScroller.startScroll(0, getScrollY(), 0, -distance);
} else {
mScroller.startScroll(0, getScrollY(), 0, -mScreenHeight - distance);
}
}
postInvalidate();
}
return super.onTouchEvent(event);
}
其中ACTION_UP這段代碼是處理彈性滑動的
case MotionEvent.ACTION_UP:
mEnd = getScrollY();
int distance = mEnd - mStart;
if (distance > 0) { // 向上滑動
if (distance < mScreenHeight / 3) {
Log.d(TAG, "onTouchEvent: distance < screen/3");
// 回到原來位置
mScroller.startScroll(0, getScrollY(), 0, -distance);
} else {
// 滾到屏幕的剩余位置
mScroller.startScroll(0, getScrollY(), 0, mScreenHeight - distance);
}
} else { // 向下滑動
if (-distance < mScreenHeight / 3) {
mScroller.startScroll(0, getScrollY(), 0, -distance);
} else {
mScroller.startScroll(0, getScrollY(), 0, -mScreenHeight - distance);
}
}
postInvalidate();
完整代碼
public class ScrollViewGroup extends ViewGroup {
private static final String TAG = "ScrollView";
private Scroller mScroller;
private int mScreenHeight; // 窗口高度
private int mLastY;
private int mStart;
private int mEnd;
public ScrollViewGroup(Context context) {
this(context, null);
}
public ScrollViewGroup(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public ScrollViewGroup(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
mScroller = new Scroller(context);
// 獲取屏幕高度
WindowManager windowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
DisplayMetrics metrics = new DisplayMetrics();
windowManager.getDefaultDisplay().getMetrics(metrics);
mScreenHeight = metrics.heightPixels;
Log.d(TAG, "ScrollViewGroup: ScreenHeight " + mScreenHeight);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
// 測量所有子View
int count = getChildCount();
for (int i = 0; i < count; i++) {
View childView = getChildAt(i);
measureChild(childView, widthMeasureSpec, heightMeasureSpec);
}
setMeasuredDimension(widthMeasureSpec, heightMeasureSpec);
}
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
// 設(shè)置ViewGroup的高度,對所有子View進(jìn)行排列
int childCount = getChildCount();
MarginLayoutParams params = (MarginLayoutParams) getLayoutParams();
params.height = mScreenHeight * childCount;
for (int i = 0; i < childCount; i++) {
View childView = getChildAt(i);
if (childView.getVisibility() != View.GONE) {
// 給每個ChildView放置在指定位置
childView.layout(l, i * mScreenHeight, r, (i + 1) * mScreenHeight);
}
}
}
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
mLastY = (int) event.getY();
mStart = getScrollY();
return true;
case MotionEvent.ACTION_MOVE:
if (!mScroller.isFinished()) {
// 終止滑動
mScroller.abortAnimation();
}
int offsetY = (int) (mLastY - event.getY());
Log.d(TAG, "onTouchEvent: getScrollY: " + getScrollY());
Log.d(TAG, "onTouchEvent: offsetY " + offsetY);
// 到達(dá)頂部,使用offset判斷方向
if (getScrollY() + offsetY < 0) { // 當(dāng)前已經(jīng)滑動的 Y 位置
offsetY = 0;
}
// 到達(dá)底部
if (getScrollY() > getHeight() - mScreenHeight && offsetY > 0) {
offsetY = 0;
}
scrollBy(0, offsetY);
// 滑動完成后,重新設(shè)置LastY位置
mLastY = (int) event.getY();
break;
case MotionEvent.ACTION_UP:
mEnd = getScrollY();
int distance = mEnd - mStart;
if (distance > 0) { // 向上滑動
if (distance < mScreenHeight / 3) {
Log.d(TAG, "onTouchEvent: distance < screen/3");
// 回到原來位置
mScroller.startScroll(0, getScrollY(), 0, -distance);
} else {
// 滾到屏幕的剩余位置
mScroller.startScroll(0, getScrollY(), 0, mScreenHeight - distance);
}
} else { // 向下滑動
if (-distance < mScreenHeight / 3) {
mScroller.startScroll(0, getScrollY(), 0, -distance);
} else {
mScroller.startScroll(0, getScrollY(), 0, -mScreenHeight - distance);
}
}
postInvalidate();
}
return super.onTouchEvent(event);
}
@Override
public void computeScroll() {
if (mScroller.computeScrollOffset()) {
scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
postInvalidate();
}
}
}
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- android開發(fā)通過Scroller實現(xiàn)過渡滑動效果操作示例
- Android使用Scroller實現(xiàn)彈性滑動效果
- Android自定義View彈性滑動Scroller詳解
- Android用Scroller實現(xiàn)一個可向上滑動的底部導(dǎo)航欄
- 詳解Android應(yīng)用開發(fā)中Scroller類的屏幕滑動功能運用
- Android Scroll實現(xiàn)彈性滑動_列表下拉彈性滑動的示例代碼
- android自定義ViewPager水平滑動彈性效果
- Android使用Handler實現(xiàn)View彈性滑動
- Android?Scroller實現(xiàn)彈性滑動效果
相關(guān)文章
Android?Jetpack組件Navigation導(dǎo)航組件的基本使用
本篇主要簡單介紹了一下?Navigation?是什么?以及使用它的流程是什么,并且結(jié)合實際案例?操作了一番,Navigation?還有很多其他用法,如條件導(dǎo)航、嵌套圖、過度動畫?等等功能?有機(jī)會再操作,需要的朋友可以參考下2022-06-06
務(wù)必掌握的Android十六進(jìn)制狀態(tài)管理最佳實踐
這篇文章主要為大家介紹了務(wù)必掌握的Android十六進(jìn)制狀態(tài)管理最佳實踐,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09
36個Android開發(fā)常用經(jīng)典代碼大全
本篇文章主要介紹了36個Android開發(fā)常用經(jīng)典代碼片段,都是實用的代碼段,具有一定的參考價值,感興趣的小伙伴們可以參考一下。2016-11-11
android 監(jiān)聽網(wǎng)絡(luò)狀態(tài)的變化及實戰(zhàn)的示例代碼
本篇文章主要介紹了android 監(jiān)聽網(wǎng)絡(luò)狀態(tài)的變化及實戰(zhàn)的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-01-01
Android 配置gradle實現(xiàn)VersionCode自增實例
今天小編就為大家分享一篇Android 配置gradle實現(xiàn)VersionCode自增實例,具有很好的 參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-03-03
Ubantu16.04進(jìn)行Android 8.0源碼編譯的流程
這篇文章主要介紹了Ubantu16.04進(jìn)行Android 8.0源碼編譯的相關(guān)資料,需要的朋友可以參考下2018-02-02

