forked from organicmaps/organicmaps-tmp
[ios][android] Replacing altitude difference to ascent and descent (#3062)
* [ios][android] Replacing altitude difference to ascent and descent The app for pedestrian and cyclists route show altitude difference between highest and lowest points. It's not too useful data in multiple cases. More informative for hikers and cyclists from values of elevation total ascent and total descent on route. I replaced altitude difference to total ascent and total descent. Routing: - Removed calculation of minRouteAltitude and maxRouteAltitude. Added calculation of totalAscent and totalDescent (map/routing_manager.cpp) iOS: - Replaced altitudeDifference to 2 separate value totalAscent and totalDescent android: - Replaced altitudeDifference to 2 separate value totalAscent and totalDescent - Removed unnessesary icon (ic_altitude_difference.webp) Signed-off-by: Arseny Novikov <arseny.n@gmail.com> * Applying style guide to map/routing_manager.cpp Signed-off-by: Arseny Novikov <arseny.n@gmail.com> * Fixes for route ascent/descent. Used unsigned var types and fix idents Changes based on pull request (organicmaps/organicmaps#3062) code review. - fixed idents - used uint32_t instead int32_t for totalAscent and totalDescent fields - replaced 2 guard variable initialization to one if Signed-off-by: Arseny Novikov <arseny.n@gmail.com> * Route ascent/descent. Fixed field description, final to java constant. Signed-off-by: Arseny Novikov <arseny.n@gmail.com> * Explicit cast from uint32_t to jint in jni for ascent/descent Signed-off-by: Arseny Novikov <arseny.n@gmail.com> * [ios] Route ascent/descent. Moved font attributes to static field Signed-off-by: Arseny Novikov <arseny.n@gmail.com>
This commit is contained in:
parent
eb4bd4f378
commit
e6aaadf28f
13 changed files with 61 additions and 54 deletions
|
@ -1230,12 +1230,12 @@ Java_com_mapswithme_maps_Framework_nativeGenerateRouteAltitudeChartBits(JNIEnv *
|
|||
}
|
||||
|
||||
vector<uint8_t> imageRGBAData;
|
||||
int32_t minRouteAltitude = 0;
|
||||
int32_t maxRouteAltitude = 0;
|
||||
uint32_t totalAscent = 0;
|
||||
uint32_t totalDescent = 0;
|
||||
measurement_utils::Units units = measurement_utils::Units::Metric;
|
||||
if (!fr->GetRoutingManager().GenerateRouteAltitudeChart(
|
||||
width, height, altitudes, routePointDistanceM, imageRGBAData,
|
||||
minRouteAltitude, maxRouteAltitude, units))
|
||||
totalAscent, totalDescent, units))
|
||||
{
|
||||
LOG(LWARNING, ("Can't generate route altitude image."));
|
||||
return nullptr;
|
||||
|
@ -1245,13 +1245,13 @@ Java_com_mapswithme_maps_Framework_nativeGenerateRouteAltitudeChartBits(JNIEnv *
|
|||
jclass const routeAltitudeLimitsClass = env->GetObjectClass(routeAltitudeLimits);
|
||||
ASSERT(routeAltitudeLimitsClass, ());
|
||||
|
||||
static jfieldID const minRouteAltitudeField = env->GetFieldID(routeAltitudeLimitsClass, "minRouteAltitude", "I");
|
||||
ASSERT(minRouteAltitudeField, ());
|
||||
env->SetIntField(routeAltitudeLimits, minRouteAltitudeField, minRouteAltitude);
|
||||
static jfieldID const totalAscentField = env->GetFieldID(routeAltitudeLimitsClass, "totalAscent", "I");
|
||||
ASSERT(totalAscentField, ());
|
||||
env->SetIntField(routeAltitudeLimits, totalAscentField, static_cast<jint>(totalAscent));
|
||||
|
||||
static jfieldID const maxRouteAltitudeField = env->GetFieldID(routeAltitudeLimitsClass, "maxRouteAltitude", "I");
|
||||
ASSERT(maxRouteAltitudeField, ());
|
||||
env->SetIntField(routeAltitudeLimits, maxRouteAltitudeField, maxRouteAltitude);
|
||||
static jfieldID const totalDescentField = env->GetFieldID(routeAltitudeLimitsClass, "totalDescent", "I");
|
||||
ASSERT(totalDescentField, ());
|
||||
env->SetIntField(routeAltitudeLimits, totalDescentField, static_cast<jint>(totalDescent));
|
||||
|
||||
static jfieldID const isMetricUnitsField = env->GetFieldID(routeAltitudeLimitsClass, "isMetricUnits", "Z");
|
||||
ASSERT(isMetricUnitsField, ());
|
||||
|
|
Binary file not shown.
Before Width: | Height: | Size: 88 B |
Binary file not shown.
Before Width: | Height: | Size: 118 B |
Binary file not shown.
Before Width: | Height: | Size: 154 B |
Binary file not shown.
Before Width: | Height: | Size: 114 B |
Binary file not shown.
Before Width: | Height: | Size: 194 B |
|
@ -112,8 +112,8 @@ public class Framework
|
|||
|
||||
public static class RouteAltitudeLimits
|
||||
{
|
||||
public int minRouteAltitude;
|
||||
public int maxRouteAltitude;
|
||||
public int totalAscent;
|
||||
public int totalDescent;
|
||||
public boolean isMetricUnits;
|
||||
}
|
||||
|
||||
|
|
|
@ -264,17 +264,10 @@ final class RoutingBottomMenuController implements View.OnClickListener
|
|||
{
|
||||
mAltitudeChart.setImageBitmap(bm);
|
||||
UiUtils.show(mAltitudeChart);
|
||||
String meter = mAltitudeDifference.getResources().getString(R.string.meter);
|
||||
String foot = mAltitudeDifference.getResources().getString(R.string.foot);
|
||||
mAltitudeDifference.setText(String.format(Locale.getDefault(), "%d %s",
|
||||
limits.maxRouteAltitude - limits.minRouteAltitude,
|
||||
limits.isMetricUnits ? meter : foot));
|
||||
Drawable icon = ContextCompat.getDrawable(mContext,
|
||||
R.drawable.ic_altitude_difference);
|
||||
int colorAccent = ContextCompat.getColor(mContext,
|
||||
UiUtils.getStyledResourceId(mContext, R.attr.colorAccent));
|
||||
mAltitudeDifference.setCompoundDrawablesRelativeWithIntrinsicBounds(Graphics.tint(icon, colorAccent),
|
||||
null, null, null);
|
||||
final String unit = limits.isMetricUnits ? mAltitudeDifference.getResources().getString(R.string.meter) : mAltitudeDifference.getResources().getString(R.string.foot);
|
||||
mAltitudeDifference.setText(String.format(Locale.getDefault(), "▲ %d %s ▼ %d %s",
|
||||
limits.totalAscent, unit,
|
||||
limits.totalDescent, unit));
|
||||
UiUtils.show(mAltitudeDifference);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -39,6 +39,12 @@ final class BaseRoutePreviewStatus: SolidTouchView {
|
|||
|
||||
weak var navigationInfo: MWMNavigationDashboardEntity?
|
||||
|
||||
static let elevationAttributes: [NSAttributedString.Key: Any] =
|
||||
[
|
||||
.foregroundColor: UIColor.linkBlue(),
|
||||
.font: UIFont.medium14()
|
||||
]
|
||||
|
||||
var elevation: NSAttributedString? {
|
||||
didSet {
|
||||
updateResultsLabel()
|
||||
|
@ -118,15 +124,11 @@ final class BaseRoutePreviewStatus: SolidTouchView {
|
|||
if MWMRouter.hasRouteAltitude() {
|
||||
heightBox.isHidden = false
|
||||
MWMRouter.routeAltitudeImage(for: heightProfileImage.frame.size,
|
||||
completion: { image, elevation in
|
||||
self.heightProfileImage.image = image
|
||||
guard let elevation = elevation else { return }
|
||||
let attributes: [NSAttributedString.Key: Any] =
|
||||
[
|
||||
.foregroundColor: UIColor.linkBlue(),
|
||||
.font: UIFont.medium14()
|
||||
]
|
||||
self.elevation = NSAttributedString(string: "▲▼ \(elevation)", attributes: attributes)
|
||||
completion: { image, totalAscent, totalDescent in
|
||||
self.heightProfileImage.image = image
|
||||
if let totalAscent = totalAscent, let totalDescent = totalDescent {
|
||||
self.elevation = NSAttributedString(string: "▲ \(totalAscent) ▼ \(totalDescent)", attributes: BaseRoutePreviewStatus.elevationAttributes)
|
||||
}
|
||||
})
|
||||
} else {
|
||||
heightBox.isHidden = true
|
||||
|
|
|
@ -8,7 +8,7 @@ typedef NS_ENUM(NSInteger, MWMRoadType) {
|
|||
MWMRoadTypeMotorway
|
||||
};
|
||||
|
||||
typedef void (^MWMImageHeightBlock)(UIImage *, NSString *);
|
||||
typedef void (^MWMImageHeightBlock)(UIImage *, NSString *, NSString *);
|
||||
|
||||
@interface MWMRouter : NSObject
|
||||
|
||||
|
|
|
@ -23,7 +23,8 @@ using namespace routing;
|
|||
@interface MWMRouter () <MWMLocationObserver, MWMFrameworkRouteBuilderObserver>
|
||||
|
||||
@property(nonatomic) NSMutableDictionary<NSValue *, NSData *> *altitudeImagesData;
|
||||
@property(nonatomic) NSString *altitudeElevation;
|
||||
@property(nonatomic) NSString *totalAscent;
|
||||
@property(nonatomic) NSString *totalDescent;
|
||||
@property(nonatomic) dispatch_queue_t renderAltitudeImagesQueue;
|
||||
@property(nonatomic) uint32_t routeManagerTransactionId;
|
||||
@property(nonatomic) BOOL canAutoAddLastLocation;
|
||||
|
@ -370,13 +371,13 @@ char const *kRenderAltitudeImagesQueueLabel = "mapsme.mwmrouter.renderAltitudeIm
|
|||
NSData *imageData = router.altitudeImagesData[sizeValue];
|
||||
if (!imageData) {
|
||||
std::vector<uint8_t> imageRGBAData;
|
||||
int32_t minRouteAltitude = 0;
|
||||
int32_t maxRouteAltitude = 0;
|
||||
uint32_t totalAscent = 0;
|
||||
uint32_t totalDescent = 0;
|
||||
measurement_utils::Units units = measurement_utils::Units::Metric;
|
||||
|
||||
if (!GetFramework().GetRoutingManager().GenerateRouteAltitudeChart(width, height, *altitudes,
|
||||
*routePointDistanceM, imageRGBAData,
|
||||
minRouteAltitude, maxRouteAltitude, units)) {
|
||||
totalAscent, totalDescent, units)) {
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -386,15 +387,16 @@ char const *kRenderAltitudeImagesQueueLabel = "mapsme.mwmrouter.renderAltitudeIm
|
|||
router.altitudeImagesData[sizeValue] = imageData;
|
||||
|
||||
auto const localizedUnits = platform::GetLocalizedAltitudeUnits();
|
||||
auto const height = maxRouteAltitude - minRouteAltitude;
|
||||
router.altitudeElevation =
|
||||
@(measurement_utils::FormatAltitudeWithLocalization(height, localizedUnits.m_low).c_str());
|
||||
router.totalAscent =
|
||||
@(measurement_utils::FormatAltitudeWithLocalization(totalAscent, localizedUnits.m_low).c_str());
|
||||
router.totalDescent =
|
||||
@(measurement_utils::FormatAltitudeWithLocalization(totalDescent, localizedUnits.m_low).c_str());
|
||||
}
|
||||
|
||||
dispatch_async(dispatch_get_main_queue(), ^{
|
||||
UIImage *altitudeImage = [UIImage imageWithRGBAData:imageData width:width height:height];
|
||||
if (altitudeImage)
|
||||
block(altitudeImage, router.altitudeElevation);
|
||||
block(altitudeImage, router.totalAscent, router.totalDescent);
|
||||
});
|
||||
});
|
||||
}
|
||||
|
@ -403,7 +405,8 @@ char const *kRenderAltitudeImagesQueueLabel = "mapsme.mwmrouter.renderAltitudeIm
|
|||
auto router = self.router;
|
||||
dispatch_async(router.renderAltitudeImagesQueue, ^{
|
||||
[router.altitudeImagesData removeAllObjects];
|
||||
router.altitudeElevation = nil;
|
||||
router.totalAscent = nil;
|
||||
router.totalDescent = nil;
|
||||
});
|
||||
}
|
||||
|
||||
|
|
|
@ -1158,8 +1158,8 @@ bool RoutingManager::GenerateRouteAltitudeChart(uint32_t width, uint32_t height,
|
|||
geometry::Altitudes const & altitudes,
|
||||
vector<double> const & routePointDistanceM,
|
||||
vector<uint8_t> & imageRGBAData,
|
||||
int32_t & minRouteAltitude,
|
||||
int32_t & maxRouteAltitude,
|
||||
uint32_t & totalAscent,
|
||||
uint32_t & totalDescent,
|
||||
measurement_utils::Units & altitudeUnits) const
|
||||
{
|
||||
CHECK_EQUAL(altitudes.size(), routePointDistanceM.size(), ());
|
||||
|
@ -1170,9 +1170,17 @@ bool RoutingManager::GenerateRouteAltitudeChart(uint32_t width, uint32_t height,
|
|||
GetStyleReader().GetCurrentStyle(), imageRGBAData))
|
||||
return false;
|
||||
|
||||
auto const minMaxIt = minmax_element(altitudes.cbegin(), altitudes.cend());
|
||||
geometry::Altitude const minRouteAltitudeM = *minMaxIt.first;
|
||||
geometry::Altitude const maxRouteAltitudeM = *minMaxIt.second;
|
||||
uint32_t totalAscentM = 0;
|
||||
uint32_t totalDescentM = 0;
|
||||
int16_t delta;
|
||||
|
||||
for (size_t i = 1; i < altitudes.size(); i++) {
|
||||
delta = altitudes[i] - altitudes[i - 1];
|
||||
if (delta > 0)
|
||||
totalAscentM += delta;
|
||||
else
|
||||
totalDescentM += -delta;
|
||||
}
|
||||
|
||||
if (!settings::Get(settings::kMeasurementUnits, altitudeUnits))
|
||||
altitudeUnits = measurement_utils::Units::Metric;
|
||||
|
@ -1180,12 +1188,12 @@ bool RoutingManager::GenerateRouteAltitudeChart(uint32_t width, uint32_t height,
|
|||
switch (altitudeUnits)
|
||||
{
|
||||
case measurement_utils::Units::Imperial:
|
||||
minRouteAltitude = measurement_utils::MetersToFeet(minRouteAltitudeM);
|
||||
maxRouteAltitude = measurement_utils::MetersToFeet(maxRouteAltitudeM);
|
||||
totalAscent = measurement_utils::MetersToFeet(totalAscentM);
|
||||
totalDescent = measurement_utils::MetersToFeet(totalDescentM);
|
||||
break;
|
||||
case measurement_utils::Units::Metric:
|
||||
minRouteAltitude = minRouteAltitudeM;
|
||||
maxRouteAltitude = maxRouteAltitudeM;
|
||||
totalAscent = totalAscentM;
|
||||
totalDescent = totalDescentM;
|
||||
break;
|
||||
}
|
||||
return true;
|
||||
|
|
|
@ -277,8 +277,8 @@ public:
|
|||
/// \param altitudes route points altitude.
|
||||
/// \param routePointDistanceM distance in meters from route beginning to route points.
|
||||
/// \param imageRGBAData is bits of result image in RGBA.
|
||||
/// \param minRouteAltitude is min altitude along the route in altitudeUnits.
|
||||
/// \param maxRouteAltitude is max altitude along the route in altitudeUnits.
|
||||
/// \param totalAscent is total ascent of the route in altitudeUnits.
|
||||
/// \param totalDescent is total descent of the route in altitudeUnits.
|
||||
/// \param altitudeUnits is units (meters or feet) which is used to pass min and max altitudes.
|
||||
/// \returns If there is valid route info and the chart was generated returns true
|
||||
/// and false otherwise. If the method returns true it is guaranteed that the size of
|
||||
|
@ -288,8 +288,9 @@ public:
|
|||
bool GenerateRouteAltitudeChart(uint32_t width, uint32_t height,
|
||||
geometry::Altitudes const & altitudes,
|
||||
std::vector<double> const & routePointDistanceM,
|
||||
std::vector<uint8_t> & imageRGBAData, int32_t & minRouteAltitude,
|
||||
int32_t & maxRouteAltitude,
|
||||
std::vector<uint8_t> & imageRGBAData,
|
||||
uint32_t & totalAscent,
|
||||
uint32_t & totalDescent,
|
||||
measurement_utils::Units & altitudeUnits) const;
|
||||
|
||||
uint32_t OpenRoutePointsTransaction();
|
||||
|
|
Loading…
Add table
Reference in a new issue