[android] improve how pp extends under status bar

Signed-off-by: Arnaud Vergnet <arnaud.vergnet@mailo.com>
This commit is contained in:
Arnaud Vergnet 2023-03-18 23:31:10 +01:00 committed by Roman Tsisyk
parent 94e06bdcd1
commit c293585eb7
2 changed files with 59 additions and 3 deletions

View file

@ -4,9 +4,17 @@
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent">
<View
android:id="@+id/place_page_status_bar_background"
android:layout_width="0dp"
android:layout_height="0dp"
android:layout_gravity="center_horizontal|top"
android:background="?ppBackground"
android:visibility="gone" />
<androidx.core.widget.NestedScrollViewClickFixed
android:id="@+id/placepage"
style="?attr/bottomSheetStyle"
android:elevation="0dp"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:fillViewport="true"

View file

@ -11,6 +11,7 @@ import android.view.ViewGroup;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.core.graphics.Insets;
import androidx.core.view.ViewCompat;
import androidx.core.view.WindowInsetsCompat;
import androidx.core.widget.NestedScrollViewClickFixed;
@ -31,6 +32,7 @@ import app.organicmaps.bookmarks.data.RoadWarningMarkType;
import app.organicmaps.routing.RoutingController;
import app.organicmaps.settings.RoadType;
import app.organicmaps.util.SharingUtils;
import app.organicmaps.util.ThemeUtils;
import app.organicmaps.util.UiUtils;
import app.organicmaps.util.bottomsheet.MenuBottomSheetFragment;
import app.organicmaps.util.bottomsheet.MenuBottomSheetItem;
@ -54,10 +56,12 @@ public class PlacePageController extends Fragment implements
private BottomSheetBehavior<View> mPlacePageBehavior;
private NestedScrollViewClickFixed mPlacePage;
private ViewGroup mPlacePageContainer;
private View mPlacePageStatusBarBackground;
private ViewGroup mCoordinator;
private int mViewportMinHeight;
private int mButtonsHeight;
private int mMaxButtons;
private int mRoutingHeaderHeight;
private PlacePageViewModel mViewModel;
private int mPreviewHeight;
private int mFrameHeight;
@ -72,6 +76,8 @@ public class PlacePageController extends Fragment implements
private ValueAnimator mCustomPeekHeightAnimator;
private PlacePageRouteSettingsListener mPlacePageRouteSettingsListener;
private final Observer<Integer> mPlacePageDistanceToTopObserver = this::updateStatusBarBackground;
private final BottomSheetBehavior.BottomSheetCallback mDefaultBottomSheetCallback = new BottomSheetBehavior.BottomSheetCallback()
{
@Override
@ -114,11 +120,13 @@ public class PlacePageController extends Fragment implements
mViewportMinHeight = res.getDimensionPixelSize(R.dimen.viewport_min_height);
mButtonsHeight = (int) res.getDimension(R.dimen.place_page_buttons_height);
mMaxButtons = res.getInteger(R.integer.pp_buttons_max);
mRoutingHeaderHeight = (int) res.getDimension(ThemeUtils.getResource(requireContext(), R.attr.actionBarSize));
mCoordinator = activity.findViewById(R.id.coordinator);
mPlacePage = view.findViewById(R.id.placepage);
mPlacePageContainer = view.findViewById(R.id.placepage_container);
mPlacePageBehavior = BottomSheetBehavior.from(mPlacePage);
mPlacePageStatusBarBackground = view.findViewById(R.id.place_page_status_bar_background);
mShouldCollapse = true;
@ -133,6 +141,14 @@ public class PlacePageController extends Fragment implements
ViewCompat.setOnApplyWindowInsetsListener(mPlacePage, (v, windowInsets) -> {
mCurrentWindowInsets = windowInsets;
final Insets insets = mCurrentWindowInsets.getInsets(WindowInsetsCompat.Type.systemBars());
final ViewGroup.MarginLayoutParams layoutParams = (ViewGroup.MarginLayoutParams) mPlacePageStatusBarBackground.getLayoutParams();
// Layout calculations are heavy so we compute them once then move the view from behind the place page to the status bar
layoutParams.height = insets.top;
layoutParams.width = mPlacePage.getWidth();
// Make sure the view is centered within the insets as is the place page
layoutParams.setMargins(insets.left, 0, insets.right, 0);
mPlacePageStatusBarBackground.setLayoutParams(layoutParams);
return windowInsets;
});
}
@ -209,14 +225,19 @@ public class PlacePageController extends Fragment implements
private void setPlacePageHeightBounds()
{
final int peekHeight = calculatePeekHeight();
final Insets insets = mCurrentWindowInsets.getInsets(WindowInsetsCompat.Type.systemBars());
// Make sure the place page can reach the peek height
final int height = Math.max(peekHeight, mFrameHeight);
final int minHeight = Math.max(peekHeight, mFrameHeight);
// Prevent the place page from showing under the status bar
// If we are in planning mode, prevent going above the header
final int topInsets = insets.top + (RoutingController.get().isPlanning() ? mRoutingHeaderHeight : 0);
final int maxHeight = Math.min(minHeight + insets.bottom, mCoordinator.getHeight() - topInsets);
// Set the minimum height of the place page to prevent jumps when new data results in SMALLER content
// This cannot be set on the place page itself as it has the fitToContent property set
mPlacePageContainer.setMinimumHeight(height);
mPlacePageContainer.setMinimumHeight(minHeight);
// Set the maximum height of the place page to prevent jumps when new data results in BIGGER content
// It does not take into account the navigation bar height so we need to add it manually
mPlacePageBehavior.setMaxHeight(height + mCurrentWindowInsets.getInsets(WindowInsetsCompat.Type.systemBars()).bottom);
mPlacePageBehavior.setMaxHeight(maxHeight);
}
/**
@ -294,6 +315,7 @@ public class PlacePageController extends Fragment implements
mPreviewHeight = previewHeight;
mFrameHeight = frameHeight;
mViewModel.setPlacePageWidth(mPlacePage.getWidth());
mPlacePageStatusBarBackground.getLayoutParams().width = mPlacePage.getWidth();
// Make sure to update the peek height on the UI thread to prevent weird animation jumps
mPlacePage.post(() -> {
setPeekHeight();
@ -568,12 +590,37 @@ public class PlacePageController extends Fragment implements
close();
}
private void updateStatusBarBackground(int distanceToTop)
{
// This callback may be called before insets are updated when resuming the app
if (mCurrentWindowInsets == null)
return;
final int topInset = mCurrentWindowInsets.getInsets(WindowInsetsCompat.Type.systemBars()).top;
// Only animate the status bar background if the place page can reach it
if (mCoordinator.getHeight() - mPlacePageContainer.getHeight() < topInset)
{
final int animationStartHeight = topInset * 3;
int newHeight = 0;
if (distanceToTop < animationStartHeight)
newHeight = Math.min(topInset * (animationStartHeight - distanceToTop) / 100, topInset);
if (newHeight > 0)
{
mPlacePageStatusBarBackground.setTranslationY(distanceToTop - newHeight);
if (!UiUtils.isVisible(mPlacePageStatusBarBackground))
UiUtils.show(mPlacePageStatusBarBackground);
}
else if (UiUtils.isVisible(mPlacePageStatusBarBackground))
UiUtils.hide(mPlacePageStatusBarBackground);
}
}
@Override
public void onStart()
{
super.onStart();
mPlacePageBehavior.addBottomSheetCallback(mDefaultBottomSheetCallback);
mViewModel.getMapObject().observe(requireActivity(), this);
mViewModel.getPlacePageDistanceToTop().observe(requireActivity(), mPlacePageDistanceToTopObserver);
}
@Override
@ -590,6 +637,7 @@ public class PlacePageController extends Fragment implements
super.onStop();
mPlacePageBehavior.removeBottomSheetCallback(mDefaultBottomSheetCallback);
mViewModel.getMapObject().removeObserver(this);
mViewModel.getPlacePageDistanceToTop().removeObserver(mPlacePageDistanceToTopObserver);
}
public interface PlacePageRouteSettingsListener