封装了一些通用工具类,可以直接使用,不用每次再单独复制粘贴
- 在gradle中:
implementation 'io.github.liyuhaolol:CommonUtils:1.5.8'
- 修正FitLayout的Padding属性不生效的问题
- !!重要!!根据GooglePlay商店上架要求不再允许在代码中使用
android.view.Window.setStatusBarColor
android.view.Window.setNavigationBarColor
android.view.Window.setNavigationBarDividerColor
LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES
等代码设置属性实现沉浸式 - 全面使用安卓15的新API
EdgetoEdge
来实现沉浸式,避免出现被GooglePlay下架的问题。 - 暂时保留使用themes.xml中配置文件设置状态栏和导航栏颜色的设置。因为Android12以下的设备,如果不通过xml进行配置。冷启动启动页会有状态栏和导航栏的占位色,影响沉浸式的实现。
- 如果后续发现GooglePlay也不允许使用theme.xml来设置。则会在范例里去除。
- 沉浸式开发推荐
minSdk=26
安卓8.0来进行开发。因为低于这个版本沉浸式的实现UI上多多少少都会有不完美的地方。- 安卓5.0,安卓5.1状态栏导航栏会有阴影,状态栏文字和导航栏功能键不能进行深浅色转。
- 安卓6.0,安卓7.0,安卓7.1.1导航栏功能键不能进行深浅色转。
- 以上问题均会导致UI与设计预想不符或者影响使用,不推荐在兼容这些版本。
- 假设在
minSdk=26
时,沉浸式适配依然存在一些需要注意的地方- 安卓8.0时,themes.xml中不能控制导航栏的功能键深浅色转,必须在代码中设置导航栏的功能键色转。
- 安卓8.0到安卓11,冷启动如果想避免显示theme.xml里默认的状态栏和导航栏颜色,则必须在theme.xml中把这些颜色设置为透明。
- 安卓8.0到安卓14,想要设置沉浸式,需要使用
EdgetoEdge
的api来实现。本库封装方法为Edge2Edge.enable()
- 所以如果你的项目
minSdk=35
也就是最低版本是安卓15,则你不需要EdgetoEdge
类来实现沉浸式。因为安卓15默认强制启用沉浸式,不要写任何代码或添加任何配置。 - 加入
Edge2Edge.enable()
方法来替换此前的TranslucentUtils
类的沉浸式启用方法。 - 保留旧的
TranslucentUtils
类,删除GooglePlay禁止使用的Api,并且进行少量适配修改。确保此类用来给Dialog
实现沉浸式,不再支持给Activity
实现沉浸式,Activity
沉浸式请参考上条。
- 修正多个layout不能调用属性的问题
FullDialog
添加默认初始化activity的构造方法
- 增加多个layout,可以在布局里完成单边沉浸式样式
- 增加
window.setStatusBarContrastEnforced(false);
似乎Android没有启用这个代码逻辑,但是为了避免某些系统出现偶发问题,统一加上。 - 修改了弹出键盘的默认参数
- 适配Andriod15的沉浸式变动
- 由于
jitpack.io
存在无法查看源码的BUG,且长时间未修复。所以废弃该平台,转而将库发布在mavenCentral
上
- FullDialog修正快速多次调用show()会造成app崩溃的问题
- FullDialog增加isShowing方法
- 移除关于
setBackgroundId()
相关的所有逻辑,全部交给FullDialog
自动处理,逻辑与之前保持一致
FullDialog
调整背景点击区域响应范围,避免点击到UI内也会响应setCanceledOnTouchOutside()
的问题
- 增加继承自DialogFragment的FullDialog基类
- 默认dialog可用空间达到全屏,去除了dialog默认的padding以及对状态栏和导航栏的限制
- 默认开启了沉浸式,所以UI的实际显示区域不在安全区里,如果需求出现UI下沉到导航栏或上抬到状态栏里,需要用户自己按照需求在不布局中添加状态栏和导航栏的占位View并托管给FullDialog处理
isUIimmerseNavbar()
默认为false,所有的导航栏占位View都会占用高度,返回true则小白条状态下,导航栏占位View不会占用高度setStyleId()
非必传,设置Dialog的主题idsetBackgroundId()
非必传(但是推荐传)设置Dialog布局的根布局Id,如果不返回id,则点击灰色(空白)区域关闭Dialog方法永远无法生效,由于开启了全屏沉浸,并把显示区域拓展到全屏,Dialog原本的灰色遮罩已被完全遮罩,无法响应任何事件,故出此下策setStatusBarId()
非必传,根据需求如果需要出现顶部状态栏高度占位View的时候,则传递其id,高度由父类FullDialog
自动负责匹配setNavigationBarId()
非必传,根据需求如果出现弹窗紧靠下方导航栏,导航栏需要跟随UI的颜色时,需要添加导航栏高度占位View,则传递其id,高度由父类FullDialog
自动负责匹配
- Dialog的UI从
onCreateView()
方法中添加,与Fragment
添加布局方式一致 - 用户需要自己根据UI颜色来判读,是否需要状态栏和导航栏的文字转色,转色逻辑必须写在
onResume()
里的super
后面,写在更前面的生命周期里,FullDialog
无法获取正确的Window对象,所有设置无法生效
- 将InsetsController替换为WindowInsetsController,应该从根源上避免了日志报异常的问题
- 修正InsetsController会报NULL崩溃的问题
- 移除状态栏对老MIUI的支持
- 状态栏和导航栏文字转色增加对Android11以上新API的支持
- 修正华为系手机(华为,荣耀,Nova)高版本系统,状态栏永远不会是透明的问题。
- 增加获取线程名称的方法
- 进一步处理Context为空的问题
- NTP服务器时间加入Context为空时的逻辑,避免空指针
- NTP服务器时间加入持久化避免竞速问题
- 添加NTP服务器校对时间模块
- 增加使用View来控制软键盘的打开关闭,增强使用场景兼容性
- 获取状态栏高度换为Context增强兼容性
- 中间更新了什么不记得了,这个版本修改dp与px互相转换的方法,修正逻辑错误。px必须为int,dp可以为float
- 增加计算时间的国际化方法
- 尝试修正一些虚拟按键手机,会认为自己有物理按键,造成导航栏判断失败的问题
- 尝试修正存在物理按键的手机,导航栏判断存在错误的问题
- 重写自适应状态栏导航栏的相关代码,增强兼容性,保证可以在系统改变时跟随。
- 恢复刘海的判断,我这里没有足够的机型测试,不过错误似乎只发生在某些机型的虚拟机上,真机未发生此问题。
- 去除自定义Toast,因为Android11彻底封禁自定义Toast的可能
- 沉浸式使用刘海区域无论沉浸的是状态栏,还是导航栏
- 去除判断是否存在刘海的判断逻辑,在Android11上似乎判断的结果有问题
- 更换源到jitpack
- 修正沉浸式启动顺序错误,造成某些弱智手机OS会全部黑掉的问题
- 增加导航栏种类的判断,判断阈值是0.03
- 增加状态栏,导航栏对View的直接支持,用来兼容Fragment中的适配问题
- 增加Android9.0以上对刘海屏的判断
- 沉浸式修正对输入法的一些兼容
- 移除状态栏对flyme老版本的支持
- 移除状态栏对colorOS老版本的支持
- 移除沉浸式对activity的依赖
- 修正Android11的导航栏阴影问题
- 修正高版本Android导航栏高度问题
- 更新最新lib库
- 添加兼容性读取res文件夹内容的方法
- 添加状态栏的显示和隐藏
- 修正透明遮罩为黑色底
- 增加navigationBar的相关适配
- 修正一些错误
- 增加一些类的方法
- 修正一些类的名字,避免冲突,或者理解困难
implementation 'com.google.android.material:material'
material类
- encrypt();
/**
* @param seed 密钥
* @param cleartext 明文
* @return 密文
*/
String encrypt(String seed, String cleartext)
- decrypt();
/**
* @param seed 密钥
* @param cleartext 密文
* @return 明文
*/
String encrypt(String seed, String cleartext)
- isApkInDebug();
/**
* @param context 上下文
* @return true代表是debug,false代表是realse
*/
boolean isApkInDebug(Context context)
- checkDeviceHasNavigationBar();
/**
* @param context 上下文
* @return true代表有虚拟按键或者其他模式的导航栏,false代表没有虚拟按键等,但也不代表有物理按键
*/
boolean checkDeviceHasNavigationBar(Context context)
- getImageFromAssetsFile();
/**
* @param context 上下文
* @param fileName 图片文件名
* @return 对应图片的bitmap
*/
Bitmap getImageFromAssetsFile(Context context, String fileName)
- getImageFromRes();
/**
* @param context 上下文
* @param resId 图片资源id
* @return 对应图片的bitmap
*/
Bitmap getImageFromRes(Context context, int resId)
- isNetworkAvailable();
/**
* 最低运行版本是Andriod6.0
* @param context 上下文
* @return 对应图片的bitmap
*/
@TargetApi(Build.VERSION_CODES.M)
boolean isNetworkAvailable(Context context)
- stopGIF();
/**
* @param imageView imageview对象
*/
stopGIF(ImageView imageView)
- startGIF();
/**
* @param imageView imageview对象
*/
startGIF(ImageView imageView)
- floatToInt();
/**
* @param f float数
* @return 四舍五入后的int数
*/
int floatToInt(float f)
- dip2px();
/**
* @param context 上下文
* @param dpValue dp值
* @return px值
*/
int dip2px(Context context, float dpValue)
- px2dip();
/**
* @param context 上下文
* @param pxValue px值
* @return dp值
*/
int px2dip(Context context, float pxValue)
- getStatusBarHeight();
/**
* @param context 上下文
* @return 状态栏高度
*/
int getStatusBarHeight(Context context)
- getNavigationBarHeight();
/**
* @param context 上下文
* @return 导航栏高度
*/
int getNavigationBarHeight(Context context)
- getLineMaxNumber();
/**
* 最低运行版本是Andriod6.0
* @param text 文本内容
* @param paint 要判断的textview的textpaint类
* @param maxWidth textview所能达到的最大宽度
* @param lines 第几行
* @return 最多显示的字数
*/
@TargetApi(Build.VERSION_CODES.M)
int getLineMaxNumber(String text, TextPaint paint, int maxWidth, int lines)
- setEditTextInhibitInputSpace();
/**
* @param editText edittext对象
*/
setEditTextInhibitInputSpace(EditText editText)
- setEditTextInhibitInputSpeChat();
/**
* 特殊字符为: `~!@#$%^&*()+=|{}':;',\[\].<>/?~!@#¥%……&*()——+|{}【】‘;:”“’。,、?
* @param editText edittext对象
*/
setEditTextInhibitInputSpeChat(EditText editText)
- openKeybord();
/**
* @param mContext 上下文
*/
openKeybord(Activity mContext)
- closeKeybord();
/**
* @param mContext 上下文
*/
closeKeybord(Activity mContext)
- getGapTime();
/**
* @param time 毫秒的时间
* @return 转换为分:秒
*/
String getGapTime(long time)
/**
* 对应的方法顺序为
* 写入boolean
* 读取boolean
* 写入string
* 读取string
* 写入int
* 读取int
* 写入long
* 读取long
* 移除某一项
* 完全移除sp的内容
*/
void putBoolean(String key, boolean value, Context context)
boolean getBoolean(String key, boolean defValue, Context context)
void putString(String key, String value, Context context)
String getString(String key, String defValue, Context context)
void putInt(String key, int value, Context context)
int getInt(String key, int defValue, Context context)
void putLong(String key, long value, Context context)
long getLong(String key, int defValue, Context context)
void remove(String key, Context context)
void clear(Context context)
/**
* 只沉浸状态栏
* Android4.4的手机为阴影沉浸
* Android5.0以上为完全透明沉浸
*/
void setTranslucentTOP(Activity activity)
/**
* 只沉浸导航栏
* Android7.0以下的手机为阴影沉浸
* Android8.0以上为完全透明沉浸
*/
void setTranslucentBottom(Activity activity)
/**
* 同时沉浸状态栏和导航栏
* Android4.4的手机为阴影渐变沉浸
* Android5.0以上,Android7.0以下为完全透明状态栏,阴影沉浸导航栏
* Android8.0以上为完全透明沉浸
*/
void setTranslucentBoth(Activity activity)
/**
* 自动撑状态栏的高度
* @param statusBarId 状态栏位置的ResId
*/
void autoFitStatusBar(Activity activity, int statusBarId)
/**
* 自动撑导航栏的高度
* @param navigationBarId 导航栏位置的ResId
*/
void autoFitNavBar(Activity activity, int navigationBarId)
/**
* 同时撑导航栏和状态栏
* @param statusBarId 状态栏位置的ResId
* @param navigationBarId 导航栏位置的ResId
*/
void autoFitBothBar(Activity activity, int statusBarId, int navigationBarId)
/**
* 隐藏导航栏
* @param activity 上下文
*/
void hideNavigationBar(Activity activity)
/**
* 显示状态栏
* @param activity 上下文
*/
void showStatusBar(Activity activity)
/**
* 隐藏状态栏
* @param activity 上下文
*/
void hideStatusBar(Activity activity)
- setStatusBarMode();
/**
* 动态适配状态栏的字体颜色,大多数情况实在Android6.0以上生效
* @param activity 上下文
* @param darkFont 是否为深色字体
* @return 是否匹配成功,true代表执行了某个区块的逻辑,false代表没有匹配成功
*/
boolean setStatusBarMode(Activity activity, boolean darkFont)
- setNavBarMode();
/**
* Android8.0以上才会生效,与沉浸式工具配合使用
* @param activity 上下文
* @param darkFont 是否为深色字体
*/
void setNavBarMode(Activity activity,boolean darkFont)
-
AutoScrollView
继承自ScrollView,用来解决ScrollView嵌套RecyclerView时,在Android5.0以上设备丢失滑动惯性问题
使用方法:在xml中插入view,actvity中注册实例使用 -
EmptyItemAnimator
继承自SimpleItemAnimator,用来去除recyclerview刷新时闪烁的动画
使用方法:Recyclerview.setItemAnimator(new EmptyItemAnimator()); -
PopupWindowAndroidN
继承自PopupWindow,用来解决Android7.0这个版本popupwindows设置位置无效等问题
使用方法:与PopupWindow的使用方法一样 -
TopCropImageView
继承自AppCompatImageView,用来对图片进行TopCorp裁切,保证顶部不被裁切掉,应用场景:后台不按规范传图保证不切掉人物脑袋
使用方法:在xml中插入view,actvity中注册实例使用 -
Toast
这个Toast类是专门用来去除某些国产OS,比如MIUI,在Toast时会强制加上AppName的问题。如无需求必要性,可以不用
使用方法:与原Toast用法完全一样