根據(jù)UI提供標注圖的標注尺寸掀抹,設置baseW(基準寬度)和baseH(基準高度)
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.PrintWriter;
public class GenerateValueFiles {
private static int baseW = 360;//基準寬度
private static int baseH = 800;//基準高度
private String dirStr = "./res";
private final static String WTemplate = "<dimen name=\"x{0}\">{1}px</dimen>\n";
private final static String HTemplate = "<dimen name=\"y{0}\">{1}px</dimen>\n";
private final static String VALUE_TEMPLATE = "values-{0}x{1}";
private static final String SUPPORT_DIMESION = "320,480;"
+ "480,800;480,854;"
+ "540,854;540,960;"
+ "600,800;600,1024;"
+ "640,960;"
+ "720,1184;720,1196;720,1208;720,1280;"
+ "768,1024;768,1280;"
+ "800,1280;"
+ "1080,1700;1080,1776;1080,1794;1080,1800;1080,1812;1080,1920;"
+ "1440,2416;1440,2560;"
+ "1600,2560;";
private String supportStr = SUPPORT_DIMESION;
public GenerateValueFiles(String supportStr) {
this.supportStr += validateInput(supportStr);
System.out.println(supportStr);
File dir = new File(dirStr);
if (!dir.exists()) {
dir.mkdir();
}
System.out.println(dir.getAbsoluteFile());
}
/**
* @param supportStr
* w,h_...w,h;
* @return
*/
private String validateInput(String supportStr) {
StringBuffer sb = new StringBuffer();
String[] vals = supportStr.split("_");
int w = -1;
int h = -1;
String[] wh;
for (String val : vals) {
try {
if (val == null || val.trim().length() == 0)
continue;
wh = val.split(",");
w = Integer.parseInt(wh[0]);
h = Integer.parseInt(wh[1]);
} catch (Exception e) {
System.out.println("skip invalidate params : w,h = " + val);
continue;
}
sb.append(w + "," + h + ";");
}
return sb.toString();
}
public void generate() {
String[] vals = supportStr.split(";");
for (String val : vals) {
String[] wh = val.split(",");
generateXmlFile(Integer.parseInt(wh[0]), Integer.parseInt(wh[1]));
}
generateDefaultXmlFile(baseW,baseH);
}
private void generateXmlFile(int w, int h) {
StringBuffer sbForWidth = new StringBuffer();
sbForWidth.append("<?xml version=\"1.0\" encoding=\"utf-8\"?>\n");
sbForWidth.append("<resources>");
float cellw = w * 1.0f / baseW;
System.out.println("width : " + w + "," + baseW + "," + cellw);
for (int i = 1; i <=baseW; i++) {
sbForWidth.append(WTemplate.replace("{0}", i + "").replace("{1}",
change(cellw * i) + ""));
}
sbForWidth.append("</resources>");
StringBuffer sbForHeight = new StringBuffer();
sbForHeight.append("<?xml version=\"1.0\" encoding=\"utf-8\"?>\n");
sbForHeight.append("<resources>");
float cellh = w *1.0f/ baseW;
System.out.println("height : "+ h + "," + baseH + "," + cellh);
for (int i = 1; i <= baseH; i++) {
sbForHeight.append(HTemplate.replace("{0}", i + "").replace("{1}",
change(cellh * i) + ""));
}
sbForHeight.append("</resources>");
File fileDir = new File(dirStr + File.separator
+ VALUE_TEMPLATE.replace("{0}", h + "")//
.replace("{1}", w + ""));
fileDir.mkdir();
File layxFile = new File(fileDir.getAbsolutePath(), "lay_x.xml");
File layyFile = new File(fileDir.getAbsolutePath(), "lay_y.xml");
try {
PrintWriter pw = new PrintWriter(new FileOutputStream(layxFile));
pw.print(sbForWidth.toString());
pw.close();
pw = new PrintWriter(new FileOutputStream(layyFile));
pw.print(sbForHeight.toString());
pw.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}
private void generateDefaultXmlFile(int w, int h){
StringBuffer sbForWidth = new StringBuffer();
sbForWidth.append("<?xml version=\"1.0\" encoding=\"utf-8\"?>\n");
sbForWidth.append("<resources>");
float cellw = w * 1.0f / baseW;
System.out.println("width : " + w + "," + baseW + "," + cellw);
for (int i = 1; i < baseW; i++) {
sbForWidth.append(WTemplate.replace("{0}", i + "").replace("{1}",
change(cellw * i) + "").replace("px", "dp"));
}
sbForWidth.append(WTemplate.replace("{0}", baseW + "").replace("{1}",
w + "").replace("px", "dp"));
sbForWidth.append("</resources>");
StringBuffer sbForHeight = new StringBuffer();
sbForHeight.append("<?xml version=\"1.0\" encoding=\"utf-8\"?>\n");
sbForHeight.append("<resources>");
float cellh = h *1.0f/ baseH;
System.out.println("height : "+ h + "," + baseH + "," + cellh);
for (int i = 1; i < baseH; i++) {
sbForHeight.append(HTemplate.replace("{0}", i + "").replace("{1}",
change(cellh * i) + "").replace("px", "dp"));
}
sbForHeight.append(HTemplate.replace("{0}", baseH + "").replace("{1}",
h + "").replace("px", "dp"));
sbForHeight.append("</resources>");
File fileDir = new File(dirStr + File.separator
+ VALUE_TEMPLATE.substring(0, 6));
fileDir.mkdir();
File layxFile = new File(fileDir.getAbsolutePath(), "lay_x.xml");
File layyFile = new File(fileDir.getAbsolutePath(), "lay_y.xml");
try {
PrintWriter pw = new PrintWriter(new FileOutputStream(layxFile));
pw.print(sbForWidth.toString());
pw.close();
pw = new PrintWriter(new FileOutputStream(layyFile));
pw.print(sbForHeight.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) {
String addition = "";
try {
if (args.length >= 3) {
baseW = Integer.parseInt(args[0]);
baseH = Integer.parseInt(args[1]);
addition = args[2];
} else if (args.length >= 2) {
baseW = Integer.parseInt(args[0]);
baseH = Integer.parseInt(args[1]);
} else if (args.length >= 1) {
addition = args[0];
}
} catch (NumberFormatException e) {
System.err
.println("right input params : java -jar xxx.jar width height w,h_w,h_..._w,h;");
e.printStackTrace();
System.exit(-1);
}
new GenerateValueFiles(addition).generate();
}
}
運行以上代碼,會得到 res/values-x-x 文件沼沈,直接復制到項目的res路徑下即可锥腻。
如何使用逸寓?
xml中指定width和height時,直接使用UI標注圖中的尺寸慨蓝,如UI標注128(不區(qū)分是寬度或高度感混,它們等效的),xml中就可以使用lay_x128礼烈。
lay_y和lay_x是一樣的內容浩习,不需要的可以刪除所有的lay_x文件,唯一區(qū)別是lay_y的范圍可以自由設置(比如800)济丘,在UI中如果有一個控件的UI標注尺寸為400谱秽,在lay_x中并沒有l(wèi)ay_x400,這時你就可以使用lay_y400了摹迷。
有虛擬導航條的手機屏幕適配問題疟赊?
比如一款手機屏幕的物理尺寸為1920*1080,如果沒有虛擬導航條它適配了values-1920x1080峡碉,但是有虛擬導航條時近哟,它所適配的values文件就不是values-1920x1080了,而是高度減去虛擬導航條的values文件鲫寄,比如虛擬導航條的高度為108吉执,這時它適配的就是values-1812x1080。需要注意的是地来,虛擬導航條可以手動設置為隱藏戳玫,當隱藏狀態(tài)下時,它適配的就是values-1920x1080了未斑。
那么我們怎么知道虛擬導航條的高度咕宿?
可以寫一個測試demo運行在這款手機上,執(zhí)行如下代碼:
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.DisplayMetrics;
import android.view.Display;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
DisplayMetrics dm = new DisplayMetrics();
Display display = getWindowManager().getDefaultDisplay();
display.getMetrics(dm);
TextView mTextView = (TextView) findViewById(R.id.main_tv);
mTextView.setText("width:"+dm.widthPixels+",height:"+dm.heightPixels);
}
}
運行該app就可以在textview中顯示當前狀態(tài)下適配屏幕所需要的寬和高了蜡秽。
如果SUPPORT_DIMESION 中沒有你想要適配屏幕的尺寸府阀,你可以自行添加進去(格式為 寬,高)芽突。