[android] Add manage route functionality

Signed-off-by: Gonzalo Pesquero <gpesquero@yahoo.es>
This commit is contained in:
Gonzalo Pesquero 2025-01-29 00:20:24 +01:00 committed by Konstantin Pastbin
parent 51fe7b5662
commit 34fcdbd3e3
17 changed files with 830 additions and 2 deletions

View file

@ -1578,6 +1578,12 @@ Java_app_organicmaps_Framework_nativeAddRoutePoint(JNIEnv * env, jclass, jstring
frm()->GetRoutingManager().AddRoutePoint(std::move(data));
}
JNIEXPORT void JNICALL
Java_app_organicmaps_Framework_nativeRemoveRoutePoints(JNIEnv * env, jclass)
{
frm()->GetRoutingManager().RemoveRoutePoints();
}
JNIEXPORT void JNICALL
Java_app_organicmaps_Framework_nativeRemoveRoutePoint(JNIEnv * env, jclass,
jint markType, jint intermediateIndex)
@ -1627,6 +1633,13 @@ Java_app_organicmaps_Framework_nativeGetRoutePoints(JNIEnv * env, jclass)
});
}
JNIEXPORT void JNICALL
Java_app_organicmaps_Framework_nativeMoveRoutePoint(JNIEnv * env, jclass,
jint currentIndex, jint targetIndex)
{
frm()->GetRoutingManager().MoveRoutePoint(currentIndex, targetIndex);
}
JNIEXPORT jobject JNICALL
Java_app_organicmaps_Framework_nativeGetTransitRouteInfo(JNIEnv * env, jclass)
{

View file

@ -333,11 +333,20 @@ public class Framework
public static native int nativeGetBestRouter(double srcLat, double srcLon,
double dstLat, double dstLon);
public static void addRoutePoint(RouteMarkData point)
{
Framework.nativeAddRoutePoint(point.mTitle, point.mSubtitle, point.mPointType,
point.mIntermediateIndex, point.mIsMyPosition,
point.mLat, point.mLon);
}
public static native void nativeAddRoutePoint(String title, String subtitle,
@RoutePointInfo.RouteMarkType int markType,
int intermediateIndex, boolean isMyPosition,
double lat, double lon);
public static native void nativeRemoveRoutePoints();
public static native void nativeRemoveRoutePoint(@RoutePointInfo.RouteMarkType int markType,
int intermediateIndex);
@ -346,6 +355,9 @@ public class Framework
public static native boolean nativeCouldAddIntermediatePoint();
@NonNull
public static native RouteMarkData[] nativeGetRoutePoints();
public static native void nativeMoveRoutePoint(int currentIndex, int targetIndex);
@NonNull
public static native TransitRouteInfo nativeGetTransitRouteInfo();
/**

View file

@ -78,6 +78,7 @@ import app.organicmaps.maplayer.ToggleMapLayerFragment;
import app.organicmaps.maplayer.isolines.IsolinesManager;
import app.organicmaps.maplayer.isolines.IsolinesState;
import app.organicmaps.maplayer.subway.SubwayManager;
import app.organicmaps.routing.ManageRouteBottomSheet;
import app.organicmaps.routing.NavigationController;
import app.organicmaps.routing.NavigationService;
import app.organicmaps.routing.RoutePointInfo;
@ -111,6 +112,7 @@ import app.organicmaps.widget.menu.MainMenu;
import app.organicmaps.widget.placepage.PlacePageController;
import app.organicmaps.widget.placepage.PlacePageData;
import app.organicmaps.widget.placepage.PlacePageViewModel;
import com.google.android.material.bottomsheet.BottomSheetDialogFragment;
import com.google.android.material.dialog.MaterialAlertDialogBuilder;
import java.util.ArrayList;
@ -229,6 +231,8 @@ public class MwmActivity extends BaseMwmFragmentActivity
@NonNull
private DisplayManager mDisplayManager;
ManageRouteBottomSheet mManageRouteBottomSheet;
private boolean mRemoveDisplayListener = true;
private int mLastUiMode = Configuration.UI_MODE_TYPE_UNDEFINED;
@ -2100,6 +2104,15 @@ public class MwmActivity extends BaseMwmFragmentActivity
RoutingController.get().start();
}
@Override
public void onManageRouteOpen()
{
// Create and show 'Manage Route' Bottom Sheet panel.
mManageRouteBottomSheet = new ManageRouteBottomSheet();
mManageRouteBottomSheet.setCancelable(false);
mManageRouteBottomSheet.show(getSupportFragmentManager(), "ManageRouteBottomSheet");
}
private boolean requestBatterySaverPermission()
{
if (!PowerManagment.isSystemPowerSaveMode(this))

View file

@ -0,0 +1,281 @@
package app.organicmaps.routing;
import android.annotation.SuppressLint;
import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.RectF;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;
import androidx.annotation.NonNull;
import androidx.appcompat.content.res.AppCompatResources;
import androidx.recyclerview.widget.RecyclerView;
import app.organicmaps.R;
import app.organicmaps.bookmarks.data.MapObject;
import app.organicmaps.util.StringUtils;
import app.organicmaps.util.UiUtils;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
public class ManageRouteAdapter extends RecyclerView.Adapter<ManageRouteAdapter.ManageRouteViewHolder>
{
Context mContext;
ArrayList<RouteMarkData> mRoutePoints;
ManageRouteListener mManageRouteListener;
public interface ManageRouteListener
{
void startDrag(RecyclerView.ViewHolder viewHolder);
void showMyLocationIcon(boolean showMyLocationIcon);
void onRoutePointDeleted(RecyclerView.ViewHolder viewHolder);
}
public ManageRouteAdapter(Context context, RouteMarkData[] routeMarkData, ManageRouteListener listener)
{
mContext = context;
mRoutePoints = new ArrayList<>(Arrays.asList(routeMarkData));
mManageRouteListener = listener;
updateMyLocationIcon();
}
@NonNull
@Override
public ManageRouteViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType)
{
View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.manage_route_list_item,
parent, false);
return new ManageRouteViewHolder(view);
}
@SuppressLint("ClickableViewAccessibility")
@Override
public void onBindViewHolder(@NonNull ManageRouteViewHolder holder, int position)
{
// Set route point icon.
int iconId;
switch(mRoutePoints.get(position).mPointType)
{
case RoutePointInfo.ROUTE_MARK_START: // Starting point.
if (mRoutePoints.get(position).mIsMyPosition)
iconId = R.drawable.ic_my_position_blue;
else
iconId = R.drawable.route_point_start;
break;
case RoutePointInfo.ROUTE_MARK_INTERMEDIATE: // Intermediate stop.
TypedArray iconArray = mContext.getResources().obtainTypedArray(R.array.route_stop_icons);
iconId = iconArray.getResourceId(mRoutePoints.get(position).mIntermediateIndex,
R.drawable.route_point_01);
iconArray.recycle();
break;
case RoutePointInfo.ROUTE_MARK_FINISH: // Destination point.
iconId = R.drawable.route_point_finish;
break;
default: // Unknown route type.
iconId = R.drawable.warning_icon;
break;
}
// Set icon widget.
holder.mImageViewIcon.setImageDrawable(AppCompatResources.getDrawable(mContext, iconId));
// Set title & subtitle.
String title, subtitle;
boolean showSubtitle;
if (mRoutePoints.get(position).mIsMyPosition)
{
// My position point.
title = mContext.getString(R.string.p2p_your_location);
if (mRoutePoints.get(position).mPointType == RoutePointInfo.ROUTE_MARK_START)
{
// Hide my position coordinates if it's the starting point of the route.
subtitle = "";
showSubtitle = false;
}
else
{
subtitle = mRoutePoints.get(position).mTitle;
showSubtitle = true;
}
}
else
{
title = mRoutePoints.get(position).mTitle;
subtitle = mRoutePoints.get(position).mSubtitle;
showSubtitle = true;
}
holder.mTextViewTitle.setText(title);
holder.mTextViewSubtitle.setText(subtitle);
UiUtils.showIf(showSubtitle, holder.mTextViewSubtitle);
// Show 'Delete' icon button only if we have intermediate stops.
UiUtils.showIf(mRoutePoints.size() > 2, holder.mImageViewDelete);
// Detection of touch events on holder view.
holder.mItemView.setOnTouchListener((v, event) -> {
if (event.getAction() == MotionEvent.ACTION_DOWN)
{
RectF deleteButtonRect = new RectF(holder.mImageViewDelete.getLeft(),
holder.mImageViewDelete.getTop(),
holder.mImageViewDelete.getRight(),
holder.mImageViewDelete.getBottom());
if (holder.mImageViewDelete.isShown() && deleteButtonRect.contains(event.getX(), event.getY()))
{
// User has clicked on the 'Delete' icon button.
mManageRouteListener.onRoutePointDeleted(holder);
}
else
{
// Call start drag listener on touch.
mManageRouteListener.startDrag(holder);
}
}
return false;
});
}
@Override
public int getItemCount()
{
return mRoutePoints.size();
}
public void moveRoutePoint(int draggedItemIndex, int targetIndex)
{
if (draggedItemIndex == targetIndex) // Dragged to same spot. Do nothing.
return;
Collections.swap(mRoutePoints, draggedItemIndex, targetIndex);
updateRoutePointsData();
notifyItemMoved(draggedItemIndex, targetIndex);
}
public void deleteRoutePoint(RecyclerView.ViewHolder viewHolder)
{
mRoutePoints.remove(viewHolder.getAbsoluteAdapterPosition());
updateRoutePointsData();
notifyItemRemoved(viewHolder.getAbsoluteAdapterPosition());
}
public void setMyLocationAsStartingPoint(MapObject myLocation)
{
String latLonString = StringUtils.formatUsingUsLocale("%.6f, %.6f",
myLocation.getLat(),
myLocation.getLon());
// Replace route point in first position with 'My Position"
mRoutePoints.set(0, new RouteMarkData(latLonString, // Title.
"", // Subtitle.
RoutePointInfo.ROUTE_MARK_START, // Point type.
0, // Intermediate index.
true, // Visible.
true, // Is my position.
false, // No passed.
myLocation.getLat(), // Latitude.
myLocation.getLon())); // Longitude.
// Update data.
updateRoutePointsData();
// Update adapter.
notifyItemChanged(0);
// Update 'My position' image button.
updateMyLocationIcon();
}
private void updateMyLocationIcon()
{
boolean containsMyLocationPoint = false;
for (RouteMarkData routePoint : mRoutePoints)
{
if (routePoint.mIsMyPosition)
{
containsMyLocationPoint = true;
break;
}
}
if (mManageRouteListener != null)
mManageRouteListener.showMyLocationIcon(!containsMyLocationPoint);
}
private void updateRoutePointsData()
{
for (int pos = 0; pos < mRoutePoints.size(); pos++)
{
if (pos == 0)
{
// Set as starting point.
mRoutePoints.get(pos).mPointType = RoutePointInfo.ROUTE_MARK_START;
}
else if (pos == mRoutePoints.size() - 1)
{
// Set as finish point.
mRoutePoints.get(pos).mPointType = RoutePointInfo.ROUTE_MARK_FINISH;
}
else
{
// Set as intermediate point.
mRoutePoints.get(pos).mPointType = RoutePointInfo.ROUTE_MARK_INTERMEDIATE;
mRoutePoints.get(pos).mIntermediateIndex = pos - 1;
}
}
}
public ArrayList<RouteMarkData> getRoutePoints()
{
return mRoutePoints;
}
static class ManageRouteViewHolder extends RecyclerView.ViewHolder
{
@NonNull
public final View mItemView;
@NonNull
public final ImageView mImageViewIcon;
@NonNull
public final TextView mTextViewTitle;
@NonNull
public final TextView mTextViewSubtitle;
@NonNull
public final ImageView mImageViewDelete;
ManageRouteViewHolder(@NonNull View itemView)
{
super(itemView);
mItemView = itemView;
mImageViewIcon = itemView.findViewById(R.id.type_icon);
mTextViewTitle = itemView.findViewById(R.id.title);
mTextViewSubtitle = itemView.findViewById(R.id.subtitle);
mImageViewDelete = itemView.findViewById(R.id.delete_icon);
}
}
}

View file

@ -0,0 +1,264 @@
package app.organicmaps.routing;
import android.app.Dialog;
import android.content.res.Resources;
import android.os.Bundle;
import android.view.KeyEvent;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.FrameLayout;
import android.widget.ImageView;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.recyclerview.widget.ItemTouchHelper;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
import app.organicmaps.Framework;
import app.organicmaps.MwmApplication;
import app.organicmaps.R;
import app.organicmaps.bookmarks.data.MapObject;
import app.organicmaps.util.UiUtils;
import com.google.android.material.bottomsheet.BottomSheetBehavior;
import com.google.android.material.bottomsheet.BottomSheetDialog;
import com.google.android.material.bottomsheet.BottomSheetDialogFragment;
import com.google.android.material.divider.MaterialDividerItemDecoration;
import java.util.ArrayList;
import static androidx.recyclerview.widget.ItemTouchHelper.*;
public class ManageRouteBottomSheet extends BottomSheetDialogFragment
implements View.OnClickListener, ManageRouteAdapter.ManageRouteListener
{
ManageRouteAdapter mManageRouteAdapter;
ItemTouchHelper mTouchHelper;
ImageView mMyLocationImageView;
MapObject mMyLocation;
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container,
@Nullable Bundle savedInstanceState)
{
View v = inflater.inflate(R.layout.manage_route_bottom_sheet, container, false);
Button cancelButton = v.findViewById(R.id.btn__cancel);
cancelButton.setOnClickListener(this);
Button planButton = v.findViewById(R.id.btn__plan);
planButton.setOnClickListener(this);
mMyLocationImageView = v.findViewById(R.id.image_my_location);
mMyLocationImageView.setOnClickListener(this);
RecyclerView manageRouteList = v.findViewById(R.id.manage_route_list);
LinearLayoutManager layoutManager = new LinearLayoutManager(getContext());
manageRouteList.setLayoutManager(layoutManager);
RecyclerView.ItemDecoration decoration = new MaterialDividerItemDecoration(getContext(),
layoutManager.getOrientation());
manageRouteList.addItemDecoration(decoration);
mManageRouteAdapter = new ManageRouteAdapter(getContext(), Framework.nativeGetRoutePoints(), this);
manageRouteList.setAdapter(mManageRouteAdapter);
// Enable drag & drop in route list.
mTouchHelper = new ItemTouchHelper(new ManageRouteItemTouchHelperCallback(mManageRouteAdapter,
getResources()));
mTouchHelper.attachToRecyclerView(manageRouteList);
return v;
}
@NonNull
@Override
public Dialog onCreateDialog(@Nullable Bundle savedInstanceState)
{
Dialog dialog = super.onCreateDialog(savedInstanceState);
// Get current location.
mMyLocation = MwmApplication.from(getContext()).getLocationHelper().getMyPosition();
// Expand bottom sheet dialog.
dialog.setOnShowListener(dialogInterface -> {
FrameLayout bottomSheet = ((BottomSheetDialog) dialogInterface).findViewById(
com.google.android.material.R.id.design_bottom_sheet);
if (bottomSheet != null)
BottomSheetBehavior.from(bottomSheet).setState(BottomSheetBehavior.STATE_EXPANDED);
});
// Set key listener to detect back button pressed.
dialog.setOnKeyListener((dialog1, keyCode, event) -> {
if (keyCode == KeyEvent.KEYCODE_BACK && event.getAction() == KeyEvent.ACTION_UP)
{
// Dismiss the fragment
dismiss();
return true;
}
// Otherwise, do nothing else.
return false;
});
return dialog;
}
@Override
public void onClick(View v)
{
int buttonId = v.getId();
if (buttonId == R.id.btn__cancel)
{
// Close dialog if 'Cancel' button is pressed.
dismiss();
}
else if (buttonId == R.id.image_my_location)
{
// Set 'My Location' as starting point of the route.
if (mMyLocation != null)
mManageRouteAdapter.setMyLocationAsStartingPoint(mMyLocation);
}
else if (buttonId == R.id.btn__plan)
{
// Get route points from adapter.
ArrayList<RouteMarkData> newRoutePoints = mManageRouteAdapter.getRoutePoints();
// Make sure that the new route contains at least 2 points (start and destination).
assert(newRoutePoints.size() >= 2);
// Remove all existing route points.
Framework.nativeRemoveRoutePoints();
// First, add the destination point.
Framework.addRoutePoint(newRoutePoints.get(newRoutePoints.size() - 1));
// Secondly, add the starting point.
Framework.addRoutePoint(newRoutePoints.get(0));
// And then, add all intermediate points.
for (int pos = 1; pos < newRoutePoints.size() - 1; pos++)
Framework.addRoutePoint(newRoutePoints.get(pos));
// Intermediate route points are added sorted by distance.
// We have to make sure that they follow the requested order.
RouteMarkData[] finalRoutePoints = Framework.nativeGetRoutePoints();
for (int first = 1; first < newRoutePoints.size() - 1; first++)
{
int secondIndex = -1;
for (int second = first; second < newRoutePoints.size() - 1; second++)
{
if (finalRoutePoints[first].equals(newRoutePoints.get(second)))
{
secondIndex = second;
break;
}
}
if (secondIndex < 0)
{
// Something went bad. Intermediate point not found in the route points.
break;
}
if (first != secondIndex)
{
// Intermediate point needs to be moved.
Framework.nativeMoveRoutePoint(secondIndex, first);
// Refresh final route points.
finalRoutePoints = Framework.nativeGetRoutePoints();
}
}
// Launch route planning.
RoutingController.get().launchPlanning();
// Dismiss (close) manage route bottom sheet.
dismiss();
}
}
@Override
public void startDrag(RecyclerView.ViewHolder viewHolder)
{
// Start dragging.
mTouchHelper.startDrag(viewHolder);
}
@Override
public void showMyLocationIcon(boolean showMyLocationIcon)
{
UiUtils.showIf(showMyLocationIcon && mMyLocation != null, mMyLocationImageView);
}
@Override
public void onRoutePointDeleted(RecyclerView.ViewHolder viewHolder)
{
mManageRouteAdapter.deleteRoutePoint(viewHolder);
mManageRouteAdapter.notifyDataSetChanged();
}
private static class ManageRouteItemTouchHelperCallback extends ItemTouchHelper.Callback
{
private final ManageRouteAdapter mManageRouteAdapter;
public ManageRouteItemTouchHelperCallback(ManageRouteAdapter adapter, Resources resources)
{
mManageRouteAdapter = adapter;
}
@Override
public int getMovementFlags(@NonNull RecyclerView recyclerView,
@NonNull RecyclerView.ViewHolder viewHolder)
{
// Enable up & down dragging. No left-right swiping is enabled.
return makeMovementFlags(UP | DOWN, 0);
}
@Override
public boolean isLongPressDragEnabled()
{
return false;
}
@Override
public boolean isItemViewSwipeEnabled()
{
return false;
}
@Override
public boolean onMove(@NonNull RecyclerView recyclerView,
@NonNull RecyclerView.ViewHolder viewHolder,
@NonNull RecyclerView.ViewHolder target)
{
mManageRouteAdapter.moveRoutePoint(viewHolder.getAbsoluteAdapterPosition(),
target.getAbsoluteAdapterPosition());
return true;
}
@Override
public void onSwiped(@NonNull RecyclerView.ViewHolder viewHolder, int direction)
{
}
@Override
public void clearView(@NonNull RecyclerView recyclerView,
@NonNull RecyclerView.ViewHolder viewHolder)
{
super.clearView(recyclerView, viewHolder);
// Called when dragging action has finished.
mManageRouteAdapter.notifyDataSetChanged();
}
}
}

View file

@ -16,8 +16,8 @@ public class RouteMarkData
@Nullable
public final String mSubtitle;
@RoutePointInfo.RouteMarkType
public final int mPointType;
public final int mIntermediateIndex;
public int mPointType;
public int mIntermediateIndex;
public final boolean mIsVisible;
public final boolean mIsMyPosition;
public final boolean mIsPassed;
@ -39,4 +39,12 @@ public class RouteMarkData
mLat = lat;
mLon = lon;
}
public boolean equals(RouteMarkData other)
{
return mTitle != null && other.mTitle != null &&
mTitle.compareTo(other.mTitle) == 0 &&
mSubtitle != null && other.mSubtitle != null &&
mSubtitle.compareTo(other.mSubtitle) == 0;
}
}

View file

@ -151,6 +151,8 @@ final class RoutingBottomMenuController implements View.OnClickListener
Resources res = mContext.getResources();
mTransitViewDecorator = new DotDividerItemDecoration(dividerDrawable, res.getDimensionPixelSize(R.dimen.margin_base),
res.getDimensionPixelSize(R.dimen.margin_half));
Button manageRouteButton = altitudeChartFrame.findViewById(R.id.btn__manage_route);
manageRouteButton.setOnClickListener(this);
}
void showAltitudeChartAndRoutingDetails()
@ -485,5 +487,7 @@ final class RoutingBottomMenuController implements View.OnClickListener
int pointType = (Integer) mActionMessage.getTag();
mListener.onSearchRoutePoint(pointType);
}
else if (id == R.id.btn__manage_route && mListener != null)
mListener.onManageRouteOpen();
}
}

View file

@ -5,4 +5,5 @@ public interface RoutingBottomMenuListener
void onUseMyPositionAsStart();
void onSearchRoutePoint(@RoutePointInfo.RouteMarkType int type);
void onRoutingStart();
void onManageRouteOpen();
}

View file

@ -426,6 +426,17 @@ public class RoutingController
resetToPlanningStateIfNavigating();
}
public void launchPlanning()
{
build();
setState(State.PREPARE);
startPlanning();
if (mContainer != null)
mContainer.updateMenu();
if (mContainer != null)
mContainer.onResetToPlanningState();
}
/**
* @return False if not navigating, true otherwise
*/

View file

@ -0,0 +1,28 @@
<vector
android:height="24dp"
android:tint="#808080"
android:viewportHeight="48"
android:viewportWidth="48"
android:width="24dp"
xmlns:android="http://schemas.android.com/apk/res/android">
<path
android:fillColor="@android:color/white"
android:pathData="M06,09 A1,1 0 0 0 06,15 A1,1 0 0 0 06,09
M16,09 A1,1 0 0 0 16,15
M16,09 L42,09 L42,15 L16,15 z
M42,15 A1,1 0 0 0 42,09
M06,21 A1,1 0 0 0 06,27 A1,1 0 0 0 06,21
M16,21 A1,1 0 0 0 16,27
M16,21 L42,21 L42,27 L16,27 z
M42,27 A1,1 0 0 0 42,21
M06,33 A1,1 0 0 0 06,39 A1,1 0 0 0 06,33
M16,33 A1,1 0 0 0 16,39
M16,33 L42,33 L42,39 L16,39 z
M42,39 A1,1 0 0 0 42,33"
android:strokeWidth="1"
android:strokeColor="@android:color/white"/>
</vector>

View file

@ -0,0 +1,9 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="24dp"
android:height="24dp"
android:viewportWidth="24"
android:viewportHeight="24">
<path
android:pathData="M12,8c-2.21,0 -4,1.79 -4,4 0,2.21 1.79,4 4,4 2.21,0 4,-1.79 4,-4 0,-2.21 -1.79,-4 -4,-4ZM20.94,11c-0.46,-4.17 -3.77,-7.48 -7.94,-7.94v-2.06h-2v2.06c-4.17,0.46 -7.48,3.77 -7.94,7.94h-2.06v2h2.06c0.46,4.17 3.77,7.48 7.94,7.94v2.06h2v-2.06c4.17,-0.46 7.48,-3.77 7.94,-7.94h2.06v-2h-2.06ZM12,19c-3.87,0 -7,-3.13 -7,-7 0,-3.87 3.13,-7 7,-7 3.87,0 7,3.13 7,7 0,3.87 -3.13,7 -7,7Z"
android:fillColor="@color/base_accent"/>
</vector>

View file

@ -0,0 +1,9 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="24dp"
android:height="24dp"
android:viewportWidth="24"
android:viewportHeight="24">
<path
android:fillColor="@android:color/holo_blue_light"
android:pathData="M21,3L3,10.53v0.98l6.84,2.65L12.48,21h0.98L21,3z"/>
</vector>

View file

@ -63,6 +63,16 @@
tools:text="5 h 55 min • 1555km"
tools:visibility="visible" />
<Button
android:id="@+id/btn__manage_route"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginStart="6dp"
style="@style/MwmWidget.Button"
android:text="@string/planning_route_manage_route"
android:drawableStart="@drawable/ic_manage_route"
android:drawablePadding="6dp"/>
<Button
android:id="@+id/start"
style="@style/MwmWidget.Button.Primary"

View file

@ -82,6 +82,16 @@
android:layout_weight="10"
android:layout_gravity="center_vertical" />
<Button
android:id="@+id/btn__manage_route"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginStart="6dp"
style="@style/MwmWidget.Button"
android:text="@string/planning_route_manage_route"
android:drawableStart="@drawable/ic_manage_route"
android:drawablePadding="6dp"/>
<Button
android:id="@+id/start"
style="@style/MwmWidget.Button.Primary"

View file

@ -82,4 +82,14 @@
tools:showIn="@layout/menu_route_plan_line"
android:layout_gravity="center_vertical" />
</LinearLayout>
<Button
android:id="@+id/btn__manage_route"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
style="@style/MwmWidget.Button"
android:text="@string/planning_route_manage_route"
android:drawableStart="@drawable/ic_manage_route"
android:drawablePadding="6dp"/>
</LinearLayout>

View file

@ -0,0 +1,79 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:layout_marginBottom="@dimen/margin_half"
android:layout_marginTop="@dimen/margin_half"
android:paddingStart="@dimen/altitude_chart_container_padding_left"
android:paddingEnd="@dimen/altitude_chart_container_padding_left">
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:layout_marginTop="@dimen/margin_half"
android:layout_marginBottom="@dimen/margin_half">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentStart="true"
android:text="@string/planning_route_manage_route"
android:textSize="20sp"
android:textStyle="bold"/>
<ImageView
android:id="@+id/image_my_location"
android:layout_width="24dp"
android:layout_height="24dp"
android:layout_alignParentEnd="true"
android:visibility="invisible"
app:srcCompat="@drawable/ic_my_location_blue"/>
</RelativeLayout>
<View
android:layout_width="match_parent"
android:layout_height="@dimen/divider_height"
android:background="?dividerHorizontal"/>
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/manage_route_list"
android:layout_width="match_parent"
android:layout_height="wrap_content"
tools:itemCount="3"
tools:listitem="@layout/manage_route_list_item"/>
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:layout_marginTop="@dimen/margin_half"
android:layout_marginBottom="@dimen/margin_half">
<Button
android:id="@+id/btn__cancel"
style="@style/MwmWidget.Button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentStart="true"
android:layout_centerVertical="true"
android:text="@string/cancel"/>
<Button
android:id="@+id/btn__plan"
style="@style/MwmWidget.Button.Primary"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="end|center_vertical"
android:layout_alignParentEnd="true"
android:layout_centerVertical="true"
android:text="@string/button_plan"/>
</RelativeLayout>
</LinearLayout>

View file

@ -0,0 +1,66 @@
<androidx.constraintlayout.widget.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="60dp"
android:gravity="center_vertical">
<ImageView
android:id="@+id/type_icon"
android:layout_width="24dp"
android:layout_height="24dp"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintBottom_toBottomOf="parent"
app:srcCompat="@drawable/route_point_finish"/>
<TextView
android:id="@+id/title"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginStart="6dp"
android:layout_marginEnd="6dp"
android:ellipsize="end"
android:inputType="none"
android:maxLines="1"
android:textAppearance="@style/MwmTextAppearance.Body1"
android:textStyle="bold"
android:textAlignment="viewStart"
app:layout_constraintStart_toEndOf="@id/type_icon"
app:layout_constraintEnd_toStartOf="@id/delete_icon"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintBottom_toTopOf="@id/subtitle"
app:layout_constrainedWidth="true"
tools:text="Title"/>
<TextView
android:id="@+id/subtitle"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginStart="6dp"
android:layout_marginEnd="6dp"
android:ellipsize="end"
android:inputType="none"
android:maxLines="2"
android:textAppearance="@style/MwmTextAppearance.Body1"
android:textAlignment="viewStart"
app:layout_constraintStart_toEndOf="@id/type_icon"
app:layout_constraintEnd_toStartOf="@id/delete_icon"
app:layout_constraintTop_toBottomOf="@id/title"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constrainedWidth="true"
tools:text="Subtitle"/>
<ImageView
android:id="@+id/delete_icon"
android:layout_width="24dp"
android:layout_height="24dp"
style="@style/MwmWidget.Editor.MetadataIcon"
android:layout_margin="0dp"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintBottom_toBottomOf="parent"
app:srcCompat="@drawable/ic_delete"/>
</androidx.constraintlayout.widget.ConstraintLayout>