nestedscrollview嵌套recyclerview,android computescroll_Android Scroller與computeScr

 2023-10-06 阅读 25 评论 0

摘要:Android ViewGroup中的Scroller與computeScroll的有什么關系?nestedscrollview嵌套recyclerview、答:沒有直接的關系知道了答案,是不是意味著下文就沒必要看了,如果說對ViewGroup自定義控件不感興趣,可以不用看了。1.Scroller到底是什么&#

Android ViewGroup中的Scroller與computeScroll的有什么關系?

nestedscrollview嵌套recyclerview、答:沒有直接的關系

知道了答案,是不是意味著下文就沒必要看了,如果說對ViewGroup自定義控件不感興趣,可以不用看了。

1.Scroller到底是什么?

答:Scroller只是個計算器,提供插值計算,讓滾動過程具有動畫屬性,但它并不是UI,也不是輔助UI滑動,反而是單純地為滑動提供計算。

無論從構造方法還是其他方法,以及Scroller的屬性可知,其并不會持有View,輔助ViewGroup滑動

2.Scroller只是提供計算,那誰來調用computeScroll使得ViewGroup滑動

答:computeScroll也不是來讓ViewGroup滑動的,真正讓ViewGroup滑動的是scrollTo,scrollBy。computeScroll的作用是計算ViewGroup如何滑動。而computeScroll是通過draw來調用的。

3.computeScroll和Scroller都是計算,兩者有啥關系?

答:文章開始已作答,沒有直接的關系。computeScroll和Scroller要是飛得拉關系的話,那就是computeScroll可以參考Scroller計算結果來影響scrollTo,scrollBy,從而使得滑動發生改變。也就是Scroller不會調用computeScroll,反而是computeScroll調用Scroller。

4.滑動時連續的,如何讓Scroller的計算也是連續的?

這個就問到了什么時候調用computeScroll了,如上所說computeScroll調用Scroller,只要computeScroll調用連續,Scroller也會連續,實質上computeScroll的連續性又invalidate方法控制,scrollTo,scrollBy都會調用invalidate,而invalidate回去觸發draw,從而computeScroll被連續調用,綜上,Scroller也會被連續調用,除非invalidate停止調用。

5.computeScroll如何和Scroller的調用過程保持一致。

computeScroll參考Scroller影響scrollTo,scrollBy,實質上,為了不重復影響scrollTo,scrollBy,那么Scroller必須終止計算currX,currY。要知道計算有沒有終止,需要通過mScroller.computeScrollOffset()

6.如上問題應該說的很清楚了吧,如果不明白,請留言。

7.通過一個SlidePanel的例子,我們來深刻的了解一下

注意:在移動平臺中,要明確知道“滑動”與“滾動”的不同,具體來說,滑動和滾動的方向總是相反的。

public class SlidingPanel extends RelativeLayout {

private Context context;

private FrameLayout leftMenu;

private FrameLayout middleMenu;

private FrameLayout rightMenu;

private FrameLayout middleMask;

private Scroller mScroller;

public? final int LEFT_ID = 0xaabbcc;

public? final int MIDEELE_ID = 0xaaccbb;

public? final int RIGHT_ID = 0xccbbaa;

private boolean isSlideCompete;

private boolean isHorizontalScroll;

private Point point = new Point();

private static final int SLIDE_SLOP = 20;

public SlidingPanel(Context context) {

super(context);

initView(context);

}

public SlidingPanel(Context context, AttributeSet attrs) {

super(context, attrs);

initView(context);

}

private void initView(Context context) {

this.context = context;

mScroller = new Scroller(context, new DecelerateInterpolator());

leftMenu = new FrameLayout(context);

middleMenu = new FrameLayout(context);

rightMenu = new FrameLayout(context);

middleMask = new FrameLayout(context);

leftMenu.setBackgroundColor(Color.RED);

middleMenu.setBackgroundColor(Color.GREEN);

rightMenu.setBackgroundColor(Color.RED);

middleMask.setBackgroundColor(0x88000000);

addView(leftMenu);

addView(middleMenu);

addView(rightMenu);

addView(middleMask);

middleMask.setAlpha(0);

}

public float onMiddleMask(){

return middleMask.getAlpha();

}

@Override

public void scrollTo(int x, int y) {

super.scrollTo(x, y);

onMiddleMask();

// Log.e("getScrollX","getScrollX="+getScrollX());//可以是負值

int curX = Math.abs(getScrollX());

float scale = curX/(float)leftMenu.getMeasuredWidth();

middleMask.setAlpha(scale);

}

@Override

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

super.onMeasure(widthMeasureSpec, heightMeasureSpec);

middleMenu.measure(widthMeasureSpec, heightMeasureSpec);

middleMask.measure(widthMeasureSpec, heightMeasureSpec);

int realWidth = MeasureSpec.getSize(widthMeasureSpec);

int tempWidthMeasure = MeasureSpec.makeMeasureSpec(

(int) (realWidth * 0.8f), MeasureSpec.EXACTLY);

leftMenu.measure(tempWidthMeasure, heightMeasureSpec);

rightMenu.measure(tempWidthMeasure, heightMeasureSpec);

}

@Override

protected void onLayout(boolean changed, int l, int t, int r, int b) {

super.onLayout(changed, l, t, r, b);

middleMenu.layout(l, t, r, b);

middleMask.layout(l, t, r, b);

leftMenu.layout(l - leftMenu.getMeasuredWidth(), t, r, b);

rightMenu.layout(

l + middleMenu.getMeasuredWidth(),

t,

l + middleMenu.getMeasuredWidth()

+ rightMenu.getMeasuredWidth(), b);

}

@Override

public boolean dispatchTouchEvent(MotionEvent ev) {

if (!isSlideCompete) {

handleSlideEvent(ev);

return true;

}

if (isHorizontalScroll) {

switch (ev.getActionMasked()) {

case MotionEvent.ACTION_MOVE:

int curScrollX = getScrollX();

int dis_x = (int) (ev.getX() - point.x);

//滑動方向和滾動滾動條方向相反,因此dis_x必須取負值

int expectX = -dis_x + curScrollX;

if(dis_x>0)

{

Log.d("I","Right-Slide,Left-Scroll");//向右滑動,向左滾動

}else{

Log.d("I","Left-Slide,Right-Scroll");

}

Log.e("I","ScrollX="+curScrollX+" , X="+ev.getX()+" , dis_x="+dis_x);

//規定expectX的變化范圍

int? ? finalX = Math.max(-leftMenu.getMeasuredWidth(),Math.min(expectX, rightMenu.getMeasuredWidth()));

scrollTo(finalX, 0);

point.x = (int) ev.getX();//更新,保證滑動平滑

break;

case MotionEvent.ACTION_UP:

case MotionEvent.ACTION_CANCEL:

curScrollX = getScrollX();

if (Math.abs(curScrollX) > leftMenu.getMeasuredWidth() >> 1) {

if (curScrollX < 0) {

mScroller.startScroll(curScrollX, 0,

-leftMenu.getMeasuredWidth() - curScrollX, 0,

200);

} else {

mScroller.startScroll(curScrollX, 0,

leftMenu.getMeasuredWidth() - curScrollX, 0,

200);

}

} else {

mScroller.startScroll(curScrollX, 0, -curScrollX, 0, 200);

}

invalidate();

isHorizontalScroll = false;

isSlideCompete = false;

break;

}

} else {

switch (ev.getActionMasked()) {

case MotionEvent.ACTION_UP:

isHorizontalScroll = false;

isSlideCompete = false;

break;

default:

break;

}

}

return super.dispatchTouchEvent(ev);

}

/**

* 通過invalidate操縱,此方法通過draw方法調用

*/

@Override

public void computeScroll() {

super.computeScroll();

if (!mScroller.computeScrollOffset()) {

//計算currX,currY,并檢測是否已完成“滾動”

return;

}

int tempX = mScroller.getCurrX();

scrollTo(tempX, 0); //會重復調用invalidate

}

private void handleSlideEvent(MotionEvent ev) {

switch (ev.getAction()&MotionEvent.ACTION_MASK) {

case MotionEvent.ACTION_DOWN:

point.x = (int) ev.getX();

point.y = (int) ev.getY();

super.dispatchTouchEvent(ev);

break;

case MotionEvent.ACTION_MOVE:

int dX = Math.abs((int) ev.getX() - point.x);

int dY = Math.abs((int) ev.getY() - point.y);

if (dX >= SLIDE_SLOP && dX > dY) { // 左右滑動

isHorizontalScroll = true;

isSlideCompete = true;

point.x = (int) ev.getX();

point.y = (int) ev.getY();

} else if (dY >= SLIDE_SLOP && dY > dX) { // 上下滑動

isHorizontalScroll = false;

isSlideCompete = true;

point.x = (int) ev.getX();

point.y = (int) ev.getY();

}

break;

case MotionEvent.ACTION_UP:

case MotionEvent.ACTION_OUTSIDE:

case MotionEvent.ACTION_CANCEL:

super.dispatchTouchEvent(ev);

isHorizontalScroll = false;

isSlideCompete = false;

break;

}

}

}

版权声明:本站所有资料均为网友推荐收集整理而来,仅供学习和研究交流使用。

原文链接:https://hbdhgg.com/4/121214.html

发表评论:

本站为非赢利网站,部分文章来源或改编自互联网及其他公众平台,主要目的在于分享信息,版权归原作者所有,内容仅供读者参考,如有侵权请联系我们删除!

Copyright © 2022 匯編語言學習筆記 Inc. 保留所有权利。

底部版权信息