博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
定义你自己ViewGroup
阅读量:6859 次
发布时间:2019-06-26

本文共 4767 字,大约阅读时间需要 15 分钟。

尊重原创

好久都没有写文章了,如今利用周末的时间对一些知识进行总结。便于加深理解,今天我就来学习一下怎样实现自己定义ViewGroup

在Android中比較经常使用的布局LinearLayout,FrameLayout,RelativeLayout。

。。这些布局都是继承自ViewGroup,通过这些布局。我们差点儿能够实现Android中全部的界面开发,可是对于一些比較常见而且比較复杂的布局,使用这些基本布局开发的话,就会花大量的时间在一些反复的工作上。那么我们能不能模仿这些基本布局。依据自己的需求来实现一些自己的布局,以后须要的时候直接拿来用呢?当然是能够的。

相似扑克牌的布局

在斗地主的游戏中,我们会经常遇到相似这种布局。我就将这个布局叫级联布局吧,这个布局使用RelativeLayout和margin等熟悉是能够做出来的,可是不是非常方便,我们今天就这种需求为背景解说一下自己定义ViewGroup

在学习自己定义布局前,读者最好先了解一下Android的布局好似怎么绘画出来的,我推荐大家去了解一下:都是官网的一些文章

1、http://developer.android.com/guide/topics/ui/how-android-draws.html
2、http://developer.android.com/reference/android/view/ViewGroup.html
3、http://developer.android.com/reference/android/view/ViewGroup.LayoutParams.html
通过以上几篇文章,我们须要了解一下知识:
1、绘制布局是由两个过程组成:測量过程和布局过程。測量过程使用measure(int,int)方法完毕,遍历完毕后,全部的View的大小都确定了,布局过程使用layout方法完毕,就是通过View的大小。决定View放置在上面地方,只是通过源代码中measure和layout方法都是final类型的。所以我们是无法改写的,之所以定义成final的。就是避免开发人员破坏了布局的绘画流程,只是測量和布局的细节我们能够通过改写onMeasure和onLayout实现。
2、ViewGroup
在前面我就说过,Android中全部的布局都是继承自ViewGroup,ViewGroup就是一个View的容器,我们能够再里面放置任务的View,至于怎样放置,我们能够通过onMeasure和onLayout来定义,onMeasure在measure调用,onLayout是在layout中调用的
3、ViewGroup.LayoutParams
这个类主要是View用来告诉他的父容器它想怎么显示,如宽、高、居中等等。ViewGroup.LayoutParams里面最重要的两个參数就是width,height。假设你想使用margin属性。那么必须使用ViewGroup.MarginLayoutParams这个类,这个类继承自ViewGroup.LayoutParams。

增加了对margin属性的支持,假设你想增加很多其它的属性,能够自己定义一个LayoutParams类增加你须要的属性。其实LinearLayout等布局都是继承自ViewGroup.MarginLayoutParams,并增加了自己须要的属性。

实现自己的布局CascadeLayout.java

/** * 自己定义布局,用来实现扑克牌效果 * com.myviewgroup.CascadeLayout * @author yuanzeyao 
* create at 2014年10月19日 下午4:15:42 */public class CascadeLayout extends ViewGroup{ /** * 水平偏移距离 */ private int horizontal_space; /** * 垂直偏移距离 */ private int vertical_space; public CascadeLayout(Context context) { super(context); } public CascadeLayout(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); initAttribute(context, attrs); } public CascadeLayout(Context context, AttributeSet attrs) { super(context, attrs); initAttribute(context, attrs); } /** * 依据xml文件里的属性对horizontal_space。vertical_space来赋值 * @param context * @param attrs */ private void initAttribute(Context context,AttributeSet attrs) { TypedArray a=context.obtainStyledAttributes(attrs, R.styleable.CascadeLayout); horizontal_space=a.getDimensionPixelSize(R.styleable.CascadeLayout_horizontal_space,this.getResources().getDimensionPixelSize(R.dimen.cascade_horizontal_spacing)); vertical_space=a.getDimensionPixelSize(R.styleable.CascadeLayout_vertical_space, this.getResources().getDimensionPixelSize(R.dimen.cascade_vertical_spacing)); a.recycle(); } /** * onMeasure在measure中调用,參数各自是CascadeLayout的宽度和高度 */ @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { //使用measureChildren对全部的孩子进行測量,你也能够使用for循环,然后掉哦弄个measurechild函数进行測量 measureChildren(widthMeasureSpec, heightMeasureSpec); //将宽度和高度赋值为CascadeLayout的measureWidth。measureHeight变量 setMeasuredDimension(widthMeasureSpec, heightMeasureSpec); } /** * 在该函数中增加怎样布置各个View的逻辑 */ @Override protected void onLayout(boolean changed, int l, int t, int r, int b) { for(int i=0;i
在CascadeLayout中,我们定义了两个属性,horizontal_space,vertical_space。这两个属性用来记录该布局中每一个子View之前的垂直距离和水平距离。他们都是在构造函数中初始化。因为我们增加了属性。所以在res/values中定义attrs.xml文件

xml version="1.0" encoding="utf-8"?> <resources> <declare-styleable name="CascadeLayout"> <attr name="horizontal_space" format="dimension"></attr> <attr name="vertical_space" format="dimension"></attr> </declare-styleable> </resources>

在layout文件里使用CascadeLayout

执行效果例如以下

自己定义自己的LayoutParams

在CascadeLayout中,我们使用的是ViewGroup.LayoutParams类。可是这个类唯独width和height属性,假设我们想增加其它属性。那么我们须要自己定义LayoutParams类
以下我们自己定义自己的LayoutParams类,这个类继承ViewGroup.MarginLayoutParams,所以我自己定义的LayoutPrams具有margin的属性
在CascadeLayout中定义例如以下LayoutParams类

public static class LayoutParams extends ViewGroup.MarginLayoutParams  {    /**     * 定义在垂直方向的偏移距离,能够覆盖CascadeLayout中的vertical_space属性     */    private int layout_vertical_spacing;    public LayoutParams(Context c, AttributeSet attrs)    {      super(c, attrs);      TypedArray a = c.obtainStyledAttributes(attrs,          R.styleable.CascadeLayout_LayoutParams);      try {        layout_vertical_spacing = a            .getDimensionPixelSize(                R.styleable.CascadeLayout_LayoutParams_layout_vertical_spacing,                -1);      } finally {        a.recycle();      }    }    public LayoutParams(int width, int height)    {      super(width, height);    }   }
该LayoutParams就支持了layout_vertical_spaceing属性了。
好了,先写到这里吧,有什么不懂的欢迎大家留言。

。。

版权声明:本文博客原创文章,博客,未经同意,不得转载。

本文转自mfrbuaa博客园博客,原文链接:http://www.cnblogs.com/mfrbuaa/p/4688139.html,如需转载请自行联系原作者

你可能感兴趣的文章
临时数据库之python用sqlite3模块操作sqlite
查看>>
ODS实现过程
查看>>
CentOS6.5固定IP方式上网(NAT)
查看>>
剑破冰山—Oracle开发艺术 书评(by yangtingkun)
查看>>
如何用手机维护Mysql数据库
查看>>
REACT NATIVE 系列教程之十三】利用LISTVIEW与TEXTINPUT制作聊天/对话框&&获取组件实例常用的两种方式...
查看>>
基于CentOS 5.3平台下搭建PXE部署ESX&ESXi 4.x模板分发服务器 v1.0
查看>>
使用tornado模板引擎配合yaml构建nginx配置接口 [扩展saltstack]
查看>>
网络作者的心声-1、感谢读者,我不会太监
查看>>
WCF分布式开发常见错误解决(1):添加服务引用出错
查看>>
Nginx实战基础篇六 通过源码包编译安装部署LNMP搭建Discuz论坛
查看>>
如何根据指定软件版本制作属于自己的puppet yum源
查看>>
Linux下架设rsync服务器
查看>>
Struts2教程8:拦截器概述
查看>>
windows 屏幕坐标 窗口坐标 客户区坐标 逻辑坐标 设备坐标之间的关系及转换
查看>>
在Foreda8上试安装Apchehttpd-2.4.6.tar.gz
查看>>
基于S3C2410的VIVI移植
查看>>
Entity Framwork one to one problem
查看>>
[转] Attach、Detach和DeleteObject
查看>>
[转] C# 获取程序运行目录
查看>>