感谢鸿洋老师《Android 屏幕适配方案》这篇文章,我的屏幕适配方案是在他基础上产生的灵感。老师的文章能一定程度的解决问题。缺陷是只能适配固定的分辨率,还有些设备有虚拟按键,虚拟键不隐藏的时候就不适配了。
在android3.2以后,google提供了sw<N>dp的方式来适配屏幕,sw是smallWidth。比如sw600dp,表示屏幕长宽的最小值都大于600dp。我们可以创建如下的文件夹:
比如320dp的设备会直接查找values-sw320dp目录下的资源文件。如果是330dp的设备,会查找比他小的最接近的目录values-sw320dp。这样可以适配常见的设备,奇葩的设备也能查找最接近的尺寸,不会让控件尺寸超出屏幕范围。
一般来说UI做的设计图都是针对720x1280(xhdpi)。这个设备屏幕宽度是720/2=360dp(2是屏幕的density)。那么针对其他的设备尺寸,我们可以用360dp的尺寸生成。
sw360dp目录下dimens.xml文件如图:
我们针对其他设备可以生成对应的dp,比如480x800(mdpi)的设备屏幕的宽度是480dp,如果720p设备上标注的是10px(5dp),那么在480设备上尺寸应该是(480dp/360dp) *5dp=6.67dp。
可以在layout中这样设置:
<Button
android:id="@+id/button"
android:layout_width="@dimen/x20"
android:layout_height="@dimen/x10" />
不同宽度的设备会直接查找对应的dp,可以保证在不同的屏幕中,控件和屏幕比例相同,不会变大或变小。
这些sw<N>dp目录都是可以由程序生成的,使用非常方便。生成的代码如下:
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.PrintWriter;
public class GenerateValueFiles {
private int baseW;
private String dirStr = "./res";
private final static String WTemplate = "<dimen name=\"x{0}\">{1}dp</dimen>\n";
private final static String VALUE_TEMPLATE = "values-sw{0}dp";
private static final String supportStr = "300;320;340;360;400;480;520;600;720;";
public GenerateValueFiles(int baseW) {
this.baseW = baseW;
File dir = new File(dirStr);
if (!dir.exists()) {
dir.mkdir();
}
}
public void generate() {
String[] vals = supportStr.split(";");
for (String val : vals) {
generateXmlFile(Integer.parseInt(val));
}
}
private void generateXmlFile(int w) {
StringBuffer sbForWidth = new StringBuffer();
sbForWidth.append("<?xml version=\"1.0\" encoding=\"utf-8\"?>\n");
sbForWidth.append("<resources>");
float cellw = w *2.0f / baseW;//2表示屏幕密度xhdpi
for (int i = 1; i < baseW; i++) {
sbForWidth.append(WTemplate.replace("{0}", i + "").replace("{1}",
change(cellw * i/2) + ""));
}
sbForWidth.append(WTemplate.replace("{0}", baseW + "").replace("{1}",
w + ""));
sbForWidth.append("</resources>");
File fileDir = new File(dirStr + File.separator
+ VALUE_TEMPLATE.replace("{0}", w + ""));
fileDir.mkdir();
File layxFile = new File(fileDir.getAbsolutePath(), "dimens.xml");
try {
PrintWriter pw = new PrintWriter(new FileOutputStream(layxFile));
pw.print(sbForWidth.toString());
pw.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}
public static float change(float a) {
int temp = (int) (a * 100);
return temp / 100f;
}
public static void main(String[] args) {
//基准的屏幕最小宽度,可以根据实际情况制定。
int baseW = 720;//720*1280(xhdpi)屏幕宽度为360dp
new GenerateValueFiles(baseW).generate();
}
}
最后生成的目录如下: