1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150
|
# frozen_string_literal: true
require 'fast_gettext'
require 'yaml'
require 'locale'
module GettextSetup
class NoConfigFoundError < RuntimeError
def initialize(path)
super("No config.yaml found! (searching: #{path})")
end
end
@config = nil
@translation_repositories = {}
# `locales_path` should include:
# - config.yaml
# - a .pot file for the project
# - i18n directories for languages, each with a .po file
# - if using .mo files, an LC_MESSAGES dir in each language dir, with the .mo file in it
# valid `options` fields:
# :file_format - one of the supported backends for fast_gettext (e.g. :po, :mo, :yaml, etc.)
def self.initialize(locales_path = 'locales', options = {})
GettextSetup.initialize_config(locales_path)
# Make the translation methods available everywhere
Object.send(:include, FastGettext::Translation)
# Define our text domain, and set the path into our root. I would prefer to
# have something smarter, but we really want this up earlier even than our
# config loading happens so that errors there can be translated.
add_repository_to_chain(config['project_name'], options)
# 'chain' is the only available multi-domain type in fast_gettext 1.1.0 We should consider
# investigating 'merge' once we can bump our dependency
FastGettext.add_text_domain('master_domain', type: :chain, chain: @translation_repositories.values)
FastGettext.default_text_domain = 'master_domain'
# Likewise, be explicit in our default language choice. Available locales
# must be set prior to setting the default_locale since default locale must
# available.
FastGettext.default_available_locales = (FastGettext.default_available_locales || []) | locales
FastGettext.default_locale = default_locale
Locale.set_default(default_locale)
end
# Sets up the config class variables.
#
# Call this without calling initialize when you only need to deal with the
# translation files and you don't need runtime translation.
def self.initialize_config(locales_path = 'locales')
config_path = File.absolute_path('config.yaml', locales_path)
File.exist?(config_path) || raise(NoConfigFoundError, config_path)
@config = YAML.load_file(config_path)['gettext']
@locales_path = locales_path
end
def self.config?
raise NoConfigFoundError, File.join(locales_path, 'config.yaml') unless @config
@config
end
def self.add_repository_to_chain(project_name, options)
repository = FastGettext::TranslationRepository.build(project_name,
path: locales_path,
type: options[:file_format] || :po,
ignore_fuzzy: false)
@translation_repositories[project_name] = repository unless @translation_repositories.key? project_name
end
def self.locales_path
@locales_path ||= File.join(Dir.pwd, 'locales')
end
def self.config
@config ||= {}
end
def self.translation_repositories
@translation_repositories
end
def self.default_locale
config['default_locale'] || 'en'
end
def self.default_locale=(new_locale)
FastGettext.default_locale = new_locale
Locale.set_default(new_locale)
config['default_locale'] = new_locale
end
# Returns the locale for the current machine. This is most useful for shell
# applications that need an ACCEPT-LANGUAGE header set.
def self.candidate_locales
Locale.candidates(type: :cldr).join(',')
end
def self.clear
Locale.clear
end
def self.locales
explicit = Dir.glob(File.absolute_path('*/*.po', locales_path)).map do |x|
File.basename(File.dirname(x))
end
([default_locale] + explicit).uniq
end
# Given an HTTP Accept-Language header return the locale with the highest
# priority from it for which we have a locale available. If none exists,
# return the default locale
def self.negotiate_locale(accept_header)
unless @config
raise ArgumentError, 'No config.yaml found! Use ' \
'`GettextSetup.initialize(locales_path)` to locate your config.yaml'
end
return FastGettext.default_locale if accept_header.nil?
available_locales = accept_header.split(',').map do |locale|
pair = locale.strip.split(';q=')
pair << '1.0' unless pair.size == 2
# Ignore everything but the language itself; that means that we treat
# 'de' and 'de-DE' identical, and would use the 'de' message catalog
# for both.
pair[0] = pair[0].split('-')[0]
pair[0] = FastGettext.default_locale if pair[0] == '*'
pair
end.sort_by do |(_, qvalue)|
-1 * qvalue.to_f
end.select do |(locale, _)|
FastGettext.available_locales.include?(locale)
end
if available_locales && available_locales.first
available_locales.first.first
else
# We can't satisfy the request preference. Just use the default locale.
default_locale
end
end
# Negotiates and sets the locale based on an accept language header.
def self.negotiate_locale!(accept_header)
FastGettext.locale = negotiate_locale(accept_header)
end
end
|