Гравюра М. Эшера «Вавилонская башня», 1928
Введение
Возможно, вы готовы к тому, что ваше приложение будет многоязычным уже на старте проекта. Но скорее всего, новость о необходимости интернационализации, как это однажды уже случилось с человечеством, застанет вас в разгаре строительства Вавилонской башни. В любом случае полезно иметь при себе джентльменский набор средств, дающих шанс завершить стройку века успешно.
Спустя четыре тысячи лет после Вавилонского столпотворения технологии предлагают нам несколько замечательных инструментов. Что же у нас есть?
Во-первых, сборная солянка — абстракция локали (locale). Локаль включает не только язык, но еще и письменность, календарь, правила форматирования чисел, денежных единиц, дат и пр.
Во-вторых, Юникод. Юникод — это не просто таблица кодирования символов. Это еще и различные формы одних и тех же букв, диакритические знаки, порядок сортировки символов, правила изменения регистра, алгоритмы нормализации строк, семейство кодировок UTF и многое другое.
Все это большое подспорье. Такие возможности, как правило, уже встроены в операционные системы и доступны в стандартных библиотеках. Программисты и пользователи во всех уголках планеты благополучно применяют одни и те же операционные системы, средства разработки, базы данных. Но, увы, нет в мире совершенства… Если ваше приложение должно одновременно обслуживать пользователей на многих языках, у вас, кем бы вы ни были (аналитиком, архитектором или программистом), возникают новые потребности.
Далее мы расскажем вам о некоторых таких часто возникающих в корпоративных приложениях потребностях, основываясь на опыте нашей компании. Примеры кода в статье будут на C#. На GitHub выложен исходный код библиотеки, включающей рассматриваемые типы данных, их работоспособные реализации и не только. Несмотря на то что материал содержит некоторую специфику .NET, изложенные концепции работы с многоязычными данными будут полезны специалистам и на других платформах.
А для начала мы рекомендуем ознакомиться с предыдущим материалом, посвященным интернационализации приложений.
Условия задачи
Представим, что наше приложение должно работать сразу с несколькими языками. На любом из них в зависимости от окружения пользователя будут не только отображаться пользовательские интерфейсы, но и вводиться оперативные и справочные данные. При этом в одной сессии из нескольких вариантов локализации данных может использоваться и вариант только на одном определенном языке, и локализации для всех языков сразу.
Для примера рассмотрим доменную сущность товара, имеющую среди прочих атрибутов артикул и наименование на различных языках. Нам требуется уметь описать доменную сущность, отображать и вводить через пользовательский интерфейс записи о товарах, а также печатать ценники.
Многоязычные строки
Первая возникающая мысль о наименовании товара в доменной сущности — словарь с кодом локали в качестве ключа.
public class Product
{
public string Code { get; set; }
public IDictionary<string, string> Name { get; set; }
}
Вариант подкупает своей простотой, но сразу нарушает принципы проектирования публичных контрактов, ведь словарь IDictionary<string, string>
не имеет ясной семантики. Немного спасти ситуацию может переименование атрибута сущности Name
в MultilingualName
и использование подобного соглашения везде, где потребуется семантика многоязычного атрибута.
Если задуматься, то мы с вами наверняка обнаружим случаи, когда со строками сразу на нескольких (или всех одновременно) языках необходимо проделать одну и ту же операцию (например, привести все буквы наименования к заглавным). Казалось бы, что может быть проще?
static IDictionary<string, string> ToUpper(IDictionary<string, string> source)
{
IDictionary<string, string> destination = new Dictionary<string, string>();
foreach (var pair in source)
{
destination[pair.Key] = pair.Value.ToUpper();
}
return destination;
}
Или совсем коротко:
static IDictionary<string, string> ToUpper(IDictionary<string, string> source)
{
return source.ToDictionary(p => p.Key, p => p.Value.ToUpper());
}
Однако в код уже закралась ошибка, правда, она всемирно известная: мы не прошли The Turkey Test.
Дело в том, что для изменения регистра символов необходимо применять правила конкретного языка. И если мы его не указываем, то используется текущая локаль (локаль региональных настроек).
Здесь оговоримся, что локаль в .NET называется культурой. Для каждого потока доступны две культуры: CurrentCulture
и CurrentUICulture
. Первая применяется для форматирования чисел, дат и других региональных настроек, а вторая используется в алгоритме поиска подходящих локализованных ресурсов, таких как строки, изображения, layout пользовательских интерфейсов и пр.
Поскольку мы заведомо меняем строки для различных локалей, верный код может выглядеть так:
static IDictionary<string, string> ToUpper(IDictionary<string, string> source)
{
IDictionary<string, string> destination = new Dictionary<string, string>();
foreach (var pair in source)
{
var culture = CultureInfo.GetCultureInfo(pair.Key);
destination[pair.Key] = pair.Value.ToUpper(culture);
}
return destination;
}
Черновик нового типа данных
Эти два факта: желание следовать хорошему стилю проектирования и высокая вероятность возникновения ошибок при регулярной работе с многоязычными данными — вполне могут и должны побудить нас ввести новый тип данных — многоязычную строку.
Что же должна уметь многоязычная строка? Необходимо как минимум:
- Иметь возможность содержать значения для любых доступных локалей.
- Предоставлять список содержащихся локалей.
- Предоставлять обыкновенную строку по заданной локали.
- Возвращать строку в текущей локали при вызове
ToString()
.
Вместе с тем интуитивно кажется, что многоязычная строка своим поведением и свойствами должна быть весьма похожа на обычную строку:
- Быть иммутабельной (неизменяемой).
- Быть сериализуемой.
- Реализовывать следующие методы:
- Изменение регистра:
ToLower()
,ToUpper()
. - Проверка на пустоту и непечатные символы:
IsNullOrEmpty()
,IsNullOrWhiteSpace()
. - Объединение нескольких строк через разделитель(-и):
Join()
. - Набивка пробелами в начале и конце строки:
PadLeft()
,PadRight()
.
- Изменение регистра:
Однако многоязычная строка явно не должна поддерживать конкатенацию строк. Конкатенация в локализованных приложениях практически под запретом (по крайней мере, в рамках одного предложения), ведь порядок слов в разных языках может отличаться.
/// <summary> Многоязычная строка. </summary>
/// <remarks> Этот класс предназначен для хранения различных вариантов строки для разных культур.
/// Строки с <see langword="null"/>-значениями не сохраняются.
/// </remarks>
[Serializable]
public sealed class MultiCulturalString
{
#region Конструкторы
/// <summary> Многоязычная строка. Ctor. </summary>
private MultiCulturalString() {...}
/// <summary> Многоязычная строка. Ctor. </summary>
public MultiCulturalString(IEnumerable<KeyValuePair<CultureInfo, string>> localizedStrings)
{...}
/// <summary> Многоязычная строка. Ctor. Создает многоязычную строку со значением
/// для единственной культуры. </summary>
public MultiCulturalString(CultureInfo culture, string value)
{...}
#endregion
#region Строковые методы
/// <summary> Имеет ли <paramref name="value"/> значение <c>null</c>
/// или <see cref="MultiCulturalString"/> только лишь с пустыми значениями? </summary>
public static bool IsNullOrEmpty(MultiCulturalString value) {...}
/// <summary> Имеет ли <paramref name="value"/> значение <see langword="null"/>
/// или <see cref="MultiCulturalString"/> только лишь с пустыми
/// или непечатаемыми значениями? </summary>
public static bool IsNullOrWhiteSpace(MultiCulturalString value) {...}
/// <summary> Объединение нескольких элементов в мультикультурную строку. </summary>
public static MultiCulturalString Join(MultiCulturalString separator, params object[] args)
{...}
/// <summary> Возвращает новый экземпляр многоязычной строки со строкой
/// <paramref name="localizedString"/> для культуры <paramref name="culture"/></summary>
public MultiCulturalString SetLocalizedString(CultureInfo culture, string localizedString)
{...}
/// <summary> Слияние двух многоязычных строк с приоритетом данной </summary>
public MultiCulturalString MergeWith(MultiCulturalString other) {...}
/// <summary> Присутствует ли в данном экземпляре строка с заданной культурой. </summary>
public bool ContainsCulture(CultureInfo culture) {...}
/// <summary> Возвращает копию данной строки, приведенную к нижнему регистру.
/// Для каждой культуры преобразование производится по правилам этой культуры. </summary>
public MultiCulturalString ToLower() {...}
/// <summary> Возвращает копию данной строки, приведенную к верхнему регистру.
/// Для каждой культуры преобразование производится по правилам этой культуры. </summary>
public MultiCulturalString ToUpper() {...}
/// <summary> Возвращает новую строку, в которой знаки данного экземпляра выровнены
/// по правому краю путем добавления слева символов-заполнителей до указанной общей длины.
/// </summary>
public MultiCulturalString PadLeft(int totalWidth, char paddingChar = ' ') {...}
/// <summary> Возвращает новую строку, в которой знаки данного экземпляра выровнены
/// по левому краю путем добавления справа символов-заполнителей до указанной общей длины.
/// </summary>
public MultiCulturalString PadRight(int totalWidth, char paddingChar = ' ') {...}
#endregion
#region Перегрузки ToString()
/// <summary> Возвращает строку в UI-культуре потока </summary>
public override string ToString() {...}
/// <summary> Возвращает строку в указанной культуре </summary>
public string ToString(CultureInfo culture) {...}
#endregion
#region Свойства
/// <summary> Возвращает многоязычную строку, которая не содержит значения
/// ни для какой культуры.</summary>
public static MultiCulturalString Empty {...}
/// <summary> Получает список культур, на которые локализована данная строка. </summary>
public IEnumerable<CultureInfo> Cultures {...}
/// <summary> Является ли мультикультурная строка пустой? </summary>
public bool IsEmpty {...}
/// <summary> Содержит ли строка только пустые или непечатаемые значения? </summary>
public bool IsWhiteSpace {...}
#endregion
}
Ну а внутри класса скрываются все тот же словарь и незатейливые манипуляции с ним.
И описание товара выглядит вполне благопристойно:
public class Product
{
public string Code { get; set; }
public MultiCulturalString Name { get; set; }
}
Усовершенствования
Как только мы начнем реализовывать или использовать приведенные методы, мы столкнемся с несколькими не совсем очевидными ранее проблемами.
ToString() недостаточно
Представим, что при заведении товара мы заполнили наименование только для некоторых из требуемых языков:
var ru = CultureInfo.GetCultureInfo("ru");
var en = CultureInfo.GetCultureInfo("en");
var product = new Product
{
Code = "V0016887",
Name = new MultiCulturalString(ru, "Шоколад Алина")
.SetLocalizedString(en, "Chocolate Alina")
};
А затем запросили наименование для отсутствующего языка:
var zhHans = CultureInfo.GetCultureInfo("zh-Hans");
Console.WriteLine(product.Name.ToString(zhHans));
// ?
Какой результат вы бы ожидали получить?
Ну никак не исключение! Может быть, null
? Вероятно! Но документация по Object.ToString()
не рекомендует возвращать ни null
, ни пустую строку. А Code Contracts прямо запрещают возвращать null
.
Тем не менее нам необходимо уметь отличать ситуацию наличия для заданной локали пустой строки от случая ее отсутствия. Поэтому наш класс многоязычной строки прирастет методами GetString(...)
, которые будут уметь возвращать null
и имеют те же сигнатуры, что и методы ToString(...)
.
Форматирование
Как мы уже говорили, конкатенацию строк мы использовать не можем, поэтому строки с подстановками — наше все. Локализованные строки в подавляющем большинстве случаев содержат одни и те же подстановки для всех локалей.
Следовательно, хорошо бы уметь многоязычную строку форматировать. Что бы это значило? Ведь мы сразу догадались поддержать перегрузки GetString(CultureInfo) / ToString(CultureInfo)
. Но стандартным для .NET способом преобразования любых объектов в строковое представление с возможностью настройки (!) является реализация интерфейса IFormattable
. Если аргументы, участвующие в подстановках, реализуют этот интерфейс, то именно он будет использоваться для преобразования аргумента в строку. Таким образом, нам предстоит реализовать IFormattable
в многоязычной строке.
В качестве поставщика формата для метода IFormattable.ToString(string format, IFormatProvider formatProvider)
можно как раз использовать локаль (культуру). А первый параметр позволяет задать параметры форматирования, не зависящие от локали. Например, вы можете задать отображение доли в виде процентов на английском для Индии:
// В Индии и некоторых других странах необычная группировка цифр
// https://en.wikipedia.org/wiki/Indian_numbering_system
12345.6789.ToString("P", CultureInfo.GetCultureInfo("en-IN"));
// 12,34,567.89%
Итак, попробуем сформировать ценник для того же товара:
var ru = CultureInfo.GetCultureInfo("ru");
var en = CultureInfo.GetCultureInfo("en");
var product = new Product
{
Code = "V0016887",
Name = new MultiCulturalString(ru, "Шоколад Алина")
.SetLocalizedString(en, "Chocolate Alina")
};
IFormatProvider localizationFormatProvider = en;
Console.WriteLine(string.Format(localizationFormatProvider,
"Артикул: {0}\r\nНаименование: {1}",
product.Code,
product.Name));
// Артикул: V0016887
// Наименование: Chocolate Alina
Прекрасно, мы получили строку "Артикул: V0016887\r\nНаименование: Chocolate Alina
", как и ожидали! Теперь немного усложним задачу, добавив в ценник дату его создания и поместив пользователя в англоязычный интерфейс с русскими региональными настройками:
Thread.CurrentThread.CurrentCulture = ru;
IFormatProvider localizationFormatProvider = en;
Console.WriteLine(string.Format(localizationFormatProvider,
"Артикул: {0}\r\nНаименование: {1}\r\nДата: {2:d}",
product.Code,
product.Name,
DateTime.Now));
// Артикул: V0016887
// Наименование: Chocolate Alina
// Дата: 11/25/2016
А что рассчитывал получить читатель? Автор, например, ожидал бы получить "Артикул: V0016887\r\nНаименование: Chocolate Alina\r\nДата: 25.11.2016
".
Да-да, мы не должны забывать о разделении региональных настроек и настроек локализации.
В .NET Framework есть как минимум три стандартные реализации IFormatProvider
(CultureInfo, NumberFormatInfo, DateTimeFormatInfo
), и ни одна из них нам не подходит. Нам необходима собственная реализация, которая будет нести в себе информацию о требуемой локали для локализации, в частности, для многоязычных строк, но не будет применяться для форматирования чисел и дат. Назовем ее LocalizationFormatInfo
. Использование выглядит не сложнее, чем код ранее:
Thread.CurrentThread.CurrentCulture = ru;
IFormatProvider localizationFormatProvider = new LocalizationFormatInfo(en);
Console.WriteLine(string.Format(localizationFormatProvider,
"Артикул: {0}\r\nНаименование: {1}\r\nДата: {2:d}",
product.Code,
product.Name,
DateTime.Now));
// Артикул: V0016887
// Наименование: Chocolate Alina
// Дата: 25.11.2016
А реализация IFormattable
в MultiCulturalString
выглядит примерно так:
string IFormattable.ToString(string format, IFormatProvider formatProvider)
{
// format не используется
var formatInfo = LocalizationFormatInfo.GetInstance(formatProvider);
return ToString(formatInfo.Culture ?? CultureInfo.CurrentUICulture);
}
Зато возможность в LocalizationFormatInfo
делегировать форматирование не многоязычных строк (дат, чисел и всего чего угодно) другим поставщикам будет весьма полезна.
/// <summary> Информация о локализации объектов. </summary>
[Serializable]
public sealed class LocalizationFormatInfo : IFormatProvider
{
/// <summary> Информация о локализации объектов. </summary>
/// <param name="culture">Культура отображения форматируемого объекта.</param>
/// <param name="provider">Поставщик других форматов.</param>
public LocalizationFormatInfo(CultureInfo culture, IFormatProvider provider = null)
{
_culture = culture;
_provider = provider;
}
/// <summary> Получает объект с информацией о каком-либо формате по типу этого объекта. </summary>
public object GetFormat(Type formatType)
{
if (formatType == GetType())
{
return this;
}
if (Provider != null)
{
// Если есть другой поставщик формата, то запрашиваем сведения у него.
return Provider.GetFormat(formatType);
}
return null;
}
/// <summary> Культура отображения форматируемого объекта. Может быть null. </summary>
public CultureInfo Culture
{
get { return _culture; }
}
private readonly CultureInfo _culture;
/// <summary> Поставщик других форматов. Может быть null. </summary>
public IFormatProvider Provider
{
get { return _provider; }
}
private readonly IFormatProvider _provider;
/// <summary>
/// Получить из <paramref name="provider"/> экземпляр <see cref="LocalizationFormatInfo"/>.
/// </summary>
/// <param name="provider">Поставщик объектов форматирования. Может быть <see langword="null"/>.</param>
/// <returns>Экземпляр <see cref="LocalizationFormatInfo"/>.</returns>
public static LocalizationFormatInfo GetInstance(IFormatProvider provider)
{
LocalizationFormatInfo lfi = null;
// Сначала пытаемся получить из поставщика
if (provider != null)
{
lfi = provider.GetFormat(typeof(LocalizationFormatInfo)) as LocalizationFormatInfo;
}
return lfi ?? Default;
}
private static readonly LocalizationFormatInfo Default = new LocalizationFormatInfo(null);
}
Развитием рассмотренного примера может стать превращение строки форматирования ценника в многоязычную строку.
Поиск подходящей локализации
Давайте еще раз представим, что мы создали экземпляр товара с наименованием на русском и английском языках, а затем запросили наименование для отсутствующего китайского языка. Вопрос прежний: какой результат вы бы ожидали получить?
В предыдущем разделе мы остановились на том, что, возможно, null
приемлем.
Рассмотрим распространенные ситуации. Новые версии приложений выходят, но цикл перевода не успевает вовремя за всеми изменениями. Многие свободные продукты переводятся энтузиастами, часто локализации от старых версий используются в новых. Как следствие, далеко не все элементы пользовательского интерфейса могут быть переведены или желаемый язык пользователя вообще не поддерживается приложением.
Очевидно, что в таких случаях пустoты в интерфейсе недопустимы. Необходимо отобразить ресурсы хотя бы для какого-то языка. При этом желательно, чтобы отображаемый элемент мог быть воспринят пользователем: узнан, прочитан, но не обязательно понят или переведен.
Здесь вступает в силу разумное предположение, что для локализованного приложения есть локаль по умолчанию, для которой набор ресурсов всегда актуален и полон.
Однако с локалью по умолчанию связана еще одна проблема: для пользователя в Казахстане при отсутствии казахстанской локализации наиболее естественно отобразить ресурс для русской локали, в то время как для пользователя в Китае логично показывать ресурс для английской локали, поскольку в Китае английским языком хоть как-то владеет большая доля населения, чем русским.
В документации по локализации в .NET описывается термин resource fallback process, который можно перевести на русский как «обработка альтернативных ресурсов». Суть обработки в том, что если для текущей локали пользовательского интерфейса не найдены соответствующие ресурсы, то будет предпринята попытка найти ресурсы для родительской локали. Так, для локали en-IN
родительской будет нейтральная локаль en
(нейтральная — не содержащая специфики региона). Поэтому в большинстве случаев для хранения универсальной для диалектов одного языка локализации можно порекомендовать именно нейтральные локали. А для локали en
, в свою очередь, родительской будет инвариантная, при выборе которой произойдет попытка найти ресурсы по умолчанию, которые должны существовать всегда.
Увы, в .NET Framework логика обработки альтернативных ресурсов «зашита» глубоко в недра платформы.
Наша же задача состоит в том, чтобы научиться кастомизировать процесс поиска ресурсов. Для этого давайте введем абстракцию IResourceFallbackProcess
. Единственной ее ответственностью будет генерация удобных нам последовательностей локалей для поиска подходящих ресурсов. При этом за поиск и загрузку ресурсов (в файловой системе, БД и т. д.) отвечают совершенно другие классы, например ResourceManager.
Представим новый интерфейс:
public interface IResourceFallbackProcess
{
/// <summary>
/// Для заданной культуры возвращает цепочку культур в том порядке,
/// в котором необходимо искать ресурсы.
/// </summary>
/// <param name="initial">Начальная культура.</param>
IEnumerable<CultureInfo> GetFallbackChain(CultureInfo initial);
}
Такой интерфейс позволит нам осуществить задуманное для каждой локали пользовательских интерфейсов:
- для
initial
локалиzh-CH
мы можем вернуть цепочкуzh-CH -> zh-CHS -> zh-Hans -> zh -> en
, - для
initial
локалиkz-KZ
мы можем вернуть цепочкуkz-KZ -> kz -> ru
.
И, конечно же, IResourceFallbackProcess
нужно активно применять в многоязычной строке. Вполне уместными смотрятся перегрузки методов GetString(...) / ToString(...)
с параметрами IResourceFallbackProcess resourceFallbackProcess
и bool useFallback
, причем перегрузки без useFallback
используют значение true
, а перегрузки без resourceFallbackProcess
— некий стандартный для вашего приложения порядок поиска.
Заключение
В исходном коде нашей библиотеки приведена работоспособная реализация IResourceFallbackProcess
. Читателю может быть полезным сделать эту реализацию конфигурируемой, а также создать собственный CustomizedResourceManager
, использующий IResourceFallbackProcess
. Кроме того, можно написать расширение для Visual Studio, чтобы автоматически генерируемые для файлов ресурсов классы использовали ваш CustomizedResourceManager
.
Очевидно, мы с вами рассмотрели не все возможные «подмостки», а только самые востребованные и универсальные. Например, можно подумать о MultiCulturalStringBuilder
, а для форматирования — об IMultiCulturalFormattable
.
В следующей статье «Подвалы Вавилонской башни, или Об интернационализации БД с доступом через ORM» мы рассмотрим вопросы хранения локализованных данных в БД и доступа к ним через объектно-реляционный маппер.
Комментарии (14)
greendimka
05.12.2016 12:52+2Несколько лет назад разработал класс Multistring как раз для подобных задач. Со всеми плюшками, сериализацией, сравнением, и т.д. и т.п.
В результате — другие методы работы практически вытеснили использование этого класса.vlio
05.12.2016 13:22Интересно, а какие именно методы "победили"?
У нас многоязычная строка и обвязка живут видимо потому, что существуют не сами по себе, а тесно интегрированы с другими инфраструктурными классами/библиотеками, включая ORM.
hdfan2
05.12.2016 13:40Для Вавилонской башни, как мне кажется, больше подходит понятие «фундамент», а не «подмостки».
greendimka
05.12.2016 13:44Ну мы старались не привязывать Multistring к чему-либо, а использовать именно как отдельный тип данных.
Видели два сценария использования этого типа.
Первый: тип данных для пользовательских данных, которые сам пользователь и наполняет (сделали удобные UI-компоненты для работы с типом).
Примеры:
- Тэги. Хозяин блога хочет добавить тэги к тексту (book = книга). Без Multistring так и добавляли: #book, #книга
- есть какая-то сущность с предопределёнными атрибутами. Например атрибут "размер" (=size). Без Multistring пользователь не мог добавить перевод с русского на английский.
В обоих случаях идея провалилась по той причине, что пользователи никак не могли понять концепции нескольких строк "в одной". Были бы программистами — скорее всего поняли бы, но мы работаем с простыми людьми — они не поняли.
В результате для блогов оставили как было, для атрибутов — используем стандартные структуры данных, добавляем таблицы, и т.д. — зависит от случая.
Второй: как хранилище строк для интерфейса — не оправдались затраты на написание кода. Файлы ресурсов оказались просто удобнее, т.к. эта информация редактируется на стороне программиста, а не заказчика.
greendimka
05.12.2016 14:02Что-бы повысить удобство использования, рекомендую вам отказаться от .ToString() в той концепции, в которой вы его используете, а именно для получения текущего значения (по текущей культуре). Всё-таки .ToString() — это метод, описывающий объект, и подразумевается, что его результат от версии к версии может быть разным. Заменить .ToString() следует операторами, конвертирующими в/из System.String.
P.S. выкидывать сам .ToString() — не обязательно
vlio
05.12.2016 14:23По поводу использования операторов и переопределенного
Object.ToString()
— спасибо, принято!
Все же методы
ToString()/GetString()
использовать приходиться, по нескольку перегрузок каждого с нужными параметрами (локаль, использовать ли и какой именно fallback process, поставщик форматирования). Причем две перегрузкиToString()
нужны для корректного форматирования при подстановках.
webmasterx
А что делать если я хочу перевести: «Should contain X character(s)»? где Х — неопределенное число, которое задается позже?.. в зависимости от >1 или 1 будут разные переводы. И если в русском и английском переводе в этом случае будет всего по два соответствующих варианта, то в других языках и других предложениях, количество возможных переводов может отличаться.
vlio
Хороший вопрос.
Microsoft вводит требование локализуемости, т.е. отделение кода и ресурсов. Утверждается, что в локализуемом приложении не понадобится писать код для новых языков приложения. Наивные.
В вашем примере это явно не так. Возможно, именно для таких случаев, необходимо уметь размещать в сателлитных сборках (один из видов хранения ресурсов в .NET) стратегии по обработке конкретных ресурсов для разных языков.
webmasterx
мало что понял из вашего ответа. лексикон .net совершенно не знаком. Но из ответа я понял что никак.
Но почему бы вам не реализовать ICU Messages? (если еще нет соответвующей либы)
vlio
Сейчас никак — стандартных средств нет.
Но и случаев таких пока не было.
vlio
Порт ICU Messages найден на гитхабе: https://github.com/jeffijoe/messageformat.net
Sinatr
Это проблема любой системы перевода (не относится к решению предлагаемому в статье). Проще всего в локализируемую строку поместить токен, который программа подменяет на значение в run-time, в простейшем случае можно тупо использовать "{0}" и string.Format(localizedString, someValue), хотя вы понимаете, чем это чревато.
А по-поводу статьи (далее обращаюсь к ТС), извините, но вводить новый тип для строк и использовать что-то вроде
вообще не круто. Интересно будет посмотреть как это выглядит в конце. Плюс вы не привязываетесь ни к winforms, ни к wpf (ни к чему-то еще?). А это как раз представляет больший интерес, чем простая возможность хранить строки для MessageBox.
В данный момент мы используем решение, где локализируемые строки выглядят как простой класс с «константами»:
а «локализация» происходит за счет рефлексии. Для WPF это легко обернуть в MarkupExtension где нужные строки возвращаются по ключу (ключ — это путь к свойству, к примеру Login.Disconnect).
Проблемы с конкатенацией (на которой вы делаете упор, обьясняя необходимость нового типа) решается элементарно конкатенацией предложений или фраз с независимым от культуры разделителем (например, перевод строки). Понятно, что обьединять слова/фразы в предложение не получится (в русском, например, нужно правильно построить окончания).
Возвращать null на неизвестную строку? Боже упаси! Если строка отсутствует в файле локализации, то вернется значение по умолчанию (в нашем случае это псевдо-english, который по-хорошему следует локализовать в en-US/en-GB). Если неверен ключ, но вернется значение ключа "!!!" + key (в этом случае пользователь/тестер сможет оперативно сделать багрепорт, а программист — с легкостью найти ошибку).
vlio
На практике такой код практически не встречается, ведь ресурсы динамически подгружаются из упомянутого
CustomizedResourceManager
. Именно он нужен, например, для WinForms.Совершенно согласен. Это предположение введено в середине статьи, чтобы развить тему с
IResourceFallbackProcess
позже. А потом сказано, что пустоты на UI, конечно же, недопустимы.vlio
Поясню. Есть две задачи:
GetString(...)
.