android屏幕適配方案

根據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(不區分是寬度或高度,它們等效的),xml中就可以使用lay_x128。
lay_y和lay_x是一樣的內容,不需要的可以刪除所有的lay_x文件,唯一區別是lay_y的范圍可以自由設置(比如800),在UI中如果有一個控件的UI標注尺寸為400,在lay_x中并沒有lay_x400,這時你就可以使用lay_y400了。

有虛擬導航條的手機屏幕適配問題?

比如一款手機屏幕的物理尺寸為1920*1080,如果沒有虛擬導航條它適配了values-1920x1080,但是有虛擬導航條時,它所適配的values文件就不是values-1920x1080了,而是高度減去虛擬導航條的values文件,比如虛擬導航條的高度為108,這時它適配的就是values-1812x1080。需要注意的是,虛擬導航條可以手動設置為隱藏,當隱藏狀態下時,它適配的就是values-1920x1080了。
那么我們怎么知道虛擬導航條的高度?
可以寫一個測試demo運行在這款手機上,執行如下代碼:

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中顯示當前狀態下適配屏幕所需要的寬和高了。

如果SUPPORT_DIMESION 中沒有你想要適配屏幕的尺寸,你可以自行添加進去(格式為 寬,高)。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容