blob: 85c5296570b7406f2818f1e3686260fc17d70690 [file] [log] [blame]
// Copyright (c) 2010 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "build/build_config.h"
#if defined(OS_POSIX) && !defined(OS_MACOSX)
#include <cstdlib>
#endif
#include "app/app_paths.h"
#include "app/l10n_util.h"
#include "app/l10n_util_collator.h"
#if !defined(OS_MACOSX)
#include "app/test/data/resource.h"
#endif
#include "base/basictypes.h"
#include "base/environment.h"
#include "base/file_util.h"
#include "base/path_service.h"
#include "base/stl_util-inl.h"
#include "base/string_util.h"
#include "base/utf_string_conversions.h"
#if defined(OS_WIN)
#include "base/win_util.h"
#endif
#include "testing/gtest/include/gtest/gtest.h"
#include "testing/platform_test.h"
#include "unicode/locid.h"
namespace {
class StringWrapper {
public:
explicit StringWrapper(const std::wstring& string) : string_(string) {}
const std::wstring& string() const { return string_; }
private:
std::wstring string_;
DISALLOW_COPY_AND_ASSIGN(StringWrapper);
};
} // namespace
class L10nUtilTest : public PlatformTest {
};
#if defined(OS_WIN)
// TODO(beng): disabled until app strings move to app.
TEST_F(L10nUtilTest, DISABLED_GetString) {
std::wstring s = l10n_util::GetString(IDS_SIMPLE);
EXPECT_EQ(std::wstring(L"Hello World!"), s);
s = l10n_util::GetStringF(IDS_PLACEHOLDERS, L"chrome", L"10");
EXPECT_EQ(std::wstring(L"Hello, chrome. Your number is 10."), s);
s = l10n_util::GetStringF(IDS_PLACEHOLDERS_2, 20);
EXPECT_EQ(std::wstring(L"You owe me $20."), s);
}
#endif // defined(OS_WIN)
TEST_F(L10nUtilTest, TruncateString) {
std::wstring string(L"foooooey bxxxar baz");
// Make sure it doesn't modify the string if length > string length.
EXPECT_EQ(string, l10n_util::TruncateString(string, 100));
// Test no characters.
EXPECT_EQ(L"", l10n_util::TruncateString(string, 0));
// Test 1 character.
EXPECT_EQ(L"\x2026", l10n_util::TruncateString(string, 1));
// Test adds ... at right spot when there is enough room to break at a
// word boundary.
EXPECT_EQ(L"foooooey\x2026", l10n_util::TruncateString(string, 14));
// Test adds ... at right spot when there is not enough space in first word.
EXPECT_EQ(L"f\x2026", l10n_util::TruncateString(string, 2));
// Test adds ... at right spot when there is not enough room to break at a
// word boundary.
EXPECT_EQ(L"foooooey\x2026", l10n_util::TruncateString(string, 11));
// Test completely truncates string if break is on initial whitespace.
EXPECT_EQ(L"\x2026", l10n_util::TruncateString(L" ", 2));
}
void SetICUDefaultLocale(const std::string& locale_string) {
icu::Locale locale(locale_string.c_str());
UErrorCode error_code = U_ZERO_ERROR;
icu::Locale::setDefault(locale, error_code);
EXPECT_TRUE(U_SUCCESS(error_code));
}
#if !defined(OS_MACOSX)
// We are disabling this test on MacOS because GetApplicationLocale() as an
// API isn't something that we'll easily be able to unit test in this manner.
// The meaning of that API, on the Mac, is "the locale used by Cocoa's main
// nib file", which clearly can't be stubbed by a test app that doesn't use
// Cocoa.
TEST_F(L10nUtilTest, GetAppLocale) {
// Use a temporary locale dir so we don't have to actually build the locale
// dlls for this test.
FilePath orig_locale_dir;
PathService::Get(app::DIR_LOCALES, &orig_locale_dir);
FilePath new_locale_dir;
EXPECT_TRUE(file_util::CreateNewTempDirectory(
FILE_PATH_LITERAL("l10n_util_test"),
&new_locale_dir));
PathService::Override(app::DIR_LOCALES, new_locale_dir);
// Make fake locale files.
std::string filenames[] = {
"en-US",
"en-GB",
"fr",
"es-419",
"es",
"zh-TW",
"zh-CN",
"he",
"fil",
"nb",
"am",
};
#if defined(OS_WIN)
static const char kLocaleFileExtension[] = ".dll";
#elif defined(OS_POSIX)
static const char kLocaleFileExtension[] = ".pak";
#endif
for (size_t i = 0; i < arraysize(filenames); ++i) {
FilePath filename = new_locale_dir.AppendASCII(
filenames[i] + kLocaleFileExtension);
file_util::WriteFile(filename, "", 0);
}
// Keep a copy of ICU's default locale before we overwrite it.
icu::Locale locale = icu::Locale::getDefault();
#if defined(OS_POSIX) && !defined(OS_CHROMEOS)
scoped_ptr<base::Environment> env(base::Environment::Create());
// Test the support of LANGUAGE environment variable.
SetICUDefaultLocale("en-US");
env->SetVar("LANGUAGE", "xx:fr_CA");
EXPECT_EQ("fr", l10n_util::GetApplicationLocale(""));
env->SetVar("LANGUAGE", "xx:yy:en_gb.utf-8@quot");
EXPECT_EQ("en-GB", l10n_util::GetApplicationLocale(""));
env->SetVar("LANGUAGE", "xx:zh-hk");
EXPECT_EQ("zh-TW", l10n_util::GetApplicationLocale(""));
// We emulate gettext's behavior here, which ignores LANG/LC_MESSAGES/LC_ALL
// when LANGUAGE is specified. If no language specified in LANGUAGE is valid,
// then just fallback to the default language, which is en-US for us.
SetICUDefaultLocale("fr-FR");
env->SetVar("LANGUAGE", "xx:yy");
EXPECT_EQ("en-US", l10n_util::GetApplicationLocale(""));
env->SetVar("LANGUAGE", "/fr:zh_CN");
EXPECT_EQ("zh-CN", l10n_util::GetApplicationLocale(""));
// Make sure the follow tests won't be affected by LANGUAGE environment
// variable.
env->UnSetVar("LANGUAGE");
#endif // defined(OS_POSIX) && !defined(OS_CHROMEOS)
SetICUDefaultLocale("en-US");
EXPECT_EQ("en-US", l10n_util::GetApplicationLocale(""));
SetICUDefaultLocale("xx");
EXPECT_EQ("en-US", l10n_util::GetApplicationLocale(""));
#if defined(OS_CHROMEOS)
// ChromeOS honors preferred locale first in GetApplicationLocale(),
// defaulting to en-US, while other targets first honor other signals.
SetICUDefaultLocale("en-GB");
EXPECT_EQ("en-US", l10n_util::GetApplicationLocale(""));
SetICUDefaultLocale("en-US");
EXPECT_EQ("en-GB", l10n_util::GetApplicationLocale("en-GB"));
#else // defined(OS_CHROMEOS)
SetICUDefaultLocale("en-GB");
EXPECT_EQ("en-GB", l10n_util::GetApplicationLocale(""));
SetICUDefaultLocale("fr-CA");
EXPECT_EQ("fr", l10n_util::GetApplicationLocale(""));
SetICUDefaultLocale("es-MX");
EXPECT_EQ("es-419", l10n_util::GetApplicationLocale(""));
SetICUDefaultLocale("es-AR");
EXPECT_EQ("es-419", l10n_util::GetApplicationLocale(""));
SetICUDefaultLocale("es-ES");
EXPECT_EQ("es", l10n_util::GetApplicationLocale(""));
SetICUDefaultLocale("es");
EXPECT_EQ("es", l10n_util::GetApplicationLocale(""));
SetICUDefaultLocale("zh-HK");
EXPECT_EQ("zh-TW", l10n_util::GetApplicationLocale(""));
SetICUDefaultLocale("zh-MK");
EXPECT_EQ("zh-TW", l10n_util::GetApplicationLocale(""));
SetICUDefaultLocale("zh-SG");
EXPECT_EQ("zh-CN", l10n_util::GetApplicationLocale(""));
#endif // defined (OS_CHROMEOS)
#if defined(OS_WIN)
// We don't allow user prefs for locale on linux/mac.
SetICUDefaultLocale("en-US");
EXPECT_EQ("fr", l10n_util::GetApplicationLocale("fr"));
EXPECT_EQ("fr", l10n_util::GetApplicationLocale("fr-CA"));
SetICUDefaultLocale("en-US");
// Aliases iw, no, tl to he, nb, fil.
EXPECT_EQ("he", l10n_util::GetApplicationLocale("iw"));
EXPECT_EQ("nb", l10n_util::GetApplicationLocale("no"));
EXPECT_EQ("fil", l10n_util::GetApplicationLocale("tl"));
// es-419 and es-XX (where XX is not Spain) should be
// mapped to es-419 (Latin American Spanish).
EXPECT_EQ("es-419", l10n_util::GetApplicationLocale("es-419"));
EXPECT_EQ("es", l10n_util::GetApplicationLocale("es-ES"));
EXPECT_EQ("es-419", l10n_util::GetApplicationLocale("es-AR"));
SetICUDefaultLocale("es-AR");
EXPECT_EQ("es", l10n_util::GetApplicationLocale("es"));
SetICUDefaultLocale("zh-HK");
EXPECT_EQ("zh-CN", l10n_util::GetApplicationLocale("zh-CN"));
SetICUDefaultLocale("he");
EXPECT_EQ("en-US", l10n_util::GetApplicationLocale("en"));
// Amharic should be blocked unless OS is Vista or newer.
if (win_util::GetWinVersion() < win_util::WINVERSION_VISTA) {
SetICUDefaultLocale("am");
EXPECT_EQ("en-US", l10n_util::GetApplicationLocale(""));
SetICUDefaultLocale("en-GB");
EXPECT_EQ("en-GB", l10n_util::GetApplicationLocale("am"));
} else {
SetICUDefaultLocale("am");
EXPECT_EQ("am", l10n_util::GetApplicationLocale(""));
SetICUDefaultLocale("en-GB");
EXPECT_EQ("am", l10n_util::GetApplicationLocale("am"));
}
#endif // defined(OS_WIN)
// Clean up.
PathService::Override(app::DIR_LOCALES, orig_locale_dir);
file_util::Delete(new_locale_dir, true);
UErrorCode error_code = U_ZERO_ERROR;
icu::Locale::setDefault(locale, error_code);
}
#endif // !defined(OS_MACOSX)
TEST_F(L10nUtilTest, SortStringsUsingFunction) {
std::vector<StringWrapper*> strings;
strings.push_back(new StringWrapper(L"C"));
strings.push_back(new StringWrapper(L"d"));
strings.push_back(new StringWrapper(L"b"));
strings.push_back(new StringWrapper(L"a"));
l10n_util::SortStringsUsingMethod(L"en-US", &strings, &StringWrapper::string);
ASSERT_TRUE(L"a" == strings[0]->string());
ASSERT_TRUE(L"b" == strings[1]->string());
ASSERT_TRUE(L"C" == strings[2]->string());
ASSERT_TRUE(L"d" == strings[3]->string());
STLDeleteElements(&strings);
}
// Test upper and lower case string conversion.
TEST_F(L10nUtilTest, UpperLower) {
string16 mixed(ASCIIToUTF16("Text with UPPer & lowER casE."));
const string16 expected_lower(ASCIIToUTF16("text with upper & lower case."));
const string16 expected_upper(ASCIIToUTF16("TEXT WITH UPPER & LOWER CASE."));
string16 result = l10n_util::ToLower(mixed);
EXPECT_EQ(result, expected_lower);
result = l10n_util::ToUpper(mixed);
EXPECT_EQ(result, expected_upper);
}
TEST_F(L10nUtilTest, LocaleDisplayName) {
// TODO(jungshik): Make this test more extensive.
// Test zh-CN and zh-TW are treated as zh-Hans and zh-Hant.
string16 result = l10n_util::GetDisplayNameForLocale("zh-CN", "en", false);
EXPECT_EQ(result, ASCIIToUTF16("Chinese (Simplified Han)"));
result = l10n_util::GetDisplayNameForLocale("zh-TW", "en", false);
EXPECT_EQ(result, ASCIIToUTF16("Chinese (Traditional Han)"));
result = l10n_util::GetDisplayNameForLocale("pt-BR", "en", false);
EXPECT_EQ(result, ASCIIToUTF16("Portuguese (Brazil)"));
result = l10n_util::GetDisplayNameForLocale("es-419", "en", false);
EXPECT_EQ(result, ASCIIToUTF16("Spanish (Latin America and the Caribbean)"));
}