import android.annotation.SuppressLint
import android.annotation.TargetApi
import android.app.Activity
import android.content.Context
import android.graphics.Color
import android.os.Build
import android.view.View
import android.view.WindowManager
import androidx.annotation.ColorInt
import com.common.core.statusbar.StatusbarUtil.lightMode
import com.common.core.statusbar.StatusbarUtil.setStatusBarTransparent
/**
* 仅在Android版本6.0以上进行StatusBar适配,因为低于6.0的设备一是不多,二是不能全部使用白底黑字,所以不需要过于复杂的适配
* 6.0以下,请配置其他的Theme
*
* 如果仅在Activity中使用,只需要调用 [lightMode] 方法,并在布局中设置 fitsSystemWindow=true 即可
*
* 当需要在Activity+多Fragment场景下使用时,如果需要动态修改颜色,需要按照以下方式使用:
* 1. 在Activity setContentView后依次调用 [setStatusBarTransparent] [lightMode] 方法
* 2. 在每个Fragment的布局中增加一个View,用于占位,例如
* <View
* android:id="@+id/view1"
* android:layout_width="match_parent"
* android:layout_height="0dp"
* android:background="@color/colorPrimaryDark" />
* 3. 在Fragment的 onViewCreated 方法中,为以上View设置高度和背景颜色:
* view1.layoutParams.height = context?.let { StatusBarUtils.getStatusBarHeight(it) } ?: 0
* view1.setBackgroundColor(Color.RED)
* 4. 在Activity中的ViewPager切换事件中,还可以通过调用 [lightMode] 方法来跟随Fragment切换动态调整Statusbar的文本颜色
*/
object StatusbarUtil {
/**
* 6.0以下统统返回0
*/
@JvmStatic
fun getStatusBarHeightOnlyFromM(context: Context): Int {
if (isFromM()) {
val resourceId = context.resources.getIdentifier("status_bar_height", "dimen", "android")
if (resourceId > 0) {
return context.resources.getDimensionPixelSize(resourceId)
}
}
return 0
}
fun getStatusBarHeight(context: Context): Int {
val resourceId = context.resources.getIdentifier("status_bar_height", "dimen", "android")
if (resourceId > 0) {
return context.resources.getDimensionPixelSize(resourceId)
}
return 0
}
fun setStatusBarTransparent(activity: Activity) {
if (isFromM()) {
val decorView = activity.window.decorView
val option = View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN or View.SYSTEM_UI_FLAG_LAYOUT_STABLE
decorView.systemUiVisibility = option
activity.window.statusBarColor = Color.TRANSPARENT
}
}
fun setStatusBarColor(activity: Activity, @ColorInt color: Int) {
if (isFromM()) {
val decorView = activity.window.decorView
val option = View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN or View.SYSTEM_UI_FLAG_LAYOUT_STABLE
decorView.systemUiVisibility = option
activity.window.statusBarColor = color
}
}
fun lightMode(activity: Activity, dark: Boolean) {
if (isFromM()) {
try {
if (!MIUILightMode(activity, dark) && !FlymeLightMode(activity, dark)) {
SystemLightMode(activity, dark)
}
} catch (ignored: Exception) {
}
}
}
private fun isFromM(): Boolean {
return Build.VERSION.SDK_INT >= Build.VERSION_CODES.M
}
@TargetApi(Build.VERSION_CODES.M)
private fun SystemLightMode(activity: Activity, dark: Boolean) {
activity.window.decorView.systemUiVisibility = when {
dark -> View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN or View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR
else -> View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN or View.SYSTEM_UI_FLAG_VISIBLE
}
}
private fun FlymeLightMode(activity: Activity, dark: Boolean): Boolean {
var success = false
val window = activity.window
if (window != null) {
try {
val lp = window.attributes
val darkFlag = WindowManager.LayoutParams::class.java
.getDeclaredField("MEIZU_FLAG_DARK_STATUS_BAR_ICON")
val meizuFlags = WindowManager.LayoutParams::class.java
.getDeclaredField("meizuFlags")
darkFlag.isAccessible = true
meizuFlags.isAccessible = true
val bit = darkFlag.getInt(null)
var value = meizuFlags.getInt(lp)
value = if (dark) {
value or bit
} else {
value and bit.inv()
}
meizuFlags.setInt(lp, value)
window.attributes = lp
success = true
} catch (ignored: Exception) {
}
}
return success
}
@SuppressLint("PrivateApi")
private fun MIUILightMode(activity: Activity, dark: Boolean): Boolean {
var success = false
val window = activity.window
if (window != null) {
val clazz = window.javaClass
try {
val darkModeFlag: Int
val layoutParams = Class.forName("android.view.MiuiWindowManager\$LayoutParams")
val field = layoutParams.getField("EXTRA_FLAG_STATUS_BAR_DARK_MODE")
darkModeFlag = field.getInt(layoutParams)
val extraFlagField = clazz.getMethod("setExtraFlags", Int::class.javaPrimitiveType, Int::class.javaPrimitiveType)
if (dark) {
extraFlagField.invoke(window, darkModeFlag, darkModeFlag)//状态栏透明且黑色字体
} else {
extraFlagField.invoke(window, 0, darkModeFlag)//清除黑色字体
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
//开发版 7.7.13 及以后版本采用了系统API,旧方法无效但不会报错,所以两个方式都要加上
SystemLightMode(activity, dark)
}
success = true
} catch (ignored: Exception) {
}
}
return success
}
}
|