Android Drawable 那些不为人知的高效用法

转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/43752383,本文出自:【张鸿洋的博客】

1、概述

Drawable在我们平时的开发中,基本都会用到,而且给大家非常的有用。那么什么是Drawable呢?能够在canvas上绘制的一个玩意,而且相比于View,并不需要去考虑measure、layout,仅仅只要去考虑如何draw(canavs)。当然了,对于Drawable传统的用法,大家肯定不陌生 ,今天主要给大家带来以下几个Drawable的用法:

1、自定义Drawable,相比View来说,Drawable属于轻量级的、使用也很简单。以后自定义实现一个效果的时候,可以改变View first的思想,尝试下Drawable first。

2、自定义状态,相信大家对于State Drawable都不陌生,但是有没有尝试过去自定义一个状态呢?

3、利用Drawable提升我们的UI Perfermance , 如何利用Drawable去提升我们的UI的性能。

2、Drawable基本概念

一般情况下,除了直接使用放在Drawable下的图片,其实的Drawable的用法都和xml相关,我们可以使用shape、layer-list等标签绘制一些背景,还可以通过selector标签定义View的状态的效果等。当然了基本每个标签都对应于一个真正的实体类,关系如下:(图片来自:Cyril Mottier :master_android_drawables)

常见的用法这里就不举例了,下面开始看本文的重点。

2、自定义Drawable

关于自定义Drawable,可以通过写一个类,然后继承自Drawable , 类似于自定义View,当然了自定义Drawable的核心方法只有一个,那就是draw。那么自定义Drawable到底有什么实际的作用呢?能干什么呢?

相信大家对于圆角、圆形图片都不陌生,并且我曾经写过通过自定义View实现的方式,具体可参考:

Android BitmapShader
实战 实现圆形、圆角图片

Android
Xfermode 实战 实现圆形、圆角图片

那我今天要告诉你,不需要自定义View,自定义Drawable也能实现,而且更加简单、高效、使用范围更广(你可以作为任何View的背景)。

1、RoundImageDrawable

代码比较简单,下面看下RoundImageDrawable

[java] view
plain
 copy

  1. package com.zhy.view;
  2. import android.graphics.Bitmap;
  3. import android.graphics.BitmapShader;
  4. import android.graphics.Canvas;
  5. import android.graphics.ColorFilter;
  6. import android.graphics.Paint;
  7. import android.graphics.PixelFormat;
  8. import android.graphics.RectF;
  9. import android.graphics.Shader.TileMode;
  10. import android.graphics.drawable.Drawable;
  11. public class RoundImageDrawable extends Drawable
  12. {
  13. private Paint mPaint;
  14. private Bitmap mBitmap;
  15. private RectF rectF;
  16. public RoundImageDrawable(Bitmap bitmap)
  17. {
  18. mBitmap = bitmap;
  19. BitmapShader bitmapShader = new BitmapShader(bitmap, TileMode.CLAMP,
  20. TileMode.CLAMP);
  21. mPaint = new Paint();
  22. mPaint.setAntiAlias(true);
  23. mPaint.setShader(bitmapShader);
  24. }
  25. @Override
  26. public void setBounds(int left, int top, int right, int bottom)
  27. {
  28. super.setBounds(left, top, right, bottom);
  29. rectF = new RectF(left, top, right, bottom);
  30. }
  31. @Override
  32. public void draw(Canvas canvas)
  33. {
  34. canvas.drawRoundRect(rectF, 30, 30, mPaint);
  35. }
  36. @Override
  37. public int getIntrinsicWidth()
  38. {
  39. return mBitmap.getWidth();
  40. }
  41. @Override
  42. public int getIntrinsicHeight()
  43. {
  44. return mBitmap.getHeight();
  45. }
  46. @Override
  47. public void setAlpha(int alpha)
  48. {
  49. mPaint.setAlpha(alpha);
  50. }
  51. @Override
  52. public void setColorFilter(ColorFilter cf)
  53. {
  54. mPaint.setColorFilter(cf);
  55. }
  56. @Override
  57. public int getOpacity()
  58. {
  59. return PixelFormat.TRANSLUCENT;
  60. }
  61. }

核心代码就是draw了,but,我们只需要一行~~~~setAlpha、setColorFilter、getOpacity、draw这几个方法是必须实现的,不过除了draw以为,其他都很简单。getIntrinsicWidth、getIntrinsicHeight主要是为了在View使用wrap_content的时候,提供一下尺寸,默认为-1可不是我们希望的。setBounds就是去设置下绘制的范围。

ok,圆角图片就这么实现了,easy 不~~

看下用法:

[java] view
plain
 copy

  1. Bitmap bitmap = BitmapFactory.decodeResource(getResources(),
  2. R.drawable.mv);
  3. ImageView iv = (ImageView) findViewById(R.id.id_one);
  4. iv.setImageDrawable(new RoundImageDrawable(bitmap));

ok,贴一下我们的效果图,两个ImageView和一个TextView

可以看到,不仅仅用于ImageView去实现圆角图片,并且可以作为任何View的背景,在ImageView中的拉伸的情况,配下ScaleType即可。在其他View作为背景时,如果出现拉伸情况,请参考:Android
BitmapShader 实战 实现圆形、圆角图片
 。 足够详细了。

2、CircleImageDrawable

那么下来,我们再看看自定义圆形Drawable的写法:

[java] view
plain
 copy

  1. package com.zhy.view;
  2. import android.graphics.Bitmap;
  3. import android.graphics.BitmapShader;
  4. import android.graphics.Canvas;
  5. import android.graphics.ColorFilter;
  6. import android.graphics.Paint;
  7. import android.graphics.PixelFormat;
  8. import android.graphics.RectF;
  9. import android.graphics.Shader.TileMode;
  10. import android.graphics.drawable.Drawable;
  11. public class CircleImageDrawable extends Drawable
  12. {
  13. private Paint mPaint;
  14. private int mWidth;
  15. private Bitmap mBitmap ;
  16. public CircleImageDrawable(Bitmap bitmap)
  17. {
  18. mBitmap = bitmap ;
  19. BitmapShader bitmapShader = new BitmapShader(bitmap, TileMode.CLAMP,
  20. TileMode.CLAMP);
  21. mPaint = new Paint();
  22. mPaint.setAntiAlias(true);
  23. mPaint.setShader(bitmapShader);
  24. mWidth = Math.min(mBitmap.getWidth(), mBitmap.getHeight());
  25. }
  26. @Override
  27. public void draw(Canvas canvas)
  28. {
  29. canvas.drawCircle(mWidth / 2, mWidth / 2, mWidth / 2, mPaint);
  30. }
  31. @Override
  32. public int getIntrinsicWidth()
  33. {
  34. return mWidth;
  35. }
  36. @Override
  37. public int getIntrinsicHeight()
  38. {
  39. return mWidth;
  40. }
  41. @Override
  42. public void setAlpha(int alpha)
  43. {
  44. mPaint.setAlpha(alpha);
  45. }
  46. @Override
  47. public void setColorFilter(ColorFilter cf)
  48. {
  49. mPaint.setColorFilter(cf);
  50. }
  51. @Override
  52. public int getOpacity()
  53. {
  54. return PixelFormat.TRANSLUCENT;
  55. }
  56. }

一样出奇的简单,再看一眼效果图:

ok,关于自定义Drawable的例子over~~~接下来看自定义状态的。

上述参考了:Romain Guy‘s Blog

3、自定义Drawable State

关于Drawable State,state_pressed神马的,相信大家都掌握的特别熟练了。

那么接下来,我们有个需求,类似于邮箱,邮件以ListView形式展示,但是我们需要一个状态去标识出未读和已读:so,我们自定义一个状态state_message_readed。

效果图:

可以看到,如果是已读的邮件,我们的图标是打开状态,且有个淡红色的背景。那么如何通过自定义drawable state 实现呢?

自定义drawable state 需要分为以下几个步骤:

1、res/values/新建一个xml文件:drawable_status.xml

[html] view
plain
 copy

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <resources>
  3. <declare-styleable name="MessageStatus">
  4. <attr name="state_message_readed" format="boolean" />
  5. </declare-styleable>
  6. </resources>

2、继承Item的容器

我们这里Item选择RelativeLayout实现,我们需要继承它,然后复写它的onCreateDrawableState方法,把我们自定义的状态在合适的时候添加进去。

[java] view
plain
 copy

  1. package com.zhy.view;
  2. import com.zhy.sample.drawable.R;
  3. import android.content.Context;
  4. import android.util.AttributeSet;
  5. import android.widget.RelativeLayout;
  6. public class MessageListItem extends RelativeLayout
  7. {
  8. private static final int[] STATE_MESSAGE_READED = { R.attr.state_message_readed };
  9. private boolean mMessgeReaded = false;
  10. public MessageListItem(Context context, AttributeSet attrs)
  11. {
  12. super(context, attrs);
  13. }
  14. public void setMessageReaded(boolean readed)
  15. {
  16. if (this.mMessgeReaded != readed)
  17. {
  18. mMessgeReaded = readed;
  19. refreshDrawableState();
  20. }
  21. }
  22. @Override
  23. protected int[] onCreateDrawableState(int extraSpace)
  24. {
  25. if (mMessgeReaded)
  26. {
  27. final int[] drawableState = super
  28. .onCreateDrawableState(extraSpace + 1);
  29. mergeDrawableStates(drawableState, STATE_MESSAGE_READED);
  30. return drawableState;
  31. }
  32. return super.onCreateDrawableState(extraSpace);
  33. }
  34. }

代码不复杂,声明了一个STATE_MESSAGE_READED,然后在mMessgeReaded=true的情况下,通过onCreateDrawableState方法,加入我们自定义的状态。

类似的代码,大家可以看看CompoundButton(CheckBox父类)的源码,它有个checked状态:

[java] view
plain
 copy

  1. @Override
  2. protected int[] onCreateDrawableState(int extraSpace) {
  3. final int[] drawableState = super.onCreateDrawableState(extraSpace + 1);
  4. if (isChecked()) {
  5. mergeDrawableStates(drawableState, CHECKED_STATE_SET);
  6. }
  7. return drawableState;
  8. }

3、使用

布局文件:

[html] view
plain
 copy

  1. <com.zhy.view.MessageListItem xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools"
  3. android:layout_width="match_parent"
  4. android:layout_height="50dp"
  5. android:background="@drawable/message_item_bg" >
  6. <ImageView
  7. android:id="@+id/id_msg_item_icon"
  8. android:layout_width="30dp"
  9. android:src="@drawable/message_item_icon_bg"
  10. android:layout_height="wrap_content"
  11. android:duplicateParentState="true"
  12. android:layout_alignParentLeft="true"
  13. android:layout_centerVertical="true"
  14. />
  15. <TextView
  16. android:id="@+id/id_msg_item_text"
  17. android:layout_width="match_parent"
  18. android:layout_height="wrap_content"
  19. android:layout_centerVertical="true"
  20. android:layout_toRightOf="@id/id_msg_item_icon" />
  21. </com.zhy.view.MessageListItem>

很简单,一个图标,一个文本;

Activity

[java] view
plain
 copy

  1. package com.zhy.sample.drawable;
  2. import com.zhy.view.MessageListItem;
  3. import android.app.ListActivity;
  4. import android.os.Bundle;
  5. import android.view.LayoutInflater;
  6. import android.view.View;
  7. import android.view.ViewGroup;
  8. import android.widget.ArrayAdapter;
  9. import android.widget.TextView;
  10. public class CustomStateActivity extends ListActivity
  11. {
  12. private Message[] messages = new Message[] {
  13. new Message("Gas bill overdue", true),
  14. new Message("Congratulations, you‘ve won!", true),
  15. new Message("I love you!", false),
  16. new Message("Please reply!", false),
  17. new Message("You ignoring me?", false),
  18. new Message("Not heard from you", false),
  19. new Message("Electricity bill", true),
  20. new Message("Gas bill", true), new Message("Holiday plans", false),
  21. new Message("Marketing stuff", false), };
  22. @Override
  23. protected void onCreate(Bundle savedInstanceState)
  24. {
  25. super.onCreate(savedInstanceState);
  26. getListView().setAdapter(new ArrayAdapter<Message>(this, -1, messages)
  27. {
  28. private LayoutInflater mInflater = LayoutInflater
  29. .from(getContext());
  30. @Override
  31. public View getView(int position, View convertView, ViewGroup parent)
  32. {
  33. if (convertView == null)
  34. {
  35. convertView = mInflater.inflate(R.layout.item_msg_list,
  36. parent, false);
  37. }
  38. MessageListItem messageListItem = (MessageListItem) convertView;
  39. TextView tv = (TextView) convertView
  40. .findViewById(R.id.id_msg_item_text);
  41. tv.setText(getItem(position).message);
  42. messageListItem.setMessageReaded(getItem(position).readed);
  43. return convertView;
  44. }
  45. });
  46. }
  47. }

代码很简单,但是可以看到,我们需要在getView里面中去使用调用setMessageReaded方法,当然了其他的一些状态,肯定也要手动触发,比如在ACTION_DOWN中触发pressed等。请勿纠结咋没有使用ViewHolder什么的,自己添加下就行。

本例参考自:Example from github

4、提升我们的UI Perfermance

现在大家越来越注重性能问题,其实没必要那么在乎,但是既然大家在乎了,这里通过Cyril Mottier :master_android_drawables ppt中的一个例子来说明如果利用Drawable来提升我们的UI的性能。

大家看这样一个效果图:

布局文件:

[html] view
plain
 copy

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:background="@color/app_background"
  6. android:padding="8dp" >
  7. <ImageView
  8. android:layout_width="wrap_content"
  9. android:layout_height="wrap_content"
  10. android:layout_gravity="center"
  11. android:layout_marginBottom="24dp"
  12. android:src="@drawable/logo" />
  13. <LinearLayout
  14. android:layout_width="match_parent"
  15. android:layout_height="48dp"
  16. android:layout_gravity="bottom"
  17. android:orientation="horizontal" >
  18. <Button
  19. android:layout_width="0dp"
  20. android:layout_height="fill_parent"
  21. android:layout_weight="1"
  22. android:text="@string/sign_up" />
  23. <Button
  24. android:layout_width="0dp"
  25. android:layout_height="fill_parent"
  26. android:layout_weight="1"
  27. android:text="@string/sign_in" />
  28. </LinearLayout>
  29. </FrameLayout>

可以看到最外层是FrameLayout仅仅是为了设置背景图和padding,这样的布局相信很多人也写过。

再看看这个布局作为APP启动时,用户的直观效果:

用户首先看到一个白板,然后显示出我们的页面。接下来,我们将利用Drawable改善我们的UI性能以及用户体验。

1、首先,我们去除我们最外层的FrameLayout,然后自定义一个drawable的xml,叫做logo.xml

[html] view
plain
 copy

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <layer-list xmlns:android="http://schemas.android.com/apk/res/android" >
  3. <item>
  4. <shape android:shape="rectangle" >
  5. <solid android:color="@color/app_background" />
  6. </shape>
  7. </item>
  8. <item android:bottom="48dp">
  9. <bitmap
  10. android:gravity="center"
  11. android:src="@drawable/logo" />
  12. </item>
  13. </layer-list>

ok,这个drawable是设置了我们的背景和logo;

2、将其作为我们当前Activity的windowBackground

[html] view
plain
 copy

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <resources>
  3. <style
  4. name="Theme.Default.NoActionBar"
  5. parent="@android:style/Theme.Holo.Light.NoActionBar" >
  6. <item name="android:windowBackground">@drawable/login</item>
  7. </style>
  8. </resources>

3、设置到Activity上:

[html] view
plain
 copy

  1. <activity
  2. android:name="LoginActivity"
  3. android:theme="@style/Theme.Default.NoActionBar">

Ok,这样不仅最小化了我们的layout,现在我们的layout里面只有一个LinearLayout和两个按钮;并且提升了用户体验,现在用户的直观效果时:

是不是体验好很多,个人很喜欢这个例子~~

ok,到此我们的文章就over了~~~大多数内容参考自一些牛人写的例子,例子还是棒棒哒,大家看完本文的同时,也可以去挖掘挖掘一些东西~~

源码点击下载

时间: 2024-10-10 22:07:40

Android Drawable 那些不为人知的高效用法的相关文章

Android Drawable 那些不为人知的高效使用方法

转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/43752383,本文出自:[张鸿洋的博客] 1.概述 Drawable在我们平时的开发中.基本都会用到,并且给大家很的实用.那么什么是Drawable呢?可以在canvas上绘制的一个玩意.并且相比于View,并不须要去考虑measure.layout,只只要去考虑怎样draw(canavs). 当然了,对于Drawable传统的使用方法.大家肯定不陌生 .今天主要给大家带来下面

AndroidDrawable那些不为人知的高效用法(转载)

AndroidDrawable那些不为人知的高效用法 转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/43752383,本文出自:[张鸿洋的博客] 1.概述 Drawable在我们平时的开发中,基本都会用到,而且给大家非常的有用.那么什么是Drawable呢?能够在canvas上绘制的一个玩意,而且相比于View,并不需要去考虑measure.layout,仅仅只要去考虑如何draw(canavs).当然了,对于Drawable

Android最佳实践之高效的应用导航

设计(一)- 规划Screens和他们之间的关系 原文地址:http://developer.android.com/training/design-navigation/screen-planning.html#beyond-simplistic-design 设计和开发Android应用程序的第一个步骤是确定用户能够查看和处理应用.一旦你知道用户与之交互的应用程序之间交互什么数据,下一步就是设计交互,允许用户导航到app的不同部分,进入和退出应用程序中的界面. 这篇文章开始向你展示如何规划高

Android UI集锦——1.Android Drawable分类汇总(1/3)

Android UI集锦--1.Android Drawable分类汇总(1/3) -转载请注明出处coder-pig 本节引言: 小猪好像写了好几个专题,都没坚持写完,又忍不住开个新的专题了,因为最近打算 开始研究Android图形与图形图像处理,动画以及自定义View等,所以就顺道记录下, 最近事有点多,感觉情绪很低迷,心理压抑又找不到倾述的对象,这个时候程序猿肯定会说: "没对象,你自己new一个啊",好有道理,我竟无言以对...好吧!还是自己的那句座右铭: 没什么可以一蹴而就,

android中Handle类的用法

android中Handle类的用法 当我们在处理下载或是其他需要长时间执行的任务时,如果直接把处理函数放Activity的OnCreate或是OnStart中,会导致执行过程中整个Activity无响应,如果时间过长,程序还会挂掉.Handler就是把这些功能放到一个单独的线程里执行,与Activity互不影响. 当用户点击一个按钮时如果执行的是一个常耗时操作的话,处理不好会导致系统假死,用户体验很差,而Android则更进一步,如果任意一个Acitivity没有响应5秒钟以上就会被强制关闭,

三、android中Handle类的用法

当我们在处理下载或是其他需要长时间执行的任务时,如果直接把处理函数放Activity的OnCreate或是OnStart中,会导致执行过程中整个Activity无响应,如果时间过长,程序还会挂掉.Handler就是把这些功能放到一个单独的线程里执行,与Activity互不影响. 当用户点击一个按钮时如果执行的是一个常耗时操作的话,处理不好会导致系统假死,用户体验很差,而Android则更进一步,如果任意一个Acitivity没有响应5秒钟以上就会被强制关闭,因此我们需要另外起动一个线程来处理长耗

Android Drawable 与 LayerList综合汇总

先看需求.要求这样的效果 上代码 <?xml version="1.0" encoding="utf-8"? > <layer-list xmlns:android="http://schemas.android.com/apk/res/android" > <item> <shape android:shape="oval" > <solid android:color=

Android Selector和Shape的用法

一.Shape的用法 :shape用于设定形状,可以在selector,layout等里面使用,有6个子标签,各属性如下: 填充:设置填充的颜色 间隔:设置四个方向上的间隔 大小:设置大小 圆角:同时设置五个属性,则Radius属性无效 android:Radius="20dp"          设置四个角的半径 android:topLeftRadius="20dp"              设置左上角的半径 android:topRightRadius=&q

Android开发技巧之viewstub用法详解及实现延迟加载

这一篇是接着上面的include标签的例子来讲的,地址http://blog.csdn.net/jason0539/article/details/26131831 上一篇的布局中间就用了viewstub这个控件,现在来说一下其作用和用法 " ViewStub 是一个不可见的,大小为0的View,最佳用途就是实现View的延迟加载,避免资源浪费,在需要的时候才加载View " 需要注意的是,加载view之后,viewstub本身就会被新加载进来的view替换掉 上代码了,看完就理解了