web-dev-qa-db-ja.com

Androidデバイスでプログラムでソフトナビゲーションバーの可用性を検出しますか?

Androidプログラムでソフトナビゲーションバーを決定しようとしています。直接決定する方法が見つかりませんでした。ナビゲーションバーの可用性を確認する方法はありますか。

ソフトナビゲーションバーの画像はこちらです。

enter image description here

29

次の方法がうまくいき、多くのデバイスでテストされました。

public boolean hasNavBar (Resources resources)
    {
        int id = resources.getIdentifier("config_showNavigationBar", "bool", "Android");
        return id > 0 && resources.getBoolean(id);
    }

注:実際のデバイスでこのメソッドを検証しました

66

私が知っているように、あなたはそれを検出することができます

boolean hasSoftKey = ViewConfiguration.get(context).hasPermanentMenuKey();

ただし、API 14 +が必要でした


上記の解決策がうまくいかない場合は、以下の方法を試してください

public boolean isNavigationBarAvailable(){

        boolean hasBackKey = KeyCharacterMap.deviceHasKey(KeyEvent.KEYCODE_BACK);
        boolean hasHomeKey = KeyCharacterMap.deviceHasKey(KeyEvent.KEYCODE_HOME);

        return (!(hasBackKey && hasHomeKey));
    }
22
IshRoid

そのハックが、それはうまく動作します。それを試してみてください。

public static boolean hasSoftKeys(WindowManager windowManager){
  Display d = windowManager.getDefaultDisplay();

  DisplayMetrics realDisplayMetrics = new DisplayMetrics();
  d.getRealMetrics(realDisplayMetrics);  

  int realHeight = realDisplayMetrics.heightPixels;
  int realWidth = realDisplayMetrics.widthPixels;

  DisplayMetrics displayMetrics = new DisplayMetrics();
  d.getMetrics(displayMetrics);

  int displayHeight = displayMetrics.heightPixels;
  int displayWidth = displayMetrics.widthPixels;

  return (realWidth - displayWidth) > 0 || (realHeight - displayHeight) > 0;
}
14
Noman Rafique

受け入れられた答え は、ほとんどの実際のデバイスで正常に機能するはずですが、エミュレーターでは機能しません。

ただし、Android 4.0以降では、エミュレーターでも動作する内部APIがあります。 IWindowManager.hasNavigationBar() 。リフレクションを使用してアクセスできます。

/**
 * Returns {@code null} if this couldn't be determined.
 */
@TargetApi(Build.VERSION_CODES.ICE_CREAM_SANDWICH)
@SuppressLint("PrivateApi")
public static Boolean hasNavigationBar() {
    try {
        Class<?> serviceManager = Class.forName("Android.os.ServiceManager");
        IBinder serviceBinder = (IBinder)serviceManager.getMethod("getService", String.class).invoke(serviceManager, "window");
        Class<?> stub = Class.forName("Android.view.IWindowManager$Stub");
        Object windowManagerService = stub.getMethod("asInterface", IBinder.class).invoke(stub, serviceBinder);
        Method hasNavigationBar = windowManagerService.getClass().getMethod("hasNavigationBar");
        return (boolean)hasNavigationBar.invoke(windowManagerService);
    } catch (ClassNotFoundException | ClassCastException | NoSuchMethodException | SecurityException | IllegalAccessException | IllegalArgumentException | InvocationTargetException e) {
        Log.w("YOUR_TAG_HERE", "Couldn't determine whether the device has a navigation bar", e);
        return null;
    }
}
7
Sam

この方法を試してください。この方法で、ナビゲーションバーが存在するかどうかを検出できます。

@TargetApi(Build.VERSION_CODES.JELLY_BEAN_MR1)
public boolean hasNavBar(Context context) {
    Point realSize = new Point();
    Point screenSize = new Point();
    boolean hasNavBar = false;
    DisplayMetrics metrics = new DisplayMetrics();
    this.getWindowManager().getDefaultDisplay().getRealMetrics(metrics);
    realSize.x = metrics.widthPixels;
    realSize.y = metrics.heightPixels;
    getWindowManager().getDefaultDisplay().getSize(screenSize);
    if (realSize.y != screenSize.y) {
        int difference = realSize.y - screenSize.y;
        int navBarHeight = 0;
        Resources resources = context.getResources();
        int resourceId = resources.getIdentifier("navigation_bar_height", "dimen", "Android");
        if (resourceId > 0) {
            navBarHeight = resources.getDimensionPixelSize(resourceId);
        }
        if (navBarHeight != 0) {
            if (difference == navBarHeight) {
                hasNavBar = true;
            }
        }

    }
    return hasNavBar;

}
2
Riccardo

他の答えは私を助けません。ただし、特にAndroid P/Qの後に、ユーザーが画面からスワイプできるナビゲーションバーが表示されるかどうかを知ることは非常に役立ちます。この記事に遭遇しました https:/ /blog.stylingandroid.com/gesture-navigation-window-insets/ そしてそのような方法を作りました

fun hasNavBar(activity: Activity): Boolean {
    val temporaryHidden = activity.window.decorView.visibility and View.SYSTEM_UI_FLAG_HIDE_NAVIGATION != 0
    if (temporaryHidden) return false
    val decorView = activity.window.decorView
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        decorView.rootWindowInsets?.let{
            return it.stableInsetBottom != 0
        }
    }
    return true
}
0
parmakli

現在、正しい答えやその他は現実的ではありません。
ナビゲーションバーが非表示になっている「フルスクリーンディスプレイ->フルスクリーンジェスチャ」などのオプションがいくつかありますが、このメソッドはすべて、彼がいることを返します。

この方法を使用して、システムビューのサイズを確認することをお勧めします。 onCreateメソッド:

ViewCompat.setOnApplyWindowInsetsListener(findViewById(Android.R.id.content), 
  (v, insets) -> { 
      int navigationBarHeight = insets.getSystemWindowInsetBottom();
      return insets;
  });
0
Dmitriy Puchkov