diff --git a/android/app/src/main/java/app/organicmaps/routing/RoutingPlanController.java b/android/app/src/main/java/app/organicmaps/routing/RoutingPlanController.java index ac9600867c..d143ffec2d 100644 --- a/android/app/src/main/java/app/organicmaps/routing/RoutingPlanController.java +++ b/android/app/src/main/java/app/organicmaps/routing/RoutingPlanController.java @@ -1,19 +1,27 @@ package app.organicmaps.routing; import android.app.Activity; +import android.content.Context; import android.os.Bundle; +import android.view.Gravity; +import android.view.LayoutInflater; import android.view.View; import android.widget.CompoundButton; import android.widget.RadioButton; import android.widget.RadioGroup; import android.widget.TextView; +import android.content.SharedPreferences; +import android.widget.Toast; import androidx.annotation.DrawableRes; import androidx.annotation.IdRes; import androidx.annotation.NonNull; import androidx.annotation.Nullable; +import androidx.appcompat.widget.Toolbar; import androidx.core.view.ViewCompat; +import androidx.appcompat.app.AppCompatActivity; + import app.organicmaps.Framework; import app.organicmaps.MwmApplication; import app.organicmaps.R; @@ -181,6 +189,26 @@ public class RoutingPlanController extends ToolbarController return (mFrameHeight > 0); } + void addStopToast() { + SharedPreferences prefs = MwmApplication.prefs(requireActivity().getApplicationContext()); + SharedPreferences.Editor editor = prefs.edit(); + editor.putBoolean(String.valueOf(R.string.toast_dismiss), false); + editor.apply(); + + View progressFrame = getToolbar().findViewById(R.id.progress_frame); + LayoutInflater inflater = requireActivity().getLayoutInflater(); + View layout = inflater.inflate(R.layout.toast_custom,null); + + TextView text = layout.findViewById(R.id.toast_custom); + text.setText(R.string.add_stop_toast); + + Toast toast = new Toast(progressFrame.getContext()); + toast.setDuration(Toast.LENGTH_LONG); + toast.setGravity(Gravity.BOTTOM, 0, 160); + toast.setView(layout); + toast.show(); + } + private void updateProgressLabels() { RoutingController.BuildState buildState = RoutingController.get().getBuildState(); @@ -212,6 +240,13 @@ public class RoutingPlanController extends ToolbarController final boolean showStartButton = !RoutingController.get().isRulerRouterType(); mRoutingBottomMenuController.setStartButton(showStartButton); mRoutingBottomMenuController.showAltitudeChartAndRoutingDetails(); + + SharedPreferences prefs = MwmApplication.prefs(requireActivity().getApplicationContext()); + boolean ShowToast = prefs.getBoolean(String.valueOf(R.string.toast_dismiss), true); + + if (ShowToast) { + addStopToast(); + } } public void updateBuildProgress(int progress, @Framework.RouterType int router) @@ -349,4 +384,4 @@ public class RoutingPlanController extends ToolbarController mDrivingOptionsBtnContainer.removeOnLayoutChangeListener(this); } } -} +} \ No newline at end of file diff --git a/android/app/src/main/res/drawable/toast_custom_shape.xml b/android/app/src/main/res/drawable/toast_custom_shape.xml new file mode 100644 index 0000000000..0dac6ea816 --- /dev/null +++ b/android/app/src/main/res/drawable/toast_custom_shape.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/android/app/src/main/res/layout/toast_custom.xml b/android/app/src/main/res/layout/toast_custom.xml new file mode 100644 index 0000000000..a2bb8360fd --- /dev/null +++ b/android/app/src/main/res/layout/toast_custom.xml @@ -0,0 +1,15 @@ + + + + + \ No newline at end of file diff --git a/android/app/src/main/res/values/strings.xml b/android/app/src/main/res/values/strings.xml index 9e4d0b3eb7..31346465ef 100644 --- a/android/app/src/main/res/values/strings.xml +++ b/android/app/src/main/res/values/strings.xml @@ -612,6 +612,8 @@ Exit Add a starting point to plan a route Add a destination to plan a route + Select any place to add an intermediate stop + Don\'t show again Remove Add Stop @@ -697,6 +699,7 @@ Avoid ferries Avoid freeways + Manage route Unable to calculate route A route could not be found. This may be caused by your routing options or incomplete OpenStreetMap data. Please change your routing options and retry. Define roads to avoid @@ -2200,4 +2203,4 @@ Events Venue Auction Collectables - + \ No newline at end of file diff --git a/data/strings/strings.txt b/data/strings/strings.txt index a6ba22f89c..6bfe44d6f5 100644 --- a/data/strings/strings.txt +++ b/data/strings/strings.txt @@ -20194,6 +20194,51 @@ zh-Hans = 添加起点以规划路线 zh-Hant = 新增起點以計劃路線 + [routing_add_stop_point] + tags = ios + en = Select any place to add an intermediate stop + af = Kies enige plek om ’n tussenstop by te voeg + ar = حدد أي مكان لإضافة توقف وسيط + az = Bir ara durmaq əlavə etmək üçün hər hansı bir yeri seçin + be = Выберыце любое месца, каб дадаць пралягчы стоп + bg = Изберете която и да е точка, за да добавите междинна спирка + ca = Seleccioneu qualsevol lloc per afegir una parada intermèdia + cs = Vyberte libovolné místo pro přidání mezipřistání + da = Vælg et hvilket som helst sted for at tilføje en mellemlanding + de = Wählen Sie einen beliebigen Ort aus, um einen Zwischenstopp hinzuzufügen + el = Επιλέξτε οποιοδήποτε μέρος για να προσθέσετε μια ενδιάμεση στάση + es = Selecciona cualquier lugar para añadir una parada intermedia + et = Valige mis tahes koht, et lisada vahepeatuse punkt + eu = Hautatu edozein toki eta gehitu eten erdia + fa = یک مکان را برای اضافه کردن یک توقف وسطی انتخاب کنید + fi = Valitse mikä tahansa paikka lisätäksesi välipysäkin + fr = Sélectionnez n'importe quel endroit pour ajouter un arrêt intermédiaire + he = בחרו בכל מקום להוסיף תחנה ביניים + hi = बिच का एक स्थान जोड़ने के लिए कोई भी स्थान चुनें + hu = Válasszon ki bármilyen helyet egy köztes megálló hozzáadásához + id = Pilih tempat mana pun untuk menambahkan titik persinggahan + it = Seleziona qualsiasi posto per aggiungere una fermata intermedia + ja = 中間停留所を追加する場所を選択してください + ko = 중간 정차지를 추가할 위치를 선택하세요 + lt = Pasirinkite bet kurį vietą, kad pridėtumėte tarpinę stotelę + mr = बीचीकडील थांबायला कोणत्याही ठिकाणी निवडा + nb = Velg et hvilket som helst sted for å legge til en mellomlanding + nl = Selecteer een willekeurige plaats om een tussenstop toe te voegen + pl = Wybierz dowolne miejsce, aby dodać przystanek pośredni + pt = Selecione qualquer lugar para adicionar uma paragem intermédia + pt-BR = Selecione qualquer lugar para adicionar uma parada intermediária + ro = Selectați orice loc pentru a adăuga un punct intermediar de oprire + ru = Выберите любое место, чтобы добавить промежуточную остановку + sk = Vyberte ľubovoľné miesto pre pridanie medzipristátia + sv = Välj valfri plats för att lägga till ett mellanstopp + sw = Chagua eneo lolote kuongeza kituo cha kati + th = เลือกสถานที่ใดก็ได้เพื่อเพิ่มจุดหยุดระหว่างทาง + tr = Araya bir durak eklemek için herhangi bir yeri seçin + uk = Виберіть будь-яке місце, щоб додати проміжну зупинку + vi = Chọn bất kỳ địa điểm nào để thêm điểm dừng trung gian + zh-Hans = 选择任何地点添加中途停靠点 + zh-Hant = 選擇任何地點添加中途停靠點 + [routing_add_finish_point] tags = android,ios en = Add a destination to plan a route @@ -30345,4 +30390,4 @@ uk = Переглянути меню vi = Xem thực đơn zh-Hans = 查看菜单 - zh-Hant = 查看選單 + zh-Hant = 查看選單 \ No newline at end of file diff --git a/iphone/Maps/Bookmarks/BookmarksList/BookmarksListInfoViewController.swift b/iphone/Maps/Bookmarks/BookmarksList/BookmarksListInfoViewController.swift index 762d8fc636..de080590af 100644 --- a/iphone/Maps/Bookmarks/BookmarksList/BookmarksListInfoViewController.swift +++ b/iphone/Maps/Bookmarks/BookmarksList/BookmarksListInfoViewController.swift @@ -57,4 +57,4 @@ final class BookmarksListInfoViewController: UIViewController { let formattedText = NSAttributedString.string(withHtml: htmlText, defaultAttributes: [:]) return formattedText?.string } -} +} \ No newline at end of file diff --git a/iphone/Maps/Classes/CustomAlert/Toast/Toast.swift b/iphone/Maps/Classes/CustomAlert/Toast/Toast.swift index 44ebd057eb..a83a59954b 100644 --- a/iphone/Maps/Classes/CustomAlert/Toast/Toast.swift +++ b/iphone/Maps/Classes/CustomAlert/Toast/Toast.swift @@ -91,7 +91,43 @@ final class Toast: NSObject { userInfo: nil, repeats: false) } - + + @objc func show(in view: UIView?, alignment: Alignment, pinToSafeArea: Bool = true, withOffset offset: CGFloat ) { + guard let view = view else { return } + blurView.translatesAutoresizingMaskIntoConstraints = false + view.addSubview(blurView) + + let leadingConstraint = blurView.leadingAnchor.constraint(greaterThanOrEqualTo: view.leadingAnchor, constant: 16) + let trailingConstraint = blurView.trailingAnchor.constraint(lessThanOrEqualTo: view.trailingAnchor, constant: -16) + + NSLayoutConstraint.activate([ + leadingConstraint, + trailingConstraint + ]) + + let verticalConstraint: NSLayoutConstraint + if alignment == .bottom { + verticalConstraint = blurView.bottomAnchor.constraint(equalTo: pinToSafeArea ? view.safeAreaLayoutGuide.bottomAnchor : view.bottomAnchor, constant: offset) + } else { + verticalConstraint = blurView.topAnchor.constraint(equalTo: pinToSafeArea ? view.safeAreaLayoutGuide.topAnchor : view.topAnchor, constant: offset) + } + + NSLayoutConstraint.activate([ + verticalConstraint, + blurView.centerXAnchor.constraint(equalTo: pinToSafeArea ? view.safeAreaLayoutGuide.centerXAnchor : view.centerXAnchor) + ]) + + UIView.animate(withDuration: kDefaultAnimationDuration) { + self.blurView.alpha = 1 + } + + timer = Timer.scheduledTimer(timeInterval: 3, + target: self, + selector: #selector(hide), + userInfo: nil, + repeats: false) + } + @objc func hide() { timer?.invalidate() if self.blurView.superview != nil { @@ -101,4 +137,4 @@ final class Toast: NSObject { Self.toasts.removeAll(where: { $0 === self }) } } } -} +} \ No newline at end of file diff --git a/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/MWMNavigationInfoVIewTests.swift b/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/MWMNavigationInfoVIewTests.swift new file mode 100644 index 0000000000..8c2b1cabe8 --- /dev/null +++ b/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/MWMNavigationInfoVIewTests.swift @@ -0,0 +1,66 @@ +import XCTest +@testable import Organic_Maps_Debug + +final class MWMNavigationInfoViewTests: XCTestCase { + + var userDefaults: UserDefaults! + + override func setUp() { + super.setUp() + userDefaults = UserDefaults(suiteName: "com.example.OrganicMapsTests") // Use a test-specific suite + userDefaults.removePersistentDomain(forName: "com.example.OrganicMapsTests") // Clear UserDefaults before each test + } + + override func tearDown() { + userDefaults = nil + super.tearDown() + } + + func testAddStopToastAppearsOnlyInFirstSession() { + + // Simulate first session + var isFirstSession = false; + if ( userDefaults.set(true, forKey: "isFirstSession") != nil ){ + isFirstSession = true; + } + var hasStart = false; + var hasFinish = false; + + // + if ( userDefaults.set(true, forKey: "hasStart") != nil ){ + hasStart = true; + } + if ( userDefaults.set(true, forKey: "hasFinish") != nil ){ + hasFinish = true; + } + + if (hasStart && hasFinish && isFirstSession){ + + // Check if the toast view is visible + XCTAssertTrue(isFirstSession, "Toast view should be visible during the first session.") + + }else if ( hasStart && hasFinish && !isFirstSession){ + + // Check if the toast view is hidden + XCTAssertFalse(!isFirstSession, "Toast view should not be visible after the first session.") + } + + + // Simulate subsequent session + if ( userDefaults.set(false, forKey: "isFirstSession") != nil){ + isFirstSession = false; + } + + if (hasStart && hasFinish && isFirstSession){ + + // Check if the toast view is visible + XCTAssertTrue(!isFirstSession, "Toast view should be visible during the first session.") + + }else if ( hasStart && hasFinish && !isFirstSession){ + + // Check if the toast view is hidden + XCTAssertFalse(isFirstSession, "Toast view should not be visible after the first session.") + } + + } +} \ No newline at end of file diff --git a/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/MWMNavigationInfoView.mm b/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/MWMNavigationInfoView.mm index fe608f7dd0..becc693068 100644 --- a/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/MWMNavigationInfoView.mm +++ b/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/MWMNavigationInfoView.mm @@ -12,6 +12,8 @@ #import "UIImageView+Coloring.h" #import "location_util.h" +#include + #include "geometry/angles.hpp" namespace @@ -92,6 +94,7 @@ BOOL defaultOrientation(CGSize const &size) { @property(weak, nonatomic) MWMNavigationDashboardEntity *navigationInfo; @property(nonatomic) BOOL hasLocation; +@property(nonatomic) BOOL hasShownAddStopToast; @property(nonatomic) NSLayoutConstraint *topConstraint; @property(nonatomic) NSLayoutConstraint *leftConstraint; @@ -117,8 +120,22 @@ BOOL defaultOrientation(CGSize const &size) { BOOL const hasStart = ([MWMRouter startPoint] != nil); BOOL const hasFinish = ([MWMRouter finishPoint] != nil); + Framework::DrapeCreationParams p; + BOOL const isFirstLaunch = [FirstSession isFirstSession]; + BOOL const hasShownAddStopHint = [FirstSession hasShownAddStopToast]; self.hasLocation = ([MWMLocationManager lastLocation] != nil); + + if (hasStart && hasFinish && isFirstLaunch && !hasShownAddStopHint) { + [self setToastViewHidden:YES]; + + MWMToast *toast = [MWMToast toastWithText:L(@"routing_add_stop_point")]; + + [toast showIn:[UIApplication sharedApplication].keyWindow alignment:MWMToastAlignmentBottom pinToSafeArea:YES withOffset:-103]; + + return; + } + if (hasStart && hasFinish) { [self setToastViewHidden:YES]; return; @@ -144,6 +161,10 @@ BOOL defaultOrientation(CGSize const &size) { [toastView configWithIsStart:YES withLocationButton:NO]; } +-(BOOL)isFirstLaunch{ + return self.isFirstLaunch; +} + - (IBAction)openSearch { BOOL const isStart = self.toastView.isStart; auto searchManager = [MWMSearchManager manager]; @@ -489,4 +510,4 @@ BOOL defaultOrientation(CGSize const &size) { - (void)applyTheme { [self setSearchState:_searchState]; } -@end +@end \ No newline at end of file diff --git a/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/RoutePreview/RouteManager/RouteManagerViewController.swift b/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/RoutePreview/RouteManager/RouteManagerViewController.swift index 9fc700476f..2212d3ae81 100644 --- a/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/RoutePreview/RouteManager/RouteManagerViewController.swift +++ b/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/RoutePreview/RouteManager/RouteManagerViewController.swift @@ -207,6 +207,11 @@ final class RouteManagerViewController: MWMViewController, UITableViewDataSource }) } + @IBAction func onAddStop(_ sender: UIButton) { + viewModel.addRoutePoint() + dismiss(animated: true, completion: nil) + } + @IBAction private func gestureRecognized(_ longPress: UIGestureRecognizer) { let locationInView = gestureLocation(longPress, in: containerView) let locationInTableView = gestureLocation(longPress, in: tableView) @@ -254,4 +259,4 @@ final class RouteManagerViewController: MWMViewController, UITableViewDataSource func tableView(_: UITableView, editingStyleForRowAt _: IndexPath) -> UITableViewCell.EditingStyle { return canDeleteRow ? .delete : .none } -} +} \ No newline at end of file diff --git a/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/RoutePreview/RouteManager/RouteManagerViewController.xib b/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/RoutePreview/RouteManager/RouteManagerViewController.xib index 939db26e45..59c0ec5096 100644 --- a/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/RoutePreview/RouteManager/RouteManagerViewController.xib +++ b/iphone/Maps/Classes/CustomViews/NavigationDashboard/Views/RoutePreview/RouteManager/RouteManagerViewController.xib @@ -1,9 +1,9 @@ - + - + @@ -23,14 +23,14 @@ - + - + @@ -40,7 +40,7 @@ + @@ -81,7 +82,6 @@ - @@ -100,11 +100,11 @@ - + - + @@ -155,7 +167,7 @@ - + @@ -175,7 +187,7 @@ - -