Android handles vertical and horizontal switching, bangs, and hiding the bottom navigation bar
Horizontal and vertical screen switching processing:
-
First, you need to configure orientation for your Activity in android:configChanges so that it will not be destroyed and rebuilt when switching screens. Instead, the Activity’s public void onConfigurationChanged(Configuration newConfig) method is called back.
Configure the Activity in the manifest file that needs to handle horizontal and vertical screens:
<activity
android:name=".TestActivity"
android:configChanges="screenSize|orientation"
.
/>
Copy the code
This configuration tells the system not to re-destroy the current Activity when its horizontal and vertical screens and dimensions change, but rather to call back the onConfigurationChanged method on the current Activity. The reason for adding screenSize here is that the size of the Activity will change when the Activity switches between the horizontal and vertical screens. Just using orientation will still create a new Activity.
- After processing the configuration in the Manifest, the pair
public void onConfigurationChanged(Configuration newConfig)
Method of processing.
Due to project reasons, when the horizontal and vertical screens are switched, the WebView size should be actively informed, otherwise the WebView display will be abnormal. So you need to calculate the size that you need to set to the WebView. Since our Activity can’t scale the window, we can calculate the width and height of the entire screen minus the safe distance of the fringe area of the screen, which can be problematic on some devices!
Calculate the width and height available for the current Activity:
@Override
public void onConfigurationChanged(@NonNull Configuration newConfig) {
super.onConfigurationChanged(newConfig);
View content = findViewById(android.R.id.content);
int contentWidth = content.getWidth();
int contentHeight = content.getHeight();
if (contentWidth > 0 && contentHeight > 0) {
int realWidth, realHeight;
if (newConfig.orientation == Configuration.ORIENTATION_PORTRAIT) {
realWidth = Math.min(contentWidth, contentHeight);
realHeight = Math.max(contentWidth, contentHeight);
textView.setText("Currently portrait; \r\n" + " content width = " + content.getWidth() + " content height = " + content.getHeight()+"\r\n realWidth = "+realWidth+" realHeight = "+realHeight);
} else {
realWidth = Math.max(contentWidth, contentHeight);
realHeight = Math.min(contentWidth, contentHeight);
textView.setText("Currently landscape; \r\n" + " content width = " + content.getWidth() + " content height = " + content.getHeight()+"\r\n realWidth = "+realWidth+" realHeight = "+realHeight); }}else {
ViewTreeObserver observer = content.getViewTreeObserver();
observer.addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
@Override
public boolean onPreDraw(a) {
int realWidth, realHeight;
if (newConfig.orientation == Configuration.ORIENTATION_PORTRAIT) {
realWidth = Math.min(contentWidth, contentHeight);
realHeight = Math.max(contentWidth, contentHeight);
textView.setText("Currently portrait; \r\n" + " content width = " + content.getWidth() + " content height = " + content.getHeight()+"\r\n realWidth = "+realWidth+" realHeight = "+realHeight);
} else {
realWidth = Math.max(contentWidth, contentHeight);
realHeight = Math.min(contentWidth, contentHeight);
textView.setText("Currently landscape; \r\n" + " content width = " + content.getWidth() + " content height = " + content.getHeight()+"\r\n realWidth = "+realWidth+" realHeight = "+realHeight);
}
//observer.removeOnPreDrawListener(this); There will be such problems as: Java. Lang. An IllegalStateException This ViewTreeObserver is not alive, call getViewTreeObserver () again
content.getViewTreeObserver().removeOnPreDrawListener(this);
return true; }}); } Log.e(TAG,"onConfigurationChanged ");
}
Copy the code
This gives you the true width and height that the current Activity can use.
After handling the Android :configChanges and onConfigurationChanged method for the Activity in Xml, our Activity can be switched between vertical and horizontal screens without being recreated by the system, and we can get the actual width and height that we can use. Here’s a look at the processing of liu Hai screen.
Liu Haiping’s treatment
As the fringe screen area needs to be left out in the project, that is, if it is a device with a fringe screen, the area of the fringe screen should be set as a black area, so that no content can be displayed, so as to prevent key places from being blocked.
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.e(TAG, "onCreate");
setContentView(R.layout.activity_main);
// Tell the system that the screen does not fit so that the area of the screen does not display the content of the Activity
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) { WindowManager.LayoutParams params = getWindow().getAttributes(); params.layoutInDisplayCutoutMode = WindowManager.LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_NEVER; getWindow().setAttributes(params); }... }Copy the code
The key is params. LayoutInDisplayCutoutMode = WindowManager. LayoutParams. LAYOUT_IN_DISPLAY_CUTOUT_MODE_NEVER; This code controls the display of bangs among several other things:
- LAYOUT_IN_DISPLAY_CUTOUT_MODE_DEFAULT: This is a default attribute that is automatically used if no explicit attribute is specified. This property allows the content of an application to automatically extend to the bangs in portrait mode but not to the bangs in landscape mode.
- LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES: This property indicates that the app’s content extends to the fringe area regardless of whether the phone is in landscape or portrait mode.
- LAYOUT_IN_DISPLAY_CUTOUT_MODE_NEVER: This property indicates that the content of an application is never allowed to extend beyond the fringe area.
For more information about adaptation of the Screen can be viewed in this article Guo Lin Damen Android 9.0 system new features, adaptation of the screen device
Bottom navigation bar handling
To work with the bottom navigation bar, use the Library ImmersionBar, as well as the solution provided by the official Google documentation.
Hide the bottom virtual navigation bar and call onCreate
@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Log.e(TAG, "onCreate"); setContentView(R.layout.activity_main); / / hide the virtual navigation ImmersionBar. With (this). NavigationBarColor (android. R.c olor. Transparent) .hideBar(BarHide.FLAG_HIDE_BAR).init(); .Copy the code
If you just do this, the virtual navigation bar will be brought up again when a dialog box pops up on the page, which is not perfect. To do this, call the Activity window when it gets focus:
@Override public void onWindowFocusChanged(boolean hasFocus) { super.onWindowFocusChanged(hasFocus); if (hasFocus){ ImmersionBar.with(this) .navigationBarColor(android.R.color.transparent) .hideBar(BarHide.FLAG_HIDE_BAR).init(); }}Copy the code
This way, when the dialog pops up, the Activity gains focus when the dialog closes and hides the virtual navigation bar again, perfect.
The Demo address
OrientationDemo
Reference article:
Hide the Navigation Bar
“New features of Android 9.0 system, Adaptation for Bangs screen Devices” by Guo Lin Dashen