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:
parent
0e24cb0f78
commit
c1e9a70d59
|
@ -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) {
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -104,28 +104,32 @@ module JsLocaleHelper
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def self.clear_cache!
|
||||||
|
@loaded_translations = nil
|
||||||
|
@plugin_translations = nil
|
||||||
|
@loaded_merges = nil
|
||||||
|
end
|
||||||
|
|
||||||
def self.translations_for(locale_str)
|
def self.translations_for(locale_str)
|
||||||
if Rails.env.development?
|
clear_cache! if Rails.env.development?
|
||||||
@loaded_translations = nil
|
|
||||||
@plugin_translations = nil
|
|
||||||
@loaded_merges = nil
|
|
||||||
end
|
|
||||||
|
|
||||||
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)
|
||||||
|
|
|
@ -26,23 +26,32 @@ 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
|
||||||
JsLocaleHelper.expects(:plugin_translations)
|
before do
|
||||||
.with(any_of("en", "en_US"))
|
JsLocaleHelper.clear_cache!
|
||||||
.returns("admin_js" => {
|
JsLocaleHelper.expects(:plugin_translations)
|
||||||
"admin" => {
|
.with(any_of("en", "en_US"))
|
||||||
"site_settings" => {
|
.returns("admin_js" => {
|
||||||
"categories" => {
|
"admin" => {
|
||||||
"github_badges" => "Github Badges"
|
"site_settings" => {
|
||||||
|
"categories" => {
|
||||||
|
"github_badges" => "Github Badges"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}).at_least_once
|
||||||
}).at_least_once
|
end
|
||||||
|
|
||||||
get "/extra-locales/admin"
|
after do
|
||||||
|
JsLocaleHelper.clear_cache!
|
||||||
|
end
|
||||||
|
|
||||||
expect(response.status).to eq(200)
|
it "includes plugin translations" do
|
||||||
expect(response.body.include?("github_badges")).to eq(true)
|
get "/extra-locales/admin"
|
||||||
|
|
||||||
|
expect(response.status).to eq(200)
|
||||||
|
expect(response.body.include?("github_badges")).to eq(true)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -54,8 +54,15 @@ export default function(name, opts) {
|
||||||
andThen(() => {
|
andThen(() => {
|
||||||
return this.render(opts.template);
|
return this.render(opts.template);
|
||||||
});
|
});
|
||||||
|
|
||||||
andThen(() => {
|
andThen(() => {
|
||||||
opts.test.call(this, assert);
|
try {
|
||||||
|
opts.test.call(this, assert);
|
||||||
|
} finally {
|
||||||
|
if (opts.afterEach) {
|
||||||
|
opts.afterEach.call(opts);
|
||||||
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -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"
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -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: {
|
||||||
hbs_test0: "evil",
|
js: {
|
||||||
hbs_test1: "trout"
|
hbs_test0: "evil",
|
||||||
|
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");
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
Loading…
Reference in New Issue