关于Android中drawable必知的一些规则

前言

一入 Android 深似海,相信很多 Android 开发者深有体会,Android 系统版本的碎片化,Android 硬件设备的多样性,第三方 Rom 的不确定因素。现在想开发一个合格的商业化 App 真的不容易,先不说别的,应用的兼容性就是一项技术和耐心的双重考验,想完美适配各种情况可以说是不可能的,往往都是在人力和适配率之间寻找平衡,今天要说的 drawable 就是需要适配的一个重要角色。

配置限定符

对于不同的屏幕密度、不同的设备方向,不同的语言和区域,都会涉及到备选 drawable 资源,在运行时,Android 会检测当前设备配置并根据具体规则(后面会提到)为应用加载合适的资源。下面是可以使用的配置限定符,需要说明的是这些配置限定符不仅对 drawable 有效,对其他资源类型(如:layout 等)也有效:

移动国家代码 (MCC):mcc310, mcc310-mnc004, mcc208-mnc00

语言和区域:en, fr, en-rUS 等等

布局方向:ldrtl(从右到左)ldltr(从左到右)

smallestWidth:sw<N>dp 如:sw320dp, sw600dp, sw720dp 等等,屏幕可用高度和宽度的最小尺寸,屏幕的“最小可能尺寸”。

可用宽度:w<N>dp 如:w720dp, w1024dp 等等,指定资源应该使用的最小可用屏幕宽度,以 dp 为单位,由 <N> 值定义。在横向和纵向之间切换时,为了匹配当前实际宽度,此配置值也会随之发生变化。

可用高度:h<N>dp 如:h720dp, h1024dp 等等,指定资源应该使用的最小可用屏幕高度,以dp为单位,由 <N> 值定义。 在横向和纵向之间切换时,为了匹配当前实际高度,此配置值也会随之发生变化。

屏幕尺寸:small, normal, large, xlarge

屏幕纵横比:long 宽屏,如 WQVGA、WVGA、FWVGA;notlong 非宽屏,如 QVGA、HVGA 和 VGA

屏幕方向:port 设备处于纵向(垂直),land 设备处于横向(水平)

UI 模式:car, desk, television, appliance, watch

夜间模式:night 夜间,nontight 白天

屏幕像素密度:ldpi, mdpi, hdpi, xhdpi, xxhdpi, xxxhdpi, nodpi, tvdpi

触摸屏类型:notouch, finger

键盘可用性:keysexposed, keyshidden, keyssoft

主要文本输入法:nokeys, qwerty, 12key

导航键可用性:navexposed 导航键可供用户使用,navhidden 导航键不可用

主要非触摸导航方法:nonav, dpad, trackball, wheel

平台版本(API 级别):v3, v4, v7 等等,如 v1 对应于 API 级别 1,v4 对应于 API 级别 4

如果你认真研究下每个配置限定符,你就会体会到:想完美适配各种情况可以说是不可能的,其实我们平时最常用的也是基本必须要用的就是屏幕像素密度,这里有必要详细的说一下该参数:

ldpi:低密度屏幕;约为 120dpi。

mdpi:中等密度(传统 HVGA)屏幕;约为 160dpi。

hdpi:高密度屏幕;约为 240dpi。

xhdpi:超高密度屏幕;约为 320dpi。API 级别 8 中新增配置

xxhdpi:超超高密度屏幕;约为 480dpi。API 级别 16 中新增配置

xxxhdpi:超超超高密度屏幕使用(仅限启动器图标,请参阅“支持多个屏幕”中的注释);约为 640dpi。 API 级别 18 中新增配置

nodpi:它可用于您不希望缩放以匹配设备密度的位图资源。

tvdpi:密度介于 mdpi 和 hdpi 之间的屏幕;约为 213dpi。它并不是“主要”密度组, 主要用于电视,而大多数应用都不需要它。对于大多数应用而言,提供 mdpi 和 hdpi 资源便已足够,系统将根据需要对其进行缩放。API 级别 13 中引入了此限定符。

六个主要密度之间的缩放比为 3:4:6:8:12:16(忽略 tvdpi 密度)。因此,9x9 (ldpi) 位图相当于 12x12 (mdpi)、18x18 (hdpi)、24x24 (xhdpi) 位图,依此类推。

限定符命名规则

可以为单组资源指定多个限定符,并使用短划线分隔。例如,drawable-en-rUS-land 适用于横排美国英语设备。

这些限定符必须遵循上面列出的顺序,所以上面的列表是有顺序的。例如:错误:drawable-hdpi-port/,正确:drawable-port-hdpi/

不能嵌套备用资源目录。例如,您不能拥有 res/drawable/drawable-en/。

值不区分大小写。在处理之前,资源编译器会将目录名称转换为小写,以避免不区分大小写的文件系统出现问题。 名称中使用的任何大写字母只是为了便于认读。

对于每种限定符类型,仅支持一个值。例如,若要对西班牙语和法语使用相同的 drawable 文件,则您肯定不能拥有名为 drawable-rES-rFR/ 的目录,而是需要两个包含相应文件的资源目录。

Android 匹配最佳 drawable 规则

如果你只使用一个配置限定符,那么很好匹配,找到符合该配置的 drawable 即可,但当你同时使用多个配置限定符,且同时存在多个 drawable 目录时,匹配最佳 drawable 就没那么简单了,这里以 Android Developer 官方的例子说明,例如:现在你的应用包含如下目录:

drawable/
drawable-en/
drawable-fr-rCA/
drawable-en-port/
drawable-en-notouch-12key/
drawable-port-ldpi/
drawable-port-notouch-12key/

同时,假设目标设备的配置如下:

区域设置 = en-GB
屏幕方向 = port
屏幕像素密度 = hdpi
触摸屏类型 = notouch
主要文本输入法 = 12key

具体的匹配过程如下:

1、淘汰与设备配置冲突的资源文件:其中 drawable-fr-rCA/ 目录与 en-GB 区域设置冲突,因而被淘汰(但有个例外,屏幕像素密度是唯一一个未因冲突而被淘汰的限定符,尽管设备的屏幕密度为 hdpi,但是 drawable-port-ldpi/ 未被淘汰,因为此时每个屏幕密度均视为匹配)

2、选择在上面限定符列表中优先级最高的限定符,先从 MCC 开始,然后下移,看是否有资源目录包括此限定符,若无则看下一个限定符,在该示例中,除非达到语言限定符,否则答案始终为“否”。

3、若有,则淘汰不含此限定符的资源目录。在该示例中,系统会淘汰所有不含语言限定符的目录。所以到这一步符合要求的 drawable 还剩:

drawable-en/

drawable-en-port/

drawable-en-notouch-12key/

4、选择下一个优先级的限定符,重复执行步骤 2, 3, 4。直到只剩下一个目录,该例中应该是 port, 所以淘汰后只剩:

drawable-en-port/

有两点需要说明一下:

1、屏幕像素密度是唯一一个未因冲突而被淘汰的限定符,如果涉及的限定符是屏幕像素密度,则 Android 会选择最接近设备屏幕密度的选项。通常 Android 倾向于缩小大型原始图像,而不是放大小型原始图像。

2、如果一个符合限定符的 drawable 都没有怎么办?还能怎么办,崩溃呗!

drawable 和 mipmap 的区别和联系

现在通过 Android Studio 创建工程,默认会创建一系列 mipmap 文件夹,而不是以前的 drawable 文件夹。那么 mipmap 和 drawable 到底是什么关系?mipmap 取代了 drawable 了吗?先看一下官方说明:

drawable/
For bitmap files (PNG, JPEG, or GIF), 9-Patch image files, and XML files that describe Drawable shapes or Drawable objects that contain multiple states (normal, pressed, or focused). See the Drawable resource type.
mipmap/
For app launcher icons. The Android system retains the resources in this folder (and density-specific folders such as mipmap-xxxhdpi) regardless of the screen resolution of the device where your app is installed. This behavior allows launcher apps to pick the best resolution icon for your app to display on the home screen. For more information about using the mipmap folders, see Managing Launcher Icons as mipmap Resources.

这里先说结论:mipmap 文件夹下,仅仅建议放启动图标 (app launcher icons),也就是应用安装后,会显示在桌面的那个图标,而其他的图片资源等,还是按照以前方式,放在 drawable 文件夹下。

那么为什么要把 Launcher Icon 放在mipmap 文件夹下? 下面英文是官方解释:

Different home screen launcher apps on different devices show app launcher icons at various resolutions. When app resource optimization techniques remove resources for unused screen densities, launcher icons can wind up looking fuzzy because the launcher app has to upscale a lower-resolution icon for display. To avoid these display issues, apps should use the mipmap/ resource folders for launcher icons. The Android system preserves these resources regardless of density stripping, and ensures that launcher apps can pick icons with the best resolution for display.
Make sure launcher apps show a high-resolution icon for your app by moving all densities of your launcher icons to density-specific res/mipmap/ folders (for example res/mipmap-mdpi/ and res/mipmap-xxxhdpi/). The mipmap/ folders replace the drawable/ folders for launcher icons. For xxhpdi launcher icons, be sure to add the higher resolution xxxhdpi versions of the icons to enhance the visual experience of the icons on higher resolution devices.

这里是我的理解:很多不同的 Launcher App 采用的 Launcher Icon 的大小不一致,而在应用安装时 Android 资源优化会把 drawable 文件夹下不需要的分辨率资源删除掉,例如在 xhdpi 的设备上将 drawable-xxhdpi 下的资源删掉,当然里面的 Launcher Icon 也会被删掉,这时如果 Launcher App 采用的 Launcher Icon 大小偏大,而高分辨 xxhdpi 下的 Icon 又被删掉了,就只能把 xhdpi 下的小尺寸 Icon 进行放大显示了,这样就会造成 Launcher Icon 显示模糊。为了避免上面的问题,就引入了 mipmap,Android 会保证 mipmap 下的资源不会因为资源优化而被删除,确保大尺寸的 Launcher Icon 可以找到更合适分辨率的 Icon。所以 mipmap 是为 Launcher Icon 而生的,而其它的图片资源还是放在 drawable 文件夹下,这样有助于Android 资源优化删除无用的资源,减少应用体积。

总结

以上就是关于Android中drawable必知的一些规则的全部内容,文章内容对于Android开发者来说很实用,希望对各位有所帮助。

(0)

相关推荐

  • Android开发使用Drawable绘制圆角与圆形图案功能示例

    本文实例讲述了Android开发使用Drawable绘制圆角与圆形图案功能.分享给大家供大家参考,具体如下: 1. 创建类RoundCircleDrawable继承Drawable /** * 圆角矩形 * @Project App_View * @Package com.android.view.drawable * @author chenlin * @version 1.0 * @Date 2016年4月21日 * @Note TODO */ public class RoundCircl

  • Android编程中TextView宽度过大导致Drawable无法居中问题解决方法

    本文实例讲述了Android编程中TextView宽度过大导致Drawable无法居中问题解决方法.分享给大家供大家参考,具体如下: 在做项目的时候,很多时候我们都要用到文字和图片一起显示,一般设置TextView的DrawableLeft.DrawableRight.DrawableTop.DrawableBottom就行了.但是有一种情况是当TextView的熟悉是fill_parent或者使用权重的时候并且设置了起Gravity的ceter的时候,Drawable图片是无法一起居中的,为了

  • Android Bitmap和Drawable相互转换的简单代码

    很多开发者表示,不知道Android的Drawable和Bitmap之间如何相关转换.下面Android123给大家两种比较简单高效的方法. 一.Bitmap转Drawable 复制代码 代码如下: Bitmap bm=xxx; //xxx根据你的情况获取 BitmapDrawable bd=BitmapDrawable(bm); Android开发网提示因为BtimapDrawable是Drawable的子类,最终直接使用bd对象即可. 二. Drawable转Bitmap 转成Bitmap对

  • Android开发基于Drawable实现圆角矩形的方法

    本文实例讲述了Android开发基于Drawable实现圆角矩形的方法.分享给大家供大家参考,具体如下: 第一步:写个类继承drawable,重写里面的方法,实现的核心代码在draw里 关键技术:BitmapShader public BitmapShader(Bitmap bitmap,Shader.TileMode tileX,Shader.TileMode tileY) 调用这个方法来产生一个画有一个位图的渲染器(Shader). bitmap 在渲染器内使用的位图 tileX The t

  • Android 让自定义TextView的drawableLeft与文本一起居中

    前言 TextView的drawableLeft.drawableRight和drawableTop是一个常用.好用的属性,可以在文本的上下左右放置一个图片,而不使用更加复杂布局就能达到,我也常常喜欢用RadioButton的这几个属性实现很多效果,但是苦于不支持让drawbleLeft与文本一起居中,设置gravity为center也无济于事,终于有空研究了一下,这里与大家一起分享. 正文 一.效果图  二.实现代码 自定义控件 /** * drawableLeft与文本一起居中显示 * *

  • Android LayerDrawable使用实例

    1.在Android项目开发的时候,经常看到很漂亮的UI界面,比如,当你点击一张图片被选中的时候,覆上一张透明的图片.表示被选中,区别与那些没有被选中的.这种是如何实现的呢?答案是LayerDrawable的使用出现的效果.下面是LayerDrawable的总结,若有不正确的地方,请大家指正. 2.简单的来说,LayerDrawable继承与Drawable,Drawable就是一个可画的对象,可能是一张位图BitmapDrawable,也可能是一个图形ShapeDrawable,还有可能是一个

  • Android自定义Drawable实现圆角效果

    Drawable是一种可绘制资源的载体,如图形.图像等.在实际开发中可以作为view的背景.主要有静态和动态两种方式,静态通过xml描述使用,动态即自定义Drawable.本文实现一个圆形和圆角的背景图片效果. 效果图: 实现方式: 1.初始化一个BitmapShader着色器对象: 2.将着色器对象set给画笔: 3.在画布上绘制圆或圆角即可: 4.使用,view.setBackgroundDrawable 或者 ImageView.setImageDrawable package com.m

  • Android App开发中将View或Drawable转为Bitmap的方法

    View转换为Bitmap Android中经常会遇到把View转换为Bitmap的情形,比如,对整个屏幕视图进行截屏并生成图片:Coverflow中需要把一页一页的view转换为Bitmap.以便实现复杂的图形效果(阴影.倒影效果等):再比如一些动态的实时View为便于观察和记录数据.需要临时生成静态的Bitmap. 实现方法: 1)下面是笔者经常用的一个转换方法 public static Bitmap convertViewToBitmap(View view, int bitmapWid

  • android 放大镜ShapeDrawable妙用分享

    首先,ShapeDrawable构造的时候可以指定描画的形状, 其次,可以通过shape.getPaint().setShader();指定Shader,shader可以接受一个图片和matrix 所以问题就顺利的解决了:) 具体实现如下:[java] 复制代码 代码如下: float scale = 1.2f; int cx = 224; int cy = 357; int r = 200; // 指定形状创建一个ShapeDrawable  ShapeDrawable shape=new S

  • Android自定义Drawable实现圆形和圆角

    本文实例为大家分享了自定义Drawable实现圆形和圆角的具体代码,供大家参考,具体内容如下 圆形 package com.customview.widget; import android.graphics.Bitmap; import android.graphics.BitmapShader; import android.graphics.Canvas; import android.graphics.ColorFilter; import android.graphics.Paint;

  • android中图形图像处理之drawable用法分析

    本文实例讲述了android中图形图像处理之drawable用法.分享给大家供大家参考.具体如下: 一.如何获取 res 中的资源 数据包package:android.content.res 主要类:Resources 其主要接口按照功能,划分为以下三部分: getXXXX() 例如: int getColor(int id) Drawable getDrawable(int id) String getString(int id)  直接获取res中存放的资源 InputStream ope

  • 基于android中的各种颜色在drawable.xml中的值详解

    < drawable name="white">#FFFFFF< /drawable>< !--白色 --> < drawable name="black">#000000< /drawable>< !--黑色 --> < drawable name="ivory">#FFFFF0< /drawable>< !--象牙色 --> <

随机推荐