AutoLayout——Android屏幕自动适配

不久前看了张鸿洋大神的《Android AutoLayout全新的适配方式 堪称适配终结者》,觉得不错。


拿来用了以后,发现有些问题。


自己觉得可以写一个类似的,于是兴趣大发,动手写出了自己的AutoLayout,感觉更胜一筹。


废话不说,我们一步步开始:


一、吩咐美工UI做的事情

设计图都用一个尺寸(分辨率)设计,例如是1280x720。

所有控件,或要发生交互的地方,标注上:

1、坐标,以设计图的左上角为原点,离左边和顶部的PX距离。

2、宽度和高度,也是PX为单位。

例如下图,这是我项目里的一个页面,尺寸:1280x720


对于列表item,即是RecyclerView、ListView等等里的item,坐标原点是以item的左上角为标准,例如下图:

AutoLayout——Android屏幕自动适配_第1张图片


我这个APP是全屏,如果你的APP要有状态栏,我也建议设计图用全屏,下面第二部分会介绍状态栏的问题。


对于那些需要变化的图片,当然要让美工切出来。


至此,美工的工作就完成了。


你看,我这种项目用上AutoLayout这种自适配方法岂不是很爽。

那些控件放上去就行了,基本不用嵌套。美工也很爽,少切很多图。


二、使用AutoLayout

1、编写xml布局,调出与你设计图相同尺寸的视图:

AutoLayout——Android屏幕自动适配_第2张图片

在此尺寸设置好布局就可以,不用理其他分辨率的。

root布局设上一张背景图,然后把控件放上去,按照美工给的坐标和宽高,设置好leftMargin和topMargin,宽layout_width,高layout_height。

(1)对于那些宽高不定的控件,例如TextView,当然是wrap_content,有些需要match_parent的地方就match_parent。

(2)对于图片ImageView,一定要设置fitXY,因为如果屏幕拉伸了,而你的图片没有拉伸,就对不齐。在你的设计尺寸中,无论ScaleType是哪种类型,效果看起来都是一样的。除非你的ImageView有特殊需求,例如程序要放大缩小图片,那么你另行设置。另外要注意,同上,设置ImageView的宽高值,不要以为图片放上xml布局就宽高适合而疏忽了。

2、在程序代码中:初始化

在你的Activity中设置:

AutoUtils.setSize(this, false, 720, 1280);

//第一个参数this,是Activity对象。

//第二个参数false,是boolean变量,表示没有状态栏。如果你的APP要有状态栏,就设置为true。

//第三个参数720,是int变量,是设计尺寸的宽度。

//第四个参数1280,是int变量,是设计尺寸的高度。


如果你的APP屏幕的显示方式没有变化,你设置一次就可以。

如果有需要,你可以随时重新设置,它是灵活的。例如你有需求切换为横屏播放视频,那么你就把设计图的宽高值换一下传进去,当然你切换回竖屏时要记得重新设置回来。


(1)对于状态栏(StatusBar),如果你需要全屏,第二个参数就设置false;如果你需要有,就设置true。

其实这里有两种方案:第一种是美工的设计图都是按全屏设计,然后true或false控制有无状态栏。第二种是美工的设计图已经包含有无状态栏,然后一律用false。因为美工的设计图的状态栏高度可能不标准,所以我建议用第一种方案。

(2)对于虚拟按键(NavigationBar),有些机,例如华为有这个。我看到张鸿洋的博客有人问这个怎么处理,在我这里是自动适配的,你完全不需要理。

3、在程序代码中:自动适配

在setContentView()之后,在页面真正绘图之前:

AutoUtils.auto(this);

//参数this,是Activity对象。


就是这么简单。

(1)对于TextView的字体大小,我作了一定的缩放处理,不过正如张鸿洋所说的“灵活应对这个问题”,因为这个字体的情况有些复杂。

(2)对于那些动态布局,在代码中生成控件,或者像平移的动画效果,先处理一下它实际显示的宽高值:

int width=AutoUtils.getDisplayWidthValue(designWidthValue);//designWidthValue,是int变量,设计的宽度
int height=AutoUtils.getDisplayHeightValue(designHeightValue);//designHeightValue,是int变量,设计的高度

像动态控件的宽高,margin值,平移动画,在设置数值前调用这两个方法转换一下就可以了。

也可以整个控件按设计尺寸设置完,再AutoUtils.auto(view)一下,如下。


4、在程序代码中:列表item的处理

像RecyclerView、ListView这些,item的xml布局大部分与上面说的一致。

有一点要注意的是:item的背景图要拿出来独立用一个ImageView显示,不要在root view中设置background,因为在列表中它的拉伸不对。

然后在你的ViewHolder构造方法中:

AutoUtils.auto(view);

//参数view,是View对象。

凡是需要inflate导入view的地方,用这个方法处理一下就可以。


在张鸿洋的方法中,你inflate导入item布局时,需要传入parent(ViewGroup布局);在我这里,你传null也可以。

我开始用张鸿洋的方法时,在root view中设置background,然后AutoUtils.autoSize(convertView);在非设计尺寸的手机上就发现有问题了,首先背景图对不齐,然后那些margin也不对。我看他只是autoSize()。估计是他没有遇到有背景图的情况,所以他的item效果看起来还好。

至此,对于一般的item,高度固定的item,就完结了。


但我遇到更棘手的需求,请看下图:

AutoLayout——Android屏幕自动适配_第3张图片

我需要在红色方框中显示评论,评论长度是不定的,因此整个item的高度是不定的。


我是这样解决的,用draw9patch设置红色方框部分可拉伸,然后背景图的ImageView的layout_alignBottom属性设置为那个评论的TextView,使ImageView一同拉伸。但是评论的TextView并没有触及底部,于是我又用LinearLayout嵌套评论的TextView与另一个TextView,使ImageView的layout_alignBottom属性设置为那个LinearLayout。并且设置评论的TextView最小行数minLines,免得评论短时过分缩小。这样我成功的解决了问题。具体可以查看我的demo代码。


最后,请看下我demo的效果图,这是从我项目中抽取的页面做成的:

我的设计尺寸大小是:720x1280

左图尺寸是:768x1280,有虚拟按键

右图尺寸是:1080x1920,有虚拟按键


AutoLayout——Android屏幕自动适配_第4张图片


详细项目:

https://github.com/zhengjingle/Autolayout

你可能感兴趣的:(Android开发)