在 Android 中自定义一个电池图标,一般是采用自定义 View,在 onDraw 中采用 Canvas 去绘制 Bitmap 或者各种几何图形。但是自定义 View 对初学者来说可能会有一点难度,那么有没有更简单的办法来实现自定义电池图标呢? 实现电池图标 Drawable 我们来分析下

在 Android 中自定义一个电池图标,一般是采用自定义 View,在 onDraw 中采用 Canvas 去绘制 Bitmap 或者各种几何图形。但是自定义 View 对初学者来说可能会有一点难度,那么有没有更简单的办法来实现自定义电池图标呢?

实现电池图标 Drawable

我们来分析下绘制一个电池图标我们需要做些什么?

电池图标

如图所示,电池图标可看成有三种状态,空的,满的,介于空和满的。那我们就可以这么做,先绘制一个空的,再根据电量绘制一个半满的。我们知道 Drawable 有很多子类,LayerDrawable 可以用来绘制多个图层,正好是我们需要的,另外绘制一半的可以采用 ClipDrawable。

ClipDrawable 在 draw 方法中会通过 Canvas#clipRect(Rect rect)方法去裁剪画布。ClipDrawable 可以通过 clipOrientation 来指定裁剪的方法为 horizontal 还是 vertical,而且还可以通过 gravity 属性来指定从哪边开始裁剪。

了解这些之后我们就可以着手写 drawable 了,在 /res/drawable 下新建一个 battery.xml 如下:

android:clipOrientation="horizontal"

android:drawable="@drawable/battery_full"

android:gravity="left" />

上面两个图层最终组成我们想要的电池图标。当然,我们还需要将 battery.xml 应用到我们的 View 中 。这里我采用的是 ImageView,直接为 ImageView 指定一个android:src="@drawable/battery"即可,也可以作为 View 的 background使用。

接下来还需要在 java 代码中指定 ClipDrawable 要裁剪的区域。代码如下:

protected void onCreate(Bundle savedInstanceState){

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

ImageView imageBattery = (ImageView) findViewById(R.id.image_battery);

LayerDrawable layerDrawable = (LayerDrawable) imageBattery.getDrawable();

clipDrawable = (ClipDrawable) layerDrawable.findDrawableByLayerId(R.id.clip_drawable);

clipDrawable.setLevel(level);

}

LayerDrawable 中有个 findDrawableByLayerId(int id) 方法,可以找到指定 id 的 Drawable。有点类似于 View 中的 findViewById(int id)方法。另外 ClipDrawable#setLevel(int level),参数 level 的取值范围是 0-10000,表示要裁剪 Drawable 多大的范围。

到这里我们已经基本实现了自定义电池图标。不过我们还没有给定真实的电量值。

获取手机电量

获取手机电量只需注册一个广播即可,当手机电量改变时,我们就能在广播中收到消息。直接看代码:

public class MainActivity extends Activity{

@Override

protected void onResume(){

super.onResume();

IntentFilter intentFilter = new IntentFilter(Intent.ACTION_BATTERY_CHANGED);

registerReceiver(batteryChangedReceiver, intentFilter);

}

@Override

protected void onPause(){

super.onPause();

unregisterReceiver(batteryChangedReceiver);

}

// 电量变化广播

private BroadcastReceiver batteryChangedReceiver = new BroadcastReceiver() {

public void onReceive(Context context, Intent intent){

if (Intent.ACTION_BATTERY_CHANGED.equals(intent.getAction())) {

int level = intent.getIntExtra("level", 0);

int scale = intent.getIntExtra("scale", 100);

// 0 - 100

int power = level * 100 / scale;

// setLevel(int level): level 的范围是 0 -10000

clipDrawable.setLevel(power * 100);

}

}

}

}

至此自定义电池图标就完成的差不多了。不过细心的童鞋可能会发现,电量还没满,比如 90% 的时候电池图标已经显示是满的了。这是因为电量显示的区域只有中间某一块,而不是整个图标,如果要求比较高,我们还需要写给方法去精确的计算裁剪的区域。

// 根据自己的电池图标精确计算裁剪区域

private int calculateLevel(int progress){

int leftOffest = Utils.dip2px(this, 2);

int powerLength = Utils.dip2px(this, 26.5f);// 40 px in hdpi

int totalLength = Utils.dip2px(this, 32.5f);// 49 px in hdpi

int level = (leftOffest + powerLength * progress / 100) * 10000 / totalLength;

return level;

}

结合图片看代码:

考虑电池边界值

这些值我们可以通过ps的标尺或者其他软件来获取。同时还要注意图片放在哪个文件夹,转换成 mdpi 下像素值,作为 dp 单位。比如我把图片放在 hdpi,图片的实际宽度(totalLength)为 49px,那么在 mdpi 大约为 32.5px,32.5 即为我们的 dp值。

dp 转 px 的方法:

public class Utils{

public static int px2dip(Context context, float pxValue){

final float scale = context.getResources().getDisplayMetrics().density;

return (int) (pxValue / scale + 0.5f);

}

public static int dip2px(Context context, float dpValue){

final float scale = context.getResources().getDisplayMetrics().density;

return (int) (dpValue * scale + 0.5f);

}

}

修改 onReceive 中的clipDrawable.setLevel(power * 100) 为 clipDrawable.setLevel(calculateLevel(power))

到这里整个自定义电池图标就全部完成了,如果需要显示充电状态,可以通过 Handler 配合属性动画去完成。这不是本节的重点,源码下载!

ClipDrawable 基础:

ClipDrawable是通过设置一个Drawable的当前显示比例来裁剪出另一张Drawable,你可以通过调节这个比例来控制裁剪的宽高,以及裁剪内容占整个容器的权重,通过ClipDrawable的setLevel()方法调节显示比例可以实现类似Progress进度条的效果。ClipDrawable的level值范围在[0,10000],level的值越大裁剪的内容越少,如果level为10000时则完全显示。

存放位置:res/drawable/

使用方式:

在Java文件中:R.drawable.filename

在xml布局文件中:@[package:]drawable/filename

语法:

xmlns:android="http://schemas.android.com/apk/res/android"

android:drawable="@drawable/drawable_resource"

android:clipOrientation=["horizontal" | "vertical"]

android:gravity=["top" | "bottom" | "left" | "right" | "center_vertical" |

"fill_vertical" | "center_horizontal" | "fill_horizontal" |

"center" | "fill" | "clip_vertical" | "clip_horizontal"] />

1

2

3

4

5

6

7

8

1

2

3

4

5

6

7

8

1、Android:drawable Drawable资源。 必须。表示该ClipDrawable引用的drawable资源。

2、android:clipOrientation 裁剪的方向。 horizontal:水平方向 vertical:垂直方向

3、android:gravity 指定从哪个地方裁剪。必须是下面一个或多个值(多个值之间用”|”分隔)

top 将这个对象放在容器的顶部,不改变其大小,当clipOrientation 是”vertical”,从底部(bottom)开始裁剪

bottom 将这个对象放在容器的底部,不改变其大小。当clipOrientation 是 “vertical”,从顶部(top)开始裁剪

left 将这个对象放在容器的左部,不改变其大小。当clipOrientation 是 “horizontal”,从右边(right)开始裁剪。这也是默认情况。left 将这个对象放在容器的左部,不改变其大小。当clipOrientation 是 “horizontal”,从右边(right)开始裁剪。这也是默认情况。

right 将这个对象放在容器的右部,不改变其大小。当clipOrientation 是 “horizontal”,从左边(left)开始裁剪。

center_vertical 将对象放在垂直中间,不改变其大小。裁剪的情况和”center“一样。

fill_vertical 垂直方向上不发生裁剪。(除非drawable的level是 0,才会不可见,表示全部裁剪完)

center_horizontal 将对象放在水平中间,不改变其大小。裁剪的情况和”center“一样。

fill_horizontal 水平方向上不发生裁剪。(除非drawable的level是 0,才会不可见,表示全部裁剪完)

center 将这个对象放在水平垂直坐标的中间,不改变其大小。当clipOrientation 是 “horizontal”裁剪发生在左右。当clipOrientation是”vertical”,裁剪发生在上下。

fill 填充整个容器,不会发生裁剪。(除非drawable的level是 0,才会不可见,表示全部裁剪完)。

clip_vertical 额外的选项,它能够把它的容器的上下边界,设置为子对象的上下边缘的裁剪边界。裁剪要基于对象垂直重力设置:如果重力设置为top,则裁剪下边,如果设置为bottom,则裁剪上边,否则则上下两边都要裁剪。

clip_horizontal 额外的选项,它能够把它的容器的左右边界,设置为子对象的左右边缘的裁剪边界。裁剪要基于对象垂直重力设置:如果重力设置为right,则裁剪左边,如果设置为left,则裁剪右边,否则则左右两边都要裁剪。

android:gravity需要和android:clipOrientation配合使用,不同的组合,裁剪的效果也不同

android实现自定义图标,Android开发中用Drawable 实现自定义电池图标相关推荐

  1. 电脑图标java开发工具素材_IconJar for Mac(图标素材管理工具) v2.3.0

    IconJar for Mac是应用在Mac上的一款图标素材管理工具,为SVG和图标字体提供了令人难以置信的支持,您可以将其拖放到任何应用程序中.QuickDrag可以拖放任何大小,颜色和各种文件格式 ...

  2. Android7.08.0 电池图标分析

    这里只是简单的介绍下电池图标的显示,所以不是全面的电池分析.想做frameworks电池图标定制的可以参考下. frameworks/base/packages/SystemUI/src/com/an ...

  3. 【Android游戏开发二十三】自定义ListView【通用】适配器并实现监听控件!

    本站文章均为 李华明Himi 原创,转载务必在明显处注明: 转载自[黑米GameDev街区] 原文链接: http://www.himigame.com/android-game/374.html L ...

  4. android自定义progressbar样式,Android开发中如何实现自定义ProgressBar的样式

    Android开发中如何实现自定义ProgressBar的样式 发布时间:2020-11-20 16:08:10 来源:亿速云 阅读:294 作者:Leah Android开发中如何实现自定义Prog ...

  5. android高德自定义图标,Android 高德地图显示在线图标

    项目刚好用到高德地图,需求里有要求显示在线图标,发现高德Demo里只有显示本地图标的的例子,那么网络配置的图标要怎么显示呢,这时候可能就想到了自定义View,因为高德地图支持自定义View作为Mark ...

  6. Android Studio自定义模板 做开发竟然可以如此轻松 后篇

    ###1.概述 最近有很多人反馈,有些哥们不喜欢看文字性的东西,还有一些哥们根本就不知道我在搞啥子,那么以后我就采用博客加视频的方式,我们可以选择看视频讲解:http://pan.baidu.com/ ...

  7. android自定义工具栏,Android工具栏中的自定义图标

    我正在使用支持工具栏中定义一个自定义图标,但唯一显示的图标是左箭头-我尝试以布局和编程方式设置它,但结果是一样的. 这是我的活动 public class MainActivity extends A ...

  8. Android自定义壁纸预览界面,Android自定义动态壁纸开发(时钟)

    看到有些手机酷炫的动态壁纸,有没有好奇过他们是如何实现的,其实我们自己也可以实现. 先看效果 上图是动态壁纸钟的一个时钟. 我们先来看看 Livewallpaper(即动态墙纸)的实现,Android ...

  9. android 自定义抽屉,android – 动作栏抽屉切换自定义图标

    我正在尝试使用操作栏抽屉切换,但我希望它显示的不仅仅是菜单图标.我希望它在我的自定义菜单图标的右上角显示通知计数,但我仍然希望操作栏抽屉切换以在需要时显示后退箭头. 因此,我首先尝试在操作栏抽屉切换时 ...

  10. Android开发所需的Android SDK、开发中用到的工具、Android开发教程、Android设计规范,免费的设计素材

    AndroidDevTools AndroidSDK在线更新镜像服务器 中国科学院开源协会镜像站地址: IPV4/IPV6:mirrors.opencas.cn端口:80 IPV4/IPV6:mirr ...

最新文章

  1. [软考]信息系统项目管理师考试大纲
  2. (百度、谷歌)地图经纬度gps偏移解决办法:gps纠偏数据库纠偏
  3. react中的axios配置文件
  4. 演练 影视演员简介 0929
  5. Flink window 用法介绍
  6. at24c16如何划分出多个读写区_漫话:如何给女朋友解释为什么Windows上面的软件都想把自己安装在C盘...
  7. Java 中 Comparable 和 Comparator 比较(转)
  8. 推荐12个漂亮的CSS3按钮实现方案
  9. TracePro模拟LED手电筒整个照明系统并分析
  10. 数据结构与算法分析(五)队列
  11. NodeJS启动vue项目的坑
  12. 采云端采云链:从订单协同到采购供应链,让采购供应链互联互通
  13. 编译原理课程设计c语言,编译原理课程设计心得体会
  14. 基于php的心理测试,据说是韩国最受欢迎的心理测试~~
  15. [GIS原理] 9.1 数字高程模型DEM-特征 | 分类 | 构建 | 数据结构 | 空间内插 | DEM的未来
  16. python简单代码画皮卡丘-用python画一只可爱的皮卡丘
  17. 在线教育系统网页版,观看视频直播功能更多,更适合用户学习
  18. 同一网段的概念以及计算
  19. 为什么写论文时一定要引用论文?
  20. 通过内网连接GaussDB

热门文章

  1. 雷军现身国庆 70 周年阅兵花车!
  2. 源支付5.18源码/三网免挂/带云端/附源码搭建教程
  3. 高效的word转换成pdf转换器
  4. 如何实现网页中按钮的自动点击?
  5. 【行车路径规划】百度地图API,行车路径规划 起始点+途经点
  6. Android开发:基站定位
  7. R语言|lollipop plot
  8. 宽带按流量计费?欢迎讨论!
  9. WinCap数据包显示
  10. 最难忘的一节计算机课,我最难忘的一节课