Android 如何修复:当弹出菜单打开时,导航栏图标仍然显示(全屏-沉浸式粘滞)
我正在尝试实现一个全屏应用程序,用户无法访问状态栏和导航栏 最好我希望他们被完全删除,但从我所读到的这是不可能的,除非你根设备 所以我的问题基本上是:当我显示弹出菜单时,如何隐藏导航栏图标 之前 之后 到目前为止,我已经尝试:Android 如何修复:当弹出菜单打开时,导航栏图标仍然显示(全屏-沉浸式粘滞),android,android-fullscreen,android-immersive,android-navigation-bar,Android,Android Fullscreen,Android Immersive,Android Navigation Bar,我正在尝试实现一个全屏应用程序,用户无法访问状态栏和导航栏 最好我希望他们被完全删除,但从我所读到的这是不可能的,除非你根设备 所以我的问题基本上是:当我显示弹出菜单时,如何隐藏导航栏图标 之前 之后 到目前为止,我已经尝试: 在弹出菜单显示之前和之后调用hideNavigation() 在onCreate()、onResume()和onWindowFocusChanged()中调用hideNavigation() 请求将焦点转移到另一个视图 试图从下拉列表中清除焦点(尝试失败,没有真正找
- 在弹出菜单显示之前和之后调用
hideNavigation()
- 在
onCreate()、onResume()和onWindowFocusChanged()中调用
hideNavigation()
- 请求将焦点转移到另一个视图
- 试图从下拉列表中清除焦点(尝试失败,没有真正找到方法)
- 更改图标颜色,“假装”它将被隐藏(尝试失败,没有真正找到这样做的方法)
- 将
与hideNavigation()
结合使用(尝试失败,可能是我做得不对)Handler
- 正在尝试配置一些COSU/KIOSK模式选项(也许有办法完全禁用整个导航栏?我还没有找到隐藏“后退”按钮的方法)
实际结果:导航栏及其图标被隐藏,在弹出菜单显示后,图标被显示导航栏重新显示,因为在视图堆栈的顶部新绘制了一个新的DecorView(PopupDecorView),它不受之前设置的标志的影响 这里没有灵丹妙药,我的方法是通过反射深入WindowManagerGlobal并捕获peek视图,再次在其上应用系统标志,因此在弹出菜单显示后,它设法隐藏导航栏(仍然有一次从导航栏显示) 代码如下:
override fun onWindowFocusChanged(hasFocus: Boolean) {
super.onWindowFocusChanged(hasFocus)
if (hasFocus) {
hideNavigation()
} else {
// When PopupMenu appears, the current Activity looses the focus
setFlagsOnThePeekView() // Hijack to the current peek view, apply the Flags on it
}
}
@SuppressLint("PrivateApi")
fun setFlagsOnThePeekView() {
try {
val wmgClass = Class.forName("android.view.WindowManagerGlobal")
val wmgInstance = wmgClass.getMethod("getInstance").invoke(null)
val viewsField = wmgClass.getDeclaredField("mViews")
viewsField.isAccessible = true
val views = viewsField.get(wmgInstance) as ArrayList<View>
// When the popup appears, its decorView is the peek of the stack aka last item
views.last().apply {
systemUiVisibility = FLAGS
setOnSystemUiVisibilityChangeListener {
systemUiVisibility = FLAGS
}
}
} catch (e: Exception) {
e.printStackTrace()
}
}
覆盖WindowFocusChanged(hasFocus:Boolean){
super.onWindowFocusChanged(hasFocus)
如果(hasFocus){
隐藏
}否则{
//当弹出菜单出现时,当前活动将失去焦点
setFlagsOnThePeekView()//劫持到当前的peek视图,在其上应用标志
}
}
@SuppressLint(“PrivateApi”)
PeekView()上的趣味设置标志{
试一试{
val wmgClass=Class.forName(“android.view.WindowManagerGlobal”)
val wmgInstance=wmgClass.getMethod(“getInstance”).invoke(null)
val viewsField=wmgClass.getDeclaredField(“mViews”)
viewsField.isAccessible=true
val views=viewsField.get(wmgInstance)作为ArrayList
//当弹出窗口出现时,其decorView是堆栈的透视图,也就是最后一项
views.last().apply{
systemUiVisibility=标志
SetOnSystemActivityChangeListener{
systemUiVisibility=标志
}
}
}捕获(e:例外){
e、 printStackTrace()
}
}
工作非常出色!这是唯一一个没有错误的解决方案。很高兴它能帮上忙:-)使用受保护的类是绝对应该避免的,因为它可能在生产中的某些设备上不起作用。@SarthakMishra如我所说,这里还没有银弹解决方案,如果你知道的话,我们都在听。你的担忧是正确的,但也需要放在正确的背景下。您是否在不知道安装设备/操作系统的目的地的情况下公开启动KIOS应用程序我知道这是一个黑客。如果您能提及在生产应用程序中使用该解决方案的后果,那就更好了。这将有助于在android开发方面不是很先进的人了解他们注册的目的:)
fun HasTitleBar.initTitleBarWith(activity: Activity, resId: Int = R.id.titleBar) {
val langButton = activity.findViewById<View>(resId).findViewById<Button>(R.id.tbLanguageChoiceBtn)
val wrapper = ContextThemeWrapper(activity, R.style.MyPopupMenu)
val popupMenu = PopUpLanguageMenu(wrapper, langButton)
langButton.setOnClickListener {
activity.hideNavigation()
popupMenu.showMenu()
activity.hideNavigation()
}
}
class PopUpLanguageMenu constructor(context: Context, view: View) : PopupMenu(context, view) {
private var popupHelper: MenuPopupHelper
init {
val popUpMenu = PopupMenu(context, view).apply {
inflate(R.menu.menu_language_dropdown)
}
popupHelper = MenuPopupHelper(context, popUpMenu.menu as MenuBuilder, view)
popupHelper.run {
gravity = Gravity.END
setForceShowIcon(true)
}
}
fun showMenu() {
popupHelper.show()
}
}
override fun onWindowFocusChanged(hasFocus: Boolean) {
super.onWindowFocusChanged(hasFocus)
if (hasFocus) {
hideNavigation()
} else {
// When PopupMenu appears, the current Activity looses the focus
setFlagsOnThePeekView() // Hijack to the current peek view, apply the Flags on it
}
}
@SuppressLint("PrivateApi")
fun setFlagsOnThePeekView() {
try {
val wmgClass = Class.forName("android.view.WindowManagerGlobal")
val wmgInstance = wmgClass.getMethod("getInstance").invoke(null)
val viewsField = wmgClass.getDeclaredField("mViews")
viewsField.isAccessible = true
val views = viewsField.get(wmgInstance) as ArrayList<View>
// When the popup appears, its decorView is the peek of the stack aka last item
views.last().apply {
systemUiVisibility = FLAGS
setOnSystemUiVisibilityChangeListener {
systemUiVisibility = FLAGS
}
}
} catch (e: Exception) {
e.printStackTrace()
}
}