blob: 874a2948fbfca60ee06925c7e749b8d1be7a9dde [file] [log] [blame]
// Copyright (c) 2012 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.
#import <Foundation/Foundation.h>
#include "base/lazy_instance.h"
#include "base/logging.h"
#include "base/mac/bundle_locations.h"
#import "base/mac/scoped_nsobject.h"
#include "base/strings/sys_string_conversions.h"
#include "ui/base/l10n/l10n_util.h"
#include "ui/base/l10n/l10n_util_mac.h"
namespace {
base::LazyInstance<std::string>::DestructorAtExit g_overridden_locale =
LAZY_INSTANCE_INITIALIZER;
base::LazyInstance<base::scoped_nsobject<NSLocale>>::DestructorAtExit
mac_locale = LAZY_INSTANCE_INITIALIZER;
} // namespace
namespace l10n_util {
base::string16 GetDisplayNameForLocale(const std::string& locale,
const std::string& display_locale) {
NSString* display = base::SysUTF8ToNSString(display_locale);
if (mac_locale.Get() == nil ||
![[mac_locale.Get() localeIdentifier] isEqualToString:display]) {
mac_locale.Get().reset([[NSLocale alloc] initWithLocaleIdentifier:display]);
}
NSString* language = base::SysUTF8ToNSString(locale);
NSString* localized_language_name =
[mac_locale.Get() displayNameForKey:NSLocaleIdentifier value:language];
// Return localized language if system API provided it. Do not attempt to
// manually parse into error format if no |locale| was provided.
if (localized_language_name.length || !language.length) {
return base::SysNSStringToUTF16(localized_language_name);
}
NSRange script_seperator =
[language rangeOfString:@"-" options:NSBackwardsSearch];
if (!script_seperator.length) {
return base::SysNSStringToUTF16([language lowercaseString]);
}
NSString* language_component =
[language substringToIndex:script_seperator.location];
NSString* script_component = [language
substringFromIndex:script_seperator.location + script_seperator.length];
if (!script_component.length) {
return base::SysNSStringToUTF16([language_component lowercaseString]);
}
return base::SysNSStringToUTF16([NSString
stringWithFormat:@"%@ (%@)", [language_component lowercaseString],
[script_component uppercaseString]]);
}
const std::string& GetLocaleOverride() {
return g_overridden_locale.Get();
}
void OverrideLocaleWithCocoaLocale() {
// NSBundle really should only be called on the main thread.
DCHECK([NSThread isMainThread]);
// Chrome really only has one concept of locale, but Mac OS X has locale and
// language that can be set independently. After talking with Chrome UX folks
// (Cole), the best path from an experience point of view is to map the Mac OS
// X language into the Chrome locale. This way strings like "Yesterday" and
// "Today" are in the same language as raw dates like "March 20, 1999" (Chrome
// strings resources vs ICU generated strings). This also makes the Mac acts
// like other Chrome platforms.
NSArray* languageList = [base::mac::OuterBundle() preferredLocalizations];
NSString* firstLocale = languageList[0];
// Mac OS X uses "_" instead of "-", so swap to get a real locale value.
std::string locale_value =
[[firstLocale stringByReplacingOccurrencesOfString:@"_"
withString:@"-"] UTF8String];
// On disk the "en-US" resources are just "en" (http://crbug.com/25578), so
// the reverse mapping is done here to continue to feed Chrome the same values
// in all cases on all platforms. (l10n_util maps en to en-US if it gets
// passed this on the command line)
if (locale_value == "en")
locale_value = "en-US";
g_overridden_locale.Get() = locale_value;
}
// Remove the Windows-style accelerator marker and change "..." into an
// ellipsis. Returns the result in an autoreleased NSString.
NSString* FixUpWindowsStyleLabel(const base::string16& label) {
const base::char16 kEllipsisUTF16 = 0x2026;
base::string16 ret;
size_t label_len = label.length();
ret.reserve(label_len);
for (size_t i = 0; i < label_len; ++i) {
base::char16 c = label[i];
if (c == '(' && i + 3 < label_len && label[i + 1] == '&'
&& label[i + 3] == ')') {
// Strip '(&?)' patterns which means Windows-style accelerator in some
// non-English locales such as Japanese.
i += 3;
} else if (c == '&') {
if (i + 1 < label_len && label[i + 1] == '&') {
ret.push_back(c);
++i;
}
} else if (c == '.' && i + 2 < label_len && label[i + 1] == '.'
&& label[i + 2] == '.') {
ret.push_back(kEllipsisUTF16);
i += 2;
} else {
ret.push_back(c);
}
}
return base::SysUTF16ToNSString(ret);
}
NSString* GetNSString(int message_id) {
return base::SysUTF16ToNSString(l10n_util::GetStringUTF16(message_id));
}
NSString* GetNSStringF(int message_id,
const base::string16& a) {
return base::SysUTF16ToNSString(l10n_util::GetStringFUTF16(message_id,
a));
}
NSString* GetNSStringF(int message_id,
const base::string16& a,
const base::string16& b) {
return base::SysUTF16ToNSString(l10n_util::GetStringFUTF16(message_id,
a, b));
}
NSString* GetNSStringF(int message_id,
const base::string16& a,
const base::string16& b,
const base::string16& c) {
return base::SysUTF16ToNSString(l10n_util::GetStringFUTF16(message_id,
a, b, c));
}
NSString* GetNSStringF(int message_id,
const base::string16& a,
const base::string16& b,
const base::string16& c,
const base::string16& d) {
return base::SysUTF16ToNSString(l10n_util::GetStringFUTF16(message_id,
a, b, c, d));
}
NSString* GetNSStringF(int message_id,
const base::string16& a,
const base::string16& b,
const base::string16& c,
const base::string16& d,
const base::string16& e) {
return base::SysUTF16ToNSString(
l10n_util::GetStringFUTF16(message_id, a, b, c, d, e));
}
NSString* GetNSStringF(int message_id,
const base::string16& a,
const base::string16& b,
std::vector<size_t>* offsets) {
return base::SysUTF16ToNSString(l10n_util::GetStringFUTF16(message_id,
a, b, offsets));
}
NSString* GetNSStringWithFixup(int message_id) {
return FixUpWindowsStyleLabel(l10n_util::GetStringUTF16(message_id));
}
NSString* GetNSStringFWithFixup(int message_id,
const base::string16& a) {
return FixUpWindowsStyleLabel(l10n_util::GetStringFUTF16(message_id,
a));
}
NSString* GetNSStringFWithFixup(int message_id,
const base::string16& a,
const base::string16& b) {
return FixUpWindowsStyleLabel(l10n_util::GetStringFUTF16(message_id,
a, b));
}
NSString* GetNSStringFWithFixup(int message_id,
const base::string16& a,
const base::string16& b,
const base::string16& c) {
return FixUpWindowsStyleLabel(l10n_util::GetStringFUTF16(message_id,
a, b, c));
}
NSString* GetNSStringFWithFixup(int message_id,
const base::string16& a,
const base::string16& b,
const base::string16& c,
const base::string16& d) {
return FixUpWindowsStyleLabel(l10n_util::GetStringFUTF16(message_id,
a, b, c, d));
}
NSString* GetPluralNSStringF(int message_id, int number) {
return base::SysUTF16ToNSString(l10n_util::GetPluralStringFUTF16(message_id,
number));
}
} // namespace l10n_util