i18n плюрализация - PullRequest
       40

i18n плюрализация

80 голосов
/ 29 мая 2011

Я хочу иметь возможность переводить множественные строки в i18n в рельсы. Строка может быть:

You have 2 kids

или

You have 1 kid

Я знаю, что могу использовать вспомогательный метод множественного числа, но я хочу встроить это в переводы i18n, чтобы в будущем мне не приходилось путаться с моими взглядами. Я читал, что :count так или иначе используется в переводах для множественного числа, но я не могу найти никаких реальных ресурсов о том, как это реализовано.

Обратите внимание, что я знаю, что могу передать переменную в строке перевода. Я также попробовал что-то вроде:

<%= t 'misc.kids', :kids_num => pluralize(1, 'kid') %>

Который работает нормально, но имеет фундаментальную проблему той же идеи. Мне нужно указать строку 'kid' во вспомогательном хелпере. Я не хочу этого делать, потому что это приведет к проблемам в будущем. Вместо этого я хочу сохранить все в переводе и ничего в поле зрения.

Как я могу это сделать?

Ответы [ 7 ]

166 голосов
/ 29 мая 2011

Попробуйте это:

en.yml:

en:
  misc:
    kids:
      zero: no kids
      one: 1 kid
      other: %{count} kids

В представлении:

You have <%= t('misc.kids', :count => 4) %>

Обновленный ответ для языков с множественным множественным числом (протестировано с Rails 3.0.7):

Файл config/initializers/pluralization.rb:

require "i18n/backend/pluralization" 
I18n::Backend::Simple.send(:include, I18n::Backend::Pluralization)

Файл config/locales/plurals.rb:

{:ru => 
  { :i18n => 
    { :plural => 
      { :keys => [:one, :few, :other],
        :rule => lambda { |n| 
          if n == 1
            :one
          else
            if [2, 3, 4].include?(n % 10) && 
               ![12, 13, 14].include?(n % 100) && 
               ![22, 23, 24].include?(n % 100)

              :few 
            else
              :other 
            end
          end
        } 
      } 
    } 
  } 
}

#More rules in this file: https://github.com/svenfuchs/i18n/blob/master/test/test_data/locales/plurals.rb
#(copy the file into `config/locales`)

Файл config/locales/en.yml:

en:
  kids:
    zero: en_zero
    one: en_one
    other: en_other

Файл config/locales/ru.yml:

ru:
  kids:
    zero: ru_zero
    one: ru_one
    few: ru_few
    other: ru_other

Test

$ rails c
>> I18n.translate :kids, :count => 1
=> "en_one"
>> I18n.translate :kids, :count => 3
=> "en_other"
>> I18n.locale = :ru
=> :ru
>> I18n.translate :kids, :count => 1
=> "ru_one"
>> I18n.translate :kids, :count => 3
=> "ru_few"  #works! yay! 
>> I18n.translate :kids, :count => 5
=> "ru_other"  #works! yay! 
34 голосов
/ 17 ноября 2014

Я надеюсь, что русскоязычные программисты на Ruby on Rails найдут это. Просто хочу поделиться своей очень точной русской формулой плюрализации. Он основан на спецификации Unicode . Вот только содержимое файла config/locales/plurals.rb, все остальное должно быть сделано так же, как в ответе выше.

{:ru => 
  { :i18n => 
    { :plural => 
      { :keys => [:zero, :one, :few, :many],
        :rule => lambda { |n| 
          if n == 0
            :zero
          elsif
            ( ( n % 10 ) == 1 ) && ( ( n % 100 != 11 ) )
            # 1, 21, 31, 41, 51, 61...
            :one
          elsif
            ( [2, 3, 4].include?(n % 10) \
            && ![12, 13, 14].include?(n % 100) )
            # 2-4, 22-24, 32-34...
            :few
          elsif ( (n % 10) == 0 || \
            ![5, 6, 7, 8, 9].include?(n % 10) || \
            ![11, 12, 13, 14].include?(n % 100) )
            # 0, 5-20, 25-30, 35-40...
            :many
          end
        } 
      } 
    } 
  } 
}

Носителям языка могут понравиться такие случаи, как 111 и 121. И вот результаты теста:

  • ноль: 0 запросов / куриц / яблок
  • один: 1 запрос / курица / яблоко
  • мало: 3 запроса / курицы / яблока
  • много: 5 запросов / куриц / яблок
  • один: 101 запрос / курица / яблоко
  • мало: 102 запроса / курицы / яблока
  • много: 105 запросов / куриц / яблок
  • много: 111 запросов / куриц / яблок
  • много: 119 запросов / куриц / яблок
  • один: 121 запрос / курица / яблоко
  • мало: 122 запроса / курицы / яблока
  • много: 125 запросов / куриц / яблок

Спасибо за первоначальный ответ!

11 голосов
/ 29 мая 2011

Во-первых, помните, что количество множественных форм зависит от языка , для английского - два, для румынского - 3, а для арабского - 6!.

Если вы хотитебыть в состоянии правильно использовать множественные формы, которые вы должны использовать gettext.

Для Ruby и rails вы должны проверить это http://www.yotabanana.com/hiki/ruby-gettext-howto-rails.html

8 голосов
/ 23 мая 2012

Rails 3 справляется с этим с учетом CLDR и переменной интерполяции подсчета. Смотри http://guides.rubyonrails.org/i18n.html#pluralization

# in view
t('actors', :count => @movie.actors.size)

# locales file, i.e. config/locales/en.yml
en:
  actors:
    one: Actor
    other: Actors
2 голосов
/ 25 мая 2014

На самом деле есть альтернатива громоздкому подходу i18n. Решение называется Tr8n.

Ваш код будет просто:

 <%= tr("You have {num || kid}", num: 1) %>

Вот и все. Не нужно извлекать ключи из кода и хранить их в пакетах ресурсов, не нужно реализовывать правила плюрализации для каждого языка. Tr8n поставляется с числовыми контекстными правилами для всех языков. Это также идет с гендерными правилами, правилами списка и языковыми случаями.

Полное определение приведенного выше ключа перевода на самом деле будет выглядеть так:

 <%= tr("You have {num:number || one: kid, other: kids}", num: 1) %>

Но поскольку мы хотим сэкономить пространство и время, num автоматически сопоставляется с числовыми правилами, и нет необходимости предоставлять все опции для значений правил. Tr8n поставляется с плюрализаторами и инфлекторами, которые сделают всю работу за вас на лету.

Перевод вашего ключа на русский, будет просто:

 "У вас есть {num || ребенок, ребенка, детей}"

Между прочим, ваш перевод будет неточным на языках, в которых есть гендерные правила. Например, на иврите вам фактически нужно было бы указать как минимум 2 перевода для вашего примера, так как «вы» будет отличаться в зависимости от пола пользователя, просматривающего просмотр. Tr8n справляется с этим очень хорошо. Вот транслитерация еврейских переводов:

 "Yesh leha yeled ahad" with {context: {viewing_user: male, num: one}}
 "Yesh leha {num} yeladim" with {context: {viewing_user: male, num: other}}
 "Yesh lah yeled ahad" with {context: {viewing_user: female, num: one}}
 "Yesh lah {num} yeladim" with {context: {viewing_user: female, num: other}}

Так что вашему единственному английскому ключу в данном случае требуется 4 перевода. Все переводы выполняются в контексте - вам не нужно разбивать предложение. Tr8n имеет механизм для сопоставления одного ключа с несколькими переводами на основе языка и контекста - все делается на лету.

И последнее. Что если бы вам пришлось сделать часть подсчета жирным шрифтом? Это было бы просто:

<%= tr("You have [bold: {num || kid}]", num: 1, bold: "<strong>{$0}</strong>") %>

На всякий случай, если вы захотите переопределить ваш «жирный» позже - это будет очень просто - вам не нужно будет просматривать все ваши файлы YAML и изменять их - вы просто делаете это в одном месте.

Чтобы узнать больше, посмотрите здесь:

https://github.com/tr8n/tr8n_rails_clientsdk

Раскрытие информации: я являюсь разработчиком и сопровождающим фреймворка Tr8n и всех его библиотек.

0 голосов
/ 30 августа 2018

О Redmine.Если вы копируете правила файла множественного использования в config / locales / как множественное число.rb и другие, отличные от имени локали (ru.rb, pl.rb и т. Д.), Они не будут работать.Вы должны переименовать правила файла в «locale» .rb или изменить метод в файле /lib/redmine/i18n.rb

def init_translations(locale)
  locale = locale.to_s
  paths = ::I18n.load_path.select {|path| File.basename(path, '.*') == locale}
  load_translations(paths)
  translations[locale] ||= {}
end

, и если у вас более старая версия Redmine, добавьте

module Implementation
        include ::I18n::Backend::Base
        **include ::I18n::Backend::Pluralization**
0 голосов
/ 05 марта 2017

английский

Это просто работает из коробки

en.yml

en:
  kid:
    one: '1 kid'
    other: '%{count} kids'

Использование (конечно, вы можете пропустить I18n в файле просмотра):

> I18n.t :kid, count: 1
 => "1 kid"

> I18n.t :kid, count: 3
 => "3 kids"

Русский (и другие языки с множественными формами множественного числа)

Установите rails-18n gem и добавьте переводы в свои .yml файлы, как в примере :

ru.yml

ru:
  kid:
    zero: 'нет детей'
    one: '%{count} ребенок'
    few: '%{count} ребенка'
    many: '%{count} детей'
    other: 'дети'

Использование:

> I18n.t :kid, count: 0
 => "нет детей"

> I18n.t :kid, count: 1
 => "1 ребенок"

> I18n.t :kid, count: 3
 => "3 ребенка"

> I18n.t :kid, count: 5
 => "5 детей"

> I18n.t :kid, count: 21
 => "21 ребенок"

> I18n.t :kid, count: 114
 => "114 детей"

> I18n.t :kid, count: ''
 => "дети"
...