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可以在
drawable
或color
中使用,如果在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内容:
<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>
参考
Android 自定义view中根据状态修改drawable图片的更多相关文章
- Android 自定义view中的属性,命名空间,以及tools标签
昨日看到有人在知乎上问这3个琐碎的小知识点,今天索性就整理了一下,其实这些知识点并不难,但是很多开发者平时很少注意到这些, 导致的后果就是开发的时候 经常会被ide报错,开发效率很低,或者看开源代码的 ...
- android 自定义view中findViewById为空的解决办法
网上说的都是在super(context, attrs);构造函数这里少加了一个字段, 其实根本不只这一个原因,属于view生命周期的应该知道,如果你在 自定义view的构造函数里面调用findVie ...
- Android 自定义 View 圆形进度条总结
Android 自定义圆形进度条总结 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 微信公众号:牙锅子 源码:CircleProgress 文中如有纰漏,欢迎大家留言指出. 最近 ...
- Android自定义View研究--View中的原点坐标和XML中布局自定义View时View触摸原点问题
这里只做个汇总~.~独一无二 文章出处:http://blog.csdn.net/djy1992/article/details/9715047 Android自定义View研究--View中的原点坐 ...
- 【朝花夕拾】Android自定义View篇之(六)Android事件分发机制(中)从源码分析事件分发逻辑及经常遇到的一些“诡异”现象
前言 转载请注明,转自[https://www.cnblogs.com/andy-songwei/p/11039252.html]谢谢! 在上一篇文章[[朝花夕拾]Android自定义View篇之(五 ...
- Android 自定义View及其在布局文件中的使用示例(三):结合Android 4.4.2_r1源码分析onMeasure过程
转载请注明出处 http://www.cnblogs.com/crashmaker/p/3549365.html From crash_coder linguowu linguowu0622@gami ...
- Android 自定义View及其在布局文件中的使用示例(二)
转载请注明出处 http://www.cnblogs.com/crashmaker/p/3530213.html From crash_coder linguowu linguowu0622@gami ...
- Android查缺补漏(View篇)--自定义 View 中 wrap_content 无效的解决方案
自定义 View 中 wrap_content 无效的解决方案 做过自定义 View 的童鞋都会发现,直接继承 View 的自定义控件需要重写 onMeasure() 方法,并设置 wrap_cont ...
- android 自定义view 前的基础知识
本篇文章是自己自学自定义view前的准备,具体参考资料来自 Android LayoutInflater原理分析,带你一步步深入了解View(一) Android视图绘制流程完全解析,带你一步步深入了 ...
- Android自定义View
转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/24252901 很多的Android入门程序猿来说对于Android自定义View ...
随机推荐
- CommunityToolkit.Mvvm8.1 viewmodel源生成器写法(3)
本系列文章导航 https://www.cnblogs.com/aierong/p/17300066.html https://github.com/aierong/WpfDemo (自我Demo地址 ...
- Ubuntu18搭建vue3
第一步我们可以先更新源(我所有的步骤都在root账户下操作的) sudo apt-get update 然后安装node sudo apt-get install nodejs 安装成功后可以查看版本 ...
- CF1738EBalance Addicts
CF1738EBalance Addicts 原题: CF1738EBalance Addicts 目录 CF1738EBalance Addicts 题目大意 做法 思路 注意 code 题目大意 ...
- 可视化大屏:mapbox+vue全攻略
如题图,mapbox是一个支持真3D地形展示的webGIS框架,与常用的Leaflet.Cesium和Openlayers并称四大框架,本文将介绍mapbox-gl js 在 vue 中的用法. 为什 ...
- [C++提高编程] 3.5 stack容器
文章目录 3.5 stack容器 3.5.1 stack 基本概念 3.5.2 stack 常用接口 3.5 stack容器 3.5.1 stack 基本概念 概念:stack是一种先进后出(Firs ...
- NC51101 Lost Cows
题目链接 题目 题目描述 \(N (2 \leq N \leq 8,000)\) cows have unique brands in the range 1..N. In a spectacular ...
- OceanBase的学习与使用
OceanBase的学习与使用 简介 1. OceanBase数据库 注意这一块下载的其实是rpm包. 一般是通过下面的OAT或者是OCP工具进行安装. 有x86还有ARM两种架构. 虽然是el7结尾 ...
- Feign踩坑源码分析--@FeignClient注入容器
一. @EnableFeignClients 1.1.类介绍 从上面注释可以看出是扫描声明了@FeignClient接口的类,还引入了 FeignClientsRegistrar类,从字面意思可以看出 ...
- Vue2积分商城项目
一.清空项目非必要文件和用户片段,路径提示的配置 views 下面的文件只保留 Home.vue ,其余删除,删除 components/HelloWorld.vue,并且 Home.vue 中不再引 ...
- 2022-04-22:给你两个正整数数组 nums 和 target ,两个数组长度相等。 在一次操作中,你可以选择两个 不同 的下标 i 和 j , 其中 0 <= i, j < nums.leng
2022-04-22:给你两个正整数数组 nums 和 target ,两个数组长度相等. 在一次操作中,你可以选择两个 不同 的下标 i 和 j , 其中 0 <= i, j < num ...