FIX: Fallback locale was not available for extra translations

Translations from fallback locales were not sent to the client
for admin_js and wizard_js.
This commit is contained in:
Gerhard Schlager 2019-05-23 21:23:31 +02:00
parent 0e24cb0f78
commit c1e9a70d59
8 changed files with 135 additions and 66 deletions

View File

@ -28,24 +28,6 @@ I18n.isValidNode = function(obj, node, undefined) {
return obj[node] !== null && obj[node] !== undefined; return obj[node] !== null && obj[node] !== undefined;
}; };
function checkExtras(origScope, sep, extras) {
if (!extras || extras.length === 0) { return; }
for (var i = 0; i < extras.length; i++) {
var messages = extras[i];
scope = origScope.split(sep);
if (scope[0] === 'js') { scope.shift(); }
while (messages && scope.length > 0) {
currentScope = scope.shift();
messages = messages[currentScope];
}
if (messages !== undefined) { return messages; }
}
}
I18n.lookup = function(scope, options) { I18n.lookup = function(scope, options) {
options = options || {}; options = options || {};
@ -64,17 +46,26 @@ I18n.lookup = function(scope, options) {
scope = options.scope.toString() + this.SEPARATOR + scope; scope = options.scope.toString() + this.SEPARATOR + scope;
} }
var origScope = "" + scope; var originalScope = scope;
scope = scope.split(this.SEPARATOR);
scope = origScope.split(this.SEPARATOR); if (scope.length > 0 && scope[0] !== "js") {
scope.unshift("js");
}
while (messages && scope.length > 0) { while (messages && scope.length > 0) {
currentScope = scope.shift(); currentScope = scope.shift();
messages = messages[currentScope]; messages = messages[currentScope];
} }
if (messages === undefined) { if (messages === undefined && this.extras && this.extras[locale]) {
messages = checkExtras(origScope, this.SEPARATOR, this.extras); messages = this.extras[locale];
scope = originalScope.split(this.SEPARATOR);
while (messages && scope.length > 0) {
currentScope = scope.shift();
messages = messages[currentScope];
}
} }
if (messages === undefined) { if (messages === undefined) {

View File

@ -40,21 +40,20 @@ class ExtraLocalesController < ApplicationController
translations = JsLocaleHelper.translations_for(locale_str) translations = JsLocaleHelper.translations_for(locale_str)
for_key = {} translations.keys.each do |l|
translations.values.each { |v| for_key.deep_merge!(v[bundle_str]) if v.has_key?(bundle_str) } translations[l].keys.each do |k|
bundle_translations = translations[l].delete(k)
translations[l].deep_merge!(bundle_translations) if k == bundle_str
end
end
js = "" js = ""
if for_key.present? if translations.present?
if plugin_for_key = JsLocaleHelper.plugin_translations(locale_str)[bundle_str]
for_key.deep_merge!(plugin_for_key)
end
js = <<~JS.squish js = <<~JS.squish
(function() { (function() {
if (window.I18n) { if (window.I18n) {
window.I18n.extras = window.I18n.extras || []; window.I18n.extras = #{translations.to_json};
window.I18n.extras.push(#{for_key.to_json});
} }
})(); })();
JS JS

View File

@ -104,28 +104,32 @@ module JsLocaleHelper
end end
end end
def self.translations_for(locale_str) def self.clear_cache!
if Rails.env.development?
@loaded_translations = nil @loaded_translations = nil
@plugin_translations = nil @plugin_translations = nil
@loaded_merges = nil @loaded_merges = nil
end end
def self.translations_for(locale_str)
clear_cache! if Rails.env.development?
locale_sym = locale_str.to_sym locale_sym = locale_str.to_sym
I18n.with_locale(locale_sym) do translations = I18n.with_locale(locale_sym) do
if locale_sym == :en if locale_sym == :en
load_translations(locale_sym) load_translations(locale_sym)
else else
load_translations_merged(*I18n.fallbacks[locale_sym]) load_translations_merged(*I18n.fallbacks[locale_sym])
end end
end end
Marshal.load(Marshal.dump(translations))
end end
def self.output_locale(locale) def self.output_locale(locale)
locale_str = locale.to_s locale_str = locale.to_s
fallback_locale_str = LocaleSiteSetting.fallback_locale(locale_str)&.to_s fallback_locale_str = LocaleSiteSetting.fallback_locale(locale_str)&.to_s
translations = Marshal.load(Marshal.dump(translations_for(locale_str))) translations = translations_for(locale_str)
message_formats = remove_message_formats!(translations, locale) message_formats = remove_message_formats!(translations, locale)
mf_locale, mf_filename = find_message_format_locale([locale_str], fallback_to_english: true) mf_locale, mf_filename = find_message_format_locale([locale_str], fallback_to_english: true)

View File

@ -26,7 +26,9 @@ describe ExtraLocalesController do
expect(response.status).to eq(404) expect(response.status).to eq(404)
end end
it "includes plugin translations" do context "with plugin" do
before do
JsLocaleHelper.clear_cache!
JsLocaleHelper.expects(:plugin_translations) JsLocaleHelper.expects(:plugin_translations)
.with(any_of("en", "en_US")) .with(any_of("en", "en_US"))
.returns("admin_js" => { .returns("admin_js" => {
@ -38,7 +40,13 @@ describe ExtraLocalesController do
} }
} }
}).at_least_once }).at_least_once
end
after do
JsLocaleHelper.clear_cache!
end
it "includes plugin translations" do
get "/extra-locales/admin" get "/extra-locales/admin"
expect(response.status).to eq(200) expect(response.status).to eq(200)
@ -46,3 +54,4 @@ describe ExtraLocalesController do
end end
end end
end end
end

View File

@ -54,8 +54,15 @@ export default function(name, opts) {
andThen(() => { andThen(() => {
return this.render(opts.template); return this.render(opts.template);
}); });
andThen(() => { andThen(() => {
try {
opts.test.call(this, assert); opts.test.call(this, assert);
} finally {
if (opts.afterEach) {
opts.afterEach.call(opts);
}
}
}); });
}); });
} }

View File

@ -99,12 +99,68 @@ QUnit.test("translations", assert => {
}); });
QUnit.test("extra translations", assert => { QUnit.test("extra translations", assert => {
I18n.extras = [{ admin: { title: "Discourse Admin" } }]; I18n.locale = "pl_PL";
I18n.extras = {
en: {
admin: {
dashboard: {
title: "Dashboard",
backup_count: {
one: "%{count} backup",
other: "%{count} backups"
}
},
web_hooks: {
events: {
incoming: {
one: "There is a new event.",
other: "There are %{count} new events."
}
}
}
}
},
pl_PL: {
admin: {
dashboard: {
title: "Raporty"
},
web_hooks: {
events: {
incoming: {
one: "Istnieje nowe wydarzenie",
few: "Istnieją %{count} nowe wydarzenia.",
many: "Istnieje %{count} nowych wydarzeń.",
other: "Istnieje %{count} nowych wydarzeń."
}
}
}
}
}
};
I18n.pluralizationRules.pl_PL = function(n) {
if (n === 1) return "one";
if (n % 10 >= 2 && n % 10 <= 4) return "few";
if (n % 10 === 0) return "many";
return "other";
};
assert.equal( assert.equal(
I18n.t("admin.title"), I18n.t("admin.dashboard.title"),
"Discourse Admin", "Raporty",
"it check extra translations when they exists" "it uses extra translations when they exists"
);
assert.equal(
I18n.t("admin.web_hooks.events.incoming", { count: 2 }),
"Istnieją 2 nowe wydarzenia.",
"it uses pluralized extra translation when it exists"
);
assert.equal(
I18n.t("admin.dashboard.backup_count", { count: 2 }),
"2 backups",
"it falls back to English and uses extra translations when they exists"
); );
}); });

View File

@ -258,6 +258,8 @@ widgetTest("handlebars d-icon", {
}); });
widgetTest("handlebars i18n", { widgetTest("handlebars i18n", {
_translations: I18n.translations,
template: `{{mount-widget widget="hbs-i18n-test" args=args}}`, template: `{{mount-widget widget="hbs-i18n-test" args=args}}`,
beforeEach() { beforeEach() {
@ -268,15 +270,21 @@ widgetTest("handlebars i18n", {
<a href title={{i18n "hbs_test0"}}>test</a> <a href title={{i18n "hbs_test0"}}>test</a>
` `
}); });
I18n.extras = [ I18n.translations = {
{ en: {
js: {
hbs_test0: "evil", hbs_test0: "evil",
hbs_test1: "trout" hbs_test1: "trout"
} }
]; }
};
this.set("args", { key: "hbs_test1" }); this.set("args", { key: "hbs_test1" });
}, },
afterEach() {
I18n.translations = this._translations;
},
test(assert) { test(assert) {
// comin up // comin up
assert.equal(find("span.string").text(), "evil"); assert.equal(find("span.string").text(), "evil");

View File

@ -1,10 +1,5 @@
(function() { (function() {
if (typeof I18n !== "undefined") { if (typeof I18n !== "undefined") {
var oldI18nlookup = I18n.lookup;
I18n.lookup = function(scope, options) {
return oldI18nlookup.apply(this, ["js." + scope, options]);
};
// Default format for storage units // Default format for storage units
var oldI18ntoHumanSize = I18n.toHumanSize; var oldI18ntoHumanSize = I18n.toHumanSize;
I18n.toHumanSize = function(number, options) { I18n.toHumanSize = function(number, options) {