Android自定义ViewGroup实现受边界限制的滚动操作(3)

上一篇文章《自定义viewgroup(2)》地址:http://www.jb51.net/article/100610.htm

代码

package com.example.libingyuan.horizontallistview.ScrollViewGroup;

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;

/**
 * 自定义ViewGroup
 * 在滚动的基础上,增加了边界限制
 */
public class ScrollViewGroup extends ViewGroup {
 //滚动计算辅助类
 private Scroller mScroller;
 //手指落点的X坐标
 private float mLastMotionX = 0;
 //屏幕宽度
 private int screenWidth;

 /**
  * 使用new关键字创建对象的时候调用
  * @param context 上下文
  */
 public ScrollViewGroup(Context context) {
  this(context, null);
 }
 /**
  * 在XML文件中使用的时候调用
  * @param context 上下文
  * @param attrs 属性:如 android:layout_width="wrap_content"
  */
 public ScrollViewGroup(Context context, AttributeSet attrs) {
  this(context, attrs, 0);
 }

 /**
  * 在xml文件中调用,并且使用了自定义属性的时候调用
  * @param context 上下文
  * @param attrs 属性:如 android:layout_width="wrap_content"
  * @param defStyleAttr 自定义属性的id
  */
 public ScrollViewGroup(Context context, AttributeSet attrs, int defStyleAttr) {
  super(context, attrs, defStyleAttr);
  init(context);
 }

 /**
  * 初始化方法
  * 初始化滚动辅助类Scroller以及计算出屏幕宽度
  * @param context
  */
 private void init(Context context) {
  mScroller = new Scroller(context);
  WindowManager manager = (WindowManager) context
    .getSystemService(Context.WINDOW_SERVICE);
  DisplayMetrics outMetrics = new DisplayMetrics();
  manager.getDefaultDisplay().getMetrics(outMetrics);
  screenWidth = outMetrics.widthPixels;
 }

 /**
  * 滚动时需要重写的方法,用于控制滚动
  */
 @Override
 public void computeScroll() {
  //判断滚动时候停止
  if (mScroller.computeScrollOffset()) {
   //滚动到指定的位置
   scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
   //这句话必须写,否则不能实时刷新
   postInvalidate();
  }
 }

 /**
  * 手指触屏事件监听
  * @param event
  * @return
  */
 @Override
 public boolean onTouchEvent(MotionEvent event) {
  // TODO Auto-generated method stub
  int action = event.getAction();
  float x = event.getX();
  switch (action) {
   case MotionEvent.ACTION_DOWN:
    if (!mScroller.isFinished()) {
     mScroller.abortAnimation();
    }
    mLastMotionX = event.getX();
    break;
   case MotionEvent.ACTION_MOVE:
    float delt = mLastMotionX - x;
    mLastMotionX = x;
    scrollBy((int) delt, 0);
    break;
   case MotionEvent.ACTION_UP:
    View lastChild=getChildAt(getChildCount()-1);
    int finalyChild= (int) (lastChild.getX()+lastChild.getWidth()-screenWidth);
    if (getScrollX()<0){
     scrollTo(0,0);
    }
    if (getScrollX()>=finalyChild)
     scrollTo(finalyChild,0);
    invalidate();
    break;

   default:
    break;
  }

  return true;
 }

 @Override
 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  //重新设置宽高
  this.setMeasuredDimension(measureWidth(widthMeasureSpec, heightMeasureSpec), measureHeight(widthMeasureSpec, heightMeasureSpec));
 }

  /**
  * 测量宽度
  */
 private int measureWidth(int widthMeasureSpec, int heightMeasureSpec) {
  // 宽度
  int sizeWidth = MeasureSpec.getSize(widthMeasureSpec);
  int modeWidth = MeasureSpec.getMode(widthMeasureSpec);
  //父控件的宽(wrap_content)
  int width = 0;
  int childCount = getChildCount();

  //重新测量子view的宽度,以及最大高度
  for (int i = 0; i < childCount; i++) {
   View child = getChildAt(i);
   measureChild(child, widthMeasureSpec, heightMeasureSpec);
   MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
   int childWidth = child.getMeasuredWidth() + lp.leftMargin + lp.rightMargin;
   width += childWidth;
  }
  return modeWidth == MeasureSpec.EXACTLY ? sizeWidth : width;
 }

 /**
  * 测量高度
  */
 private int measureHeight(int widthMeasureSpec, int heightMeasureSpec) {
  //高度
  int sizeHeight = MeasureSpec.getSize(heightMeasureSpec);
  int modeHeight = MeasureSpec.getMode(heightMeasureSpec);
  //父控件的高(wrap_content)
  int height = 0;
  int childCount = getChildCount();

  //重新测量子view的宽度,以及最大高度
  for (int i = 0; i < childCount; i++) {
   View child = getChildAt(i);
   measureChild(child, widthMeasureSpec, heightMeasureSpec);
   MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
   int childHeight = child.getMeasuredHeight() + lp.topMargin + lp.bottomMargin;
   height += childHeight;
  }
  height = height / childCount;
  return modeHeight == MeasureSpec.EXACTLY ? sizeHeight : height;
 }

 /**
  * 给子布局设定位置
  */
 @Override
 protected void onLayout(boolean changed, int l, int t, int r, int b) {
  int childLeft = 0;//子View左边的间距
  int childWidth;//子View的宽度
  int height = getHeight();//屏幕的宽度
  int childCount = getChildCount();//子View的数量
  for (int i = 0; i < childCount; i++) {
   View child = getChildAt(i);
   MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
   childWidth = child.getMeasuredWidth() + lp.leftMargin + lp.rightMargin;
   child.layout(childLeft, 0, childLeft + childWidth, height);
   childLeft += childWidth;
  }
 }

 @Override
 public LayoutParams generateLayoutParams(AttributeSet attrs) {
  return new MarginLayoutParams(getContext(), attrs);
 }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • android创建手势识别示例代码

    这篇的内容使用到的是android.gesture包,具体的例子参考的是Sample中GestureBuilder程序. 1.手势创建手势创建主要用到GestureOverlayView和GestureLibrary.GestureOverlayView的父类为android.widget.FrameLayout,是手势绘图区.GestureLibrary类主要对手势进行保存.删除等操作的,存放手势的仓库.下面给出创建手势的例子,如下图,可以定义如图手势打开csdn.net 1.1.创建绘图区

  • Android自定义viewgroup可滚动布局 GestureDetector手势监听(5)

    这篇效果和上一篇:http://www.jb51.net/article/100638.htm的效果是一样的,但是不再在OnTouchEvent中写代码,而是使用系统自带的类GestureDetector来监听手势以及滑动事件等等,它内置了滑动,点击,长按等事件,而且有快速滑动,比较方便,比自己写的细节处理要好. 代码: package com.example.libingyuan.horizontallistview.ScrollViewGroup; import android.conten

  • Android App开发中自定义View和ViewGroup的实例教程

    View Android所有的控件都是View或者View的子类,它其实表示的就是屏幕上的一块矩形区域,用一个Rect来表示,left,top表示View相对于它的parent View的起点,width,height表示View自己的宽高,通过这4个字段就能确定View在屏幕上的位置,确定位置后就可以开始绘制View的内容了. View绘制过程 View的绘制可以分为下面三个过程: Measure View会先做一次测量,算出自己需要占用多大的面积.View的Measure过程给我们暴露了一个

  • android使用gesturedetector手势识别示例分享

    复制代码 代码如下: public class MyGestureLintener extends SimpleOnGestureListener {private Context context;public MyGestureLintener(Context context) {    super();    this.context = context;} // 单击,触摸屏按下时立刻触发/*@Overridepublic boolean onDown(MotionEvent e) {  

  • Android自定义ViewGroup的实现方法

    在android中提供了常见的几种ViewGroup的实现,包括LinearLayout.Relativeayout.FrameLayout等.这些ViewGroup可以满足我们一般的开发需求,但是对于界面要求复杂的,这几个布局就显得捉襟见肘了.所以自定义的ViewGroup在我们接触过的应用中比比皆是. 要想实现一个自定义的ViewGroup,第一步是学会自定义属性,这些自定义的属性将让我们配置布局文件的时候更加的灵活.自定义属性是在value目录下声明一个attrs.xml文件. <?xml

  • Android自定义ViewGroup实现可滚动的横向布局(2)

    上一篇文章自定义viewgroup(1)地址:http://www.jb51.net/article/100608.htm 这里直接代码: package com.example.libingyuan.horizontallistview.ScrollViewGroup; import android.content.Context; import android.util.AttributeSet; import android.view.MotionEvent; import android

  • Android自定义ViewGroup之实现FlowLayout流式布局

    整理总结自鸿洋的博客:http://blog.csdn.net/lmj623565791/article/details/38352503/  一.FlowLayout介绍  所谓FlowLayout,就是控件根据ViewGroup的宽,自动的往右添加,如果当前行剩余空间不足,则自动添加到下一行.有点像所有的控件都往左飘的感觉,第一行满了,往第二行飘~所以也叫流式布局.Android并没有提供流式布局,但是某些场合中,流式布局还是非常适合使用的,比如关键字标签,搜索热词列表等,比如下图: git

  • Android通过手势实现的缩放处理实例代码

    网络上传言HTC的HERO-ROM支持多点触摸的论证大多源于浏览网页和图片时,能像IPhone一样通过手势来控制页面的大小.下面的例子是利用现有的API实现HERO浏览图片和网页的缩放功能. 主要原理是onTouchEvent事件中的参数MotionEvent,它有一个getSize()方法.在一个点的时候,该方法永远返回0,而在两个触电的时候,该方法则根据两点相对位置变化而返回不同的值.我们只需计算出两点之间的距离变化,距离的大小表明我们希望目标变化的趋势.而getX()和getY()方法则永

  • android开发之为activity增加左右手势识别示例

    android开发中为activity增加左右手势识别,如右滑关闭当前页面. 复制代码 代码如下: /* *  for左右手势 *  1.复制下面的内容到目标Activity *  2.目标Activity的onCreate()调用initGesture() *  3.目标Activity需implements OnTouchListener, OnGestureListener */   private GestureDetector mGestureDetector;   private i

  • Android应用中实现手势控制图片缩放的完全攻略

    一.概述 现在app中,图片预览功能肯定是少不了的,用户基本已经形成条件反射,看到小图,点击看大图,看到大图两个手指开始进行放大,放大后,开始移动到指定部位~~~ 我相信看图的整个步骤,大家或者说用户应该不希望被打断把~~~"我擦,竟然不能放大,什么玩意,卸了~~" , "我擦,竟然不能移动,留有何用,卸了~~". 哈~所以对于图片的预览,一来,我们要让用户爽:二来,我们作为开发者,也得知道如何实现~~~ 想要做到图片支持多点触控,自由的进行缩放.平移,需要了解几个

随机推荐