diff --git a/install/data/defaults.json b/install/data/defaults.json
index c471db6b89..86e63882cd 100644
--- a/install/data/defaults.json
+++ b/install/data/defaults.json
@@ -35,5 +35,6 @@
"allowPrivateGroups": 1,
"unreadCutoff": 2,
"bookmarkThreshold": 5,
- "topicsPerList": 20
+ "topicsPerList": 20,
+ "autoDetectLang": 1
}
diff --git a/public/language/en-GB/admin/general/languages.json b/public/language/en-GB/admin/general/languages.json
index da45cade2c..bdd57849b3 100644
--- a/public/language/en-GB/admin/general/languages.json
+++ b/public/language/en-GB/admin/general/languages.json
@@ -1,5 +1,6 @@
{
"language-settings": "Language Settings",
"description": "The default language determines the language settings for all users who are visiting your forum.
Individual users can override the default language on their account settings page.",
- "default-language": "Default Language"
+ "default-language": "Default Language",
+ "auto-detect": "Auto Detect Language Setting for Guests"
}
\ No newline at end of file
diff --git a/public/src/modules/helpers.js b/public/src/modules/helpers.js
index 3f02a86758..e1cf620607 100644
--- a/public/src/modules/helpers.js
+++ b/public/src/modules/helpers.js
@@ -176,7 +176,8 @@
}).join('');
};
- helpers.localeToHTML = function (locale) {
+ helpers.localeToHTML = function (locale, fallback) {
+ locale = locale || fallback || 'en-GB';
return locale.replace('_', '-');
};
diff --git a/src/controllers/admin/languages.js b/src/controllers/admin/languages.js
index 0ac4e98e99..e2d848ddae 100644
--- a/src/controllers/admin/languages.js
+++ b/src/controllers/admin/languages.js
@@ -18,6 +18,7 @@ languagesController.get = function (req, res, next) {
res.render('admin/general/languages', {
languages: languages,
+ autoDetectLang: parseInt(meta.config.autoDetectLang, 10) === 1,
});
});
};
diff --git a/src/languages.js b/src/languages.js
index c4c3d5ae0e..520ae8bba1 100644
--- a/src/languages.js
+++ b/src/languages.js
@@ -4,7 +4,7 @@ var fs = require('fs');
var path = require('path');
var async = require('async');
-var Languages = {};
+var Languages = module.exports;
var languagesPath = path.join(__dirname, '../build/public/language');
Languages.init = function (next) {
@@ -27,10 +27,13 @@ Languages.get = function (language, namespace, callback) {
});
};
-Languages.list = function (callback) {
- var languages = [];
+var codeCache = null;
+Languages.listCodes = function (callback) {
+ if (codeCache && codeCache.length) {
+ return callback(null, codeCache);
+ }
- fs.readdir(languagesPath, function (err, files) {
+ fs.readFile(path.join(languagesPath, 'metadata.json'), function (err, buffer) {
if (err && err.code === 'ENOENT') {
return callback(null, []);
}
@@ -38,43 +41,59 @@ Languages.list = function (callback) {
return callback(err);
}
- async.each(files, function (folder, next) {
- fs.stat(path.join(languagesPath, folder), function (err, stat) {
- if (err) {
- return next(err);
- }
+ var parsed;
+ try {
+ parsed = JSON.parse(buffer.toString());
+ } catch (e) {
+ return callback(e);
+ }
- if (!stat.isDirectory()) {
- return next();
- }
-
- var configPath = path.join(languagesPath, folder, 'language.json');
-
- fs.readFile(configPath, function (err, buffer) {
- if (err && err.code !== 'ENOENT') {
- return next(err);
- }
- if (buffer) {
- var lang = JSON.parse(buffer.toString());
- if (lang.name && lang.code && lang.dir) {
- languages.push(lang);
- }
- }
- next();
- });
- });
- }, function (err) {
- if (err) {
- return callback(err);
- }
- // Sort alphabetically
- languages = languages.sort(function (a, b) {
- return a.code > b.code ? 1 : -1;
- });
-
- callback(err, languages);
- });
+ var langs = parsed.languages;
+ codeCache = langs;
+ callback(null, langs);
});
};
-module.exports = Languages;
+var listCache = null;
+Languages.list = function (callback) {
+ if (listCache && listCache.length) {
+ return callback(null, listCache);
+ }
+
+ Languages.listCodes(function (err, codes) {
+ if (err) {
+ return callback(err);
+ }
+
+ async.map(codes, function (folder, next) {
+ var configPath = path.join(languagesPath, folder, 'language.json');
+
+ fs.readFile(configPath, function (err, buffer) {
+ if (err && err.code === 'ENOENT') {
+ return next();
+ }
+ if (err) {
+ return next(err);
+ }
+ try {
+ var lang = JSON.parse(buffer.toString());
+ next(null, lang);
+ } catch (e) {
+ next(e);
+ }
+ });
+ }, function (err, languages) {
+ if (err) {
+ return callback(err);
+ }
+
+ // filter out invalid ones
+ languages = languages.filter(function (lang) {
+ return lang.code && lang.name && lang.dir;
+ });
+
+ listCache = languages;
+ callback(null, languages);
+ });
+ });
+};
diff --git a/src/meta/languages.js b/src/meta/languages.js
index b420445f16..395711cba7 100644
--- a/src/meta/languages.js
+++ b/src/meta/languages.js
@@ -102,6 +102,25 @@ function getTranslationTree(callback) {
});
},
+ // save a list of languages to `${buildLanguagesPath}/metadata.json`
+ // avoids readdirs later on
+ function (ref, next) {
+ async.waterfall([
+ function (next) {
+ mkdirp(buildLanguagesPath, next);
+ },
+ function (x, next) {
+ fs.writeFile(path.join(buildLanguagesPath, 'metadata.json'), JSON.stringify({
+ languages: ref.languages.sort(),
+ namespaces: ref.namespaces.sort(),
+ }), next);
+ },
+ function (next) {
+ next(null, ref);
+ },
+ ], next);
+ },
+
// for each language and namespace combination,
// run through core and all plugins to generate
// a full translation hash
diff --git a/src/middleware/header.js b/src/middleware/header.js
index 70c0755def..0eb9cc9a1f 100644
--- a/src/middleware/header.js
+++ b/src/middleware/header.js
@@ -133,6 +133,7 @@ module.exports = function (middleware) {
templateValues.customJS = templateValues.useCustomJS ? meta.config.customJS : '';
templateValues.maintenanceHeader = parseInt(meta.config.maintenanceMode, 10) === 1 && !results.isAdmin;
templateValues.defaultLang = meta.config.defaultLang || 'en-GB';
+ templateValues.userLang = res.locals.config.userLang;
templateValues.privateUserInfo = parseInt(meta.config.privateUserInfo, 10) === 1;
templateValues.privateTagListing = parseInt(meta.config.privateTagListing, 10) === 1;
diff --git a/src/views/admin/general/languages.tpl b/src/views/admin/general/languages.tpl
index 310d1a366d..747c5d43af 100644
--- a/src/views/admin/general/languages.tpl
+++ b/src/views/admin/general/languages.tpl
@@ -16,6 +16,17 @@
+
+