Android 自定义view中根据状态修改drawable图片

这篇具有很好参考价值的文章主要介绍了Android 自定义view中根据状态修改drawable图片。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

原文地址:Android 自定义view中根据状态修改drawable图片 - Stars-One的杂货小窝

本文涉及知识点:

  • Android里的selector图片使用
  • 底部导航栏的使用
  • 自定义view的步骤了解

建议有以上基础有助于帮助你理解本篇文章....

起因,由于UI那边的实现,不是按照的Material Design风格设计的,设计的底部导航栏图标和文本在同一行,原本想用官方的BottomNavigation组件也没法使用,只好自己仿造地写个自定义组件

正常BottomNavigation组件,是读取menu文件来设置图标和文本从而实现数据

在自定义View中,如何根据状态去修改drawble图片?

说明

从menu菜单文件得知:通过icon属性设置一个drawble对象即可实现图标

如果你给的drawable对象为一个selector,那么在selector中正确声明了不同状态下的drawable,那么就能够实现图标在选中和未选中的图标变更,具体可以参考我之前的文章,Android BottomNavigation底部导航栏使用 - Stars-One的杂货小窝

这里xml里的selector,其实最终被Android里解析处理,得到一个StateListDrawable对象

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

仿照实现导航栏切换图标功能

前面的一些自定义view的步骤在此略过,主要讲解核心的东西

我们需要自定义view去读取menu文件里的数据,得到icon的drawble文件,并根据不同状态去取这个drawable里的图片,之后去更改我们的imageview即可

获取菜单文件icon内容:

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

获取不同状态的drawable:

先说下思路,我们view中有一个状态位标明当前是哪个item选择,当item为选择的时候,我们让item的图标展示已选中状态的drawable,反之则相反


//view中的一个状态表示
val viewIsCheck = false

if (icon is StateListDrawable) {
    val drawable = if(viewIsCheck){
        getDrawable(icon, android.R.attr.state_checked)
    }else{
        //加个-号,则表示反过来的状态(即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的扩展方法,方便之后调用,已收录在我的库中stars-one/XAndroidUtil: 封装自己常用的一些Android的组件或工具

/**
 * 获取不同状态的drawable
 * @param flag 可选数值如下
- [android.R.attr.state_pressed]:按钮被按下时的状态。
- [android.R.attr.state_focused]:视图获取焦点时的状态。
- [android.R.attr.state_selected]:视图被选中时的状态。
- [android.R.attr.state_checked]:用于可选中的视图,表示视图处于选中状态。
- [android.R.attr.state_enabled]:视图可用时的状态。
- [android.R.attr.state_hovered]:视图被悬停时的状态。
- [android.R.attr.state_activated]:用于用作活动项目的视图。
 *
 */
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
    }
}

其他补充

动态构造StateListDrawable对象

上面说到的都是从xml里读取,既然是一个类,那么我们也可以通过写代码的方式快速构造出一个StateListDrawable对象

// 创建 StateListDrawable
val stateListDrawable = StateListDrawable()

// 添加按下状态的 Drawable
val pressedDrawable = resources.getDrawable(R.drawable.pressed_bg, null)
stateListDrawable.addState(intArrayOf(android.R.attr.state_pressed), pressedDrawable)

// 添加默认状态的 Drawable
val defaultDrawable = resources.getDrawable(R.drawable.default_bg, null)
stateListDrawable.addState(intArrayOf(), defaultDrawable)

// 将 StateListDrawable 设置为 View 的背景
view.background = stateListDrawable

自定义view的reference类型

如果需要自定义view,可以在xml中设置一个menu菜单,可以声明一个属性为reference,如下代码:

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

之后在代码里使用getResourceId方法可以读取属性数据:

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);
  
}

关于颜色ColorStateList

上文也提到,我们也可以在color的资源文件夹中使用selector,这里也补充下如何读取吧

在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)

封装的扩展方法,获取某个状态的color:

/**
 * 获取selector某个状态的color
 * - selector里定义`androird:state_pressed="true"`,即为`android.R.attr.state_pressed`
 * - selector里定义`androird:state_pressed="false"`,即为`-android.R.attr.state_pressed`
 *
 * @param flag 可选数值如下: 前面加个`-`,标示为状态为false
- [android.R.attr.state_pressed]:按钮被按下时的状态。
- [android.R.attr.state_focused]:视图获取焦点时的状态。
- [android.R.attr.state_selected]:视图被选中时的状态。
- [android.R.attr.state_checked]:用于可选中的视图,表示视图处于选中状态。
- [android.R.attr.state_enabled]:视图可用时的状态。
- [android.R.attr.state_hovered]:视图被悬停时的状态。
- [android.R.attr.state_activated]:用于用作活动项目的视图。
 *
 */
fun ColorStateList.getColorForState(@AttrRes flag: Int, @ColorInt defaultColor: Int): Int {
    val array = intArrayOf(flag)
    return getColorForState(array, defaultColor)
}

动态构造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)

关于ColorStateListDrawable类型

这个类名和上面的有些类型,但其是一个drawable类型,xml文件位于drawable文件夹中

与StateListDrawable有些区别的是,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内容:文章来源地址https://www.toymoban.com/news/detail-537455.html

<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>

参考

  • Get State Drawable from StateListDrawable for Custom View on Android API<29 - Stack Overflow

到了这里,关于Android 自定义view中根据状态修改drawable图片的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Android 自定义View 之 圆环进度条

      很多时候我们会使用进度条,而Android默认的进度条是长条的,从左至右。而在日常开发中,有时候UI为了让页面更美观,就需要用到圆环进度条,那么本文就是通过自定义写一个圆环进度条,首先看一下效果图:   关于自定义View的基础知识就不再做过多的讲解了,我

    2024年02月10日
    浏览(47)
  • Android:绘制自定义View人脸识别框

    项目开发需要自定义View实现一个人脸框,代码实现很平常,一些细节记录一下,方便以后查阅。 代码实现: FaceView.java 注意: 这里我把FaceView的layout_width、layout_height都设置成了\\\"match_parent\\\" 这里面有个 坑 容易踩到 本次开发时,xml中FaceView控件往上一直到第一层父布局宽、高

    2024年02月09日
    浏览(64)
  • Android 自定义View之圆形进度条

    很多场景下都用到这种进度条,有的还带动画效果, 今天我也来写一个。 写之前先拆解下它的组成: 底层圆形 上层弧形 中间文字 那我们要做的就是: 绘制底层圆形; 在同位置绘制上层弧形,但颜色不同; 在中心点绘制文本,显示进度。 按照这个目标,学习下自定义Vi

    2024年02月09日
    浏览(44)
  • Android 自定义View 之 Dialog弹窗

      在日常开发中用到弹窗是比较多的,常用于提示作用,比如错误操作提示,余额不足提示,退出登录提示等,还有用于数据展示的弹窗,上拉弹窗等等,主要为了简化在日常开发中的使用。   Android中的Dialog弹窗是一种用于展示特定信息或者在用户需要进行某些操作时

    2024年02月16日
    浏览(43)
  • [Android]自定义RecyclerView中View的动画

    官方有一个默认Item动画类DafaultItemAnimator,其中 DefaultItemAnimator 继承了SimpleItemAnimator 继承了 RecyclerView.ItemAnimator SimpleItemAnimator 它是一个包装类,用来判断当前的ViewHolder到底是执行移动、移除、添加或者改变等行为。 DefaultItemAnimator 是执行具体动画类,它负责将viewHolder初始化

    2024年02月11日
    浏览(67)
  • Android自定义View之游戏摇杆键盘实现(一)

    public class RemoteViewBg { private Bitmap bitmapBg; public RemoteViewBg(Bitmap bitmap) { bitmapBg = bitmap; } //背景的绘图函数 public void draw(Canvas canvas, Paint paint, Rect src0 ,Rect dst0 ) { canvas.drawBitmap(bitmapBg, src0, dst0, paint); } } 重写系统的触摸时间,判断触摸点在背景范围内还是背景范围外 @Override public b

    2024年04月17日
    浏览(55)
  • Android 自定义View实战—制作一个简易输入框

    这次我们来做一个简易输入框,可以用于密码输入和验证码输入。 依然在EasyView中进行创建,在 com.easy.view 下新建一个 EasyEditText ,继承自 View ,实现里面的构造方法。 ① 构造方法 然后我们继承自 View ,重写里面的构造方法,代码如下: 下面就可以增加样式了。 ② XML样式

    2024年02月10日
    浏览(37)
  • Android自定义View之游戏摇杆键盘实现(一),快手android面试经验

    public class RemoteViewBg { private Bitmap bitmapBg; public RemoteViewBg(Bitmap bitmap) { bitmapBg = bitmap; } //背景的绘图函数 public void draw(Canvas canvas, Paint paint, Rect src0 ,Rect dst0 ) { canvas.drawBitmap(bitmapBg, src0, dst0, paint); } } 重写系统的触摸时间,判断触摸点在背景范围内还是背景范围外 @Override public b

    2024年04月12日
    浏览(46)
  • Android音视频剪辑器自定义View实战!

    Android音视频剪辑器自定义View实战! - 掘金   Android音视频剪辑器自定义View实战! - 掘金 话不多说,先上一个代码完成效果。 动图好像录成横屏的了,也没找到调整反转 GIF 的位置,下面再补一张设计稿静态图吧 最近这几年音视频应用越来越广泛,随之而来的音视频相关的需

    2024年02月12日
    浏览(42)
  • 【Android】自定义View onDraw()方法会调用两次

    自定义了View后,在构造函数中设置画笔颜色,发现它没起效,但是在onDraw()里设置颜色就会起效,出问题的代码如下: 我在构造函数中设置的画笔颜色是红色,但是实际画出的线是黑色的(画笔默认颜色是黑色),分析了一下发现是paint.reset()的问题,reset就是画笔重置设置嘛

    2024年01月19日
    浏览(32)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包