diff --git a/android/app/.attach_pid35553 b/android/app/.attach_pid35553 new file mode 100644 index 0000000000..e69de29bb2 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..c0d4d1d38f 100644 --- a/android/app/src/main/java/app/organicmaps/routing/NavigationController.java +++ b/android/app/src/main/java/app/organicmaps/routing/NavigationController.java @@ -249,7 +249,7 @@ public class NavigationController implements TrafficManager.TrafficCallback, { final Location location = LocationHelper.from(mFrame.getContext()).getSavedLocation(); if (location == null) { - mSpeedLimit.setSpeedLimitMps(0); + mSpeedLimit.setSpeedLimitMps(-1); return; } mSpeedLimit.setCurrentSpeed(location.getSpeed()); 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..23e8d240db 100644 --- a/android/app/src/main/java/app/organicmaps/widget/SpeedLimitView.java +++ b/android/app/src/main/java/app/organicmaps/widget/SpeedLimitView.java @@ -1,21 +1,28 @@ package app.organicmaps.widget; +import android.animation.Animator; +import android.animation.AnimatorListenerAdapter; +import android.animation.ValueAnimator; +import android.annotation.SuppressLint; 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.RectF; import android.graphics.Typeface; import android.util.AttributeSet; import android.util.Pair; import android.view.View; +import android.view.WindowInsets; import androidx.annotation.ColorInt; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.core.content.ContextCompat; +import app.organicmaps.Framework; import app.organicmaps.R; import app.organicmaps.util.StringUtils; @@ -28,17 +35,27 @@ public class SpeedLimitView extends View @ColorInt int TEXT_COLOR = Color.BLACK; @ColorInt + int UNLIMITED_BORDER_COLOR = Color.BLACK; + @ColorInt int TEXT_ALERT_COLOR = Color.WHITE; - float BORDER_WIDTH_RATIO = 0.1f; + float BORDER_WIDTH_RATIO = 0.2f; + + float BORDER_RADIUS_RATIO = 0.95f; } @ColorInt private final int mBackgroundColor; + @ColorInt + private final int mBackgroundColorWidget; + @ColorInt private final int mBorderColor; + @ColorInt + private final int mUnlimitedBorderColor; + @ColorInt private final int mAlertColor; @@ -53,7 +70,23 @@ public class SpeedLimitView extends View @NonNull private final Paint mSignBorderPaint; @NonNull + private final Paint mSignUnlimitedBorderPaint; + @NonNull private final Paint mTextPaint; + @NonNull + private final Paint mCurrentSpeedBorderPaint; + @NonNull + private final Paint mCurrentSpeedBackgroundPaint; + @NonNull + private final Paint mCurrentSpeedTextPaint; + @NonNull + private final Paint mCurrentSpeedUnitsTextPaint; // Added for units text + @NonNull + private final Paint mBackgroundPaint; + @NonNull + private final Paint mWidgetBackgroundPaint; + private float mSpeedLimitAlpha; + private float mWidth; private float mHeight; @@ -64,18 +97,37 @@ public class SpeedLimitView extends View private double mSpeedLimitMps; @Nullable private String mSpeedLimitStr; + @Nullable + private String mCurrentSpeedStr; + @Nullable + private String mCurrentSpeedUnitsStr; - private double mCurrentSpeed; + private double mCurrentSpeedMps; + /** + * We’ll store the bottom system navigation bar (or gesture bar) inset here. + * Instead of subtracting it from the height, we’ll apply a vertical translation + * so the entire widget moves up. + */ + private int mBottomSystemWindowInset = 0; + private int mLeftSystemWindowInset = 0; + + @SuppressLint("SwitchIntDef") 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)) { + mBackgroundColorWidget = data.getColor(R.styleable.SpeedLimitView_WidgetBackgroundColor, switch (Framework.nativeGetMapStyle()) { + case Framework.MAP_STYLE_DARK, Framework.MAP_STYLE_VEHICLE_DARK, Framework.MAP_STYLE_OUTDOORS_DARK -> + Color.DKGRAY; + default -> Color.WHITE; + }); mBackgroundColor = data.getColor(R.styleable.SpeedLimitView_BackgroundColor, DefaultValues.BACKGROUND_COLOR); mBorderColor = data.getColor(R.styleable.SpeedLimitView_borderColor, ContextCompat.getColor(context, R.color.base_red)); + mUnlimitedBorderColor = data.getColor(R.styleable.SpeedLimitView_unlimitedBorderColor, DefaultValues.UNLIMITED_BORDER_COLOR); 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); @@ -83,7 +135,8 @@ public class SpeedLimitView extends View { mSpeedLimitMps = data.getInt(R.styleable.SpeedLimitView_editModeSpeedLimit, -1); mSpeedLimitStr = mSpeedLimitMps > 0 ? String.valueOf(((int) mSpeedLimitMps)) : null; - mCurrentSpeed = data.getInt(R.styleable.SpeedLimitView_editModeCurrentSpeed, -1); + mCurrentSpeedMps = data.getInt(R.styleable.SpeedLimitView_editModeCurrentSpeed, -1); + mCurrentSpeedStr = mCurrentSpeedMps > 0 ? String.valueOf(((int) mCurrentSpeedMps)) : null; } } @@ -94,11 +147,49 @@ public class SpeedLimitView extends View mSignBorderPaint.setColor(mBorderColor); mSignBorderPaint.setStrokeWidth(mBorderWidth); mSignBorderPaint.setStyle(Paint.Style.STROKE); + mSignUnlimitedBorderPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + mSignUnlimitedBorderPaint.setColor(mUnlimitedBorderColor); + mSignUnlimitedBorderPaint.setStrokeWidth(mBorderWidth); + mSignUnlimitedBorderPaint.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)); + + mCurrentSpeedBackgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + mCurrentSpeedBackgroundPaint.setColor(mBackgroundColor); + mCurrentSpeedTextPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + mCurrentSpeedTextPaint.setColor(mTextColor); + mCurrentSpeedTextPaint.setTextAlign(Paint.Align.CENTER); + mCurrentSpeedTextPaint.setTypeface(Typeface.create(Typeface.DEFAULT, Typeface.BOLD)); + mCurrentSpeedUnitsTextPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + mCurrentSpeedUnitsTextPaint.setColor(mTextColor); + mCurrentSpeedUnitsTextPaint.setTextAlign(Paint.Align.CENTER); + mCurrentSpeedUnitsTextPaint.setTypeface(Typeface.create(Typeface.DEFAULT, Typeface.NORMAL)); + mCurrentSpeedBorderPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + mCurrentSpeedBorderPaint.setColor(Color.BLACK); + mCurrentSpeedBorderPaint.setStyle(Paint.Style.STROKE); + mBackgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + mBackgroundPaint.setColor(mBackgroundColor); + mWidgetBackgroundPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + mWidgetBackgroundPaint.setColor(mBackgroundColorWidget); + mSpeedLimitMps = -1.0; + } + + /** + * Capture bottom navigation bar insets (gesture area). + * We’ll use this to translate the canvas up. + */ + @Override + @NonNull + public WindowInsets onApplyWindowInsets(@NonNull WindowInsets insets) + { + mBottomSystemWindowInset = insets.getSystemWindowInsetBottom(); + mLeftSystemWindowInset = insets.getSystemWindowInsetLeft(); + // Request a re-draw so we can apply the translation in onDraw(). + invalidate(); + return super.onApplyWindowInsets(insets); } public void setSpeedLimitMps(final double speedLimitMps) @@ -106,65 +197,192 @@ public class SpeedLimitView extends View if (mSpeedLimitMps == speedLimitMps) return; + boolean hadSpeedLimit = mSpeedLimitMps >= 0; mSpeedLimitMps = speedLimitMps; - if (mSpeedLimitMps <= 0) - { - mSpeedLimitStr = null; - setVisibility(GONE); - return; + boolean hasSpeedLimit = mSpeedLimitMps >= 0; + + if (mSpeedLimitMps >= 0) { + final Pair speedLimitAndUnits = StringUtils.nativeFormatSpeedAndUnits(mSpeedLimitMps); + mSpeedLimitStr = speedLimitAndUnits.first; } - final Pair speedLimitAndUnits = StringUtils.nativeFormatSpeedAndUnits(mSpeedLimitMps); - setVisibility(VISIBLE); - mSpeedLimitStr = speedLimitAndUnits.first; + if (hadSpeedLimit != hasSpeedLimit) { + animateSpeedLimitAlpha(hasSpeedLimit); + } else { + invalidate(); + } configureTextSize(); invalidate(); } - public void setCurrentSpeed(final double currentSpeed) - { - mCurrentSpeed = currentSpeed; + private void animateSpeedLimitAlpha(final boolean fadeIn) { + float startAlpha = fadeIn ? 0f : 1f; + float endAlpha = fadeIn ? 1f : 0f; + ValueAnimator animator = ValueAnimator.ofFloat(startAlpha, endAlpha); + animator.setDuration(500); + animator.addUpdateListener(animation -> { + mSpeedLimitAlpha = (float) animation.getAnimatedValue(); + invalidate(); + }); + animator.addListener(new AnimatorListenerAdapter() { + @Override + public void onAnimationEnd(Animator animation) { + if (!fadeIn) { + mSpeedLimitStr = null; + } + } + }); + animator.start(); + } + public void setCurrentSpeed(final double currentSpeedMps) { + mCurrentSpeedMps = currentSpeedMps; + final Pair speedAndUnits = StringUtils.nativeFormatSpeedAndUnits(mCurrentSpeedMps); + if (mCurrentSpeedMps < 0) + mCurrentSpeedStr = "0"; + else + mCurrentSpeedStr = speedAndUnits.first; + mCurrentSpeedUnitsStr = speedAndUnits.second; + configureTextSize(); invalidate(); } - @Override - protected void onDraw(@NonNull Canvas canvas) - { - super.onDraw(canvas); + private void draw_widgets(@NonNull Canvas canvas, float cx, float cx_or_cy, float cy, boolean isLandscape) { + final boolean alert = mSpeedLimitMps > 0 && mSpeedLimitStr != null && Integer.parseInt(mCurrentSpeedStr) > Integer.parseInt(mSpeedLimitStr); //TODO do better? + float second_cx = (isLandscape) ? cx_or_cy : cx; + float second_cy = (isLandscape) ? cy : cx_or_cy; + // Draw combined background and speed limit sign with layer alpha + if (mSpeedLimitAlpha > 0f) { + int saveCount = canvas.saveLayerAlpha(0, 0, getWidth(), getHeight(), (int) (mSpeedLimitAlpha * 255)); + drawCombinedBackground(canvas, cx, cx_or_cy, cy); + // Draw speed limit sign and text + if (mSpeedLimitStr != null) + drawSpeedLimitSign(canvas, cx, second_cy); - 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); + canvas.restoreToCount(saveCount); + } + // Draw current speed sign and text (always visible) + if (mCurrentSpeedStr != null) + drawCurrentSpeedSign(canvas, second_cx, cy, alert); } - private void drawSign(@NonNull Canvas canvas, float cx, float cy, boolean alert) - { - if (alert) - mSignBackgroundPaint.setColor(mAlertColor); - else - mSignBackgroundPaint.setColor(mBackgroundColor); + @Override + protected void onDraw(@NonNull Canvas canvas) { + super.onDraw(canvas); + canvas.save(); + canvas.translate(mLeftSystemWindowInset, -mBottomSystemWindowInset); + final boolean isLandscape = mWidth > mHeight; - canvas.drawCircle(cx, cy, mBackgroundRadius, mSignBackgroundPaint); - if (!alert) - { - mSignBorderPaint.setStrokeWidth(mBorderWidth); - canvas.drawCircle(cx, cy, mBorderRadius, mSignBorderPaint); + if (isLandscape) { + final float cy = mHeight / 2; + float gap = mWidth * 0.02f; + final float totalWidth = 4 * mBackgroundRadius + gap; + final float startX = (mWidth - totalWidth) / 2; + final float currentSpeedCx = startX + mBackgroundRadius; + final float speedLimitCx = currentSpeedCx + 2 * mBackgroundRadius + gap; + draw_widgets(canvas, speedLimitCx, currentSpeedCx, cy, true); + } else { + final float cx = mWidth / 2; + float gap = mHeight * 0.02f; + final float totalHeight = 4 * mBackgroundRadius + gap; + final float startY = (mHeight - totalHeight) / 2; + final float speedLimitCy = startY + mBackgroundRadius; + final float currentSpeedCy = speedLimitCy + 2 * mBackgroundRadius + gap; + draw_widgets(canvas, cx, speedLimitCy, currentSpeedCy, false); } } - private void drawText(@NonNull Canvas canvas, float cx, float cy, boolean alert) + private void drawCombinedBackground(Canvas canvas, float cx, float cx_or_cy, float cy) { + final boolean isLandscape = mWidth > mHeight; + float left; + float right; + float top; + float bottom; + + if (isLandscape) { + top = cy - mBackgroundRadius; + bottom = cy + mBackgroundRadius; + left = cx_or_cy - mBackgroundRadius; + right = cx + mBackgroundRadius; + } else { + left = cx - mBackgroundRadius; + right = cx + mBackgroundRadius; + top = cx_or_cy - mBackgroundRadius; + bottom = cy + mBackgroundRadius; + } + RectF rect = new RectF(left, top, right, bottom); + float cornerRadius = mBackgroundRadius; + canvas.drawRoundRect(rect, cornerRadius, cornerRadius, mWidgetBackgroundPaint); + } + + private void drawUnlimitedSign(@NonNull Canvas canvas, float cx, float cy) { + // Paint for stripes + Paint stripesPaint = new Paint(Paint.ANTI_ALIAS_FLAG); + stripesPaint.setColor(Color.BLACK); + stripesPaint.setStyle(Paint.Style.STROKE); + + // Thickness of the stripes can be relative to the border width or radius + float stripeThickness = mBorderWidth * 0.4f; + stripesPaint.setStrokeWidth(stripeThickness); + + float r = mBackgroundRadius; + + // Direction vector for the diagonal lines (top-right to bottom-left) + float dx = -1.0f / (float)Math.sqrt(2); + float dy = 1.0f / (float)Math.sqrt(2); + + // Perpendicular vector to D (rotate by +90°) + float px = -dy; + float py = dx; + + // Increase lineSpacing to spread stripes further apart + float lineSpacing = r * 0.12f; // slightly larger than 0.1f + + // Use a scale factor to shorten the lines + float lineScale = 0.8f; // reduce from full radius to 85% radius + + int[] offsets = {-2, -1, 0, 1, 2}; + + // Draw 5 parallel stripes + for (int i : offsets) { + float ox = i * lineSpacing * px; + float oy = i * lineSpacing * py; + + // Start and end points of the line, shortened by lineScale + float sx = cx + dx * r * lineScale + ox; + float sy = cy + dy * r * lineScale + oy; + float ex = cx - dx * r * lineScale + ox; + float ey = cy - dy * r * lineScale + oy; + + canvas.drawLine(sx, sy, ex, ey, stripesPaint); + } + } + + + private void drawSpeedLimitSign(@NonNull Canvas canvas, float cx, float cy) { + // Draw the sign background and border as before + mSignBackgroundPaint.setColor(mBackgroundColor); + canvas.drawCircle(cx, cy, mBackgroundRadius, mSignBackgroundPaint); + + if (mSpeedLimitMps == 0) { + // Unlimited speed scenario: Draw the special no-limit pattern + mSignUnlimitedBorderPaint.setStrokeWidth(mBorderWidth); + canvas.drawCircle(cx, cy, mBorderRadius * DefaultValues.BORDER_RADIUS_RATIO, mSignUnlimitedBorderPaint); + drawUnlimitedSign(canvas, cx, cy); + } + else if (mSpeedLimitStr != null && mSpeedLimitMps > 0) { + // Normal speed limit scenario: Draw the text + mSignBorderPaint.setStrokeWidth(mBorderWidth); + canvas.drawCircle(cx, cy, mBorderRadius * DefaultValues.BORDER_RADIUS_RATIO, mSignBorderPaint); + drawSpeedLimitText(canvas, cx, cy); + } + } + + private void drawSpeedLimitText(@NonNull Canvas canvas, float cx, float cy) { if (mSpeedLimitStr == null) return; - if (alert) - mTextPaint.setColor(mTextAlertColor); - else - mTextPaint.setColor(mTextColor); + mTextPaint.setColor(mTextColor); final Rect textBounds = new Rect(); mTextPaint.getTextBounds(mSpeedLimitStr, 0, mSpeedLimitStr.length(), textBounds); @@ -172,32 +390,106 @@ public class SpeedLimitView extends View canvas.drawText(mSpeedLimitStr, cx, textY, mTextPaint); } + private void drawCurrentSpeedSign(@NonNull Canvas canvas, float cx, float cy, boolean alert) { + // Change background color based on alert state + mCurrentSpeedBackgroundPaint.setColor(mBackgroundColor); + // Draw current speed circle (background) + canvas.drawCircle(cx, cy, mBackgroundRadius, mCurrentSpeedBackgroundPaint); + // Draw border around current speed circle + if (alert) { + mCurrentSpeedBackgroundPaint.setColor(mAlertColor); + canvas.drawCircle(cx, cy, mBackgroundRadius * DefaultValues.BORDER_RADIUS_RATIO * 0.95f, mCurrentSpeedBackgroundPaint); + } + mCurrentSpeedBorderPaint.setStrokeWidth(mBorderWidth / 2); + canvas.drawCircle(cx, cy, mBorderRadius, mCurrentSpeedBorderPaint); + drawCurrentSpeedText(canvas, cx, cy, alert); + } + + private void drawCurrentSpeedText(@NonNull Canvas canvas, float cx, float cy, boolean alert) { + if (mCurrentSpeedStr == null || mCurrentSpeedUnitsStr == null) + return; + // Change text color based on alert state + if (alert) { + mCurrentSpeedTextPaint.setColor(mTextAlertColor); + mCurrentSpeedUnitsTextPaint.setColor(mTextAlertColor); + } else { + mCurrentSpeedTextPaint.setColor(mTextColor); + mCurrentSpeedUnitsTextPaint.setColor(mTextColor); + } + final Rect speedTextBounds = new Rect(); + mCurrentSpeedTextPaint.getTextBounds(mCurrentSpeedStr, 0, mCurrentSpeedStr.length(), speedTextBounds); + final Rect unitsTextBounds = new Rect(); + mCurrentSpeedUnitsTextPaint.getTextBounds(mCurrentSpeedUnitsStr, 0, mCurrentSpeedUnitsStr.length(), unitsTextBounds); + // Position speed text + float speedTextY = cy - (float) speedTextBounds.height() / 2 + speedTextBounds.height(); + // Position units text + float unitsTextY = speedTextY + unitsTextBounds.height() + 10f; + // Draw speed text + canvas.drawText(mCurrentSpeedStr, cx, speedTextY, mCurrentSpeedTextPaint); + // Draw units text + canvas.drawText(mCurrentSpeedUnitsStr, cx, unitsTextY, mCurrentSpeedUnitsTextPaint); + } + @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()); + final float paddingX = getPaddingLeft() + getPaddingRight(); + final float paddingY = getPaddingTop() + getPaddingBottom(); + mWidth = w - paddingX; + mHeight = h - paddingY; - mWidth = (float) w - paddingX; - mHeight = (float) h - paddingY; - mBackgroundRadius = Math.min(mWidth, mHeight) / 2; - mBorderWidth = mBackgroundRadius * 2 * DefaultValues.BORDER_WIDTH_RATIO; + boolean isLandscape = mWidth > mHeight; + float gap = (isLandscape) ? mWidth * 0.1f : mHeight * 0.1f; + // Compute maximum possible radius + mBackgroundRadius = (isLandscape) ? (mWidth - gap) / 4 : (mHeight - gap) / 4; + // Ensure the radius does not exceed half of the width + mBackgroundRadius = (isLandscape) ? Math.min(mBackgroundRadius, mHeight / 2) : Math.min(mBackgroundRadius, mWidth / 2); + mBorderWidth = mBackgroundRadius * 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; + private void configureTextSize() { + // Use reference strings to keep text size consistent + configureTextSizeForString(mTextPaint); // For speed limit + configureCurrentSpeedTextSize(); // For current speed and units + } + private void configureCurrentSpeedTextSize() { + final float textRadius = mBorderRadius * 0.75f - mCurrentSpeedBorderPaint.getStrokeWidth(); + final float availableHeight = 2 * textRadius; + float speedTextHeightRatio = 0.75f; + float unitsTextHeightRatio = 0.2f; + float speedTextHeight = availableHeight * speedTextHeightRatio; + float unitsTextHeight = availableHeight * unitsTextHeightRatio; + float speedTextSize = findMaxTextSizeForHeight("299", mCurrentSpeedTextPaint, speedTextHeight); + float unitsTextSize = findMaxTextSizeForHeight("km/h", mCurrentSpeedUnitsTextPaint, unitsTextHeight); + mCurrentSpeedTextPaint.setTextSize(speedTextSize); + mCurrentSpeedUnitsTextPaint.setTextSize(unitsTextSize); + } + private float findMaxTextSizeForHeight(String text, Paint paint, float targetHeight) { + float lowerBound = 0; + float upperBound = targetHeight; + float textSize = targetHeight; + final Rect textBounds = new Rect(); + while (upperBound - lowerBound > 1f) { + textSize = (lowerBound + upperBound) / 2; + paint.setTextSize(textSize); + paint.getTextBounds(text, 0, text.length(), textBounds); + if (textBounds.height() <= targetHeight) + lowerBound = textSize; + else + upperBound = textSize; + } + return lowerBound; + } - final String text = mSpeedLimitStr; + private void configureTextSizeForString(Paint textPaint) { final float textRadius = mBorderRadius - mBorderWidth; final float textMaxSize = 2 * textRadius; - final float textMaxSizeSquared = (float) Math.pow(textMaxSize, 2); + final float textMaxSizeSquared = textMaxSize * textMaxSize; float lowerBound = 0; float upperBound = textMaxSize; @@ -207,15 +499,15 @@ public class SpeedLimitView extends View while (lowerBound <= upperBound) { textSize = (lowerBound + upperBound) / 2; - mTextPaint.setTextSize(textSize); - mTextPaint.getTextBounds(text, 0, text.length(), textBounds); + textPaint.setTextSize(textSize); + textPaint.getTextBounds("180", 0, 3, textBounds); - if (Math.pow(textBounds.width(), 2) + Math.pow(textBounds.height(), 2) <= textMaxSizeSquared) + if (textBounds.width() * textBounds.width() + textBounds.height() * textBounds.height() <= textMaxSizeSquared) lowerBound = textSize + 1; else upperBound = textSize - 1; } - mTextPaint.setTextSize(Math.max(1, textSize)); + textPaint.setTextSize(Math.max(1, textSize)); } -} +} \ No newline at end of file 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 9e2b8612b7..631d9de698 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 @@ -33,9 +33,6 @@ public class NavMenu private final View mHeaderFrame; private final ImageView mTts; - private final View mSpeedViewContainer; - private final TextView mSpeedValue; - private final TextView mSpeedUnits; private final TextView mTimeHourValue; private final TextView mTimeHourUnits; private final TextView mTimeMinuteValue; @@ -94,10 +91,6 @@ public class NavMenu } }); - // Bottom frame - mSpeedViewContainer = bottomFrame.findViewById(R.id.speed_view_container); - mSpeedValue = bottomFrame.findViewById(R.id.speed_value); - mSpeedUnits = bottomFrame.findViewById(R.id.speed_dimen); mTimeHourValue = bottomFrame.findViewById(R.id.time_hour_value); mTimeHourUnits = bottomFrame.findViewById(R.id.time_hour_dimen); mTimeMinuteValue = bottomFrame.findViewById(R.id.time_minute_value); @@ -213,20 +206,6 @@ public class NavMenu return; Pair speedAndUnits = StringUtils.nativeFormatSpeedAndUnits(last.getSpeed()); - mSpeedValue.setText(speedAndUnits.first); - - if (info.speedLimitMps > 0.0 && last.getSpeed() > info.speedLimitMps) - { - if (info.isSpeedCamLimitExceeded()) - mSpeedValue.setTextColor(ContextCompat.getColor(mActivity, R.color.white_primary)); - else - mSpeedValue.setTextColor(ContextCompat.getColor(mActivity, R.color.base_red)); - } - else - mSpeedValue.setTextColor(ThemeUtils.getColor(mActivity, android.R.attr.textColorPrimary)); - - mSpeedUnits.setText(speedAndUnits.second); - mSpeedViewContainer.setActivated(info.isSpeedCamLimitExceeded()); } public void update(@NonNull RoutingInfo info) diff --git a/android/app/src/main/res/layout-h400dp/map_buttons_layout_navigation.xml b/android/app/src/main/res/layout-h400dp/map_buttons_layout_navigation.xml index f20833f98b..1aa9f18332 100644 --- a/android/app/src/main/res/layout-h400dp/map_buttons_layout_navigation.xml +++ b/android/app/src/main/res/layout-h400dp/map_buttons_layout_navigation.xml @@ -25,29 +25,6 @@ android:padding="@dimen/nav_frame_padding" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintStart_toStartOf="parent"> - - - + + + + diff --git a/android/app/src/main/res/layout-h400dp/map_buttons_layout_planning.xml b/android/app/src/main/res/layout-h400dp/map_buttons_layout_planning.xml index b80f476643..c3b7eeda03 100644 --- a/android/app/src/main/res/layout-h400dp/map_buttons_layout_planning.xml +++ b/android/app/src/main/res/layout-h400dp/map_buttons_layout_planning.xml @@ -18,21 +18,6 @@ android:clipToPadding="false" android:padding="@dimen/nav_frame_padding" app:layout_constraintStart_toStartOf="parent"> - - + + + + @@ -23,7 +23,7 @@ style="@style/MwmWidget.MapButton.Search" android:layout_alignParentTop="true" android:layout_centerHorizontal="true" - android:layout_marginStart="@dimen/margin_half_plus" + android:layout_marginEnd="@dimen/margin_half_plus" android:layout_marginTop="@dimen/margin_half" android:contentDescription="@string/category_fuel" app:srcCompat="@drawable/ic_routing_fuel_on" /> @@ -31,13 +31,13 @@ android:id="@+id/search_parking" style="@style/MwmWidget.MapButton.Search" android:layout_marginTop="@dimen/margin_base_plus" - android:layout_toEndOf="@id/search_fuel" + android:layout_toStartOf="@id/search_fuel" android:contentDescription="@string/category_parking" app:srcCompat="@drawable/ic_routing_parking_on" /> 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..a24b9b88e5 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 @@ -7,7 +7,9 @@ android:layout_width="match_parent" android:layout_height="match_parent" tools:background="#20FF0000" - tools:showIn="@layout/layout_nav"> + tools:showIn="@layout/layout_nav" + android:clipToPadding="false" + android:clipChildren="false"> + android:id="@+id/nav_speed_limit" + android:layout_width="110dp" + android:layout_height="60dp" + app:editModeCurrentSpeed="90" + app:editModeSpeedLimit="120" + app:layout_constraintStart_toStartOf="parent" + app:layout_constraintBottom_toBottomOf="parent" + android:layout_marginBottom="88dp" /> diff --git a/android/app/src/main/res/layout/layout_nav_bottom_numbers.xml b/android/app/src/main/res/layout/layout_nav_bottom_numbers.xml index 50016a8444..ad0e92915f 100644 --- a/android/app/src/main/res/layout/layout_nav_bottom_numbers.xml +++ b/android/app/src/main/res/layout/layout_nav_bottom_numbers.xml @@ -13,42 +13,6 @@ android:layout_height="match_parent" android:layout_weight="0.5"/> - - - - - - - - - - - + tools:showIn="@layout/layout_nav" + android:clipToPadding="false" + android:clipChildren="false"> + app:layout_constraintStart_toStartOf="parent" + app:layout_constraintBottom_toBottomOf="parent" + android:layout_marginBottom="140dp" /> diff --git a/android/app/src/main/res/layout/map_buttons_layout_navigation.xml b/android/app/src/main/res/layout/map_buttons_layout_navigation.xml index ad3fac1b19..f00ac58add 100644 --- a/android/app/src/main/res/layout/map_buttons_layout_navigation.xml +++ b/android/app/src/main/res/layout/map_buttons_layout_navigation.xml @@ -25,29 +25,6 @@ android:padding="@dimen/nav_frame_padding" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintStart_toStartOf="parent"> - - - + + + - - + + + + + app:layout_constraintEnd_toEndOf="parent" /> \ No newline at end of file diff --git a/android/app/src/main/res/layout/map_buttons_search_frame.xml b/android/app/src/main/res/layout/map_buttons_search_frame.xml index bb5b4a255d..45d4712542 100644 --- a/android/app/src/main/res/layout/map_buttons_search_frame.xml +++ b/android/app/src/main/res/layout/map_buttons_search_frame.xml @@ -13,46 +13,46 @@ android:id="@+id/search_frame" android:layout_width="286dp" android:layout_height="48dp" - android:layout_alignParentStart="true" + android:layout_alignParentEnd="true" android:background="?searchLayoutBackground" android:orientation="horizontal" android:padding="@dimen/margin_quarter" android:visibility="invisible" tools:visibility="visible"> + android:id="@+id/search_parking" + style="@style/MwmWidget.MapButton.Search" + android:layout_gravity="center_vertical" + android:layout_marginEnd="@dimen/margin_half" + android:contentDescription="@string/category_parking" + app:srcCompat="@drawable/ic_routing_parking_on" /> + android:id="@+id/search_eat" + style="@style/MwmWidget.MapButton.Search" + android:layout_gravity="center_vertical" + android:layout_marginEnd="@dimen/margin_half" + android:contentDescription="@string/category_eat" + app:srcCompat="@drawable/ic_routing_eat_on" /> + android:id="@+id/search_food" + style="@style/MwmWidget.MapButton.Search" + android:layout_gravity="center_vertical" + android:layout_marginEnd="@dimen/margin_half" + android:contentDescription="@string/category_shopping" + app:srcCompat="@drawable/ic_routing_food_on" /> + android:id="@+id/search_atm" + style="@style/MwmWidget.MapButton.Search" + android:layout_gravity="center_vertical" + android:layout_marginEnd="@dimen/margin_half" + android:contentDescription="@string/category_atm" + app:srcCompat="@drawable/ic_routing_atm_on" /> + android:id="@+id/search_fuel" + style="@style/MwmWidget.MapButton.Search" + android:layout_gravity="center_vertical" + android:layout_marginEnd="50dp" + android:contentDescription="@string/category_fuel" + app:srcCompat="@drawable/ic_routing_fuel_on" /> diff --git a/android/app/src/main/res/values/attrs.xml b/android/app/src/main/res/values/attrs.xml index e5120a01d5..a9e0ac7dae 100644 --- a/android/app/src/main/res/values/attrs.xml +++ b/android/app/src/main/res/values/attrs.xml @@ -2,7 +2,9 @@ + +