// Copyright (c) 2006-2008 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 "base/i18n/rtl.h"
#include <algorithm>
#include "base/file_path.h"
#include "base/string_util.h"
#include "base/utf_string_conversions.h"
#include "base/sys_string_conversions.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "testing/platform_test.h"
namespace {
base::i18n::TextDirection GetTextDirection(const char* locale_name) {
return base::i18n::GetTextDirectionForLocale(locale_name);
}
}
class RTLTest : public PlatformTest {
};
TEST_F(RTLTest, GetFirstStrongCharacterDirection) {
// Test pure LTR string.
std::wstring string(L"foo bar");
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT,
base::i18n::GetFirstStrongCharacterDirection(string));
// Test bidi string in which the first character with strong directionality
// is a character with type L.
string.assign(L"foo \x05d0 bar");
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT,
base::i18n::GetFirstStrongCharacterDirection(string));
// Test bidi string in which the first character with strong directionality
// is a character with type R.
string.assign(L"\x05d0 foo bar");
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT,
base::i18n::GetFirstStrongCharacterDirection(string));
// Test bidi string which starts with a character with weak directionality
// and in which the first character with strong directionality is a character
// with type L.
string.assign(L"!foo \x05d0 bar");
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT,
base::i18n::GetFirstStrongCharacterDirection(string));
// Test bidi string which starts with a character with weak directionality
// and in which the first character with strong directionality is a character
// with type R.
string.assign(L",\x05d0 foo bar");
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT,
base::i18n::GetFirstStrongCharacterDirection(string));
// Test bidi string in which the first character with strong directionality
// is a character with type LRE.
string.assign(L"\x202a \x05d0 foo bar");
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT,
base::i18n::GetFirstStrongCharacterDirection(string));
// Test bidi string in which the first character with strong directionality
// is a character with type LRO.
string.assign(L"\x202d \x05d0 foo bar");
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT,
base::i18n::GetFirstStrongCharacterDirection(string));
// Test bidi string in which the first character with strong directionality
// is a character with type RLE.
string.assign(L"\x202b foo \x05d0 bar");
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT,
base::i18n::GetFirstStrongCharacterDirection(string));
// Test bidi string in which the first character with strong directionality
// is a character with type RLO.
string.assign(L"\x202e foo \x05d0 bar");
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT,
base::i18n::GetFirstStrongCharacterDirection(string));
// Test bidi string in which the first character with strong directionality
// is a character with type AL.
string.assign(L"\x0622 foo \x05d0 bar");
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT,
base::i18n::GetFirstStrongCharacterDirection(string));
// Test a string without strong directionality characters.
string.assign(L",!.{}");
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT,
base::i18n::GetFirstStrongCharacterDirection(string));
// Test empty string.
string.assign(L"");
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT,
base::i18n::GetFirstStrongCharacterDirection(string));
// Test characters in non-BMP (e.g. Phoenician letters. Please refer to
// http://demo.icu-project.org/icu-bin/ubrowse?scr=151&b=10910 for more
// information).
#if defined(WCHAR_T_IS_UTF32)
string.assign(L" ! \x10910" L"abc 123");
#elif defined(WCHAR_T_IS_UTF16)
string.assign(L" ! \xd802\xdd10" L"abc 123");
#else
#error wchar_t should be either UTF-16 or UTF-32
#endif
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT,
base::i18n::GetFirstStrongCharacterDirection(string));
#if defined(WCHAR_T_IS_UTF32)
string.assign(L" ! \x10401" L"abc 123");
#elif defined(WCHAR_T_IS_UTF16)
string.assign(L" ! \xd801\xdc01" L"abc 123");
#else
#error wchar_t should be either UTF-16 or UTF-32
#endif
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT,
base::i18n::GetFirstStrongCharacterDirection(string));
}
TEST_F(RTLTest, WrapPathWithLTRFormatting) {
const wchar_t* kTestData[] = {
// Test common path, such as "c:\foo\bar".
L"c:/foo/bar",
// Test path with file name, such as "c:\foo\bar\test.jpg".
L"c:/foo/bar/test.jpg",
// Test path ending with punctuation, such as "c:\(foo)\bar.".
L"c:/(foo)/bar.",
// Test path ending with separator, such as "c:\foo\bar\".
L"c:/foo/bar/",
// Test path with RTL character.
L"c:/\x05d0",
// Test path with 2 level RTL directory names.
L"c:/\x05d0/\x0622",
// Test path with mixed RTL/LTR directory names and ending with punctuation.
L"c:/\x05d0/\x0622/(foo)/b.a.r.",
// Test path without driver name, such as "/foo/bar/test/jpg".
L"/foo/bar/test.jpg",
// Test path start with current directory, such as "./foo".
L"./foo",
// Test path start with parent directory, such as "../foo/bar.jpg".
L"../foo/bar.jpg",
// Test absolute path, such as "//foo/bar.jpg".
L"//foo/bar.jpg",
// Test path with mixed RTL/LTR directory names.
L"c:/foo/\x05d0/\x0622/\x05d1.jpg",
// Test empty path.
L""
};
for (unsigned int i = 0; i < arraysize(kTestData); ++i) {
FilePath path;
#if defined(OS_WIN)
std::wstring win_path(kTestData[i]);
std::replace(win_path.begin(), win_path.end(), '/', '\\');
path = FilePath(win_path);
std::wstring wrapped_expected =
std::wstring(L"\x202a") + win_path + L"\x202c";
#else
path = FilePath(base::SysWideToNativeMB(kTestData[i]));
std::wstring wrapped_expected =
std::wstring(L"\x202a") + kTestData[i] + L"\x202c";
#endif
string16 localized_file_path_string;
base::i18n::WrapPathWithLTRFormatting(path, &localized_file_path_string);
std::wstring wrapped_actual = UTF16ToWide(localized_file_path_string);
EXPECT_EQ(wrapped_expected, wrapped_actual);
}
}
typedef struct {
std::wstring raw_filename;
std::wstring display_string;
} StringAndLTRString;
TEST_F(RTLTest, GetDisplayStringInLTRDirectionality) {
const StringAndLTRString test_data[] = {
{ L"test", L"\x202atest\x202c" },
{ L"test.html", L"\x202atest.html\x202c" },
{ L"\x05d0\x05d1\x05d2", L"\x202a\x05d0\x05d1\x05d2\x202c" },
{ L"\x05d0\x05d1\x05d2.txt", L"\x202a\x05d0\x05d1\x05d2.txt\x202c" },
{ L"\x05d0"L"abc", L"\x202a\x05d0"L"abc\x202c" },
{ L"\x05d0"L"abc.txt", L"\x202a\x05d0"L"abc.txt\x202c" },
{ L"abc\x05d0\x05d1", L"\x202a"L"abc\x05d0\x05d1\x202c" },
{ L"abc\x05d0\x05d1.jpg", L"\x202a"L"abc\x05d0\x05d1.jpg\x202c" },
};
for (unsigned int i = 0; i < arraysize(test_data); ++i) {
string16 input = WideToUTF16(test_data[i].raw_filename);
string16 expected = base::i18n::GetDisplayStringInLTRDirectionality(input);
if (base::i18n::IsRTL())
EXPECT_EQ(expected, WideToUTF16(test_data[i].display_string));
else
EXPECT_EQ(expected, input);
}
}
TEST_F(RTLTest, GetTextDirection) {
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT, GetTextDirection("ar"));
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT, GetTextDirection("ar_EG"));
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT, GetTextDirection("he"));
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT, GetTextDirection("he_IL"));
// iw is an obsolete code for Hebrew.
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT, GetTextDirection("iw"));
// Although we're not yet localized to Farsi and Urdu, we
// do have the text layout direction information for them.
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT, GetTextDirection("fa"));
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT, GetTextDirection("ur"));
#if 0
// Enable these when we include the minimal locale data for Azerbaijani
// written in Arabic and Dhivehi. At the moment, our copy of
// ICU data does not have entries for them.
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT, GetTextDirection("az_Arab"));
// Dhivehi that uses Thaana script.
EXPECT_EQ(base::i18n::RIGHT_TO_LEFT, GetTextDirection("dv"));
#endif
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT, GetTextDirection("en"));
// Chinese in China with '-'.
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT, GetTextDirection("zh-CN"));
// Filipino : 3-letter code
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT, GetTextDirection("fil"));
// Russian
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT, GetTextDirection("ru"));
// Japanese that uses multiple scripts
EXPECT_EQ(base::i18n::LEFT_TO_RIGHT, GetTextDirection("ja"));
}