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 plaincopy在CODE上查看代碼片派生到我的代碼片
  1. package com.zhy.view;  
  2.   
  3. import android.graphics.Bitmap;  
  4. import android.graphics.BitmapShader;  
  5. import android.graphics.Canvas;  
  6. import android.graphics.ColorFilter;  
  7. import android.graphics.Paint;  
  8. import android.graphics.PixelFormat;  
  9. import android.graphics.RectF;  
  10. import android.graphics.Shader.TileMode;  
  11. import android.graphics.drawable.Drawable;  
  12.   
  13. public class RoundImageDrawable extends Drawable  
  14. {  
  15.   
  16.     private Paint mPaint;  
  17.     private Bitmap mBitmap;  
  18.   
  19.     private RectF rectF;  
  20.   
  21.     public RoundImageDrawable(Bitmap bitmap)  
  22.     {  
  23.         mBitmap = bitmap;  
  24.         BitmapShader bitmapShader = new BitmapShader(bitmap, TileMode.CLAMP,  
  25.                 TileMode.CLAMP);  
  26.         mPaint = new Paint();  
  27.         mPaint.setAntiAlias(true);  
  28.         mPaint.setShader(bitmapShader);  
  29.     }  
  30.   
  31.     @Override  
  32.     public void setBounds(int left, int top, int right, int bottom)  
  33.     {  
  34.         super.setBounds(left, top, right, bottom);  
  35.         rectF = new RectF(left, top, right, bottom);  
  36.     }  
  37.   
  38.     @Override  
  39.     public void draw(Canvas canvas)  
  40.     {  
  41.         canvas.drawRoundRect(rectF, 3030, mPaint);  
  42.     }  
  43.   
  44.     @Override  
  45.     public int getIntrinsicWidth()  
  46.     {  
  47.         return mBitmap.getWidth();  
  48.     }  
  49.   
  50.     @Override  
  51.     public int getIntrinsicHeight()  
  52.     {  
  53.         return mBitmap.getHeight();  
  54.     }  
  55.   
  56.     @Override  
  57.     public void setAlpha(int alpha)  
  58.     {  
  59.         mPaint.setAlpha(alpha);  
  60.     }  
  61.   
  62.     @Override  
  63.     public void setColorFilter(ColorFilter cf)  
  64.     {  
  65.         mPaint.setColorFilter(cf);  
  66.     }  
  67.   
  68.     @Override  
  69.     public int getOpacity()  
  70.     {  
  71.         return PixelFormat.TRANSLUCENT;  
  72.     }  
  73.   
  74. }  

核心代碼就是draw了,but,我們只需要一行~~~~setAlpha、setColorFilter、getOpacity、draw這幾個方法是必須實現的,不過除了draw以爲,其他都很簡單。getIntrinsicWidth、getIntrinsicHeight主要是爲了在View使用wrap_content的時候,提供一下尺寸,默認爲-1可不是我們希望的。setBounds就是去設置下繪製的範圍。

ok,圓角圖片就這麼實現了,easy 不~~

看下用法:

[java] view plaincopy在CODE上查看代碼片派生到我的代碼片
  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 plaincopy在CODE上查看代碼片派生到我的代碼片
  1. package com.zhy.view;  
  2.   
  3. import android.graphics.Bitmap;  
  4. import android.graphics.BitmapShader;  
  5. import android.graphics.Canvas;  
  6. import android.graphics.ColorFilter;  
  7. import android.graphics.Paint;  
  8. import android.graphics.PixelFormat;  
  9. import android.graphics.RectF;  
  10. import android.graphics.Shader.TileMode;  
  11. import android.graphics.drawable.Drawable;  
  12.   
  13. public class CircleImageDrawable extends Drawable  
  14. {  
  15.   
  16.     private Paint mPaint;  
  17.     private int mWidth;  
  18.     private Bitmap mBitmap ;   
  19.   
  20.     public CircleImageDrawable(Bitmap bitmap)  
  21.     {  
  22.         mBitmap = bitmap ;   
  23.         BitmapShader bitmapShader = new BitmapShader(bitmap, TileMode.CLAMP,  
  24.                 TileMode.CLAMP);  
  25.         mPaint = new Paint();  
  26.         mPaint.setAntiAlias(true);  
  27.         mPaint.setShader(bitmapShader);  
  28.         mWidth = Math.min(mBitmap.getWidth(), mBitmap.getHeight());  
  29.     }  
  30.   
  31.     @Override  
  32.     public void draw(Canvas canvas)  
  33.     {  
  34.         canvas.drawCircle(mWidth / 2, mWidth / 2, mWidth / 2, mPaint);  
  35.     }  
  36.   
  37.     @Override  
  38.     public int getIntrinsicWidth()  
  39.     {  
  40.         return mWidth;  
  41.     }  
  42.   
  43.     @Override  
  44.     public int getIntrinsicHeight()  
  45.     {  
  46.         return mWidth;  
  47.     }  
  48.   
  49.     @Override  
  50.     public void setAlpha(int alpha)  
  51.     {  
  52.         mPaint.setAlpha(alpha);  
  53.     }  
  54.   
  55.     @Override  
  56.     public void setColorFilter(ColorFilter cf)  
  57.     {  
  58.         mPaint.setColorFilter(cf);  
  59.     }  
  60.   
  61.     @Override  
  62.     public int getOpacity()  
  63.     {  
  64.         return PixelFormat.TRANSLUCENT;  
  65.     }  
  66.   
  67. }  

一樣出奇的簡單,再看一眼效果圖:


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 plaincopy在CODE上查看代碼片派生到我的代碼片
  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 plaincopy在CODE上查看代碼片派生到我的代碼片
  1. package com.zhy.view;  
  2.   
  3. import com.zhy.sample.drawable.R;  
  4.   
  5. import android.content.Context;  
  6. import android.util.AttributeSet;  
  7. import android.widget.RelativeLayout;  
  8.   
  9. public class MessageListItem extends RelativeLayout  
  10. {  
  11.   
  12.     private static final int[] STATE_MESSAGE_READED = { R.attr.state_message_readed };  
  13.     private boolean mMessgeReaded = false;  
  14.   
  15.     public MessageListItem(Context context, AttributeSet attrs)  
  16.     {  
  17.         super(context, attrs);  
  18.     }  
  19.   
  20.     public void setMessageReaded(boolean readed)  
  21.     {  
  22.         if (this.mMessgeReaded != readed)  
  23.         {  
  24.             mMessgeReaded = readed;  
  25.             refreshDrawableState();  
  26.         }  
  27.     }  
  28.   
  29.     @Override  
  30.     protected int[] onCreateDrawableState(int extraSpace)  
  31.     {  
  32.         if (mMessgeReaded)  
  33.         {  
  34.             final int[] drawableState = super  
  35.                     .onCreateDrawableState(extraSpace + 1);  
  36.             mergeDrawableStates(drawableState, STATE_MESSAGE_READED);  
  37.             return drawableState;  
  38.         }  
  39.         return super.onCreateDrawableState(extraSpace);  
  40.     }  
  41.   
  42. }  

代碼不復雜,聲明瞭一個STATE_MESSAGE_READED,然後在mMessgeReaded=true的情況下,通過onCreateDrawableState方法,加入我們自定義的狀態。

類似的代碼,大家可以看看CompoundButton(CheckBox父類)的源碼,它有個checked狀態:

[java] view plaincopy在CODE上查看代碼片派生到我的代碼片
  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 plaincopy在CODE上查看代碼片派生到我的代碼片
  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.   
  7.     <ImageView  
  8.         android:id="@+id/id_msg_item_icon"  
  9.         android:layout_width="30dp"  
  10.         android:src="@drawable/message_item_icon_bg"  
  11.         android:layout_height="wrap_content"  
  12.         android:duplicateParentState="true"  
  13.         android:layout_alignParentLeft="true"  
  14.         android:layout_centerVertical="true"  
  15.       />  
  16.   
  17.     <TextView  
  18.         android:id="@+id/id_msg_item_text"  
  19.         android:layout_width="match_parent"  
  20.         android:layout_height="wrap_content"  
  21.         android:layout_centerVertical="true"  
  22.         android:layout_toRightOf="@id/id_msg_item_icon" />  
  23.   
  24. </com.zhy.view.MessageListItem>  

很簡單,一個圖標,一個文本;

Activity

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

代碼很簡單,但是可以看到,我們需要在getView裏面中去使用調用setMessageReaded方法,當然了其他的一些狀態,肯定也要手動觸發,比如在ACTION_DOWN中觸發pressed等。請勿糾結咋沒有使用ViewHolder什麼的,自己添加下就行。

本例參考自:Example from github 

4、提升我們的UI Perfermance 

現在大家越來越注重性能問題,其實沒必要那麼在乎,但是既然大家在乎了,這裏通過Cyril Mottier :master_android_drawables ppt中的一個例子來說明如果利用Drawable來提升我們的UI的性能。

大家看這樣一個效果圖:


佈局文件:

[html] view plaincopy在CODE上查看代碼片派生到我的代碼片
  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.   
  8.     <ImageView  
  9.         android:layout_width="wrap_content"  
  10.         android:layout_height="wrap_content"  
  11.         android:layout_gravity="center"  
  12.         android:layout_marginBottom="24dp"  
  13.         android:src="@drawable/logo" />  
  14.   
  15.     <LinearLayout  
  16.         android:layout_width="match_parent"  
  17.         android:layout_height="48dp"  
  18.         android:layout_gravity="bottom"  
  19.         android:orientation="horizontal" >  
  20.   
  21.         <Button  
  22.             android:layout_width="0dp"  
  23.             android:layout_height="fill_parent"  
  24.             android:layout_weight="1"  
  25.             android:text="@string/sign_up" />  
  26.   
  27.         <Button  
  28.             android:layout_width="0dp"  
  29.             android:layout_height="fill_parent"  
  30.             android:layout_weight="1"  
  31.             android:text="@string/sign_in" />  
  32.     </LinearLayout>  
  33.   
  34. </FrameLayout>  

可以看到最外層是FrameLayout僅僅是爲了設置背景圖和padding,這樣的佈局相信很多人也寫過。

再看看這個佈局作爲APP啓動時,用戶的直觀效果:


用戶首先看到一個白板,然後顯示出我們的頁面。接下來,我們將利用Drawable改善我們的UI性能以及用戶體驗。

1、首先,我們去除我們最外層的FrameLayout,然後自定義一個drawable的xml,叫做logo.xml

[html] view plaincopy在CODE上查看代碼片派生到我的代碼片
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <layer-list xmlns:android="http://schemas.android.com/apk/res/android" >  
  3.     <item>  
  4.   
  5.         <shape android:shape="rectangle" >  
  6.             <solid android:color="@color/app_background" />  
  7.         </shape>  
  8.     </item>  
  9.   
  10.     <item android:bottom="48dp">  
  11.         <bitmap  
  12.             android:gravity="center"  
  13.             android:src="@drawable/logo" />  
  14.     </item>  
  15. </layer-list>  

ok,這個drawable是設置了我們的背景和logo;

2、將其作爲我們當前Activity的windowBackground

[html] view plaincopy在CODE上查看代碼片派生到我的代碼片
  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 plaincopy在CODE上查看代碼片派生到我的代碼片
  1. <activity  
  2. android:name="LoginActivity"  
  3. android:theme="@style/Theme.Default.NoActionBar">  

Ok,這樣不僅最小化了我們的layout,現在我們的layout裏面只有一個LinearLayout和兩個按鈕;並且提升了用戶體驗,現在用戶的直觀效果時:


是不是體驗好很多,個人很喜歡這個例子~~


ok,到此我們的文章就over了~~~大多數內容參考自一些牛人寫的例子,例子還是棒棒噠,大家看完本文的同時,也可以去挖掘挖掘一些東西~~


源碼點擊下載


發佈了11 篇原創文章 · 獲贊 10 · 訪問量 3萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章