Ich kämpfe darum, das Konzept zu verstehen, dass es fitsSystemWindows
je nach Ansicht unterschiedliche Dinge bewirkt. Laut offizieller Dokumentation ist es ein
Boolesches internes Attribut zum Anpassen des Ansichtslayouts basierend auf Systemfenstern wie der Statusleiste. Wenn true, wird der Abstand dieser Ansicht angepasst , um Platz für die Systemfenster zu lassen .
View.java
Wenn ich nun die Klasse überprüfe, kann ich sehen, dass bei Einstellung auf true
die Fenstereinfügungen (Statusleiste, Navigationsleiste ...) auf die Ansichtsauffüllungen angewendet werden, was gemäß der oben zitierten Dokumentation funktioniert. Dies ist der relevante Teil des Codes:
private boolean fitSystemWindowsInt(Rect insets) {
if ((mViewFlags & FITS_SYSTEM_WINDOWS) == FITS_SYSTEM_WINDOWS) {
mUserPaddingStart = UNDEFINED_PADDING;
mUserPaddingEnd = UNDEFINED_PADDING;
Rect localInsets = sThreadLocal.get();
if (localInsets == null) {
localInsets = new Rect();
sThreadLocal.set(localInsets);
}
boolean res = computeFitSystemWindows(insets, localInsets);
mUserPaddingLeftInitial = localInsets.left;
mUserPaddingRightInitial = localInsets.right;
internalSetPadding(localInsets.left, localInsets.top,
localInsets.right, localInsets.bottom);
return res;
}
return false;
}
Mit dem neuen Materialdesign gibt es neue Klassen, die diese Flagge in großem Umfang nutzen, und hier kommt die Verwirrung. In vielen Quellen fitsSystemWindows
wird als Flag zum Setzen der Ansicht hinter den Systemleisten erwähnt. Siehe hier .
Die Dokumentation in ViewCompat.java
for setFitsSystemWindows
sagt:
Legt fest, ob diese Ansicht Systembildschirmdekorationen wie die Statusleiste berücksichtigen soll, und fügt deren Inhalt ein. Das heißt, Sie steuern, ob die Standardimplementierung von {@link View # fitSystemWindows (Rect)} ausgeführt wird. Weitere Informationen finden Sie in dieser Methode .
Bedeutet das fitsSystemWindows
einfach, dass die Funktion fitsSystemWindows()
ausgeführt wird? Die neuen Materialklassen scheinen dies nur zum Zeichnen unter der Statusleiste zu verwenden. Wenn wir uns DrawerLayout.java
den Code ansehen , können wir Folgendes sehen:
if (ViewCompat.getFitsSystemWindows(this)) {
IMPL.configureApplyInsets(this);
mStatusBarBackground = IMPL.getDefaultStatusBarBackground(context);
}
...
public static void configureApplyInsets(View drawerLayout) {
if (drawerLayout instanceof DrawerLayoutImpl) {
drawerLayout.setOnApplyWindowInsetsListener(new InsetsListener());
drawerLayout.setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_STABLE
| View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN);
}
}
Und wir sehen das gleiche Muster im neuen CoordinatorLayout
oder AppBarLayout
.
Funktioniert das nicht genau umgekehrt wie die Dokumentation für fitsSystemWindows
? In den letzten Fällen bedeutet dies , hinter die Systemstangen zu ziehen .
Wenn Sie jedoch möchten FrameLayout
, dass sich a hinter die Statusleiste zieht, reicht die Einstellung fitsSystemWindows
auf true nicht aus, da die Standardimplementierung das tut, was ursprünglich dokumentiert wurde. Sie müssen es überschreiben und die gleichen Flags wie die anderen genannten Klassen hinzufügen. Vermisse ich etwas
CoordinatorLayout
dieses Flag, verwenden, um zu schließen, ob sie hinter der Statusleiste malen sollen oder nicht. Das ist FrameLayout
zum Beispiel nicht der Fall .