Skip to content
Permalink
Browse files
Start localization
Trac 30310
  • Loading branch information
arlolra committed Aug 14, 2019
1 parent 0b55fd3 commit ecadbe3d869ae3dcdde02d24b6246859621bd34a
Showing with 115 additions and 26 deletions.
  1. +1 −0 .gitignore
  2. +42 −11 proxy/init-badge.js
  3. +4 −3 proxy/make.js
  4. +32 −0 proxy/static/_locales/en_US/messages.json
  5. +3 −3 proxy/static/embed.html
  6. +1 −1 proxy/static/index.html
  7. +12 −0 proxy/static/popup.js
  8. +18 −7 proxy/webext/embed.js
  9. +2 −1 proxy/webext/manifest.json
@@ -19,5 +19,6 @@ proxy/webext/popup.js
proxy/webext/embed.html
proxy/webext/embed.css
proxy/webext/assets/
proxy/webext/_locales/
ignore/
npm-debug.log
@@ -4,6 +4,20 @@
UI
*/

class Messages {
constructor(json) {
this.json = json;
}
getMessage(m, ...rest) {
let message = this.json[m].message;
return message.replace(/\$(\d+)/g, (...args) => {
return rest[Number(args[1]) - 1];
});
}
}

let messages = null;

class BadgeUI extends UI {

constructor() {
@@ -16,28 +30,31 @@ class BadgeUI extends UI {
missingFeature(missing) {
this.popup.setEnabled(false);
this.popup.setActive(false);
this.popup.setStatusText("Snowflake is off");
this.popup.setStatusText(messages.getMessage('popupStatusOff'));
this.popup.setStatusDesc(missing, true);
this.popup.hideButton();
}

turnOn() {
const clients = this.active ? 1 : 0;
this.popup.setChecked(true);
this.popup.setToggleText('Turn Off');
this.popup.setStatusText(`${clients} client${(clients !== 1) ? 's' : ''} connected.`);
this.popup.setToggleText(messages.getMessage('popupTurnOff'));
if (clients > 0) {
this.popup.setStatusText(messages.getMessage('popupStatusOn', String(clients)));
} else {
this.popup.setStatusText(messages.getMessage('popupStatusReady'));
}
// FIXME: Share stats from webext
const total = 0;
this.popup.setStatusDesc(`Your snowflake has helped ${total} user${(total !== 1) ? 's' : ''} circumvent censorship in the last 24 hours.`);
this.popup.setStatusDesc('');
this.popup.setEnabled(true);
this.popup.setActive(this.active);
}

turnOff() {
this.popup.setChecked(false);
this.popup.setToggleText('Turn On');
this.popup.setStatusText("Snowflake is off");
this.popup.setStatusDesc("");
this.popup.setToggleText(messages.getMessage('popupTurnOn'));
this.popup.setStatusText(messages.getMessage('popupStatusOff'));
this.popup.setStatusDesc('');
this.popup.setEnabled(false);
this.popup.setActive(false);
}
@@ -108,12 +125,12 @@ var debug, snowflake, config, broker, ui, log, dbg, init, update, silenceNotific
ui = new BadgeUI();

if (!Util.hasWebRTC()) {
ui.missingFeature("WebRTC feature is not detected.");
ui.missingFeature(messages.getMessage('popupWebRTCOff'));
return;
}

if (!Util.hasCookies()) {
ui.missingFeature("Cookies are not enabled.");
ui.missingFeature(messages.getMessage('badgeCookiesOff'));
return;
}

@@ -153,6 +170,20 @@ var debug, snowflake, config, broker, ui, log, dbg, init, update, silenceNotific
return null;
};

window.onload = init;
window.onload = function() {
const lang = 'en_US';
fetch(`./_locales/${lang}/messages.json`)
.then((res) => {
if (!res.ok) { return; }
return res.json();
})
.then((json) => {
messages = new Messages(json);
Popup.fill(document.body, (m) => {
return messages.getMessage(m);
});
init();
});
}

}());
@@ -32,7 +32,8 @@ var SHARED_FILES = [
'embed.html',
'embed.css',
'popup.js',
'assets'
'assets',
'_locales',
];

var concatJS = function(outDir, init, outFile) {
@@ -67,7 +68,7 @@ task('test', 'snowflake unit tests', function() {
});

task('build', 'build the snowflake proxy', function() {
execSync('rm -r build');
execSync('rm -rf build');
execSync('cp -r ' + STATIC + '/ build/');
concatJS('build', 'badge', 'embed.js');
console.log('Snowflake prepared.');
@@ -87,7 +88,7 @@ task('node', 'build the node binary', function() {
});

task('clean', 'remove all built files', function() {
execSync('rm -r build test spec/support');
execSync('rm -rf build test spec/support');
});

var cmd = process.argv[2];
@@ -0,0 +1,32 @@
{
"appDesc": {
"message": "Snowflake is a WebRTC pluggable transport for Tor."
},
"popupTurnOn": {
"message": "Turn On"
},
"popupTurnOff": {
"message": "Turn Off"
},
"popupLearnMore": {
"message": "Learn more"
},
"popupStatusOff": {
"message": "Snowflake is off"
},
"popupStatusOn": {
"message": "Number of users currently connected: $1"
},
"popupStatusReady": {
"message": "Your Snowflake is ready to help users circumvent censorship!"
},
"popupWebRTCOff": {
"message": "WebRTC feature is not detected."
},
"popupDescOn": {
"message": "Number of users your Snowflake has helped circumvent censorship in the last 24 hours: $1"
},
"badgeCookiesOff": {
"message": "Cookies are not enabled."
}
}
@@ -11,18 +11,18 @@
<body>
<div id="active">
<div id="statusimg"></div>
<p id="statustext">Snowflake is off</p>
<p id="statustext">__MSG_popupStatusOff__</p>
<p id="statusdesc"></p>
</div>
<div class="b button">
<label id="toggle" for="enabled">Turn On</label>
<label id="toggle" for="enabled">__MSG_popupTurnOn__</label>
<label class="switch">
<input id="enabled" type="checkbox" />
<span class="slider round"></span>
</label>
</div>
<div class="b learn">
<a target="_blank" href="https://snowflake.torproject.org/">Learn more</a>
<a target="_blank" href="https://snowflake.torproject.org/">__MSG_popupLearnMore__</a>
</div>
</body>
</html>
@@ -86,7 +86,7 @@ <h3>EMBED</h3>

<p>Which looks like this:</p>

<iframe src="embed.html" width="320px" height="200px" frameborder="0" scrolling="no"></iframe>
<iframe src="embed.html" width="320px" height="240px" frameborder="0" scrolling="no"></iframe>

</div>
</body>
@@ -38,4 +38,16 @@ class Popup {
setToggleText(txt) {
document.getElementById('toggle').innerText = txt;
}
static fill(n, func) {
switch(n.nodeType) {
case 3: { // Node.TEXT_NODE
const m = /^__MSG_([^_]*)__$/.exec(n.nodeValue);
if (m) { n.nodeValue = func(m[1]); }
break;
}
case 1: // Node.ELEMENT_NODE
n.childNodes.forEach(c => Popup.fill(c, func));
break;
}
}
}
@@ -1,5 +1,12 @@
/* global chrome, Popup */

// Fill i18n in HTML
window.onload = () => {
Popup.fill(document.body, (m) => {
return chrome.i18n.getMessage(m);
});
};

const port = chrome.runtime.connect({
name: "popup"
});
@@ -11,8 +18,8 @@ port.onMessage.addListener((m) => {
if (missingFeature) {
popup.setEnabled(false);
popup.setActive(false);
popup.setStatusText("Snowflake is off");
popup.setStatusDesc("WebRTC feature is not detected.", true);
popup.setStatusText(chrome.i18n.getMessage('popupStatusOff'));
popup.setStatusDesc(chrome.i18n.getMessage('popupWebRTCOff'), true);
popup.hideButton();
return;
}
@@ -21,13 +28,17 @@ port.onMessage.addListener((m) => {

if (enabled) {
popup.setChecked(true);
popup.setToggleText('Turn Off');
popup.setStatusText(`${clients} client${(clients !== 1) ? 's' : ''} connected.`);
popup.setStatusDesc(`Your snowflake has helped ${total} user${(total !== 1) ? 's' : ''} circumvent censorship in the last 24 hours.`);
popup.setToggleText(chrome.i18n.getMessage('popupTurnOff'));
if (clients > 0) {
popup.setStatusText(chrome.i18n.getMessage('popupStatusOn', String(clients)));
} else {
popup.setStatusText(chrome.i18n.getMessage('popupStatusReady'));
}
popup.setStatusDesc((total > 0) ? chrome.i18n.getMessage('popupDescOn', String(total)) : '');
} else {
popup.setChecked(false);
popup.setToggleText('Turn On');
popup.setStatusText("Snowflake is off");
popup.setToggleText(chrome.i18n.getMessage('popupTurnOn'));
popup.setStatusText(chrome.i18n.getMessage('popupStatusOff'));
popup.setStatusDesc("");
}
popup.setEnabled(enabled);
@@ -2,7 +2,8 @@
"manifest_version": 2,
"name": "Snowflake",
"version": "0.0.9",
"description": "Snowflake is a WebRTC pluggable transport for Tor.",
"description": "__MSG_appDesc__",
"default_locale": "en_US",
"background": {
"scripts": ["snowflake.js"],
"persistent": true

0 comments on commit ecadbe3

Please sign in to comment.