diff --git a/i/css/i.css b/i/css/i.css new file mode 100644 index 0000000..7f28428 --- /dev/null +++ b/i/css/i.css @@ -0,0 +1,46 @@ +body { + background: #f8f8f8; + color: #000000; + font-size: 1em; + font-family: sans-serif; +} +h3 { + font-size: 1.5em; + font-weight: 400; + line-height: 1.1; + color: #444; +} + +a { + color: #1b8250; +} + +li { + padding-bottom: 0.75em; +} + +.btn:hover, .btn:focus, .btn.focus { + color: #666; + text-decoration: none; +} + +.btn-primary { + background: #43a047; + color: #f2f2f2; +} +.btn-primary:hover, .btn-primary:focus, .btn-primary:active, .btn-primary.active, .open > .dropdown-toggle.btn-primary { + background: #388e3c; + color: #000000; + box-shadow: none; +} +.btn-primary:active, .btn-primary.active { + background: #2e7d32; + box-shadow: none; +} +.text-center { + text-align: center; +} +.hint { + font-size: 0.9em; + color: #444; +} diff --git a/i/css/styles.css b/i/css/styles.css new file mode 100644 index 0000000..abdd5f7 --- /dev/null +++ b/i/css/styles.css @@ -0,0 +1,66 @@ +/* Mostly cherrypicked from bootstrap https://getbootstrap.com/css/ */ +.btn { + display: inline-block; + margin-bottom: 0; + font-weight: normal; + text-align: center; + vertical-align: middle; text-transform: uppercase; + border-right: none; + border-bottom: none; + color: #666; + text-decoration: none; + transition: all .2s; + touch-action: manipulation; + cursor: pointer; + border: 1px solid transparent; + white-space: nowrap; + padding: 6px 16px; + font-size: 13px; + line-height: 1.846; + border-radius: 3px; + -webkit-user-select: none; + -moz-user-select: none; +} + +.btn:hover, .btn:focus, .btn.focus { + color: #666; + text-decoration: none; +} + +textarea, textarea.form-control, input.form-control, input[type="text"], input[type="password"], input[type="email"], input[type="number"], .form-control[type="text"], .form-control[type="password"], .form-control[type="email"], .form-control[type="tel"] { + padding: 0; + border: none; + border-radius: 0; + -webkit-box-shadow: inset 0 -1px 0 #ddd; + box-shadow: inset 0 -1px 0 #ddd; + font-size: 1.1em; +} + +.form-control { + display: block; + width: 100%; + height: 37px; + line-height: 1.846; + color: #666; + background-color: transparent; + background-image: none; + -webkit-box-shadow: inset 0 1px 1px rgba(0,0,0,0.075); + box-shadow: inset 0 1px 1px rgba(0,0,0,0.075); + -webkit-transition: border-color ease-in-out .15s,-webkit-box-shadow ease-in-out .15s; + transition: border-color ease-in-out .15s,box-shadow ease-in-out .15s; +} + +input, button { + -webkit-font-smoothing: antialiased; + letter-spacing: .1px; + text-rendering: optimizeLegibility; +} + +a { + text-decoration: none; +} + +a:hover, a:focus { + color: #165d16; + text-decoration: underline; +} diff --git a/i/index.html b/i/index.html new file mode 100644 index 0000000..b50b7e2 --- /dev/null +++ b/i/index.html @@ -0,0 +1,37 @@ + + + + + + + + + + + + +
+ +

+

+ +

+

+ + + +

+

+
+ + + + + diff --git a/i/js/i18n-text.js b/i/js/i18n-text.js new file mode 100644 index 0000000..3003f14 --- /dev/null +++ b/i/js/i18n-text.js @@ -0,0 +1,6 @@ +/*! + i18n-text - v0.4.3 - 2014-08-14 + https://vogdb@bitbucket.org/vogdb/i18n-text.git + Copyright (c) 2014 Sanin Aleksey aka vogdb; Licensed WTFPL +*/ +!function(a){function b(a){return"[object Object]"===Object.prototype.toString.call(a)}function c(a,b){return a.replace(/{{([\w]+)}}/g,function(a,c){var d=b[c];return d?d:a})}function d(a,b){throw new Error(c(a,b))}function e(a){return-1!==a.indexOf("file://")}function f(a){return e(a)||e(window.location.href)&&a.indexOf(!1)}var g={parseAndEval:function(a,b){return a.replace(/{{([^{}]+#[^}]*(?:|.*#.*)+)}}/g,function(a,c){return g.eval(c,b)})},eval:function(a,b){for(var c=a.split("|"),e=c.length-1;e>=0;e--){var f=c[e].split("#"),i=f[0];if(g.isValidFormula(i)||d(h.error.INVALID_PLURAL,{plural:i}),g.evalFormula(i,b))return f[1]}return a},isValidFormula:function(a){return/[ \dn<>=*-+!?]+/g.test(a)},evalFormula:function(a,b){return new Function("n","return "+a)(b)}},h=function(a){this._loadedLocales={},this._currentLocale=null,this._msgPath=null,this._subscribers={},this._init(a)};h.prototype.getLocale=function(){return this._currentLocale},h.prototype.setLocale=function(a){this.hasLocale(a)?this._setLocale(a):(this.once(h.event.LOCALE_LOAD,function(b){b.error||b.locale!==a||this._setLocale(b.locale)}.bind(this)),this.loadLocale(a))},h.prototype._setLocale=function(a){this._currentLocale=a,this._fire(h.event.LOCALE_CHANGE,{locale:a})},h.prototype.hasLocale=function(a){return!!this._loadedLocales[a]},h.prototype.loadLocale=function(a,b){b=b||this._msgPath;var d=b+"/"+a+".json";h.loadFile({url:d,success:function(b){b.length>0?(this._loadedLocales[a]={},this._mergeKeys(JSON.parse(b),this._loadedLocales[a],""),this._fire(h.event.LOCALE_LOAD,{locale:a})):this._fire(h.event.LOCALE_LOAD,{error:c(h.error.EMPTY_MESSAGES,{file:d})})}.bind(this),error:function(a){this._fire(h.event.LOCALE_LOAD,{error:c(h.error.NO_MESSAGES,{file:d,error:a})})}.bind(this)})},h.prototype._mergeKeys=function(a,c,d){for(var e in a)if(b(a[e])){var f;f=d?d+h.KEY_SEPARATOR+e:e,this._mergeKeys(a[e],c,f)}else{var g;g=d?d+h.KEY_SEPARATOR+e:e,c[g]=a[e]}},h.prototype.text=function(a,e,f){arguments[1]&&!b(arguments[1])&&(f=arguments[1],e=void 0);var i=f||this.getLocale();i||d(h.error.NO_LOCALE_IS_SET),this.hasLocale(i)||this.loadLocale(i);var j=this._loadedLocales[i][a];return void 0===j&&d(h.error.INVALID_KEY,{key:a,locale:i}),e?(void 0!==e.n&&(j=g.parseAndEval(j,e.n)),c(j,e)):j},h.prototype.on=function(a,b){this._subscribers[a]||(this._subscribers[a]=[]),this._subscribers[a].push(b)},h.prototype._fire=function(a,b){if(this._subscribers[a])for(var c=this._subscribers[a],d=c.length-1;d>=0;d--)c[d].call(null,b)},h.prototype.off=function(a,b){if(this._subscribers[a]){if(b){var c=this._subscribers[a].indexOf(b);this._subscribers[a].splice(c,1)}b&&0!==this._subscribers.length||delete this._subscribers[a]}},h.prototype.once=function(a,b){this.on(a,function(c){this.off(a,b),b(c)}.bind(this))},h.prototype._init=function(a){a||d(h.error.NO_OPTS),a.path?this._msgPath=a.path:d(h.error.NO_PATH)},h.loadFile=function(a){var b=new XMLHttpRequest;b.open("GET",a.url),b.onreadystatechange=function(){4===b.readyState&&(200==b.status||f(a.url)&&b.responseText.length>0?a.success(b.responseText):a.error(b.responseText))},b.send(null)},h.error={NO_OPTS:"Options are not present",NO_PATH:"Options path is not present",NO_MESSAGES:"{{file}} is unreachable. Error {{error}}",EMPTY_MESSAGES:"{{file}} is empty",INVALID_KEY:"{{key}} key is not present in locale {{locale}}",NO_LOCALE_IS_SET:"Locale is not set.",INVALID_PLURAL:"Invalid plural form: {{plural}}"},h.event={LOCALE_LOAD:"localeload",LOCALE_CHANGE:"localechange"},h.KEY_SEPARATOR=".","function"==typeof define&&define.amd?define(function(){return h}):a.I18nText=h}("undefined"==typeof window?this:window); \ No newline at end of file diff --git a/i/js/main.js b/i/js/main.js new file mode 100644 index 0000000..f201015 --- /dev/null +++ b/i/js/main.js @@ -0,0 +1,121 @@ +(function() { + 'use strict'; + + var initialized = false; + var i18n; + + // i18n key prefix for MUC ("muc.") or 1:1 chat ("chat.") + var key_prefix; + var display_data = null; + + function show_clients(client_array) { + var list = document.getElementById('client_list'); + for (var id = 0; id < client_array.length; id++) { + var item = document.createElement('li'); + item.innerHTML = client_array[id]; + list.appendChild(item); + } + } + + function load_clients(url) { + var request = new XMLHttpRequest(); + request.open('GET', url); + request.onreadystatechange = function () { + if (request.readyState === 4) { + if (request.status === 200 || (isLocalFileRequest(url) && request.responseText.length > 0)) { + show_clients(JSON.parse(request.responseText)); + } + } + }; + request.send(null); + } + + function load_hash() { + var muc = false; + key_prefix = "chat."; + var jid = window.location.search || window.location.hash; + jid = decodeURIComponent(jid.substring(jid.indexOf('#') + 1, jid.length)); + + if (jid === "") + jid = "decentralized@conference.moemoekyun.moe?join" + + try { + base_decoded = window.atob(jid); + if (base_decoded.search('@') >= 0) + jid = base_decoded; + } catch (err) { + // ignore error, JID wasn't base64 encoded + } + if (jid.search("\\?join") >= 0) { + muc = true; + key_prefix = "muc."; + } + + // TODO: proper error checking / display / Creation of invitations + if (jid.search("@") <= 0) return {jid: jid, name: jid}; + + var name = jid.split("@")[0]; + name = name.charAt(0).toUpperCase() + name.slice(1); + + return {jid: jid, name: name}; + } + + function translate_ui() { + // translation + document.title = i18n.text(key_prefix + 'title', display_data); + // MUC/chat specific + ['heading', 'button'].forEach(function(id) { + document.getElementById(id).innerHTML = i18n.text(key_prefix + id, display_data); + }); + // and agnostic + ['clients', 'recommend', 'checkfulllist', 'xmppis'].forEach(function(id) { + document.getElementById(id).innerHTML = i18n.text(id, display_data); + }); + } + + function rehash() { + display_data = load_hash(); + document.getElementById('button').href = "xmpp:" + display_data.jid; + document.getElementById('url_in').value = "xmpp:" + display_data.jid; + translate_ui(); + } + + function load_done() { + if (initialized) return; + initialized = true; + + // load i18n and perform translation + i18n = new I18nText({path: 'lang'}); + i18n.once(I18nText.event.LOCALE_CHANGE, function (data) { + rehash(); + }); + i18n.setLocale('ru'); + + // functionality + if (navigator.userAgent.indexOf("Android") >= 0) { + load_clients("json/clients_Android.json") + } + else if (navigator.userAgent.indexOf("Linux") >= 0) { + load_clients("json/clients_Linux.json") + } + else if (navigator.userAgent.indexOf("iPhone") >= 0) { + load_clients("json/clients_iOS.json") + } + else { + load_clients("json/clients_Windows.json") + } + + window.addEventListener("hashchange", rehash, false); + document.getElementById("url_in").addEventListener("focus", function(event) { + event.target.select(); + }); + } + + // Wait for the DOM to be ready + document.addEventListener('DOMContentLoaded', load_done, false); + document.onreadystatechange = function() { + if (document.readyState === 'interactive') { + load_done(); + } + }; +})(); diff --git a/i/json/clients_Android.json b/i/json/clients_Android.json new file mode 100644 index 0000000..49c6069 --- /dev/null +++ b/i/json/clients_Android.json @@ -0,0 +1,3 @@ +[ + "Pix-Art Messenger - современный, полнофункциональный и сфокусированный на удобстве пользования" +] diff --git a/i/json/clients_Linux.json b/i/json/clients_Linux.json new file mode 100644 index 0000000..351c626 --- /dev/null +++ b/i/json/clients_Linux.json @@ -0,0 +1,5 @@ +[ + "Dino - modern, clean and GNOME integrated", + "Kaidan - modern, convergent and cross-platform", + "Gajim - полнофункциональный" +] diff --git a/i/json/clients_Windows.json b/i/json/clients_Windows.json new file mode 100644 index 0000000..0c7240f --- /dev/null +++ b/i/json/clients_Windows.json @@ -0,0 +1,3 @@ +[ + "Gajim - полнофункционый" +] diff --git a/i/json/clients_iOS.json b/i/json/clients_iOS.json new file mode 100644 index 0000000..63e5a29 --- /dev/null +++ b/i/json/clients_iOS.json @@ -0,0 +1,3 @@ +[ + "ChatSecure - Encrypted Messenger for iOS" +] diff --git a/i/lang/ru.json b/i/lang/ru.json new file mode 100644 index 0000000..bf652ad --- /dev/null +++ b/i/lang/ru.json @@ -0,0 +1,19 @@ +{ + "chat": { + "title": "Приглашение от {{name}}", + "heading": "{{name}} хочет связаться с вами", + "button": "Добавить {{name}} в список контактов", + "":"" + }, + "muc": { + "title": "Приглашение в {{name}}", + "heading": "Вы были приглашены в {{name}}", + "button": "Вступить в групповой чат {{name}}", + "":"" + }, + "clients": "Чтобы это произошло, вам нужно установить и настроить джаббер клиент, после этого посетить эту страницу снова.", + "recommend": "Мы рекомендуем один из:", + "checkfulllist": "Джаббер (XMPP) - это удобная и безопасная форма обмена сообщениями. Вы можете выбрать один из множества клиентов и иметь свободный выбор любого сервера для связи со всеми.", + "xmppis": "Основано на i.kaidan.im", + "":"" +}