欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android?自定義view中根據(jù)狀態(tài)修改drawable圖片

 更新時(shí)間:2023年07月10日 08:37:24   作者:Stars-one  
這篇文章主要介紹了Android?自定義view中根據(jù)狀態(tài)修改drawable圖片的相關(guān)資料,需要的朋友可以參考下

本文涉及知識(shí)點(diǎn):

  • Android里的selector圖片使用
  • 底部導(dǎo)航欄的使用
  • 自定義view的步驟了解

建議有以上基礎(chǔ)有助于幫助你理解本篇文章....

起因,由于UI那邊的實(shí)現(xiàn),不是按照的Material Design風(fēng)格設(shè)計(jì)的,設(shè)計(jì)的底部導(dǎo)航欄圖標(biāo)和文本在同一行,原本想用官方的BottomNavigation組件也沒法使用,只好自己仿造地寫個(gè)自定義組件

正常BottomNavigation組件,是讀取menu文件來設(shè)置圖標(biāo)和文本從而實(shí)現(xiàn)數(shù)據(jù)

在自定義View中,如何根據(jù)狀態(tài)去修改drawble圖片?

說明

從menu菜單文件得知:通過icon屬性設(shè)置一個(gè)drawble對象即可實(shí)現(xiàn)圖標(biāo)

如果你給的drawable對象為一個(gè)selector,那么在selector中正確聲明了不同狀態(tài)下的drawable,那么就能夠?qū)崿F(xiàn)圖標(biāo)在選中和未選中的圖標(biāo)變更,具體可以參考我之前的文章,Android BottomNavigation底部導(dǎo)航欄使用 - Stars-One的雜貨小窩

這里xml里的selector,其實(shí)最終被Android里解析處理,得到一個(gè)StateListDrawable對象

PS selector可以在drawablecolor中使用,如果在color中使用,得到的就是ColorStateList對象

仿照實(shí)現(xiàn)導(dǎo)航欄切換圖標(biāo)功能

前面的一些自定義view的步驟在此略過,主要講解核心的東西

我們需要自定義view去讀取menu文件里的數(shù)據(jù),得到icon的drawble文件,并根據(jù)不同狀態(tài)去取這個(gè)drawable里的圖片,之后去更改我們的imageview即可

獲取菜單文件icon內(nèi)容:

val menuRes = R.menu.test_menu

val popupMenu = PopupMenu(context, View(context))
popupMenu.inflate(menuRes)
val menu = popupMenu.menu

//得到menu后,使用此對象獲取icon或label等屬性
val icon =  menu.children.first().icon

獲取不同狀態(tài)的drawable:

先說下思路,我們view中有一個(gè)狀態(tài)位標(biāo)明當(dāng)前是哪個(gè)item選擇,當(dāng)item為選擇的時(shí)候,我們讓item的圖標(biāo)展示已選中狀態(tài)的drawable,反之則相反

//view中的一個(gè)狀態(tài)表示
val viewIsCheck = false

if (icon is StateListDrawable) {
    val drawable = if(viewIsCheck){
        getDrawable(icon, android.R.attr.state_checked)
    }else{
        //加個(gè)-號(hào),則表示反過來的狀態(tài)(即xml里的android:state_checked屬性為false)
        val drawable = getDrawable(icon, -android.R.attr.state_checked)
    }	
    myBottomNavItem.ivIcon.load(drawable)
}


fun getDrawable(icon: StateListDrawable, flag: Int): Drawable {
    return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
        val index = icon.findStateDrawableIndex(intArrayOf(flag))
        icon.getStateDrawable(index)!!
    } else {
        icon.state = IntArray(android.R.attr.state_checked)
        icon.current
    }
}

工具方法封裝

這里稍微把上面的工具方法getDrawable寫成了StateListDrawable的擴(kuò)展方法,方便之后調(diào)用,已收錄在我的庫中stars-one/XAndroidUtil: 封裝自己常用的一些Android的組件或工具

/**
 * 獲取不同狀態(tài)的drawable
 * @param flag 可選數(shù)值如下
- [android.R.attr.state_pressed]:按鈕被按下時(shí)的狀態(tài)。
- [android.R.attr.state_focused]:視圖獲取焦點(diǎn)時(shí)的狀態(tài)。
- [android.R.attr.state_selected]:視圖被選中時(shí)的狀態(tài)。
- [android.R.attr.state_checked]:用于可選中的視圖,表示視圖處于選中狀態(tài)。
- [android.R.attr.state_enabled]:視圖可用時(shí)的狀態(tài)。
- [android.R.attr.state_hovered]:視圖被懸停時(shí)的狀態(tài)。
- [android.R.attr.state_activated]:用于用作活動(dòng)項(xiàng)目的視圖。
 *
 */
fun StateListDrawable.getXStateDrawable(flag: Int): Drawable {
    val icon =this
    return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
        val index = icon.findStateDrawableIndex(intArrayOf(flag))
        icon.getStateDrawable(index)!!
    } else {
        icon.state = IntArray(android.R.attr.state_checked)
        icon.current
    }
}

其他補(bǔ)充

動(dòng)態(tài)構(gòu)造StateListDrawable對象

上面說到的都是從xml里讀取,既然是一個(gè)類,那么我們也可以通過寫代碼的方式快速構(gòu)造出一個(gè)StateListDrawable對象

// 創(chuàng)建 StateListDrawable
val stateListDrawable = StateListDrawable()

// 添加按下狀態(tài)的 Drawable
val pressedDrawable = resources.getDrawable(R.drawable.pressed_bg, null)
stateListDrawable.addState(intArrayOf(android.R.attr.state_pressed), pressedDrawable)

// 添加默認(rèn)狀態(tài)的 Drawable
val defaultDrawable = resources.getDrawable(R.drawable.default_bg, null)
stateListDrawable.addState(intArrayOf(), defaultDrawable)

// 將 StateListDrawable 設(shè)置為 View 的背景
view.background = stateListDrawable

自定義view的reference類型

如果需要自定義view,可以在xml中設(shè)置一個(gè)menu菜單,可以聲明一個(gè)屬性為reference,如下代碼:

 <declare-styleable name="SettingItem">
        <attr name="mymenu" format="reference"/>
</declare-styleable>

之后在代碼里使用getResourceId方法可以讀取屬性數(shù)據(jù):

val ta = context.obtainStyledAttributes(attrs, R.styleable.CustomView);
val menuResourceId = ta.getResourceId(R.styleable.CustomView_menuAttr, 0);
ta.recycle();

//得到菜單文件    
if (menuResourceId != 0) {
    // 加載菜單資源
    val mMenu = PopupMenu(mContext, null).getMenu();
    val inflater = MenuInflater(mContext);
    inflater.inflate(menuResourceId, mMenu);
  
}

關(guān)于顏色ColorStateList

上文也提到,我們也可以在color的資源文件夾中使用selector,這里也補(bǔ)充下如何讀取吧

在color資源文件夾定義文件custom_color_state_list.xml

<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:color="#FF0000" android:state_pressed="true" />
    <item android:color="#00FF00" android:state_enabled="true" />
    <item android:color="#0000FF" />
</selector>
// 從資源文件中讀取 ColorStateList 對象
val colorStateList = ContextCompat.getColorStateList(context, R.color.custom_color_state_list)

// 使用 ColorStateList 對象
textView.setTextColor(colorStateList)

封裝的擴(kuò)展方法,獲取某個(gè)狀態(tài)的color:

/**
 * 獲取selector某個(gè)狀態(tài)的color
 * - selector里定義`androird:state_pressed="true"`,即為`android.R.attr.state_pressed`
 * - selector里定義`androird:state_pressed="false"`,即為`-android.R.attr.state_pressed`
 *
 * @param flag 可選數(shù)值如下: 前面加個(gè)`-`,標(biāo)示為狀態(tài)為false
- [android.R.attr.state_pressed]:按鈕被按下時(shí)的狀態(tài)。
- [android.R.attr.state_focused]:視圖獲取焦點(diǎn)時(shí)的狀態(tài)。
- [android.R.attr.state_selected]:視圖被選中時(shí)的狀態(tài)。
- [android.R.attr.state_checked]:用于可選中的視圖,表示視圖處于選中狀態(tài)。
- [android.R.attr.state_enabled]:視圖可用時(shí)的狀態(tài)。
- [android.R.attr.state_hovered]:視圖被懸停時(shí)的狀態(tài)。
- [android.R.attr.state_activated]:用于用作活動(dòng)項(xiàng)目的視圖。
 *
 */
fun ColorStateList.getColorForState(@AttrRes flag: Int, @ColorInt defaultColor: Int): Int {
    val array = intArrayOf(flag)
    return getColorForState(array, defaultColor)
}

動(dòng)態(tài)構(gòu)造ColorStateList對象有兩種方法:

  • ColorStateList.valueOf()
  • ColorStateList.createFromInt()
//第一種方法
val colors = intArrayOf(Color.RED, Color.GREEN, Color.BLUE)
val states = arrayOf(
    intArrayOf(android.R.attr.state_enabled),
    intArrayOf(android.R.attr.state_pressed),
    intArrayOf()
)

val colorStateList = ColorStateList(states, colors)

//第二種方法
val colors = intArrayOf(Color.RED, Color.GREEN, Color.BLUE)
val states = arrayOf(
    intArrayOf(android.R.attr.state_enabled),
    intArrayOf(android.R.attr.state_pressed),
    intArrayOf()
)

val defaultColor = Color.BLACK

var colorStateList = ColorStateList.createFromInt(states, colors)
colorStateList = colorStateList.withDefaultColor(defaultColor)

關(guān)于ColorStateListDrawable類型

這個(gè)類名和上面的有些類型,但其是一個(gè)drawable類型,xml文件位于drawable文件夾中

與StateListDrawable有些區(qū)別的是,drawable屬性是使用的shape

<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:drawable="@drawable/shape_pressed" android:state_pressed="true" />
    <item android:drawable="@drawable/shape_enabled" android:state_enabled="true" />
    <item android:drawable="@drawable/shape_default" />
</selector>

shape_pressed內(nèi)容:

<shape xmlns:android="http://schemas.android.com/apk/res/android">
    <solid android:color="#FF0000" />
    <corners android:radius="8dp" />
    <stroke
        android:width="2dp"
        android:color="#000000" />
</shape>

到此這篇關(guān)于Android 自定義view中根據(jù)狀態(tài)修改drawable圖片的文章就介紹到這了,更多相關(guān)Android 自定義view 修改drawable圖片內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Android自定義ViewPager指示器

    Android自定義ViewPager指示器

    這篇文章主要為大家詳細(xì)介紹了Android自定義ViewPager指示器的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-11-11
  • Android?WebView升級(jí)詳細(xì)操作指南

    Android?WebView升級(jí)詳細(xì)操作指南

    Android的WebView差異化很嚴(yán)重,下面這篇文章主要給大家介紹了關(guān)于Android?WebView升級(jí)的相關(guān)資料,文中通過代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2024-07-07
  • Android ProgressBar實(shí)現(xiàn)進(jìn)度條效果

    Android ProgressBar實(shí)現(xiàn)進(jìn)度條效果

    這篇文章主要為大家詳細(xì)介紹了Android ProgressBar實(shí)現(xiàn)進(jìn)度條效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-04-04
  • Kotlin Fragment的具體使用詳解

    Kotlin Fragment的具體使用詳解

    Fragment是Android3.0后引入的一個(gè)新的API,他出現(xiàn)的初衷是為了適應(yīng)大屏幕的平板電腦, 當(dāng)然現(xiàn)在他仍然是平板APP UI設(shè)計(jì)的寵兒,而且我們普通手機(jī)開發(fā)也會(huì)加入這個(gè)Fragment, 我們可以把他看成一個(gè)小型的Activity,又稱Activity片段
    2022-10-10
  • 一個(gè)簡單的Android圓弧刷新動(dòng)畫

    一個(gè)簡單的Android圓弧刷新動(dòng)畫

    這篇文章主要為大家詳細(xì)介紹了一個(gè)簡單的Android圓弧刷新動(dòng)畫,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-09-09
  • Android 動(dòng)態(tài)改變布局實(shí)例詳解

    Android 動(dòng)態(tài)改變布局實(shí)例詳解

    這篇文章主要介紹了Android 動(dòng)態(tài)改變布局實(shí)例詳解的相關(guān)資料,這里舉例說明如何實(shí)現(xiàn)動(dòng)態(tài)改變布局的例子,幫助大家學(xué)習(xí)理解,需要的朋友可以參考下
    2016-11-11
  • Android藍(lán)牙庫FastBle的基礎(chǔ)入門使用

    Android藍(lán)牙庫FastBle的基礎(chǔ)入門使用

    這篇文章主要給大家介紹了關(guān)于Android藍(lán)牙庫FastBle的基礎(chǔ)入門使用的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2018-07-07
  • Android HTTP發(fā)送請求和接收響應(yīng)的實(shí)例代碼

    Android HTTP發(fā)送請求和接收響應(yīng)的實(shí)例代碼

    Android HTTP請求和接收響應(yīng)實(shí)例完整的Manifest文件如下,感興趣的朋友可以參考下哈,希望對大家有所幫助
    2013-06-06
  • Flutter 給列表增加下拉刷新和上滑加載更多功能

    Flutter 給列表增加下拉刷新和上滑加載更多功能

    在實(shí)際的 App 中,下拉刷新和上滑加載更多是非常常見的交互形式。在 Flutter 中,有 flutter_easyrefresh開源插件用于實(shí)現(xiàn)下拉刷新和上滑加載更多。本篇介紹了有狀態(tài)組件和 flutter_easyrefresh 的基本應(yīng)用,同時(shí)使用模擬的方式完成了異步數(shù)據(jù)加載。
    2021-05-05
  • 直接拿來用的Android刮獎(jiǎng)控件

    直接拿來用的Android刮獎(jiǎng)控件

    這篇文章主要為大家分享了可以直接拿來用的Android刮獎(jiǎng)控件,非常棒的刮獎(jiǎng)控件,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-09-09

最新評(píng)論