热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

安卓学习笔记之自定义ViewGroup

涉及的知识点-ViewGroup的测量与布局-View的测量与布局-滑动冲突的处理-VelocityTracker滑动速率跟踪-Scroller实现弹性滑动-屏幕宽高的获取等实现步

涉及的知识点

- ViewGroup的测量与布局- View的测量与布局- 滑动冲突的处理- VelocityTracker滑动速率跟踪- Scroller实现弹性滑动- 屏幕宽高的获取等



实现步骤

1. 创建MyScrollView继承ViewGroup,实现构造与方法2. 在onMeasure方法中对子View进行测量,同时计算出ViewGroup的宽高,并通过setMeasuredDimension设置3. 在onLayout方法中进行布局4. 在onInterceptTouchEvent方法中进行滑动冲突的处理5. 在onTouchEvent方法中进行滑动事件的处理6. 实现弹性滑动



具体实现


继承ViewGroup

public class MyScrollView extends ViewGroup {/*** 可视为点击事件的距离,视为滑动的临界值*/private int touchSlop;private Scroller mScroller;/*** 屏幕宽度*/private int screenHeight;/*** 滑动速度跟踪类*/private VelocityTracker velocityTracker;/*** 滑动的起始坐标*/private int startX, startY;/*** getScrollY的起始与结束值*/private int startScrollY, endScrollY;public MyScrollView(Context context) {this(context, null);}public MyScrollView(Context context, AttributeSet attrs) {this(context, attrs, 0);}public MyScrollView(Context context, AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);touchSlop = ViewConfiguration.get(context).getScaledTouchSlop();mScroller = new Scroller(context, new OvershootInterpolator()); // 添加InterpolatorscreenHeight = getScreenHeight(context);}

/*** 获取屏幕高*/private int getScreenHeight(Context context) {Display d = ((Activity) context).getWindowManager().getDefaultDisplay();Point outSize = new Point();d.getSize(outSize); // outSize保存着屏幕的宽高return outSize.y;}

onMeasure中进行测量

@Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {super.onMeasure(widthMeasureSpec, heightMeasureSpec);final int measureHeight = measureHeight(heightMeasureSpec); // 通过模式测量的高度final int measureWidth = measureWidth(widthMeasureSpec); // 通过模式测量的宽度int totalHeight = 0; // 实际的总高度int maxWidth = 0; // 实际的最大宽度measureChildren(widthMeasureSpec, heightMeasureSpec); //测量子viewint childCount = getChildCount(); //子view数量for (int i = 0; i int height = child.getMeasuredHeight();int width = child.getMeasuredWidth();totalHeight += height;maxWidth = Math.max(maxWidth, child.getMeasuredWidth() + child.getPaddingLeft() + child.getPaddingRight());}//比较出最大宽度与最大高度,并设置给MyScrollViewsetMeasuredDimension(Math.max(measureWidth, maxWidth), Math.max(measureHeight, totalHeight));}/*** 通过widthMeasureSpec得到的宽度** @param widthMeasureSpec* @return*/private int measureWidth(int widthMeasureSpec) {int mode = MeasureSpec.getMode(widthMeasureSpec);int specSize = MeasureSpec.getSize(widthMeasureSpec);int finalSize = 200; //给定默认值switch (mode) {case MeasureSpec.EXACTLY:finalSize = specSize;break;case MeasureSpec.AT_MOST:finalSize = Math.min(finalSize, specSize);break;}return finalSize;}/*** 通过heightMeasureSpec得到的高度** @param heightMeasureSpec* @return*/private int measureHeight(int heightMeasureSpec) {int mode = MeasureSpec.getMode(heightMeasureSpec);int specSize = MeasureSpec.getSize(heightMeasureSpec);int finalSize = 200;switch (mode) {case MeasureSpec.EXACTLY:finalSize = specSize;break;case MeasureSpec.AT_MOST:finalSize = Math.min(finalSize, specSize);break;}return finalSize;}

onLayout方法中进行布局

@Overrideprotected void onLayout(boolean changed, int l, int t, int r, int b) {int childCount = getChildCount();
// int paddingTop = 0;
// for (int i = 0; i
// View child = getChildAt(i);
// int height = child.getMeasuredHeight();
// int childLeft = l + child.getPaddingLeft();
// child.layout(childLeft, paddingTop, childLeft + child.getMeasuredWidth(), paddingTop + child.getMeasuredHeight());
// paddingTop += height;
// lastChildY = paddingTop;
// }for (int i = 0; i // 这里让每一个child都填充了屏幕
child.layout(l, screenHeight * i, r, screenHeight * (i + 1));}}

onInterceptTouchEvent滑动冲突处理

/*** 滑动冲突处理,* 当Y方向滑动大于X方向滑动距离,并且Y方向滑动距离大于touchSlop时,拦截事件* 其他情况不拦截** @param ev* @return*/@Overridepublic boolean onInterceptTouchEvent(MotionEvent ev) {boolean result = false;switch (ev.getAction()) {case MotionEvent.ACTION_DOWN:startX = (int) ev.getX();startY = (int) ev.getY();break;case MotionEvent.ACTION_MOVE:int dX = (int) (ev.getX() - startX);int dY = (int) (ev.getY() - startY);if (Math.abs(dY) > Math.abs(dX) && Math.abs(dY) > touchSlop) { //y方向大于x方法距离,且y方向滑动大于touchSlopresult = true;} else {result = false;}break;}return result;}

onTouchEvent滑动事件的处理

&#64;Overridepublic boolean onTouchEvent(MotionEvent ev) {switch (ev.getAction()) {case MotionEvent.ACTION_DOWN:if (velocityTracker &#61;&#61; null) {velocityTracker &#61; VelocityTracker.obtain();} else {velocityTracker.clear(); // 重置到初始状态}velocityTracker.addMovement(ev); // You should call this for the initial ACTION_DOWNif (!mScroller.isFinished())mScroller.abortAnimation();startX &#61; (int) ev.getX();startY &#61; (int) ev.getY();startScrollY &#61; getScrollY();break;case MotionEvent.ACTION_MOVE:velocityTracker.addMovement(ev); // Add a user&#39;s movement to the tracker.velocityTracker.computeCurrentVelocity(50); // 计算当前速率&#xff0c;按每50毫秒int dX &#61; (int) (ev.getX() - startX);int dY &#61; (int) (ev.getY() - startY);if (Math.abs(getScrollY()) > screenHeight * (getChildCount() - 1) && dY <0) { // 滑动边界限制return false;}scrollBy(0, -dY);break;case MotionEvent.ACTION_UP:endScrollY &#61; getScrollY();int deltaScrollY &#61; endScrollY - startScrollY;int page &#61; Math.abs(endScrollY / screenHeight); // 当前页if (Math.abs(velocityTracker.getYVelocity()) > 100 && page !&#61; getChildCount() - 1) { //滑动速率大于100if (velocityTracker.getYVelocity() > 100) { //下滑smoothScroll(0, getScrollY(), 0, -getScrollY() &#43; page * screenHeight, 500); // 上一页} else if (-velocityTracker.getYVelocity() > 100) { // 上滑smoothScroll(0, getScrollY(), 0, -getScrollY() &#43; (page &#43; 1) * screenHeight, 500); // 下一页}} else {if (deltaScrollY <0) { // 下拉if (Math.abs(deltaScrollY) 3) { //小于屏幕1/3时&#xff0c;回滚smoothScroll(0, getScrollY(), 0, -deltaScrollY, deltaScrollY * 2);} else { // 自动滑动到下一页smoothScroll(0, getScrollY(), 0, -getScrollY() &#43; page * screenHeight, deltaScrollY * 2);}} else { //上拉if (deltaScrollY > screenHeight / 3) { //大于屏幕1/3时&#xff0c;自动滑动到下一页smoothScroll(0, getScrollY(), 0, -getScrollY() &#43; (page &#43; 1) * screenHeight, deltaScrollY * 2);} else { //回滚smoothScroll(0, getScrollY(), 0, -deltaScrollY, deltaScrollY * 2);}}}break;}startX &#61; (int) ev.getX();startY &#61; (int) ev.getY();return true;}

弹性滑动的实现

/*** 弹性滑动* &#64;param startX* &#64;param startY* &#64;param dX* &#64;param dY* &#64;param duration*/private void smoothScroll(int startX, int startY, int dX, int dY, int duration) {if (Math.abs(duration) > 800) duration &#61; 800;mScroller.startScroll(startX, startY, dX, dY, Math.abs(duration));invalidate();}&#64;Overridepublic void computeScroll() {super.computeScroll();if (mScroller.computeScrollOffset()) {scrollTo(mScroller.getCurrX(), mScroller.getCurrY());invalidate();}}

源码&#xff1a; http://download.csdn.net/detail/qq_28261343/9600431


推荐阅读
  • 本文介绍如何在 Android 中通过代码模拟用户的点击和滑动操作,包括参数说明、事件生成及处理逻辑。详细解析了视图(View)对象、坐标偏移量以及不同类型的滑动方式。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • 本文介绍如何使用 Android 的 Canvas 和 View 组件创建一个简单的绘图板应用程序,支持触摸绘画和保存图片功能。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 本文详细介绍了Akka中的BackoffSupervisor机制,探讨其在处理持久化失败和Actor重启时的应用。通过具体示例,展示了如何配置和使用BackoffSupervisor以实现更细粒度的异常处理。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 本文介绍如何使用布局文件在Android应用中排列多行TextView和Button,使其占据屏幕的特定比例,并提供示例代码以帮助理解和实现。 ... [详细]
  • 在现代Web应用中,当用户滚动到页面底部时,自动加载更多内容的功能变得越来越普遍。这种无刷新加载技术不仅提升了用户体验,还优化了页面性能。本文将探讨如何实现这一功能,并介绍一些实际应用案例。 ... [详细]
  • 本文详细介绍了Java中org.neo4j.helpers.collection.Iterators.single()方法的功能、使用场景及代码示例,帮助开发者更好地理解和应用该方法。 ... [详细]
  • Explore how Matterverse is redefining the metaverse experience, creating immersive and meaningful virtual environments that foster genuine connections and economic opportunities. ... [详细]
  • 深入理解Java中的volatile、内存屏障与CPU指令
    本文详细探讨了Java中volatile关键字的作用机制,以及其与内存屏障和CPU指令之间的关系。通过具体示例和专业解析,帮助读者更好地理解多线程编程中的同步问题。 ... [详细]
  • 本文详细介绍了Java中org.eclipse.ui.forms.widgets.ExpandableComposite类的addExpansionListener()方法,并提供了多个实际代码示例,帮助开发者更好地理解和使用该方法。这些示例来源于多个知名开源项目,具有很高的参考价值。 ... [详细]
  • Android 渐变圆环加载控件实现
    本文介绍了如何在 Android 中创建一个自定义的渐变圆环加载控件,该控件已在多个知名应用中使用。我们将详细探讨其工作原理和实现方法。 ... [详细]
author-avatar
手机用户2502887521
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有