Android自定义viewgroup 使用adapter适配数据(6)

在自定义viewgroup(5):http://www.jb51.net/article/100639.htm的基础上,添加使用adapter来适配数据,这样更加的方便,这里只是使用adapter适配数据,不能更新。

package com.example.libingyuan.horizontallistview.ScrollViewGroup; import android.content.Context; import android.util.AttributeSet; import android.util.DisplayMetrics; import android.util.Log; import android.view.GestureDetector; import android.view.MotionEvent; import android.view.View; import android.view.ViewGroup; import android.view.WindowManager; import android.widget.BaseAdapter; import android.widget.GridView; import android.widget.Scroller; /** * 自定义ViewGroup(横向滚动) */ public class ScrollViewGroup extends ViewGroup { //滚动计算辅助类 private Scroller mScroller; //屏幕宽度 private int screenWidth; //可以移动的最大距离 private int mMaxDistance; //自定义手势监听类 private ScrollTouchLisener mTouchLisener; //手势监听 private GestureDetector mDetector; private BaseAdapter mAdapter; /** * 使用new关键字创建对象的时候调用 */ public ScrollViewGroup(Context context) { this(context, null); } /** * 在XML文件中使用的时候调用 */ public ScrollViewGroup(Context context, AttributeSet attrs) { this(context, attrs, 0); } /** * 在xml文件中调用,并且使用了自定义属性的时候调用 */ public ScrollViewGroup(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); init(context); } /** * 初始化方法 * 初始化滚动辅助类Scroller以及计算出屏幕宽度 */ 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; //手势指示器初始化 mTouchLisener = new ScrollTouchLisener(); mDetector = new GestureDetector(context, mTouchLisener); } /** * 滚动时需要重写的方法,用于控制滚动 */ @Override public void computeScroll() { //判断滚动时候停止 if (mScroller.computeScrollOffset()) { //滚动到指定的位置 scrollTo(mScroller.getCurrX(), mScroller.getCurrY()); //这句话必须写,否则不能实时刷新 invalidate(); } } /** * 手指触屏事件监听 */ @Override public boolean onTouchEvent(MotionEvent event) { mDetector.onTouchEvent(event); if (event.getAction() == MotionEvent.ACTION_UP) { this.onUp(event); } return true; } public void setAdapter(BaseAdapter adapter) { this.mAdapter = adapter; requestLayout(); } /* *测量方法,测量父布局的宽度和高度 */ @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { removeAllViews(); for (int i = 0; i < mAdapter.getCount(); i++) { View child = mAdapter.getView(i, null, null); addView(child); } //重新设置宽高 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); // LayoutParams lp = child.getLayoutParams(); /* MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams(); int childWidth = child.getMeasuredWidth() + lp.leftMargin + lp.rightMargin; width += childWidth;*/ width += child.getMeasuredWidth(); } 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 += child.getMeasuredHeight(); } 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;*/ childWidth = child.getMeasuredWidth(); child.layout(childLeft, 0, childLeft + childWidth, height); childLayout(child); childLeft += childWidth; } } /** *让子View的子View和子View大小一样 */ private void childLayout(View child){ if (child==null){ throw new IllegalStateException("ScrollViewGroup must has one child"); } if (child instanceof ViewGroup){ if(((ViewGroup)child).getChildCount()>0) throw new IllegalStateException("view can host only one direct child"); ((ViewGroup) child).getChildAt(0).layout(0,0,child.getWidth(),child.getHeight()); } } @Override public LayoutParams generateLayoutParams(AttributeSet attrs) { return new MarginLayoutParams(getContext(), attrs); } /* *按下事件 ACTION_DOWN */ public boolean onDown(MotionEvent e) { //得到最后一个子View View lastChild = getChildAt(getChildCount() - 1); //获取滑动的最大滑动距离(最后一个Child的右边框的坐标减去屏幕的宽度) int finalyChild = (int) (lastChild.getX() + lastChild.getWidth() - screenWidth); mMaxDistance = finalyChild; //如果停止滚动则取消动画(即手指按下就停止滚动) if (!mScroller.isFinished()) { mScroller.abortAnimation(); } return false; } /* *抬起事件 ACTION_UP */ public boolean onUp(MotionEvent e) { //如果滑动的距离小于第一个控件的最左边(0)则回弹至(0,0)点 if (getScrollX() <= 0) { scrollTo(0, 0); } //如果滑动的距离大于最大可滑动距离则滑动到最后一个子View if (getScrollX() >= mMaxDistance) { scrollTo(mMaxDistance, 0); } //刷新界面 invalidate(); return false; } /* *ACTION_DOWN 、短按不移动 */ public void onShowPress(MotionEvent e) { } /* *短按ACTION_DOWN、ACTION_UP */ public boolean onSingleTapUp(MotionEvent e) { return false; } /* *ACTION_DOWN 、慢滑动 */ public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { //滚动 scrollBy((int) distanceX, 0); return false; } // ACTION_DOWN 、长按不滑动 public void onLongPress(MotionEvent e) { } /* *ACTION_DOWN 、快滑动、 ACTION_UP */ public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) { mScroller.fling(getScrollX(), 0, (int) -velocityX, 0, 0, mMaxDistance, 0, 0); return false; } /** * 自定义手势监听类 */ private class ScrollTouchLisener implements GestureDetector.OnGestureListener { //按下事件 @Override public boolean onDown(MotionEvent e) { return ScrollViewGroup.this.onDown(e); } //单击事件 @Override public void onShowPress(MotionEvent e) { ScrollViewGroup.this.onShowPress(e); } //手指抬起事件 @Override public boolean onSingleTapUp(MotionEvent e) { return ScrollViewGroup.this.onSingleTapUp(e); } //滚动事件 @Override public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { return ScrollViewGroup.this.onScroll(e1, e2, distanceX, distanceY); } //长按事件 @Override public void onLongPress(MotionEvent e) { ScrollViewGroup.this.onLongPress(e); } //滑动事件 @Override public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) { return ScrollViewGroup.this.onFling(e1, e2, velocityX, velocityY); } } }

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

时间: 2024-10-01 05:08:38

Android自定义viewgroup 使用adapter适配数据(6)的相关文章

Android自定义viewgroup 使用adapter适配数据(6)_Android

在自定义viewgroup(5):http://www.jb51.net/article/100639.htm的基础上,添加使用adapter来适配数据,这样更加的方便,这里只是使用adapter适配数据,不能更新. package com.example.libingyuan.horizontallistview.ScrollViewGroup; import android.content.Context; import android.util.AttributeSet; import a

Android自定义ViewGroup实现标签流容器FlowLayout_Android

本篇文章讲的是Android 自定义ViewGroup之实现标签流式布局-FlowLayout,开发中我们会经常需要实现类似于热门标签等自动换行的流式布局的功能,网上也有很多这样的FlowLayout,但不影响我对其的学习.和往常一样,主要还是想总结一下自定义ViewGroup的开发过程以及一些需要注意的地方. 按照惯例,我们先来看看效果图 一.写代码之前,有几个是问题是我们先要弄清楚的: 1.什么是ViewGroup:从名字上来看,它可以被翻译为控件组,言外之意是ViewGroup内部包含了许

viewgroup 传参-android自定义ViewGroup的问题

问题描述 android自定义ViewGroup的问题 在名为Demo的activity中用到了一个继承ViewGroup的类MyView来布局,具体是这样的 在Demo的布局文件xml中, 在Demo的代码中 MyView scroll = (MyView) findViewById(R.id.view1); 在MyView中的构造函数 public MyView(Context context, AttributeSet attrs) { //各类操作 } 问题是Demo需要给scroll传

Android自定义ViewGroup打造各种风格的SlidingMenu_Android

上篇给大家介绍QQ5.0侧滑菜单的视频课程,对于侧滑的时的动画效果的实现有了新的认识,似乎打通了任督二脉,目前可以实现任意效果的侧滑菜单了,感谢鸿洋大大!! 用的是HorizontalScrollView来实现的侧滑菜单功能,HorizontalScrollView的好处是为我们解决了滑动功能,处理了滑动冲突问题,让我们使用起来非常方便,但是滑动和冲突处理都是android中的难点,是我们应该掌握的知识点,掌握了这些,我们可以不依赖于系统的API,随心所欲打造我们想要的效果,因此这篇文章我将直接

Android自定义ViewGroup实现绚丽的仿支付宝咻一咻雷达脉冲效果_Android

去年春节的时候支付宝推行的集福娃活动着实火的不能再火了,更给力的是春晚又可以全民参与咻一咻集福娃活动,集齐五福就可平分亿元大红包,只可惜没有敬业福--那时候在家没事写了个咻一咻插件,只要到了咻一咻的时间点插件就可以自动的点击咻一咻来咻红包,当时只是纯粹练习这部分技术代码没有公开,后续计划写篇关于插件这方面的文章,扯远了(*^__^*) --我们知道在支付宝的咻一咻页面有个雷达扩散的动画效果,当时感觉动画效果非常棒,于是私下尝试着实现了类似的效果,后来在github发现有大神也写有类似效果,于是读

Android自定义ViewGroup打造各种风格的SlidingMenu

上篇给大家介绍QQ5.0侧滑菜单的视频课程,对于侧滑的时的动画效果的实现有了新的认识,似乎打通了任督二脉,目前可以实现任意效果的侧滑菜单了,感谢鸿洋大大!! 用的是HorizontalScrollView来实现的侧滑菜单功能,HorizontalScrollView的好处是为我们解决了滑动功能,处理了滑动冲突问题,让我们使用起来非常方便,但是滑动和冲突处理都是android中的难点,是我们应该掌握的知识点,掌握了这些,我们可以不依赖于系统的API,随心所欲打造我们想要的效果,因此这篇文章我将直接

Android自定义ViewGroup横向布局(1)_Android

最近学习自定义viewgroup,我的目标是做一个可以很想滚动的listview,使用adapter填充数据,并且使用adapter.notifyDataSetChanged()更新数据. 不过一口吃不成一个胖子(我吃成这样可是好几年的积累下来的~~~~),我们一步一步来,这篇笔记首先写一个横向的布局. 代码: package com.example.libingyuan.horizontallistview.ScrollViewGroup; import android.content.Con

Android自定义ViewGroup横向布局(1)

最近学习自定义viewgroup,我的目标是做一个可以很想滚动的listview,使用adapter填充数据,并且使用adapter.notifyDataSetChanged()更新数据. 不过一口吃不成一个胖子(我吃成这样可是好几年的积累下来的~~~~),我们一步一步来,这篇笔记首先写一个横向的布局. 代码: package com.example.libingyuan.horizontallistview.ScrollViewGroup; import android.content.Con

Android自定义ViewGroup实现带箭头的圆角矩形菜单_Android

本文和大家一起做一个带箭头的圆角矩形菜单,大概长下面这个样子:  要求顶上的箭头要对准菜单锚点,菜单项按压反色,菜单背景色和按压色可配置. 最简单的做法就是让UX给个三角形的图片往上一贴,但是转念一想这样是不是太low了点,而且不同分辨率也不太好适配,干脆自定义一个ViewGroup吧! 自定义ViewGroup其实很简单,基本都是按一定的套路来的.  一.定义一个attrs.xml就是声明一下你的这个自定义View有哪些可配置的属性,将来使用的时候可以自由配置.这里声明了7个属性,分别是:箭头