diff --git a/android/app/src/main/cpp/app/organicmaps/util/StringUtils.cpp b/android/app/src/main/cpp/app/organicmaps/util/StringUtils.cpp index e99a34b0ed..833f2fa238 100644 --- a/android/app/src/main/cpp/app/organicmaps/util/StringUtils.cpp +++ b/android/app/src/main/cpp/app/organicmaps/util/StringUtils.cpp @@ -49,6 +49,12 @@ Java_app_organicmaps_util_StringUtils_nativeFilterContainsNormalized(JNIEnv * en return jni::ToJavaStringArray(env, filtered); } +JNIEXPORT jint JNICALL Java_app_organicmaps_util_StringUtils_nativeFormatSpeed( + JNIEnv * env, jclass thiz, jdouble metersPerSecond) +{ + return measurement_utils::FormatSpeed(metersPerSecond, measurement_utils::GetMeasurementUnits()); +} + JNIEXPORT jobject JNICALL Java_app_organicmaps_util_StringUtils_nativeFormatSpeedAndUnits( JNIEnv * env, jclass thiz, jdouble metersPerSecond) { 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 843793731f..f5bfa85e5c 100644 --- a/android/app/src/main/java/app/organicmaps/routing/NavigationController.java +++ b/android/app/src/main/java/app/organicmaps/routing/NavigationController.java @@ -16,6 +16,7 @@ import app.organicmaps.Framework; import app.organicmaps.R; import app.organicmaps.location.LocationHelper; import app.organicmaps.maplayer.traffic.TrafficManager; +import app.organicmaps.util.StringUtils; import app.organicmaps.util.UiUtils; import app.organicmaps.util.Utils; import app.organicmaps.widget.LanesView; @@ -249,10 +250,10 @@ public class NavigationController implements TrafficManager.TrafficCallback, { final Location location = LocationHelper.from(mFrame.getContext()).getSavedLocation(); if (location == null) { - mSpeedLimit.setSpeedLimitMps(0); + mSpeedLimit.setSpeedLimit(0, false); return; } - mSpeedLimit.setCurrentSpeed(location.getSpeed()); - mSpeedLimit.setSpeedLimitMps(info.speedLimitMps); + final boolean speedLimitExceeded = info.speedLimitMps < location.getSpeed(); + mSpeedLimit.setSpeedLimit(StringUtils.nativeFormatSpeed(info.speedLimitMps), speedLimitExceeded); } } diff --git a/android/app/src/main/java/app/organicmaps/util/StringUtils.java b/android/app/src/main/java/app/organicmaps/util/StringUtils.java index 8f991fb30f..00129f8105 100644 --- a/android/app/src/main/java/app/organicmaps/util/StringUtils.java +++ b/android/app/src/main/java/app/organicmaps/util/StringUtils.java @@ -49,6 +49,7 @@ public class StringUtils public static native boolean nativeContainsNormalized(String str, String substr); public static native String[] nativeFilterContainsNormalized(String[] strings, String substr); + public static native int nativeFormatSpeed(double metersPerSecond); public static native Pair nativeFormatSpeedAndUnits(double metersPerSecond); public static native Distance nativeFormatDistance(double meters); @NonNull diff --git a/android/app/src/main/java/app/organicmaps/widget/LanesView.java b/android/app/src/main/java/app/organicmaps/widget/LanesView.java index 3436e032fd..b1a1a5daa0 100644 --- a/android/app/src/main/java/app/organicmaps/widget/LanesView.java +++ b/android/app/src/main/java/app/organicmaps/widget/LanesView.java @@ -59,14 +59,14 @@ public class LanesView extends View try (TypedArray data = context.getTheme().obtainStyledAttributes(attrs, R.styleable.LanesView, 0, 0)) { - backgroundColor = getAttrColor(data, R.styleable.LanesView_backgroundColor, DefaultValues.BACKGROUND_COLOR); - mActiveLaneTintColor = getAttrColor(data, R.styleable.LanesView_activeLaneTintColor, DefaultValues.ACTIVE_LANE_TINT_COLOR); - mInactiveLaneTintColor = getAttrColor(data, R.styleable.LanesView_inactiveLaneTintColor, DefaultValues.INACTIVE_LANE_TINT_COLOR); - mCornerRadius = (int) Math.max(data.getDimension(R.styleable.LanesView_cornerRadius, DefaultValues.CORNER_RADIUS), 0.0f); + backgroundColor = getAttrColor(data, R.styleable.LanesView_lanesBackgroundColor, DefaultValues.BACKGROUND_COLOR); + mActiveLaneTintColor = getAttrColor(data, R.styleable.LanesView_lanesActiveLaneTintColor, DefaultValues.ACTIVE_LANE_TINT_COLOR); + mInactiveLaneTintColor = getAttrColor(data, R.styleable.LanesView_lanesInactiveLaneTintColor, DefaultValues.INACTIVE_LANE_TINT_COLOR); + mCornerRadius = (int) Math.max(data.getDimension(R.styleable.LanesView_lanesCornerRadius, DefaultValues.CORNER_RADIUS), 0.0f); if (isInEditMode()) { - final int lanesCount = Math.max(1, data.getInt(R.styleable.LanesView_editModeLanesCount, DefaultValues.LANES_COUNT)); + final int lanesCount = Math.max(1, data.getInt(R.styleable.LanesView_lanesEditModeLanesCount, DefaultValues.LANES_COUNT)); createLanesForEditMode(lanesCount); } } diff --git a/android/app/src/main/java/app/organicmaps/widget/SpeedLimitView.java b/android/app/src/main/java/app/organicmaps/widget/SpeedLimitView.java index 4cc8b2ff43..4e71f55c5d 100644 --- a/android/app/src/main/java/app/organicmaps/widget/SpeedLimitView.java +++ b/android/app/src/main/java/app/organicmaps/widget/SpeedLimitView.java @@ -8,16 +8,14 @@ import android.graphics.Paint; import android.graphics.Rect; import android.graphics.Typeface; import android.util.AttributeSet; -import android.util.Pair; +import android.view.MotionEvent; 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 { @@ -26,6 +24,10 @@ public class SpeedLimitView extends View @ColorInt int BACKGROUND_COLOR = Color.WHITE; @ColorInt + int BORDER_COLOR = Color.RED; + @ColorInt + int ALERT_COLOR = Color.RED; + @ColorInt int TEXT_COLOR = Color.BLACK; @ColorInt int TEXT_ALERT_COLOR = Color.WHITE; @@ -61,29 +63,28 @@ public class SpeedLimitView extends View private float mBorderRadius; private float mBorderWidth; - private double mSpeedLimitMps; - @Nullable - private String mSpeedLimitStr; - - private double mCurrentSpeed; + private int mSpeedLimit = 0; + @NonNull + private String mSpeedLimitStr = "0"; + private boolean mAlert = false; public SpeedLimitView(Context context, @Nullable AttributeSet attrs) { super(context, attrs); try (TypedArray data = context.getTheme() - .obtainStyledAttributes(attrs, R.styleable.SpeedLimitView, 0, 0)) + .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); + mBackgroundColor = data.getColor(R.styleable.SpeedLimitView_speedLimitBackgroundColor, DefaultValues.BACKGROUND_COLOR); + mBorderColor = data.getColor(R.styleable.SpeedLimitView_speedLimitBorderColor, DefaultValues.BORDER_COLOR); + mAlertColor = data.getColor(R.styleable.SpeedLimitView_speedLimitAlertColor, DefaultValues.ALERT_COLOR); + mTextColor = data.getColor(R.styleable.SpeedLimitView_speedLimitTextColor, DefaultValues.TEXT_COLOR); + mTextAlertColor = data.getColor(R.styleable.SpeedLimitView_speedLimitTextAlertColor, 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); + mSpeedLimit = data.getInt(R.styleable.SpeedLimitView_speedLimitEditModeSpeedLimit, 60); + mSpeedLimitStr = Integer.toString(mSpeedLimit); + mAlert = data.getBoolean(R.styleable.SpeedLimitView_speedLimitEditModeAlert, false); } } @@ -101,29 +102,19 @@ public class SpeedLimitView extends View mTextPaint.setTypeface(Typeface.create(Typeface.DEFAULT, Typeface.BOLD)); } - public void setSpeedLimitMps(final double speedLimitMps) + public void setSpeedLimit(final int speedLimit, boolean alert) { - if (mSpeedLimitMps == speedLimitMps) - return; + final boolean speedLimitChanged = mSpeedLimit != speedLimit; - mSpeedLimitMps = speedLimitMps; - if (mSpeedLimitMps <= 0) + mSpeedLimit = speedLimit; + mAlert = alert; + + if (speedLimitChanged) { - mSpeedLimitStr = null; - setVisibility(GONE); - return; + mSpeedLimitStr = Integer.toString(mSpeedLimit); + configureTextSize(); } - final Pair speedLimitAndUnits = StringUtils.nativeFormatSpeedAndUnits(mSpeedLimitMps); - setVisibility(VISIBLE); - mSpeedLimitStr = speedLimitAndUnits.first; - configureTextSize(); - invalidate(); - } - - public void setCurrentSpeed(final double currentSpeed) - { - mCurrentSpeed = currentSpeed; invalidate(); } @@ -132,13 +123,15 @@ public class SpeedLimitView extends View { super.onDraw(canvas); - final boolean alert = mCurrentSpeed > mSpeedLimitMps && mSpeedLimitMps > 0; + final boolean validSpeedLimit = mSpeedLimit > 0; + if (!validSpeedLimit) + return; final float cx = mWidth / 2; final float cy = mHeight / 2; - drawSign(canvas, cx, cy, alert); - drawText(canvas, cx, cy, alert); + drawSign(canvas, cx, cy, mAlert); + drawText(canvas, cx, cy, mAlert); } private void drawSign(@NonNull Canvas canvas, float cx, float cy, boolean alert) @@ -158,9 +151,6 @@ public class SpeedLimitView extends View private void drawText(@NonNull Canvas canvas, float cx, float cy, boolean alert) { - if (mSpeedLimitStr == null) - return; - if (alert) mTextPaint.setColor(mTextAlertColor); else @@ -172,6 +162,26 @@ public class SpeedLimitView extends View canvas.drawText(mSpeedLimitStr, cx, textY, mTextPaint); } + @Override + public boolean onTouchEvent(@NonNull MotionEvent event) + { + final float cx = mWidth / 2; + final float cy = mHeight / 2; + if (Math.pow(event.getX() - cx, 2) + Math.pow(event.getY() - cy, 2) <= Math.pow(mBackgroundRadius, 2)) + { + performClick(); + return true; + } + return false; + } + + @Override + public boolean performClick() + { + super.performClick(); + return false; + } + @Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { @@ -191,9 +201,6 @@ public class SpeedLimitView extends View // 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; 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 07fef56f84..3ed8acfe81 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 @@ -116,20 +116,19 @@ app:layout_constraintStart_toEndOf="@+id/nav_speed_limit" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintTop_toBottomOf="@id/street_frame" - app:activeLaneTintColor="?navLaneArrowActiveColor" - app:inactiveLaneTintColor="?navLaneArrowInactiveColor" - app:backgroundColor="?colorAccent" - app:cornerRadius="@dimen/margin_quarter" - app:editModeLanesCount="10" + app:lanesActiveLaneTintColor="?navLaneArrowActiveColor" + app:lanesInactiveLaneTintColor="?navLaneArrowInactiveColor" + app:lanesBackgroundColor="?colorAccent" + app:lanesCornerRadius="@dimen/margin_quarter" + app:lanesEditModeLanesCount="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 e86c6ac075..93ea3b9771 100644 --- a/android/app/src/main/res/layout/layout_nav_top.xml +++ b/android/app/src/main/res/layout/layout_nav_top.xml @@ -117,20 +117,19 @@ 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" - app:backgroundColor="?navLanesBackgroundColor" - app:cornerRadius="@dimen/margin_quarter" - app:editModeLanesCount="5" + app:lanesActiveLaneTintColor="?navLaneArrowActiveColor" + app:lanesInactiveLaneTintColor="?navLaneArrowInactiveColor" + app:lanesBackgroundColor="?navLanesBackgroundColor" + app:lanesCornerRadius="@dimen/margin_quarter" + app:lanesEditModeLanesCount="5" tools:visibility="visible" /> diff --git a/android/app/src/main/res/values/attrs.xml b/android/app/src/main/res/values/attrs.xml index e5120a01d5..1de353baaf 100644 --- a/android/app/src/main/res/values/attrs.xml +++ b/android/app/src/main/res/values/attrs.xml @@ -1,23 +1,23 @@ - - - - - + + + + + - - + + - - - - + + + + - + diff --git a/android/app/src/main/res/values/styles.xml b/android/app/src/main/res/values/styles.xml index 8c8925ba5b..2442db2747 100644 --- a/android/app/src/main/res/values/styles.xml +++ b/android/app/src/main/res/values/styles.xml @@ -5,6 +5,11 @@ +