diff --git a/android/app/src/main/java/app/organicmaps/routing/NavigationController.java b/android/app/src/main/java/app/organicmaps/routing/NavigationController.java index 1c4daa7360..843793731f 100644 --- a/android/app/src/main/java/app/organicmaps/routing/NavigationController.java +++ b/android/app/src/main/java/app/organicmaps/routing/NavigationController.java @@ -1,5 +1,6 @@ package app.organicmaps.routing; +import android.location.Location; import android.text.TextUtils; import android.view.View; import android.widget.ImageView; @@ -13,11 +14,13 @@ import androidx.core.view.ViewCompat; import androidx.core.view.WindowInsetsCompat; import app.organicmaps.Framework; import app.organicmaps.R; +import app.organicmaps.location.LocationHelper; import app.organicmaps.maplayer.traffic.TrafficManager; import app.organicmaps.util.UiUtils; import app.organicmaps.util.Utils; -import app.organicmaps.util.WindowInsetUtils; import app.organicmaps.widget.LanesView; +import app.organicmaps.widget.SpeedLimitView; +import app.organicmaps.util.WindowInsetUtils; import app.organicmaps.widget.menu.NavMenu; import com.google.android.material.bottomsheet.BottomSheetBehavior; @@ -38,6 +41,8 @@ public class NavigationController implements TrafficManager.TrafficCallback, @NonNull private final LanesView mLanesView; + @NonNull + private final SpeedLimitView mSpeedLimit; private final NavMenu mNavMenu; View.OnClickListener mOnSettingsClickListener; @@ -75,6 +80,8 @@ public class NavigationController implements TrafficManager.TrafficCallback, mLanesView = topFrame.findViewById(R.id.lanes); + mSpeedLimit = topFrame.findViewById(R.id.nav_speed_limit); + // Show a blank view below the navbar to hide the menu content final View navigationBarBackground = mFrame.findViewById(R.id.nav_bottom_sheet_nav_bar); final View nextTurnContainer = mFrame.findViewById(R.id.nav_next_turn_container); @@ -103,11 +110,13 @@ public class NavigationController implements TrafficManager.TrafficCallback, else UiUtils.hide(mCircleExit); - UiUtils.showIf(info.nextCarDirection.containsNextTurn(), mNextNextTurnFrame); + UiUtils.visibleIf(info.nextCarDirection.containsNextTurn(), mNextNextTurnFrame); if (info.nextCarDirection.containsNextTurn()) info.nextCarDirection.setNextTurnDrawable(mNextNextTurnImage); mLanesView.setLanes(info.lanes); + + updateSpeedLimit(info); } private void updatePedestrian(@NonNull RoutingInfo info) @@ -236,4 +245,14 @@ public class NavigationController implements TrafficManager.TrafficCallback, RoutingController.get().cancel(); } + private void updateSpeedLimit(@NonNull final RoutingInfo info) + { + final Location location = LocationHelper.from(mFrame.getContext()).getSavedLocation(); + if (location == null) { + mSpeedLimit.setSpeedLimitMps(0); + return; + } + mSpeedLimit.setCurrentSpeed(location.getSpeed()); + mSpeedLimit.setSpeedLimitMps(info.speedLimitMps); + } } diff --git a/android/app/src/main/java/app/organicmaps/widget/SpeedLimitView.java b/android/app/src/main/java/app/organicmaps/widget/SpeedLimitView.java new file mode 100644 index 0000000000..4cc8b2ff43 --- /dev/null +++ b/android/app/src/main/java/app/organicmaps/widget/SpeedLimitView.java @@ -0,0 +1,221 @@ +package app.organicmaps.widget; + +import android.content.Context; +import android.content.res.TypedArray; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.Paint; +import android.graphics.Rect; +import android.graphics.Typeface; +import android.util.AttributeSet; +import android.util.Pair; +import android.view.View; + +import androidx.annotation.ColorInt; +import androidx.annotation.NonNull; +import androidx.annotation.Nullable; +import androidx.core.content.ContextCompat; + +import app.organicmaps.R; +import app.organicmaps.util.StringUtils; + +public class SpeedLimitView extends View +{ + private interface DefaultValues + { + @ColorInt + int BACKGROUND_COLOR = Color.WHITE; + @ColorInt + int TEXT_COLOR = Color.BLACK; + @ColorInt + int TEXT_ALERT_COLOR = Color.WHITE; + + float BORDER_WIDTH_RATIO = 0.1f; + } + + @ColorInt + private final int mBackgroundColor; + + @ColorInt + private final int mBorderColor; + + @ColorInt + private final int mAlertColor; + + @ColorInt + private final int mTextColor; + + @ColorInt + private final int mTextAlertColor; + + @NonNull + private final Paint mSignBackgroundPaint; + @NonNull + private final Paint mSignBorderPaint; + @NonNull + private final Paint mTextPaint; + + private float mWidth; + private float mHeight; + private float mBackgroundRadius; + private float mBorderRadius; + private float mBorderWidth; + + private double mSpeedLimitMps; + @Nullable + private String mSpeedLimitStr; + + private double mCurrentSpeed; + + public SpeedLimitView(Context context, @Nullable AttributeSet attrs) + { + super(context, attrs); + + try (TypedArray data = context.getTheme() + .obtainStyledAttributes(attrs, R.styleable.SpeedLimitView, 0, 0)) + { + mBackgroundColor = data.getColor(R.styleable.SpeedLimitView_BackgroundColor, DefaultValues.BACKGROUND_COLOR); + mBorderColor = data.getColor(R.styleable.SpeedLimitView_borderColor, ContextCompat.getColor(context, R.color.base_red)); + mAlertColor = data.getColor(R.styleable.SpeedLimitView_alertColor, ContextCompat.getColor(context, R.color.base_red)); + mTextColor = data.getColor(R.styleable.SpeedLimitView_textColor, DefaultValues.TEXT_COLOR); + mTextAlertColor = data.getColor(R.styleable.SpeedLimitView_textAlertColor, DefaultValues.TEXT_ALERT_COLOR); + if (isInEditMode()) + { + mSpeedLimitMps = data.getInt(R.styleable.SpeedLimitView_editModeSpeedLimit, -1); + mSpeedLimitStr = mSpeedLimitMps > 0 ? String.valueOf(((int) mSpeedLimitMps)) : null; + mCurrentSpeed = data.getInt(R.styleable.SpeedLimitView_editModeCurrentSpeed, -1); + } + } + + mSignBackgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + mSignBackgroundPaint.setColor(mBackgroundColor); + + mSignBorderPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + mSignBorderPaint.setColor(mBorderColor); + mSignBorderPaint.setStrokeWidth(mBorderWidth); + mSignBorderPaint.setStyle(Paint.Style.STROKE); + + mTextPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + mTextPaint.setColor(mTextColor); + mTextPaint.setTextAlign(Paint.Align.CENTER); + mTextPaint.setTypeface(Typeface.create(Typeface.DEFAULT, Typeface.BOLD)); + } + + public void setSpeedLimitMps(final double speedLimitMps) + { + if (mSpeedLimitMps == speedLimitMps) + return; + + mSpeedLimitMps = speedLimitMps; + if (mSpeedLimitMps <= 0) + { + mSpeedLimitStr = null; + setVisibility(GONE); + return; + } + + final Pair speedLimitAndUnits = StringUtils.nativeFormatSpeedAndUnits(mSpeedLimitMps); + setVisibility(VISIBLE); + mSpeedLimitStr = speedLimitAndUnits.first; + configureTextSize(); + invalidate(); + } + + public void setCurrentSpeed(final double currentSpeed) + { + mCurrentSpeed = currentSpeed; + invalidate(); + } + + @Override + protected void onDraw(@NonNull Canvas canvas) + { + super.onDraw(canvas); + + final boolean alert = mCurrentSpeed > mSpeedLimitMps && mSpeedLimitMps > 0; + + final float cx = mWidth / 2; + final float cy = mHeight / 2; + + drawSign(canvas, cx, cy, alert); + drawText(canvas, cx, cy, alert); + } + + private void drawSign(@NonNull Canvas canvas, float cx, float cy, boolean alert) + { + if (alert) + mSignBackgroundPaint.setColor(mAlertColor); + else + mSignBackgroundPaint.setColor(mBackgroundColor); + + canvas.drawCircle(cx, cy, mBackgroundRadius, mSignBackgroundPaint); + if (!alert) + { + mSignBorderPaint.setStrokeWidth(mBorderWidth); + canvas.drawCircle(cx, cy, mBorderRadius, mSignBorderPaint); + } + } + + private void drawText(@NonNull Canvas canvas, float cx, float cy, boolean alert) + { + if (mSpeedLimitStr == null) + return; + + if (alert) + mTextPaint.setColor(mTextAlertColor); + else + mTextPaint.setColor(mTextColor); + + final Rect textBounds = new Rect(); + mTextPaint.getTextBounds(mSpeedLimitStr, 0, mSpeedLimitStr.length(), textBounds); + final float textY = cy - textBounds.exactCenterY(); + canvas.drawText(mSpeedLimitStr, cx, textY, mTextPaint); + } + + @Override + protected void onSizeChanged(int w, int h, int oldw, int oldh) + { + super.onSizeChanged(w, h, oldw, oldh); + + final float paddingX = (float) (getPaddingLeft() + getPaddingRight()); + final float paddingY = (float) (getPaddingTop() + getPaddingBottom()); + + mWidth = (float) w - paddingX; + mHeight = (float) h - paddingY; + mBackgroundRadius = Math.min(mWidth, mHeight) / 2; + mBorderWidth = mBackgroundRadius * 2 * DefaultValues.BORDER_WIDTH_RATIO; + mBorderRadius = mBackgroundRadius - mBorderWidth / 2; + configureTextSize(); + } + + // Apply binary search to determine the optimal text size that fits within the circular boundary. + private void configureTextSize() + { + if (mSpeedLimitStr == null) + return; + + final String text = mSpeedLimitStr; + final float textRadius = mBorderRadius - mBorderWidth; + final float textMaxSize = 2 * textRadius; + final float textMaxSizeSquared = (float) Math.pow(textMaxSize, 2); + + float lowerBound = 0; + float upperBound = textMaxSize; + float textSize = textMaxSize; + final Rect textBounds = new Rect(); + + while (lowerBound <= upperBound) + { + textSize = (lowerBound + upperBound) / 2; + mTextPaint.setTextSize(textSize); + mTextPaint.getTextBounds(text, 0, text.length(), textBounds); + + if (Math.pow(textBounds.width(), 2) + Math.pow(textBounds.height(), 2) <= textMaxSizeSquared) + lowerBound = textSize + 1; + else + upperBound = textSize - 1; + } + + mTextPaint.setTextSize(Math.max(1, textSize)); + } +} diff --git a/android/app/src/main/java/app/organicmaps/widget/menu/NavMenu.java b/android/app/src/main/java/app/organicmaps/widget/menu/NavMenu.java index 763a3c86d1..9e2b8612b7 100644 --- a/android/app/src/main/java/app/organicmaps/widget/menu/NavMenu.java +++ b/android/app/src/main/java/app/organicmaps/widget/menu/NavMenu.java @@ -213,14 +213,7 @@ public class NavMenu return; Pair speedAndUnits = StringUtils.nativeFormatSpeedAndUnits(last.getSpeed()); - - if (info.speedLimitMps > 0.0) - { - Pair speedLimitAndUnits = StringUtils.nativeFormatSpeedAndUnits(info.speedLimitMps); - mSpeedValue.setText(speedAndUnits.first + "\u202F/\u202F" + speedLimitAndUnits.first); - } - else - mSpeedValue.setText(speedAndUnits.first); + mSpeedValue.setText(speedAndUnits.first); if (info.speedLimitMps > 0.0 && last.getSpeed() > info.speedLimitMps) { diff --git a/android/app/src/main/res/layout-land/layout_nav_top.xml b/android/app/src/main/res/layout-land/layout_nav_top.xml index 94c07dd6c9..07fef56f84 100644 --- a/android/app/src/main/res/layout-land/layout_nav_top.xml +++ b/android/app/src/main/res/layout-land/layout_nav_top.xml @@ -110,12 +110,11 @@ android:id="@+id/lanes" android:layout_width="0dp" android:layout_height="68dp" - android:layout_marginEnd="@dimen/margin_half" - android:layout_marginTop="@dimen/margin_half" + android:layout_margin="@dimen/margin_half" android:padding="@dimen/margin_half" android:visibility="gone" + app:layout_constraintStart_toEndOf="@+id/nav_speed_limit" app:layout_constraintEnd_toEndOf="parent" - app:layout_constraintStart_toEndOf="@id/nav_next_turn_container" app:layout_constraintTop_toBottomOf="@id/street_frame" app:activeLaneTintColor="?navLaneArrowActiveColor" app:inactiveLaneTintColor="?navLaneArrowInactiveColor" @@ -123,4 +122,14 @@ app:cornerRadius="@dimen/margin_quarter" app:editModeLanesCount="10" tools:visibility="visible" /> + + diff --git a/android/app/src/main/res/layout/layout_nav_top.xml b/android/app/src/main/res/layout/layout_nav_top.xml index 64d19a315c..e86c6ac075 100644 --- a/android/app/src/main/res/layout/layout_nav_top.xml +++ b/android/app/src/main/res/layout/layout_nav_top.xml @@ -94,7 +94,7 @@ android:layout_alignEnd="@id/nav_next_turn_frame" android:background="?navNextNextTurnFrame" android:elevation="@dimen/nav_elevation" - android:visibility="gone" + android:visibility="invisible" tools:visibility="visible"> + + diff --git a/android/app/src/main/res/values/attrs.xml b/android/app/src/main/res/values/attrs.xml index 60530de713..e5120a01d5 100644 --- a/android/app/src/main/res/values/attrs.xml +++ b/android/app/src/main/res/values/attrs.xml @@ -1,5 +1,16 @@ + + + + + + + + + + +