這篇文章給大家介紹Android中怎么通過自定義ViewGroup實(shí)現(xiàn)一個(gè)彈性滑動(dòng)效果,內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對(duì)大家能有所幫助。
創(chuàng)新互聯(lián)長期為超過千家客戶提供的網(wǎng)站建設(shè)服務(wù),團(tuán)隊(duì)從業(yè)經(jīng)驗(yàn)10年,關(guān)注不同地域、不同群體,并針對(duì)不同對(duì)象提供差異化的產(chǎn)品和服務(wù);打造開放共贏平臺(tái),與合作伙伴共同營造健康的互聯(lián)網(wǎng)生態(tài)環(huán)境。為烏翠企業(yè)提供專業(yè)的成都網(wǎng)站設(shè)計(jì)、成都網(wǎng)站制作,烏翠網(wǎng)站改版等技術(shù)服務(wù)。擁有十載豐富建站經(jīng)驗(yàn)和眾多成功案例,為您定制開發(fā)。
實(shí)現(xiàn)原理
onMeasure()中測(cè)量所有子View
@Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { // 測(cè)量所有子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的高度,對(duì)所有子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) { // 給每個(gè)ChildView放置在指定位置 childView.layout(l, i * mScreenHeight, r, (i + 1) * mScreenHeight); } } }
onTouchEvent()中處理滑動(dòng)
@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()) { // 終止滑動(dòng) 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)滑動(dòng)的 Y 位置 offsetY = 0; } // 到達(dá)底部 if (getScrollY() > getHeight() - mScreenHeight && offsetY > 0) { offsetY = 0; } scrollBy(0, offsetY); // 滑動(dòng)完成后,重新設(shè)置LastY位置 mLastY = (int) event.getY(); break; case MotionEvent.ACTION_UP: mEnd = getScrollY(); int distance = mEnd - mStart; if (distance > 0) { // 向上滑動(dòng) 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 { // 向下滑動(dòng) 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這段代碼是處理彈性滑動(dòng)的
case MotionEvent.ACTION_UP: mEnd = getScrollY(); int distance = mEnd - mStart; if (distance > 0) { // 向上滑動(dòng) 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 { // 向下滑動(dòng) 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) { // 測(cè)量所有子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的高度,對(duì)所有子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) { // 給每個(gè)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()) { // 終止滑動(dòng) 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)滑動(dòng)的 Y 位置 offsetY = 0; } // 到達(dá)底部 if (getScrollY() > getHeight() - mScreenHeight && offsetY > 0) { offsetY = 0; } scrollBy(0, offsetY); // 滑動(dòng)完成后,重新設(shè)置LastY位置 mLastY = (int) event.getY(); break; case MotionEvent.ACTION_UP: mEnd = getScrollY(); int distance = mEnd - mStart; if (distance > 0) { // 向上滑動(dòng) 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 { // 向下滑動(dòng) 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(); } } }
關(guān)于Android中怎么通過自定義ViewGroup實(shí)現(xiàn)一個(gè)彈性滑動(dòng)效果就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。
文章標(biāo)題:Android中怎么通過自定義ViewGroup實(shí)現(xiàn)一個(gè)彈性滑動(dòng)效果
網(wǎng)頁地址:http://bm7419.com/article28/gijjcp.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供營銷型網(wǎng)站建設(shè)、網(wǎng)站設(shè)計(jì)、手機(jī)網(wǎng)站建設(shè)、云服務(wù)器、電子商務(wù)、虛擬主機(jī)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)