Set decimal separator depending on locale #4029

Merged
root merged 1 commit from decimal into master 2023-09-19 08:01:30 +00:00
15 changed files with 314 additions and 111 deletions

View file

@ -163,5 +163,14 @@ jobs:
# world_feed_integration_tests - https://github.com/organicmaps/organicmaps/issues/215
CTEST_EXCLUDE_REGEX: "drape_tests|generator_integration_tests|opening_hours_integration_tests|opening_hours_supported_features_tests|routing_benchmarks|routing_integration_tests|routing_quality_tests|search_quality_tests|storage_integration_tests|shaders_tests|world_feed_integration_tests"
run: |
sudo locale-gen en_US
sudo locale-gen en_US.UTF-8
sudo locale-gen es_ES
sudo locale-gen es_ES.UTF-8
sudo locale-gen fr_FR
sudo locale-gen fr_FR.UTF-8
sudo locale-gen ru_RU
sudo locale-gen ru_RU.UTF-8
sudo update-locale
Review

I think it's better to move this part to a separate job. Probably near the dependencies job or before the tests job with a comment that it's required only for testing

I think it's better to move this part to a separate job. Probably near the dependencies job or before the tests job with a comment that it's required only for testing
Review

Hi @AndrewShkrob! I've removed the UTs for the ToStringPrecision() function, so these calls to sudo locale-gen XX_YY are no longer needed. Thanks anyway for your comment.

Hi @AndrewShkrob! I've removed the UTs for the `ToStringPrecision()` function, so these calls to `sudo locale-gen XX_YY` are no longer needed. Thanks anyway for your comment.
Review

Hello again @AndrewShkrob! As I've included back again the UTs for the ToStringPrecision() function, I've included the sudo locale-gen xx_YY calls in the Github workflow for Linux, in the same place that I initially used.

You suggested to move this part to a separate job, but I'm not sure how to do it.

Currently, Linux tests are run in the last step 'Tests' of the 'linux-matrix' job.

I'd appreciate if you could provide a piece of code of your proposal for this change.

Hello again @AndrewShkrob! As I've included back again the UTs for the `ToStringPrecision()` function, I've included the `sudo locale-gen xx_YY` calls in the Github workflow for Linux, in the same place that I initially used. You suggested to move this part to a separate job, but I'm not sure how to do it. Currently, Linux tests are run in the last step 'Tests' of the 'linux-matrix' job. I'd appreciate if you could provide a piece of code of your proposal for this change.
biodranik commented 2023-09-04 00:27:10 +00:00 (Migrated from github.com)
Review
  1. You can move these calls to the tests section.
  2. Is there a way to detect if locale is installed and skip the test if it's not?
  3. Is there a way to get a locale-specific delimiter from a test and avoid ifdef?
1. You can move these calls to the tests section. 2. Is there a way to detect if locale is installed and skip the test if it's not? 3. Is there a way to get a locale-specific delimiter from a test and avoid ifdef?
Review
  1. I still don't know where you want these calls to be placed. They already are located in the 'Tests' step of the 'linux-matrix' job, and I don't see any other tests section in this 'linux-check.yaml" file. Please provide a diff file or some direct instructions (like: insert these calls in line XX of file YY), otherwise I'm lost regarding this change request.

  2. I've added some try/catch in the GetLocale() function in Linux to detect errors while loading the locales, so now the tests don't crash and we can skip the tests for the missing locales. Under macOS, GetLocale() never fails, even if a wrong locale is requested.

  3. I've removed the #ifdefs for grouping separator checking in different OS's by adding a AddGroupingSeparators() function that uses locale's grouping separator for the tests.

1. I still don't know where you want these calls to be placed. They already are located in the 'Tests' step of the 'linux-matrix' job, and I don't see any other tests section in this 'linux-check.yaml" file. Please provide a diff file or some direct instructions (like: insert these calls in line XX of file YY), otherwise I'm lost regarding this change request. 2. I've added some try/catch in the `GetLocale()` function in Linux to detect errors while loading the locales, so now the tests don't crash and we can skip the tests for the missing locales. Under macOS, `GetLocale()` never fails, even if a wrong locale is requested. 3. I've removed the `#ifdefs` for grouping separator checking in different OS's by adding a `AddGroupingSeparators()` function that uses locale's grouping separator for the tests.
ln -s ../data data
ctest -LE "fixture" -E "$CTEST_EXCLUDE_REGEX" --output-on-failure

View file

@ -39,6 +39,7 @@
#include "platform/country_file.hpp"
#include "platform/local_country_file.hpp"
#include "platform/local_country_file_utils.hpp"
#include "platform/locale.hpp"
#include "platform/location.hpp"
#include "platform/localization.hpp"
#include "platform/measurement_utils.hpp"
@ -1294,6 +1295,10 @@ Java_app_organicmaps_Framework_nativeGenerateRouteAltitudeChartBits(JNIEnv * env
totalDescent = measurement_utils::MetersToFeet(totalDescent);
}
jni::TScopedLocalRef const totalAscentString(env, jni::ToJavaString(env, ToStringPrecision(totalAscent, 0)));
jni::TScopedLocalRef const totalDescentString(env, jni::ToJavaString(env, ToStringPrecision(totalDescent, 0)));
// Passing route limits.
// Do not use jni::GetGlobalClassRef, because this class is used only to init static fieldId vars.
static jclass const routeAltitudeLimitsClass = env->GetObjectClass(routeAltitudeLimits);
@ -1307,6 +1312,14 @@ Java_app_organicmaps_Framework_nativeGenerateRouteAltitudeChartBits(JNIEnv * env
ASSERT(totalDescentField, ());
env->SetIntField(routeAltitudeLimits, totalDescentField, static_cast<jint>(totalDescent));
static jfieldID const totalAscentStringField = env->GetFieldID(routeAltitudeLimitsClass, "totalAscentString", "Ljava/lang/String;");
ASSERT(totalAscentStringField, ());
env->SetObjectField(routeAltitudeLimits, totalAscentStringField, totalAscentString.get());
static jfieldID const totalDescentStringField = env->GetFieldID(routeAltitudeLimitsClass, "totalDescentString", "Ljava/lang/String;");
ASSERT(totalDescentStringField, ());
env->SetObjectField(routeAltitudeLimits, totalDescentStringField, totalDescentString.get());
static jfieldID const isMetricUnitsField = env->GetFieldID(routeAltitudeLimitsClass, "isMetricUnits", "Z");
ASSERT(isMetricUnitsField, ());
env->SetBooleanField(routeAltitudeLimits, isMetricUnitsField, units == Units::Metric);

View file

@ -50,8 +50,18 @@ Locale GetCurrentLocale()
"()Ljava/lang/String;");
jni::ScopedLocalRef currencyCode(env, env->CallStaticObjectMethod(g_utilsClazz, getCurrencyCodeId));
static jmethodID const getDecimalSeparatorId = jni::GetStaticMethodID(env, g_utilsClazz, "getDecimalSeparator",
"()Ljava/lang/String;");
jni::ScopedLocalRef decimalSeparatorChar(env, env->CallStaticObjectMethod(g_utilsClazz, getDecimalSeparatorId));
static jmethodID const getGroupingSeparatorId = jni::GetStaticMethodID(env, g_utilsClazz, "getGroupingSeparator",
"()Ljava/lang/String;");
jni::ScopedLocalRef groupingSeparatorChar(env, env->CallStaticObjectMethod(g_utilsClazz, getGroupingSeparatorId));
return {jni::ToNativeString(env, static_cast<jstring>(languageCode.get())),
jni::ToNativeString(env, static_cast<jstring>(countryCode.get())),
currencyCode.get() ? jni::ToNativeString(env, static_cast<jstring>(currencyCode.get())) : ""};
currencyCode.get() ? jni::ToNativeString(env, static_cast<jstring>(currencyCode.get())) : "",
jni::ToNativeString(env, static_cast<jstring>(decimalSeparatorChar.get())),
jni::ToNativeString(env, static_cast<jstring>(groupingSeparatorChar.get()))};
}
} // namespace platform

View file

@ -120,6 +120,8 @@ public class Framework
{
public int totalAscent;
public int totalDescent;
public String totalAscentString;
public String totalDescentString;
public boolean isMetricUnits;
}

View file

@ -338,9 +338,8 @@ final class RoutingBottomMenuController implements View.OnClickListener
mAltitudeChart.setImageBitmap(bm);
UiUtils.show(mAltitudeChart);
final String unit = limits.isMetricUnits ? mAltitudeDifference.getResources().getString(R.string.m) : mAltitudeDifference.getResources().getString(R.string.ft);
mAltitudeDifference.setText(String.format(Locale.getDefault(), "↗ %d %s ↘ %d %s",
limits.totalAscent, unit,
limits.totalDescent, unit));
mAltitudeDifference.setText("" + limits.totalAscentString + " " + unit +
"" + limits.totalDescentString + " " + unit);
UiUtils.show(mAltitudeDifference);
}
}

View file

@ -53,6 +53,7 @@ import java.io.IOException;
import java.io.Serializable;
import java.lang.ref.WeakReference;
import java.text.DateFormat;
import java.text.DecimalFormatSymbols;
import java.text.NumberFormat;
import java.text.SimpleDateFormat;
import java.util.Arrays;
@ -495,6 +496,18 @@ public class Utils
return Locale.getDefault().getLanguage();
}
@NonNull
public static String getDecimalSeparator()
{
return String.valueOf(DecimalFormatSymbols.getInstance().getDecimalSeparator());
}
@NonNull
public static String getGroupingSeparator()
{
return String.valueOf(DecimalFormatSymbols.getInstance().getGroupingSeparator());
}
@Nullable
public static Currency getCurrencyForLocale(@NonNull Locale locale)
{

View file

@ -3,6 +3,7 @@
#include "geometry/mercator.hpp"
#include "geometry/angles.hpp"
#include "platform/locale.hpp"
#include "platform/localization.hpp"
#include "platform/settings.hpp"
#include "platform/measurement_utils.hpp"
@ -31,10 +32,8 @@
*/
- (NSString*) valueAsString {
if (self.value > 9.999)
return [NSString stringWithFormat:@"%.0f", self.value];
else
return [NSString stringWithFormat:@"%.1f", self.value];
auto const outString = measurement_utils::ToStringPrecision(self.value, self.value >= 10.0 ? 0 : 1);
return [NSString stringWithUTF8String:outString.c_str()];
}
- (instancetype)initAsSpeed:(double) mps {

View file

@ -1,5 +1,6 @@
#include "distance.hpp"
#include "platform/locale.hpp"
#include "platform/localization.hpp"
#include "platform/measurement_utils.hpp"
@ -138,9 +139,7 @@ std::string Distance::GetDistanceString() const
if (m_distance < 10.0 && IsHighUnits())
precision = 1;
std::ostringstream os;
os << std::fixed << std::setprecision(precision) << m_distance;
return os.str();
return ToStringPrecision(m_distance, precision);
}
std::string Distance::GetUnitsString() const
@ -194,7 +193,7 @@ std::string Distance::ToString() const
if (!IsValid())
return "";
return GetDistanceString() + " " + GetUnitsString();
return GetDistanceString() + kNarrowNonBreakingSpace + GetUnitsString();
}
std::string DebugPrint(Distance::Units units)

View file

@ -4,13 +4,19 @@
namespace platform
{
std::string const kNonBreakingSpace = "\u00A0";
std::string const kNarrowNonBreakingSpace = "\u202F";
biodranik commented 2023-01-19 20:44:04 +00:00 (Migrated from github.com)
Review
  char m_decimalSeparator;
  char m_groupingSeparator;
```suggestion char m_decimalSeparator; char m_groupingSeparator; ```
Review

Corrected !!

Corrected !!
struct Locale
{
public:
std::string m_language;
std::string m_country;
std::string m_currency;
std::string m_decimalSeparator;
std::string m_groupingSeparator;
};
Locale GetCurrentLocale();
bool GetLocale(std::string localeName, Locale& result);
} // namespace platform
biodranik commented 2023-08-16 20:49:52 +00:00 (Migrated from github.com)
Review

Can a decimal separator also be a non-ASCII symbol? Is there a full list somewhere?

Can a decimal separator also be a non-ASCII symbol? Is there a full list somewhere?
Review

Yes, according to this Wikipedia article, there's also an Arabic decimal separator (U+066B).

Here's a screenshot of the place info panel with locate set to Farsi/Persian, where the decimal and grouping separators are both non-ASCII chars:
Captura desde 2023-08-17 00-08-14

Yes, according to [this](https://en.wikipedia.org/wiki/Decimal_separator) Wikipedia article, there's also an Arabic decimal separator (U+066B). Here's a screenshot of the place info panel with locate set to Farsi/Persian, where the decimal and grouping separators are both non-ASCII chars: ![Captura desde 2023-08-17 00-08-14](https://github.com/organicmaps/organicmaps/assets/25888296/cceb989c-00ed-435b-9c64-a33706e18622)

View file

@ -4,11 +4,28 @@
namespace platform
{
biodranik commented 2023-01-19 20:39:10 +00:00 (Migrated from github.com)
Review

Can an empty grouping separator be perfectly valid in some cases?
If yes, can we support these cases without performance degradation? Maybe by using char(0)?

Can an empty grouping separator be perfectly valid in some cases? If yes, can we support these cases without performance degradation? Maybe by using char(0)?
biodranik commented 2023-01-19 20:45:34 +00:00 (Migrated from github.com)
Review

There is no need to create a temporary string, NSString can be used directly, right?

There is no need to create a temporary string, NSString can be used directly, right?
Review

Yes, empty grouping separators are valid, so I will modify the code to support these situations using a char(0) as grouping separator.

Yes, empty grouping separators are valid, so I will modify the code to support these situations using a `char(0)` as grouping separator.
Review

Modified code to use NSString directly...

Modified code to use NSString directly...
Review

Implemented use of char(0) for empty group separator

Implemented use of `char(0)` for empty group separator
Locale GetCurrentLocale()
Locale NSLocale2Locale(NSLocale *locale)
{
Review

I'm not sure that default value for groupingSeparator is OK. In my country we don't use commas: "10 123 m" or "1234 m".

I think, it's better to leave it empty because if it happens "somehow" that there is no info about groupingSeparator, some people may be confused by the printed values.

I'm not sure that default value for `groupingSeparator` is OK. In my country we don't use commas: "10 123 m" or "1234 m". I think, it's better to leave it empty because if it happens "somehow" that there is no info about `groupingSeparator`, some people may be confused by the printed values.
Review

Yes, maybe an empty grouping separator may be more common across different locales, though this default value will be overwritten by system's configuration in most of the times.

Your comment also confirms that in many languages, the grouping separator is only used in values with more than 4 digits, keeping digits with 4 digits with no grouping separator ("10 123 m" and "1234 m"). Maybe we shall also keep this behavior, which is also suggested as International Standard according to this Wikipedia article about digit grouping. This International Standard also suggests to use always the narrow no-break space char as the grouping separator.

My proposal for the grouping separator would be then: set empty "" (no grouping separator) as default if the system does not provide any & apply grouping separators only on numbers with more that 4 digits.

Yes, maybe an empty grouping separator may be more common across different locales, though this default value will be overwritten by system's configuration in most of the times. Your comment also confirms that in many languages, the grouping separator is only used in values with more than 4 digits, keeping digits with 4 digits with no grouping separator ("10 123 m" and "1234 m"). Maybe we shall also keep this behavior, which is also _suggested_ as International Standard according to [this](https://en.wikipedia.org/wiki/Decimal_separator#Digit_grouping) Wikipedia article about digit grouping. This International Standard also _suggests_ to use always the narrow no-break space char as the grouping separator. My proposal for the grouping separator would be then: set empty "" (no grouping separator) as default if the system does not provide any & apply grouping separators *only* on numbers with more that 4 digits.
biodranik commented 2023-08-17 22:16:47 +00:00 (Migrated from github.com)
Review

Thanks! Looks like a good default is:

  • Use group separator for >4 chars only
  • Use a non-breaking space

P.S. Is non-breaking space used after a number and before the unit name? Should we use it too?

Thanks! Looks like a good default is: - Use group separator for >4 chars only - Use a non-breaking space P.S. Is non-breaking space used after a number and before the unit name? Should we use it too?
Review

@biodranik: I assume that you suggest to use as default the narrow non-breaking space (NNBSP, U+202F), instead of the standard non-breaking space (NBSP, U+00A0).

In this Wikipedia article, it's indeed stated that the NBSP's are also used between the value and the units, so yes, we can also use them there instead of the regular whitespace.

In a first commit I've set the NNBSP as default for iOS and linux and I've modified the UTs accordingly, but after pushing the commit, the iOS tests have failed because the grouping separator is configured to the ',' character. The workaround is to set the default character in linux and the UTs with the ',' character.

I've amended the commit according to these last comments.

@biodranik: I assume that you suggest to use as default the **narrow** non-breaking space (NNBSP, U+202F), instead of the **standard** non-breaking space (NBSP, U+00A0). In [this](https://en.wikipedia.org/wiki/Non-breaking_space#Uses_and_variations) Wikipedia article, it's indeed stated that the NBSP's are also used between the value and the units, so yes, we can also use them there instead of the regular whitespace. In a first commit I've set the NNBSP as default for iOS and linux and I've modified the UTs accordingly, but after pushing the commit, the iOS tests have failed because the grouping separator is configured to the ',' character. The workaround is to set the default character in linux and the UTs with the ',' character. I've amended the commit according to these last comments.
NSLocale * locale = [NSLocale currentLocale];
return {locale.languageCode ? [locale.languageCode UTF8String] : "",
locale.countryCode ? [locale.countryCode UTF8String] : "",
locale.currencyCode ? [locale.currencyCode UTF8String] : ""};
locale.currencyCode ? [locale.currencyCode UTF8String] : "",
locale.decimalSeparator ? [locale.decimalSeparator UTF8String] : ".",
locale.groupingSeparator ? [locale.groupingSeparator UTF8String] : kNarrowNonBreakingSpace};
}
Locale GetCurrentLocale()
{
return NSLocale2Locale([NSLocale currentLocale]);
}
bool GetLocale(std::string localeName, Locale& result)
{
NSLocale *loc = [[NSLocale alloc] initWithLocaleIdentifier: @(localeName.c_str())];
if (!loc)
return false;
result = NSLocale2Locale(loc);
return true;
}
} // namespace platform

View file

@ -1,10 +1,41 @@
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
#include "platform/locale.hpp"
#include <locale>
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
biodranik commented 2023-01-19 20:47:17 +00:00 (Migrated from github.com)
Review

This should be a bit faster:

  return {{}, {}, {}, '.', ','};
This should be a bit faster: ```suggestion return {{}, {}, {}, '.', ','}; ```
biodranik commented 2023-01-19 20:48:27 +00:00 (Migrated from github.com)
Review

Can we properly add the Linux code? https://www.systutorials.com/docs/linux/man/7-locale/

Can we properly add the Linux code? https://www.systutorials.com/docs/linux/man/7-locale/
Review

Replaced "" with {}...

Regarding the Linux code, I don't exactly understand what functionality you want to implement here...

Replaced "" with {}... Regarding the Linux code, I don't exactly understand what functionality you want to implement here...
Review

Same here

Same here
Review

Same as before, I agree to set the empty "" grouping separator as default, though this change affects the unit testing code, so I will have to change back again the unit tests for distances.

Same as before, I agree to set the empty "" grouping separator as default, though this change affects the unit testing code, so I will have to change back again the unit tests for distances.
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
namespace platform
{
Locale StdLocale2Locale(std::locale loc)
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
{
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
return {"",
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
"",
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
std::use_facet<std::moneypunct<char, true>>(loc).curr_symbol(),
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
std::string(1, std::use_facet<std::numpunct<char>>(loc).decimal_point()),
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
std::string(1, std::use_facet<std::numpunct<char>>(loc).thousands_sep())};
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
}
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
Locale GetCurrentLocale()
{
return {"", "", ""};
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
// Environment's default locale.
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
std::locale loc;
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
return StdLocale2Locale(loc);
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
}
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
bool GetLocale(std::string const localeName, Locale& result)
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
{
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
try
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
{
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
std::locale loc(localeName);
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
result = StdLocale2Locale(loc);
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
return true;
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
}
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
catch(...)
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
{
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
return false;
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
}
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
}
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
} // namespace platform

biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!
biodranik commented 2023-09-06 21:44:04 +00:00 (Migrated from github.com)
Review

Why setting a global locale required? Does std::locale(localeName) work directly?

Why setting a global locale required? Does `std::locale(localeName)` work directly?
Review

You're right! There's no need to use std::locale::global(). Will change accordingly...

You're right! There's no need to use `std::locale::global()`. Will change accordingly...
Review

Commit amended!!

Commit amended!!

View file

@ -1,5 +1,5 @@
#include "platform/locale.hpp"
#include "platform/measurement_utils.hpp"
#include "platform/settings.hpp"
#include "geometry/mercator.hpp"
@ -18,19 +18,45 @@
namespace measurement_utils
{
using namespace platform;
using namespace settings;
using namespace std;
using namespace strings;
namespace
{
string ToStringPrecision(double d, int pr)
{
// We assume that the app will be restarted if a user changes device's locale.
static Locale const loc = GetCurrentLocale();
return ToStringPrecisionLocale(loc, d, pr);
}
biodranik commented 2023-01-19 20:55:02 +00:00 (Migrated from github.com)
Review
  // We assume that the app will be restarted if a user changes device's locale.
  static Locale const loc = GetCurrentLocale();
```suggestion // We assume that the app will be restarted if a user changes device's locale. static Locale const loc = GetCurrentLocale(); ```
Review

Modified !!

Modified !!
string ToStringPrecisionLocale(Locale loc, double d, int pr)
{
stringstream ss;
ss << setprecision(pr) << fixed << d;
return ss.str();
string out = ss.str();
// std::locale does not work on Android NDK, so decimal and grouping (thousands) separator
// shall be customized manually here.
if (pr)
{
// Value with decimals. Set locale decimal separator.
if (loc.m_decimalSeparator != ".")
out.replace(out.size() - pr - 1, 1, loc.m_decimalSeparator);
}
else
{
// Value with no decimals. Check if it's equal or bigger than 10000 to
// insert the grouping (thousands) separator characters.
biodranik commented 2023-01-19 20:51:57 +00:00 (Migrated from github.com)
Review
    int const precision = round(highV * 10) / 10 >= 10.0 ? 0 : 1;
```suggestion int const precision = round(highV * 10) / 10 >= 10.0 ? 0 : 1; ```
biodranik commented 2023-01-19 20:59:32 +00:00 (Migrated from github.com)
Review

Can we assume here that the '.' will always be at out.size() - 2 for better performance?
How 3.0 is printed? As 3.0 or as 3?

Is it possible to add unit tests for this and other cases?

Can we assume here that the '.' will always be at `out.size() - 2` for better performance? How 3.0 is printed? As 3.0 or as 3? Is it possible to add unit tests for this and other cases?
biodranik commented 2023-01-19 20:59:47 +00:00 (Migrated from github.com)
Review

We don't use {} for one-liners.

We don't use {} for one-liners.
biodranik commented 2023-01-19 21:02:36 +00:00 (Migrated from github.com)
Review

Mac, Linux, and maybe Windows may properly format the string using the system's locale. Can we at least mention it here? If you see that it should not be corrected on Mac, for example, then an #ifdef may be needed.

Mac, Linux, and maybe Windows may properly format the string using the system's locale. Can we at least mention it here? If you see that it should not be corrected on Mac, for example, then an `#ifdef` may be needed.
biodranik commented 2023-01-19 21:03:09 +00:00 (Migrated from github.com)
Review

We also try to use . at the end of comment sentences. ;-)

We also try to use . at the end of comment sentences. ;-)
biodranik commented 2023-01-19 21:03:58 +00:00 (Migrated from github.com)
Review

ditto, the grouping may be already applied by mac and Linux.

ditto, the grouping may be already applied by mac and Linux.
biodranik commented 2023-01-19 21:20:24 +00:00 (Migrated from github.com)
Review
      if (highV >= 1000.0 && out.size() >= 4)
        out.insert(out.size() - 3, 1, loc.m_grouping_separator);
    }

Is it possible to have two or more grouping separators?

```suggestion if (highV >= 1000.0 && out.size() >= 4) out.insert(out.size() - 3, 1, loc.m_grouping_separator); } ``` Is it possible to have two or more grouping separators?
biodranik commented 2023-01-19 21:26:18 +00:00 (Migrated from github.com)
Review
    // Add units.
    return out + ' ' + high;
```suggestion // Add units. return out + ' ' + high; ```
Review

Modified !!

Modified !!
Review

Yes, instead of std::replace() we could assume that the decimal separator is at out.size() - 2. I will change the code accordingly...
And yes, 3.0 is printed as 3.0

Yes, instead of `std::replace()` we could assume that the decimal separator is at `out.size() - 2`. I will change the code accordingly... And yes, 3.0 is printed as 3.0
Review

And regarding the unit tests, please confirm that the idea is to add to measurement_tests.cpp tests of the FormatDistanceImpl() function to check the functionality of the decimal and grouping separators replacement based on different locales.

And regarding the unit tests, please confirm that the idea is to add to `measurement_tests.cpp` tests of the `FormatDistanceImpl()` function to check the functionality of the decimal and grouping separators replacement based on different locales.
Review

Corrected !!

Corrected !!
Review

I've added #ifdef so the decimal and grouping separators and "manually" replaced only on Android OS.
I've also added a comment explaining this.

I've added `#ifdef` so the decimal and grouping separators and "manually" replaced only on Android OS. I've also added a comment explaining this.
Review

The added #ifdef applies for both decimal and grouping separator...

The added `#ifdef` applies for both decimal and grouping separator...
Review

Yes, I can change the code so we can handle two or mode grouping separators...

Yes, I can change the code so we can handle two or mode grouping separators...
Review

Modified !!

Modified !!
Review

Done!! I've also moved this piece of code to the function ToStringPrecision() so speed strings are also localized...

Done!! I've also moved this piece of code to the function `ToStringPrecision()` so speed strings are also localized...
if (out.size() > 4 && !loc.m_groupingSeparator.empty())
for (int pos = out.size() - 3; pos > 0; pos -= 3)
out.insert(pos, loc.m_groupingSeparator);
}
return out;
}
} // namespace
std::string DebugPrint(Units units)
{

View file

@ -1,6 +1,7 @@
#pragma once
#include "geometry/point2d.hpp"
#include "platform/locale.hpp"
#include <string>
@ -62,4 +63,6 @@ bool OSMDistanceToMeters(std::string const & osmRawValue, double & outMeters);
std::string OSMDistanceToMetersString(std::string const & osmRawValue,
bool supportZeroAndNegativeValues = true,
int digitsAfterComma = 2);
std::string ToStringPrecision(double d, int pr);
std::string ToStringPrecisionLocale(platform::Locale loc, double d, int pr);
} // namespace measurement_utils

View file

@ -6,6 +6,11 @@
namespace platform
{
std::string MakeDistanceStr(std::string const & value, std::string const & unit)
{
return value + kNarrowNonBreakingSpace + unit;
}
struct ScopedSettings
{
/// Saves/restores previous units and sets new units for a scope.
@ -45,7 +50,7 @@ UNIT_TEST(Distance_CreateFormatted)
TEST_EQUAL(d.GetUnits(), Distance::Units::Meters, ());
TEST_ALMOST_EQUAL_ULPS(d.GetDistance(), 100.0, ());
TEST_EQUAL(d.GetDistanceString(), "100", ());
TEST_EQUAL(d.ToString(), "100 m", ());
TEST_EQUAL(d.ToString(), MakeDistanceStr("100", "m"), ());
}
{
ScopedSettings guard(measurement_utils::Units::Imperial);
@ -54,7 +59,7 @@ UNIT_TEST(Distance_CreateFormatted)
TEST_EQUAL(d.GetUnits(), Distance::Units::Feet, ());
TEST_ALMOST_EQUAL_ULPS(d.GetDistance(), 330.0, ());
TEST_EQUAL(d.GetDistanceString(), "330", ());
TEST_EQUAL(d.ToString(), "330 ft", ());
TEST_EQUAL(d.ToString(), MakeDistanceStr("330", "ft"), ());
}
}
@ -67,7 +72,25 @@ UNIT_TEST(Distance_CreateAltitudeFormatted)
TEST_EQUAL(d.GetUnits(), Distance::Units::Meters, ());
TEST_ALMOST_EQUAL_ULPS(d.GetDistance(), 5.0, ());
TEST_EQUAL(d.GetDistanceString(), "5", ());
TEST_EQUAL(d.ToString(), "5 m", ());
TEST_EQUAL(d.ToString(), MakeDistanceStr("5", "m"), ());
}
{
ScopedSettings guard(measurement_utils::Units::Metric);
Distance d = Distance::CreateAltitudeFormatted(8849);
TEST_EQUAL(d.GetUnits(), Distance::Units::Meters, ());
TEST_ALMOST_EQUAL_ULPS(d.GetDistance(), 8849.0, ());
TEST_EQUAL(d.GetDistanceString(), "8849", ());
TEST_EQUAL(d.ToString(), MakeDistanceStr("8849", "m"), ());
}
{
ScopedSettings guard(measurement_utils::Units::Metric);
Distance d = Distance::CreateAltitudeFormatted(12345);
TEST_EQUAL(d.GetUnits(), Distance::Units::Meters, ());
TEST_ALMOST_EQUAL_ULPS(d.GetDistance(), 12345.0, ());
TEST_EQUAL(d.GetDistanceString(), "12,345", ());
TEST_EQUAL(d.ToString(), MakeDistanceStr("12,345", "m"), ());
}
{
ScopedSettings guard(measurement_utils::Units::Imperial);
@ -75,8 +98,8 @@ UNIT_TEST(Distance_CreateAltitudeFormatted)
Distance d = Distance::CreateAltitudeFormatted(10000);
TEST_EQUAL(d.GetUnits(), Distance::Units::Feet, ());
TEST_ALMOST_EQUAL_ULPS(d.GetDistance(), 32808.0, ());
TEST_EQUAL(d.GetDistanceString(), "32808", ());
TEST_EQUAL(d.ToString(), "32808 ft", ());
TEST_EQUAL(d.GetDistanceString(), "32,808", ());
TEST_EQUAL(d.ToString(), MakeDistanceStr("32,808", "ft"), ());
}
}
@ -173,7 +196,7 @@ UNIT_TEST(Distance_ToPlatformUnitsFormatted)
TEST_EQUAL(newDistance.GetUnits(), Distance::Units::Meters, (d.ToString()));
TEST_ALMOST_EQUAL_ULPS(newDistance.GetDistance(), 3.0, (d.ToString()));
TEST_EQUAL(newDistance.GetDistanceString(), "3", (d.ToString()));
TEST_EQUAL(newDistance.ToString(), "3 m", (d.ToString()));
TEST_EQUAL(newDistance.ToString(), MakeDistanceStr("3", "m"), (d.ToString()));
d = Distance{11, Distance::Units::Kilometers};
newDistance = d.ToPlatformUnitsFormatted();
@ -181,7 +204,7 @@ UNIT_TEST(Distance_ToPlatformUnitsFormatted)
TEST_EQUAL(newDistance.GetUnits(), Distance::Units::Kilometers, (d.ToString()));
TEST_ALMOST_EQUAL_ULPS(newDistance.GetDistance(), 11.0, (d.ToString()));
TEST_EQUAL(newDistance.GetDistanceString(), "11", (d.ToString()));
TEST_EQUAL(newDistance.ToString(), "11 km", (d.ToString()));
TEST_EQUAL(newDistance.ToString(), MakeDistanceStr("11", "km"), (d.ToString()));
}
{
@ -193,7 +216,7 @@ UNIT_TEST(Distance_ToPlatformUnitsFormatted)
TEST_EQUAL(newDistance.GetUnits(), Distance::Units::Feet, (d.ToString()));
TEST_ALMOST_EQUAL_ULPS(newDistance.GetDistance(), 11.0, (d.ToString()));
TEST_EQUAL(newDistance.GetDistanceString(), "11", (d.ToString()));
TEST_EQUAL(newDistance.ToString(), "11 ft", (d.ToString()));
TEST_EQUAL(newDistance.ToString(), MakeDistanceStr("11", "ft"), (d.ToString()));
d = Distance{11, Distance::Units::Kilometers};
newDistance = d.ToPlatformUnitsFormatted();
@ -201,7 +224,7 @@ UNIT_TEST(Distance_ToPlatformUnitsFormatted)
TEST_EQUAL(newDistance.GetUnits(), Distance::Units::Miles, (d.ToString()));
TEST_ALMOST_EQUAL_ULPS(newDistance.GetDistance(), 6.8, (d.ToString()));
TEST_EQUAL(newDistance.GetDistanceString(), "6.8", (d.ToString()));
TEST_EQUAL(newDistance.ToString(), "6.8 mi", (d.ToString()));
TEST_EQUAL(newDistance.ToString(), MakeDistanceStr("6.8", "mi"), (d.ToString()));
}
}
@ -238,97 +261,99 @@ UNIT_TEST(Distance_FormattedDistance)
// clang-format off
TestData testData[] = {
// From Meters to Meters
{Distance(0, Units::Meters), 0, Units::Meters, "0", "0 m"},
{Distance(0.3, Units::Meters), 0, Units::Meters, "0", "0 m"},
{Distance(0.9, Units::Meters), 1, Units::Meters, "1", "1 m"},
{Distance(1, Units::Meters), 1, Units::Meters, "1", "1 m"},
{Distance(1.234, Units::Meters), 1, Units::Meters, "1", "1 m"},
{Distance(9.99, Units::Meters), 10, Units::Meters, "10", "10 m"},
{Distance(10.01, Units::Meters), 10, Units::Meters, "10", "10 m"},
{Distance(10.4, Units::Meters), 10, Units::Meters, "10", "10 m"},
{Distance(10.5, Units::Meters), 11, Units::Meters, "11", "11 m"},
{Distance(10.51, Units::Meters), 11, Units::Meters, "11", "11 m"},
{Distance(64.2, Units::Meters), 64, Units::Meters, "64", "64 m"},
{Distance(99, Units::Meters), 99, Units::Meters, "99", "99 m"},
{Distance(100, Units::Meters), 100, Units::Meters, "100", "100 m"},
{Distance(101, Units::Meters), 100, Units::Meters, "100", "100 m"},
{Distance(109, Units::Meters), 110, Units::Meters, "110", "110 m"},
{Distance(991, Units::Meters), 990, Units::Meters, "990", "990 m"},
{Distance(0, Units::Meters), 0, Units::Meters, "0", MakeDistanceStr("0", "m")},
biodranik commented 2023-08-17 22:18:03 +00:00 (Migrated from github.com)
Review

If we change defaults to >4 chars for group separators, these tests will change, right?

If we change defaults to >4 chars for group separators, these tests will change, right?
Review

Yes, the UTs have to be changed.

Yes, the UTs have to be changed.
{Distance(0.3, Units::Meters), 0, Units::Meters, "0", MakeDistanceStr("0", "m")},
{Distance(0.9, Units::Meters), 1, Units::Meters, "1", MakeDistanceStr("1", "m")},
{Distance(1, Units::Meters), 1, Units::Meters, "1", MakeDistanceStr("1", "m")},
{Distance(1.234, Units::Meters), 1, Units::Meters, "1", MakeDistanceStr("1", "m")},
{Distance(9.99, Units::Meters), 10, Units::Meters, "10", MakeDistanceStr("10", "m")},
{Distance(10.01, Units::Meters), 10, Units::Meters, "10", MakeDistanceStr("10", "m")},
{Distance(10.4, Units::Meters), 10, Units::Meters, "10", MakeDistanceStr("10", "m")},
{Distance(10.5, Units::Meters), 11, Units::Meters, "11", MakeDistanceStr("11", "m")},
{Distance(10.51, Units::Meters), 11, Units::Meters, "11", MakeDistanceStr("11", "m")},
{Distance(64.2, Units::Meters), 64, Units::Meters, "64", MakeDistanceStr("64", "m")},
{Distance(99, Units::Meters), 99, Units::Meters, "99", MakeDistanceStr("99", "m")},
{Distance(100, Units::Meters), 100, Units::Meters, "100", MakeDistanceStr("100", "m")},
{Distance(101, Units::Meters), 100, Units::Meters, "100", MakeDistanceStr("100", "m")},
{Distance(109, Units::Meters), 110, Units::Meters, "110", MakeDistanceStr("110", "m")},
{Distance(991, Units::Meters), 990, Units::Meters, "990", MakeDistanceStr("990", "m")},
// From Kilometers to Kilometers
biodranik commented 2023-08-16 20:56:04 +00:00 (Migrated from github.com)
Review
  1. Can tests for a decimal separator also be added?
  2. Is it hard to add tests for altitude formatting?
  3. What if tests are run on a French locale?
1. Can tests for a decimal separator also be added? 2. Is it hard to add tests for altitude formatting? 3. What if tests are run on a French locale?
Review
  1. Can tests for a decimal separator also be added?

Do you mean to test decimal separators on different locales? With the current implementation in ToStringPrecision(), GetCurrentLocale() is only called once at start-up, so we could only make tests with a single locale.

  1. Is it hard to add tests for altitude formatting?

We already have a couple of tests for altitude formatting in UNIT_TEST(Distance_CreateAltitudeFormatted) in distance_tests.cpp. I've included tough a third one for high altitudes in meters.

  1. What if tests are run on a French locale?

Current implementation of GetCurrentLocale() in linux does not read system's locale, and forces decimal and grouping separators to '.' and ',' respectively, so tests run flawlessly even with Spanish or French system locales.

> 1. Can tests for a decimal separator also be added? Do you mean to test decimal separators on different locales? With the current implementation in `ToStringPrecision()`, `GetCurrentLocale()` is only called once at start-up, so we could only make tests with a single locale. > 2. Is it hard to add tests for altitude formatting? We already have a couple of tests for altitude formatting in `UNIT_TEST(Distance_CreateAltitudeFormatted)` in `distance_tests.cpp`. I've included tough a third one for high altitudes in meters. > 3. What if tests are run on a French locale? Current implementation of `GetCurrentLocale()` in linux does not read system's locale, and forces decimal and grouping separators to '.' and ',' respectively, so tests run flawlessly even with Spanish or French system locales.
biodranik commented 2023-08-17 22:18:42 +00:00 (Migrated from github.com)
Review

1/ I meant test cases where a decimal separator is used.

1/ I meant test cases where a decimal separator is used.
{Distance(0, Units::Kilometers), 0, Units::Meters, "0", "0 m"},
{Distance(0.3, Units::Kilometers), 300, Units::Meters, "300", "300 m"},
{Distance(1.234, Units::Kilometers), 1.2, Units::Kilometers, "1.2", "1.2 km"},
{Distance(10, Units::Kilometers), 10, Units::Kilometers, "10", "10 km"},
{Distance(11, Units::Kilometers), 11, Units::Kilometers, "11", "11 km"},
{Distance(54, Units::Kilometers), 54, Units::Kilometers, "54", "54 km"},
{Distance(99.99, Units::Kilometers), 100, Units::Kilometers, "100", "100 km"},
{Distance(100.01, Units::Kilometers), 100, Units::Kilometers, "100", "100 km"},
{Distance(115, Units::Kilometers), 115, Units::Kilometers, "115", "115 km"},
{Distance(999, Units::Kilometers), 999, Units::Kilometers, "999", "999 km"},
{Distance(1000, Units::Kilometers), 1000, Units::Kilometers, "1000", "1000 km"},
{Distance(1049.99, Units::Kilometers), 1050, Units::Kilometers, "1050", "1050 km"},
{Distance(1050, Units::Kilometers), 1050, Units::Kilometers, "1050", "1050 km"},
{Distance(1050.01, Units::Kilometers), 1050, Units::Kilometers, "1050", "1050 km"},
{Distance(1234, Units::Kilometers), 1234, Units::Kilometers, "1234", "1234 km"},
{Distance(0, Units::Kilometers), 0, Units::Meters, "0", MakeDistanceStr("0", "m")},
{Distance(0.3, Units::Kilometers), 300, Units::Meters, "300", MakeDistanceStr("300", "m")},
{Distance(1.234, Units::Kilometers), 1.2, Units::Kilometers, "1.2", MakeDistanceStr("1.2", "km")},
{Distance(10, Units::Kilometers), 10, Units::Kilometers, "10", MakeDistanceStr("10", "km")},
{Distance(11, Units::Kilometers), 11, Units::Kilometers, "11", MakeDistanceStr("11", "km")},
{Distance(54, Units::Kilometers), 54, Units::Kilometers, "54", MakeDistanceStr("54", "km")},
{Distance(99.99, Units::Kilometers), 100, Units::Kilometers, "100", MakeDistanceStr("100", "km")},
{Distance(100.01, Units::Kilometers), 100, Units::Kilometers, "100", MakeDistanceStr("100", "km")},
{Distance(115, Units::Kilometers), 115, Units::Kilometers, "115", MakeDistanceStr("115", "km")},
{Distance(999, Units::Kilometers), 999, Units::Kilometers, "999", MakeDistanceStr("999", "km")},
{Distance(1000, Units::Kilometers), 1000, Units::Kilometers, "1000", MakeDistanceStr("1000", "km")},
{Distance(1049.99, Units::Kilometers), 1050, Units::Kilometers, "1050", MakeDistanceStr("1050", "km")},
{Distance(1050, Units::Kilometers), 1050, Units::Kilometers, "1050", MakeDistanceStr("1050", "km")},
{Distance(1050.01, Units::Kilometers), 1050, Units::Kilometers, "1050", MakeDistanceStr("1050", "km")},
{Distance(1234, Units::Kilometers), 1234, Units::Kilometers, "1234", MakeDistanceStr("1234", "km")},
{Distance(12345, Units::Kilometers), 12345, Units::Kilometers, "12,345", MakeDistanceStr("12,345", "km")},
// From Feet to Feet
{Distance(0, Units::Feet), 0, Units::Feet, "0", "0 ft"},
{Distance(1, Units::Feet), 1, Units::Feet, "1", "1 ft"},
{Distance(9.99, Units::Feet), 10, Units::Feet, "10", "10 ft"},
{Distance(10.01, Units::Feet), 10, Units::Feet, "10", "10 ft"},
{Distance(95, Units::Feet), 95, Units::Feet, "95", "95 ft"},
{Distance(125, Units::Feet), 130, Units::Feet, "130", "130 ft"},
{Distance(991, Units::Feet), 990, Units::Feet, "990", "990 ft"},
{Distance(0, Units::Feet), 0, Units::Feet, "0", MakeDistanceStr("0", "ft")},
{Distance(1, Units::Feet), 1, Units::Feet, "1", MakeDistanceStr("1", "ft")},
{Distance(9.99, Units::Feet), 10, Units::Feet, "10", MakeDistanceStr("10", "ft")},
{Distance(10.01, Units::Feet), 10, Units::Feet, "10", MakeDistanceStr("10", "ft")},
{Distance(95, Units::Feet), 95, Units::Feet, "95", MakeDistanceStr("95", "ft")},
{Distance(125, Units::Feet), 130, Units::Feet, "130", MakeDistanceStr("130", "ft")},
{Distance(991, Units::Feet), 990, Units::Feet, "990", MakeDistanceStr("990", "ft")},
// From Miles to Miles
{Distance(0, Units::Miles), 0, Units::Feet, "0", "0 ft"},
{Distance(0.1, Units::Miles), 530, Units::Feet, "530", "530 ft"},
{Distance(1, Units::Miles), 1.0, Units::Miles, "1.0", "1.0 mi"},
{Distance(1.234, Units::Miles), 1.2, Units::Miles, "1.2", "1.2 mi"},
{Distance(9.99, Units::Miles), 10, Units::Miles, "10", "10 mi"},
{Distance(10.01, Units::Miles), 10, Units::Miles, "10", "10 mi"},
{Distance(11, Units::Miles), 11, Units::Miles, "11", "11 mi"},
{Distance(54, Units::Miles), 54, Units::Miles, "54", "54 mi"},
{Distance(145, Units::Miles), 145, Units::Miles, "145", "145 mi"},
{Distance(999, Units::Miles), 999, Units::Miles, "999", "999 mi"},
{Distance(1149.99, Units::Miles), 1150, Units::Miles, "1150", "1150 mi"},
{Distance(1150, Units::Miles), 1150, Units::Miles, "1150", "1150 mi"},
{Distance(1150.01, Units::Miles), 1150, Units::Miles, "1150", "1150 mi"},
{Distance(0, Units::Miles), 0, Units::Feet, "0", MakeDistanceStr("0", "ft")},
{Distance(0.1, Units::Miles), 530, Units::Feet, "530", MakeDistanceStr("530", "ft")},
{Distance(1, Units::Miles), 1.0, Units::Miles, "1.0", MakeDistanceStr("1.0", "mi")},
{Distance(1.234, Units::Miles), 1.2, Units::Miles, "1.2", MakeDistanceStr("1.2", "mi")},
{Distance(9.99, Units::Miles), 10, Units::Miles, "10", MakeDistanceStr("10", "mi")},
{Distance(10.01, Units::Miles), 10, Units::Miles, "10", MakeDistanceStr("10", "mi")},
{Distance(11, Units::Miles), 11, Units::Miles, "11", MakeDistanceStr("11", "mi")},
{Distance(54, Units::Miles), 54, Units::Miles, "54", MakeDistanceStr("54", "mi")},
{Distance(145, Units::Miles), 145, Units::Miles, "145", MakeDistanceStr("145", "mi")},
{Distance(999, Units::Miles), 999, Units::Miles, "999", MakeDistanceStr("999", "mi")},
{Distance(1149.99, Units::Miles), 1150, Units::Miles, "1150", MakeDistanceStr("1150", "mi")},
{Distance(1150, Units::Miles), 1150, Units::Miles, "1150", MakeDistanceStr("1150", "mi")},
{Distance(1150.01, Units::Miles), 1150, Units::Miles, "1150", MakeDistanceStr("1150", "mi")},
{Distance(12345.0, Units::Miles), 12345, Units::Miles, "12,345", MakeDistanceStr("12,345", "mi")},
// From Meters to Kilometers
{Distance(999, Units::Meters), 1.0, Units::Kilometers, "1.0", "1.0 km"},
{Distance(1000, Units::Meters), 1.0, Units::Kilometers, "1.0", "1.0 km"},
{Distance(1001, Units::Meters), 1.0, Units::Kilometers, "1.0", "1.0 km"},
{Distance(1100, Units::Meters), 1.1, Units::Kilometers, "1.1", "1.1 km"},
{Distance(1140, Units::Meters), 1.1, Units::Kilometers, "1.1", "1.1 km"},
{Distance(1151, Units::Meters), 1.2, Units::Kilometers, "1.2", "1.2 km"},
{Distance(1500, Units::Meters), 1.5, Units::Kilometers, "1.5", "1.5 km"},
{Distance(1549.9, Units::Meters), 1.5, Units::Kilometers, "1.5", "1.5 km"},
{Distance(1550, Units::Meters), 1.6, Units::Kilometers, "1.6", "1.6 km"},
{Distance(1551, Units::Meters), 1.6, Units::Kilometers, "1.6", "1.6 km"},
{Distance(9949, Units::Meters), 9.9, Units::Kilometers, "9.9", "9.9 km"},
{Distance(9992, Units::Meters), 10, Units::Kilometers, "10", "10 km"},
{Distance(10000, Units::Meters), 10, Units::Kilometers, "10", "10 km"},
{Distance(10499.9, Units::Meters), 10, Units::Kilometers, "10", "10 km"},
{Distance(10501, Units::Meters), 11, Units::Kilometers, "11", "11 km"},
{Distance(101'001, Units::Meters), 101, Units::Kilometers, "101", "101 km"},
{Distance(101'999, Units::Meters), 102, Units::Kilometers, "102", "102 km"},
{Distance(287'386, Units::Meters), 287, Units::Kilometers, "287", "287 km"},
{Distance(999, Units::Meters), 1.0, Units::Kilometers, "1.0", MakeDistanceStr("1.0", "km")},
{Distance(1000, Units::Meters), 1.0, Units::Kilometers, "1.0", MakeDistanceStr("1.0", "km")},
{Distance(1001, Units::Meters), 1.0, Units::Kilometers, "1.0", MakeDistanceStr("1.0", "km")},
{Distance(1100, Units::Meters), 1.1, Units::Kilometers, "1.1", MakeDistanceStr("1.1", "km")},
{Distance(1140, Units::Meters), 1.1, Units::Kilometers, "1.1", MakeDistanceStr("1.1", "km")},
{Distance(1151, Units::Meters), 1.2, Units::Kilometers, "1.2", MakeDistanceStr("1.2", "km")},
{Distance(1500, Units::Meters), 1.5, Units::Kilometers, "1.5", MakeDistanceStr("1.5", "km")},
{Distance(1549.9, Units::Meters), 1.5, Units::Kilometers, "1.5", MakeDistanceStr("1.5", "km")},
{Distance(1550, Units::Meters), 1.6, Units::Kilometers, "1.6", MakeDistanceStr("1.6", "km")},
{Distance(1551, Units::Meters), 1.6, Units::Kilometers, "1.6", MakeDistanceStr("1.6", "km")},
{Distance(9949, Units::Meters), 9.9, Units::Kilometers, "9.9", MakeDistanceStr("9.9", "km")},
{Distance(9992, Units::Meters), 10, Units::Kilometers, "10", MakeDistanceStr("10", "km")},
{Distance(10000, Units::Meters), 10, Units::Kilometers, "10", MakeDistanceStr("10", "km")},
{Distance(10499.9, Units::Meters), 10, Units::Kilometers, "10", MakeDistanceStr("10", "km")},
{Distance(10501, Units::Meters), 11, Units::Kilometers, "11", MakeDistanceStr("11", "km")},
{Distance(101'001, Units::Meters), 101, Units::Kilometers, "101", MakeDistanceStr("101", "km")},
{Distance(101'999, Units::Meters), 102, Units::Kilometers, "102", MakeDistanceStr("102", "km")},
{Distance(287'386, Units::Meters), 287, Units::Kilometers, "287", MakeDistanceStr("287", "km")},
// From Feet to Miles
{Distance(999, Units::Feet), 0.2, Units::Miles, "0.2", "0.2 mi"},
{Distance(1000, Units::Feet), 0.2, Units::Miles, "0.2", "0.2 mi"},
{Distance(1150, Units::Feet), 0.2, Units::Miles, "0.2", "0.2 mi"},
{Distance(5280, Units::Feet), 1.0, Units::Miles, "1.0", "1.0 mi"},
{Distance(7920, Units::Feet), 1.5, Units::Miles, "1.5", "1.5 mi"},
{Distance(10560, Units::Feet), 2.0, Units::Miles, "2.0", "2.0 mi"},
{Distance(100'000, Units::Feet), 19, Units::Miles, "19", "19 mi"},
{Distance(285'120, Units::Feet), 54, Units::Miles, "54", "54 mi"},
{Distance(633'547, Units::Feet), 120, Units::Miles, "120", "120 mi"},
{Distance(633'600, Units::Feet), 120, Units::Miles, "120", "120 mi"},
{Distance(633'653, Units::Feet), 120, Units::Miles, "120", "120 mi"},
{Distance(999'999, Units::Feet), 189, Units::Miles, "189", "189 mi"},
{Distance(999, Units::Feet), 0.2, Units::Miles, "0.2", MakeDistanceStr("0.2", "mi")},
{Distance(1000, Units::Feet), 0.2, Units::Miles, "0.2", MakeDistanceStr("0.2", "mi")},
{Distance(1150, Units::Feet), 0.2, Units::Miles, "0.2", MakeDistanceStr("0.2", "mi")},
{Distance(5280, Units::Feet), 1.0, Units::Miles, "1.0", MakeDistanceStr("1.0", "mi")},
{Distance(7920, Units::Feet), 1.5, Units::Miles, "1.5", MakeDistanceStr("1.5", "mi")},
{Distance(10560, Units::Feet), 2.0, Units::Miles, "2.0", MakeDistanceStr("2.0", "mi")},
{Distance(100'000, Units::Feet), 19, Units::Miles, "19", MakeDistanceStr("19", "mi")},
{Distance(285'120, Units::Feet), 54, Units::Miles, "54", MakeDistanceStr("54", "mi")},
{Distance(633'547, Units::Feet), 120, Units::Miles, "120", MakeDistanceStr("120", "mi")},
{Distance(633'600, Units::Feet), 120, Units::Miles, "120", MakeDistanceStr("120", "mi")},
{Distance(633'653, Units::Feet), 120, Units::Miles, "120", MakeDistanceStr("120", "mi")},
{Distance(999'999, Units::Feet), 189, Units::Miles, "189", MakeDistanceStr("189", "mi")},
};
// clang-format on

View file

@ -4,10 +4,22 @@
#include "base/math.hpp"
#include <iostream>
#include <string>
using namespace measurement_utils;
using namespace platform;
std::string AddGroupingSeparators(std::string const & valueString, std::string const & groupingSeparator)
{
std::string out(valueString);
if (out.size() > 4 && !groupingSeparator.empty())
for (int pos = out.size() - 3; pos > 0; pos -= 3)
out.insert(pos, groupingSeparator);
return out;
}
UNIT_TEST(LatLonToDMS_Origin)
{
@ -132,3 +144,42 @@ UNIT_TEST(UnitsConversion)
TEST(base::AlmostEqualAbs(KmphToMps(3.6), 1.0, kEps), ());
TEST(base::AlmostEqualAbs(MpsToKmph(1.0), 3.6, kEps), ());
}
UNIT_TEST(ToStringPrecisionLocale)
{
double d1 = 9.8;
int pr1 = 1;
double d2 = 12345.0;
int pr2 = 0;
std::string d2String("12345");
struct TestData
{
std::string localeName;
std::string d1String;
};
TestData testData[] = {
// Locale name , Decimal
{ "en_US.UTF-8", "9.8"},
{ "es_ES.UTF-8", "9,8"},
{ "fr_FR.UTF-8", "9,8"},
{ "ru_RU.UTF-8", "9,8"}
};
for (TestData const & data : testData)
{
Locale loc;
if (!GetLocale(data.localeName, loc))
{
std::cout << "Locale '" << data.localeName << "' not found!! Skipping test..." << std::endl;
continue;
}
TEST_EQUAL(measurement_utils::ToStringPrecisionLocale(loc, d1, pr1), data.d1String, ());
TEST_EQUAL(measurement_utils::ToStringPrecisionLocale(loc, d2, pr2),
AddGroupingSeparators(d2String, loc.m_groupingSeparator), ());
}
}