Android Palette(調(diào)色板)的使用

一、Palette的簡(jiǎn)單介紹

android-support-v7-palette 里面的Palette是Android L SDK 中的新特性,可以使用 Palette 從圖像中提取出突出的顏色(主色調(diào))嘱函,獲取到顏色之后我們?cè)賹⑦@個(gè)顏色值賦給 ActionBar磁奖、狀態(tài)欄等温兼。從而達(dá)到界面色調(diào)的統(tǒng)一收恢,使界面美觀協(xié)調(diào)。

原理
通過得到一個(gè)bitmap绳泉,通過方法進(jìn)行分析逊抡,取出LightVibrantSwatch,DarkVibrantSwatch零酪,LightMutedSwatch冒嫡,DarkMutedSwatch這些樣本,然后得到rgb四苇。

Palette這個(gè)類中提取以下突出的顏色

Vibrant (有活力)
Vibrant dark(有活力 暗色)
Vibrant light(有活力 亮色)
Muted (柔和)
Muted dark(柔和 暗色)
Muted light(柔和 亮色)

創(chuàng)建方法
      //目標(biāo)bitmap
      Bitmap bm =BitmapFactory.decodeResource(getResources(),R.drawable.kale);
      //方法1
      Palette.Builder builder = Palette.from(bm);
      Palette palette=builder.generate();

      //方法2   使用異步
      builder.generate(bitmap, new Palette.PaletteAsyncListener() {  
          @Override  
          public void onGenerated(Palette palette) {     
           // Here's your generated palette
      }
二孝凌、內(nèi)部嵌套類和常用方法
Palette.Builder

生成器類,生成 Palette 實(shí)例月腋。

Palette.Filter

過濾器接口蟀架,使 Palette 有更加細(xì)膩的顏色過濾

Palette.PaletteAsyncListener

異步加載監(jiān)聽

pattle.Swatch

提供獲取結(jié)果的色彩樣本

from(List<Palette.Switch> switches)

通過預(yù)設(shè)的 Palette.Swatch 顏色樣本列表 來生成 Palette
返回值:static Palette

from(Bitmap bitmap)

通過返回 Palette.Builder 實(shí)例來構(gòu)建 Palette
返回值:static Palette.Builder

generate(Bitmap bitmap,int numColors) 、generate(Bitmap bitmap) 榆骚、

該方法已被遺棄片拍,建議用 Palette.Builder 來生成Palette
返回值:static Palette

getColorForTarget(Target target,int defaultColor)

返回一個(gè)從目標(biāo)獲取的的 rgb 色值
返回值:int

getDarkMutedColor(int defaultColor)

返回一個(gè)柔和的暗色調(diào) rgb 值
返回值:int

getDarkMutedSwatch()

返回一個(gè)柔和的暗色調(diào)樣本類
返回值:Palette.Swatch

getDarkVibrantColor(int defaultColor)

返回一個(gè)鮮明的暗色調(diào) rgb 值
返回值:int

getDomainSwatch()

返回一個(gè)主色調(diào)的樣本類
返回值:Palette.Swatch

getLightMutedColor(int defaultColor)

返回一個(gè)柔和的亮色調(diào)顏色 rgb
返回值:Palette.Swatch

getLightVibrantSwatch()

返回一個(gè)鮮明的亮色調(diào)樣本類
返回值:Palette.Swatch

三、使用樣本(swatch)

創(chuàng)建完一個(gè)實(shí)例之后妓肢,我們還需要得到一種采集的樣本(swatch)捌省,有6中樣本(swatch):

Palette.getVibrantSwatch()
Palette.getDarkVibrantSwatch()
Palette.getLightVibrantSwatch()
Palette.getMutedSwatch()
Palette.getDarkMutedSwatch()
Palette.getLightMutedSwatch()
List<Palette.Swatch> swatches = palette.getSwatches();//一次性獲得所有的swatch;

使用方法
getPopulation(): the amount of pixels which this swatch represents.
getRgb(): the RGB value of this color.
getHsl(): the HSL value of this color.
getBodyTextColor(): the RGB value of a text color which can be displayed on top of this color.
getTitleTextColor(): the RGB value of a text color which can be displayed on top of this color.

比如如果你的TextView 有個(gè)背景圖片,要想讓字體顏色能夠和背景圖片匹配碉钠,則使用getBodyTextColor()比較合適纲缓,getTitleTextColor()其實(shí)應(yīng)該和getBodyTextColor()差不多

四、Size問題

在上面的代碼中放钦,你可能注意到了可以設(shè)置palette的size色徘。size越大恭金,花費(fèi)的時(shí)間越長(zhǎng)操禀,而越小,可以選擇的色彩也越小横腿。最佳的選擇是根據(jù)image的用途:

頭像之類的颓屑,size最好在24-32之間斤寂;
風(fēng)景大圖之類的 size差不多在8-16;
默認(rèn)是16.

Bitmap bm = BitmapFactory.decodeResource(getResources(),
              R.drawable.kale);
      Palette palette = Palette.generate(bm);
      if (palette.getLightVibrantSwatch() != null) {
          //需要注意的是`getVibrantSwatch()可能會(huì)返回一個(gè)null值揪惦,
          //所以檢查一下是必須的遍搞。
          //得到不同的樣本,設(shè)置給imageview進(jìn)行顯示
          iv.setBackgroundColor(palette.getLightVibrantSwatch().getRgb());
          iv1.setBackgroundColor(palette.getDarkVibrantSwatch().getRgb());
          iv2.setBackgroundColor(palette.getLightMutedSwatch().getRgb());
          iv3.setBackgroundColor(palette.getDarkMutedSwatch().getRgb());

      }
注意

加載不能在主線程中進(jìn)行器腋,加載方式有同步加載和異步加載兩種:

1 同步

由于他們很可能會(huì)比較耗時(shí)(在分析大圖片或者所需顏色較多時(shí))溪猿,所以它們不應(yīng)該在主線程中執(zhí)行。你應(yīng)該先在別的線程中使用這兩個(gè)函數(shù)進(jìn)行解析纫塌,解析成功之后再使用诊县。

//在加載圖片的后臺(tái)線程中同步加載
Palette palette = Palette.from(bitmap).genrate();
.
.
.
2 異步

有時(shí)候你不會(huì)在加載圖片的線程(非主線程)中使用解析出的顏色,所以Palette提供了異步方法措左,他們與之前的函數(shù)的區(qū)別就是需要傳入PaletteAsyncListener依痊,提供在圖片解析完成后的回調(diào)函數(shù)。

//異步加載
Palette.from(bitmap).genrate(new PaletteAsyncListener(){
  public void onGenerated(Palette p){
      .
      .
      .
  }
});
Palette經(jīng)常用于和ViewPager怎披,F(xiàn)ragment搭配使用胸嘁,當(dāng)我們的Pager切換時(shí)伴隨著Fragment的變化,而Fragment里的內(nèi)容一般是不同的凉逛,所以每個(gè)Fragment里的一般視覺效果也是不同的性宏,所以我們可以用Palette來去提取Fragment中的主色調(diào),把這個(gè)主色調(diào)用于整體的UI風(fēng)格鱼炒。

Demo下載地址:http://download.csdn.net/detail/breeze_wf/9273313

2625875-1a5ffa9daaeb0192.gif

其他Demo
private void setPaletteColor(TextView tv, int color) {
    tv.setBackgroundColor(color);
    tv.setText(ColorUtils.toRGBHexString(color));
    tv.setTextColor(ColorUtils.parseBackgroundColor(color));
}

private void initPalette() {
    Drawable drawable = mPictureIv.getDrawable();
    Bitmap bitmap = drawableToBitmap(drawable);

    // Synchronous
    // mPalette = Palette.from(bitmap).generate();

    // Asynchronous
    Palette.from(bitmap).generate(new Palette.PaletteAsyncListener() {
        @Override
        public void onGenerated(Palette palette) {
            // Use generated instance
            int defaultColor = Color.parseColor("#b64242");

            int mVibrantColor = palette.getVibrantColor(defaultColor);
            int mDarkVibrantColor = palette.getDarkVibrantColor(defaultColor);
            int mLightVibrantColor = palette.getLightVibrantColor(defaultColor);
            int mMutedColor = palette.getMutedColor(defaultColor);
            int mDarkMutedColor = palette.getDarkMutedColor(defaultColor);
            int mLightMutedColor = palette.getLightMutedColor(defaultColor);

            setPaletteColor(mVibrantColorTv, mVibrantColor);
            setPaletteColor(mDarkVibrantColorTv, mDarkVibrantColor);
            setPaletteColor(mLightVibrantColorTv, mLightVibrantColor);
            setPaletteColor(mMutedColorTv, mMutedColor);
            setPaletteColor(mDarkMutedColorTv, mDarkMutedColor);
            setPaletteColor(mLightMutedColorTv, mLightMutedColor);

            // dominant color (主色)
            int mDominantColor = palette.getDominantColor(defaultColor);
            setPaletteColor(mDominantColorTv, mDominantColor);

            // Swatch - 色塊 // 15種
            List<Palette.Swatch> mSwatchList = palette.getSwatches();
            Toast.makeText(MainActivity.this, "Swatch num: " + mSwatchList.size(), Toast.LENGTH_SHORT).show();
            int index = -1;
            LinearLayout mSwatchesContainer = null;
            LinearLayout.LayoutParams params;
            for (Palette.Swatch swatch : mSwatchList) {
                int color = swatch.getRgb();
                index++;

                if (index % 3 == 0) {
                    mSwatchesContainer = new LinearLayout(getApplicationContext());
                    mSwatchesContainer.setOrientation(LinearLayout.HORIZONTAL);
                    params = new LinearLayout.LayoutParams(
                            LinearLayout.LayoutParams.MATCH_PARENT,
                            LinearLayout.LayoutParams.WRAP_CONTENT
                    );
                    params.topMargin = (int) DisplayUtils.dp2px(getApplicationContext(), 10);
                    mContainerLayout.addView(mSwatchesContainer, params);       //
                }

                LinearLayout mSwatchContainer = new LinearLayout(getApplicationContext());
                mSwatchContainer.setOrientation(LinearLayout.VERTICAL);
                params = new LinearLayout.LayoutParams(0, LinearLayout.LayoutParams.WRAP_CONTENT);
                params.weight = 1;
                params.gravity = Gravity.CENTER;
                if (mSwatchesContainer != null) {
                    mSwatchesContainer.addView(mSwatchContainer, params);       //
                }

                TextView mColorTv = new TextView(getApplicationContext());
                mColorTv.setTextSize(TypedValue.COMPLEX_UNIT_SP, 16);
                setPaletteColor(mColorTv, color);           //
                mColorTv.setGravity(Gravity.CENTER);
                params = new LinearLayout.LayoutParams(
                        LinearLayout.LayoutParams.MATCH_PARENT,
                        (int) DisplayUtils.dp2px(getApplicationContext(), 80)
                );
                params.gravity = Gravity.CENTER;
                mSwatchContainer.addView(mColorTv, params);                 //

                TextView mColorNameTv = new TextView(getApplicationContext());
                mColorNameTv.setTextSize(TypedValue.COMPLEX_UNIT_SP, 16);
                mColorNameTv.setText("Swatch " + index);
                mColorNameTv.setGravity(Gravity.CENTER);
                mColorNameTv.setTextColor(Color.parseColor("#333333"));
                params = new LinearLayout.LayoutParams(
                        LinearLayout.LayoutParams.WRAP_CONTENT,
                        LinearLayout.LayoutParams.WRAP_CONTENT
                );
                params.gravity = Gravity.CENTER;
                mSwatchContainer.addView(mColorNameTv, params);
            }
        }
    });
}
Drawable轉(zhuǎn)Bitmap的方法:
public static Bitmap drawableToBitmap(Drawable drawable) {
    Bitmap bitmap = Bitmap.createBitmap(
            drawable.getIntrinsicWidth(),
            drawable.getIntrinsicHeight(),
            drawable.getOpacity() != PixelFormat.OPAQUE ? Bitmap.Config.ARGB_8888 : Bitmap.Config.RGB_565);
    Canvas canvas = new Canvas(bitmap); // canvas -> bitmap
    //canvas.setBitmap(bitmap);
    drawable.setBounds(0, 0, drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight());
    drawable.draw(canvas);      // drawable -> canvas
    return bitmap;  // drawable -> canvas -> bitmap
}
ColorUtils
public class ColorUtils {

    public static int parseBackgroundColor2(int color) {
        int counter = 0;
        counter += Color.red(color) >= 128 ? 1 : 0;
        counter += Color.green(color) >= 128 ? 1 : 0;
        counter += Color.blue(color) >= 128 ? 1 : 0;
        return counter >= 2 ? Color.BLACK : Color.WHITE;
    }

    // 通過分析背景色來決定當(dāng)前文字的匹配顏色衔沼,使文字顏色自適應(yīng)背景顏色
    public static int parseBackgroundColor(int color) {
        int red = Color.red(color);
        int green = Color.green(color);
        int blue = Color.blue(color);
        if (red >= 128 && green >= 128      // 三選二
                || red >= 128 && blue >= 128
                || green >= 128 && blue >= 128) {
            return Color.rgb(0, 0, 0);
        }
        return Color.rgb(255, 255, 255);
    }

    // #FF55FF => color
    // int color = Color.parseColor("#b64242");

    // color -> #FF55FF
    public static String toRGBHexString(final int color) {
        return toRGBHexString(Color.red(color), Color.green(color), Color.blue(color));
    }

    // (r,g,b) -> #FF55FF
    public static String toRGBHexString(int red, int green, int blue) {
        return toARGBHexString(-1, red, green, blue);
    }

    // default prefix: "#"
    // (a,r,g,b) -> #FF55FF55
    public static String toARGBHexString(int alpha, int red, int green, int blue) {
        return toARGBHexString("#", alpha, red, green, blue);
    }

    public static String toARGBHexString(String prefix, int alpha, int red, int green, int blue) {
        StringBuilder sb = new StringBuilder();
        sb.append(prefix);
        if (alpha != -1) {
            String mAlphaStr = Integer.toHexString(alpha);
            sb.append(mAlphaStr.length() == 1 ? "0" + mAlphaStr : mAlphaStr);
        }
        String mRedStr = Integer.toHexString(red);
        sb.append(mRedStr.length() == 1 ? "0" + mRedStr : mRedStr);
        String mGreenStr = Integer.toHexString(green);
        sb.append(mGreenStr.length() == 1 ? "0" + mGreenStr : mGreenStr);
        String mBlueStr = Integer.toHexString(blue);
        sb.append(mBlueStr.length() == 1 ? "0" + mBlueStr : mBlueStr);
        return sb.toString().toUpperCase();
    }

}

Palette
Demo地址

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市昔瞧,隨后出現(xiàn)的幾起案子指蚁,更是在濱河造成了極大的恐慌,老刑警劉巖自晰,帶你破解...
    沈念sama閱讀 217,542評(píng)論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件凝化,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡酬荞,警方通過查閱死者的電腦和手機(jī)搓劫,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,822評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來混巧,“玉大人枪向,你說我怎么就攤上這事∵值常” “怎么了秘蛔?”我有些...
    開封第一講書人閱讀 163,912評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我深员,道長(zhǎng)负蠕,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,449評(píng)論 1 293
  • 正文 為了忘掉前任倦畅,我火速辦了婚禮遮糖,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘叠赐。我一直安慰自己欲账,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,500評(píng)論 6 392
  • 文/花漫 我一把揭開白布芭概。 她就那樣靜靜地躺著敬惦,像睡著了一般。 火紅的嫁衣襯著肌膚如雪谈山。 梳的紋絲不亂的頭發(fā)上俄删,一...
    開封第一講書人閱讀 51,370評(píng)論 1 302
  • 那天,我揣著相機(jī)與錄音奏路,去河邊找鬼畴椰。 笑死,一個(gè)胖子當(dāng)著我的面吹牛鸽粉,可吹牛的內(nèi)容都是我干的斜脂。 我是一名探鬼主播,決...
    沈念sama閱讀 40,193評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼触机,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼帚戳!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起儡首,我...
    開封第一講書人閱讀 39,074評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤片任,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后蔬胯,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體对供,經(jīng)...
    沈念sama閱讀 45,505評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,722評(píng)論 3 335
  • 正文 我和宋清朗相戀三年氛濒,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了产场。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,841評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡舞竿,死狀恐怖京景,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情骗奖,我是刑警寧澤确徙,帶...
    沈念sama閱讀 35,569評(píng)論 5 345
  • 正文 年R本政府宣布靡菇,位于F島的核電站,受9級(jí)特大地震影響米愿,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜鼻吮,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,168評(píng)論 3 328
  • 文/蒙蒙 一育苟、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧椎木,春花似錦违柏、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,783評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至畜伐,卻和暖如春馍惹,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背玛界。 一陣腳步聲響...
    開封第一講書人閱讀 32,918評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工万矾, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人慎框。 一個(gè)月前我還...
    沈念sama閱讀 47,962評(píng)論 2 370
  • 正文 我出身青樓良狈,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親笨枯。 傳聞我的和親對(duì)象是個(gè)殘疾皇子薪丁,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,781評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容