|
|
// Load #1, load as little as possible because we are in a race to get the loading screen up.
|
|
|
define([
|
|
|
'/bower_components/nthen/index.js',
|
|
|
'/api/config',
|
|
|
'jquery',
|
|
|
], function (nThen, ApiConfig, $) {
|
|
|
var common = {};
|
|
|
|
|
|
common.start = function (cfg) {
|
|
|
cfg = cfg || {};
|
|
|
var realtime = !cfg.noRealtime;
|
|
|
var secret;
|
|
|
var hashes;
|
|
|
var CpNfOuter;
|
|
|
var Cryptpad;
|
|
|
var Crypto;
|
|
|
var Cryptget;
|
|
|
var sframeChan;
|
|
|
var FilePicker;
|
|
|
|
|
|
nThen(function (waitFor) {
|
|
|
// Load #2, the loading screen is up so grab whatever you need...
|
|
|
require([
|
|
|
'/common/sframe-chainpad-netflux-outer.js',
|
|
|
'/common/cryptpad-common.js',
|
|
|
'/bower_components/chainpad-crypto/crypto.js',
|
|
|
'/common/cryptget.js',
|
|
|
'/common/sframe-channel.js',
|
|
|
'/filepicker/main.js',
|
|
|
], waitFor(function (_CpNfOuter, _Cryptpad, _Crypto, _Cryptget, SFrameChannel,
|
|
|
_FilePicker) {
|
|
|
CpNfOuter = _CpNfOuter;
|
|
|
Cryptpad = _Cryptpad;
|
|
|
Crypto = _Crypto;
|
|
|
Cryptget = _Cryptget;
|
|
|
FilePicker = _FilePicker;
|
|
|
|
|
|
if (localStorage.CRYPTPAD_URLARGS !== ApiConfig.requireConf.urlArgs) {
|
|
|
console.log("New version, flushing cache");
|
|
|
Object.keys(localStorage).forEach(function (k) {
|
|
|
if (k.indexOf('CRYPTPAD_CACHE|') !== 0) { return; }
|
|
|
delete localStorage[k];
|
|
|
});
|
|
|
localStorage.CRYPTPAD_URLARGS = ApiConfig.requireConf.urlArgs;
|
|
|
}
|
|
|
var cache = {};
|
|
|
var localStore = {};
|
|
|
Object.keys(localStorage).forEach(function (k) {
|
|
|
if (k.indexOf('CRYPTPAD_CACHE|') === 0) {
|
|
|
cache[k.slice(('CRYPTPAD_CACHE|').length)] = localStorage[k];
|
|
|
return;
|
|
|
}
|
|
|
if (k.indexOf('CRYPTPAD_STORE|') === 0) {
|
|
|
localStore[k.slice(('CRYPTPAD_STORE|').length)] = localStorage[k];
|
|
|
return;
|
|
|
}
|
|
|
});
|
|
|
|
|
|
SFrameChannel.create($('#sbox-iframe')[0].contentWindow, waitFor(function (sfc) {
|
|
|
sframeChan = sfc;
|
|
|
}), false, { cache: cache, localStore: localStore, language: Cryptpad.getLanguage() });
|
|
|
Cryptpad.ready(waitFor());
|
|
|
}));
|
|
|
}).nThen(function (waitFor) {
|
|
|
|
|
|
sframeChan.on('EV_CACHE_PUT', function (x) {
|
|
|
Object.keys(x).forEach(function (k) {
|
|
|
localStorage['CRYPTPAD_CACHE|' + k] = x[k];
|
|
|
});
|
|
|
});
|
|
|
sframeChan.on('EV_LOCALSTORE_PUT', function (x) {
|
|
|
Object.keys(x).forEach(function (k) {
|
|
|
if (typeof(x[k]) === "undefined") {
|
|
|
delete localStorage['CRYPTPAD_STORE|' + k];
|
|
|
return;
|
|
|
}
|
|
|
localStorage['CRYPTPAD_STORE|' + k] = x[k];
|
|
|
});
|
|
|
});
|
|
|
|
|
|
secret = cfg.getSecrets ? cfg.getSecrets(Cryptpad) : Cryptpad.getSecrets();
|
|
|
if (!secret.channel) {
|
|
|
// New pad: create a new random channel id
|
|
|
secret.channel = Cryptpad.createChannelId();
|
|
|
}
|
|
|
Cryptpad.getShareHashes(secret, waitFor(function (err, h) { hashes = h; }));
|
|
|
}).nThen(function () {
|
|
|
var readOnly = secret.keys && !secret.keys.editKeyStr;
|
|
|
if (!secret.keys) { secret.keys = secret.key; }
|
|
|
var parsed = Cryptpad.parsePadUrl(window.location.href);
|
|
|
if (!parsed.type) { throw new Error(); }
|
|
|
var defaultTitle = Cryptpad.getDefaultName(parsed);
|
|
|
var proxy = Cryptpad.getProxy();
|
|
|
var updateMeta = function () {
|
|
|
//console.log('EV_METADATA_UPDATE');
|
|
|
var name;
|
|
|
nThen(function (waitFor) {
|
|
|
Cryptpad.getLastName(waitFor(function (err, n) {
|
|
|
if (err) { console.log(err); }
|
|
|
name = n;
|
|
|
}));
|
|
|
}).nThen(function (/*waitFor*/) {
|
|
|
var metaObj = {
|
|
|
doc: {
|
|
|
defaultTitle: defaultTitle,
|
|
|
type: parsed.type
|
|
|
},
|
|
|
user: {
|
|
|
name: name,
|
|
|
uid: Cryptpad.getUid(),
|
|
|
avatar: Cryptpad.getAvatarUrl(),
|
|
|
profile: Cryptpad.getProfileUrl(),
|
|
|
curvePublic: proxy.curvePublic,
|
|
|
netfluxId: Cryptpad.getNetwork().webChannels[0].myID,
|
|
|
},
|
|
|
priv: {
|
|
|
accountName: Cryptpad.getAccountName(),
|
|
|
origin: window.location.origin,
|
|
|
pathname: window.location.pathname,
|
|
|
fileHost: ApiConfig.fileHost,
|
|
|
readOnly: readOnly,
|
|
|
availableHashes: hashes,
|
|
|
isTemplate: Cryptpad.isTemplate(window.location.href),
|
|
|
feedbackAllowed: Cryptpad.isFeedbackAllowed(),
|
|
|
friends: proxy.friends || {},
|
|
|
settings: proxy.settings || {},
|
|
|
isPresent: parsed.hashData && parsed.hashData.present,
|
|
|
isEmbed: parsed.hashData && parsed.hashData.embed,
|
|
|
}
|
|
|
};
|
|
|
if (cfg.addData) {
|
|
|
cfg.addData(metaObj.priv, Cryptpad);
|
|
|
}
|
|
|
sframeChan.event('EV_METADATA_UPDATE', metaObj);
|
|
|
});
|
|
|
};
|
|
|
Cryptpad.onDisplayNameChanged(updateMeta);
|
|
|
sframeChan.onReg('EV_METADATA_UPDATE', updateMeta);
|
|
|
proxy.on('change', 'settings', updateMeta);
|
|
|
|
|
|
Cryptpad.onError(function (info) {
|
|
|
console.log('error');
|
|
|
console.log(info);
|
|
|
if (info && info.type === "store") {
|
|
|
//onConnectError();
|
|
|
}
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_ANON_RPC_MESSAGE', function (data, cb) {
|
|
|
Cryptpad.anonRpcMsg(data.msg, data.content, function (err, response) {
|
|
|
cb({error: err, response: response});
|
|
|
});
|
|
|
});
|
|
|
|
|
|
var currentTitle;
|
|
|
var currentTabTitle;
|
|
|
var setDocumentTitle = function () {
|
|
|
if (!currentTabTitle) {
|
|
|
document.title = currentTitle || 'CryptPad';
|
|
|
return;
|
|
|
}
|
|
|
var title = currentTabTitle.replace(/\{title\}/g, currentTitle || 'CryptPad');
|
|
|
document.title = title;
|
|
|
};
|
|
|
sframeChan.on('Q_SET_PAD_TITLE_IN_DRIVE', function (newTitle, cb) {
|
|
|
currentTitle = newTitle;
|
|
|
setDocumentTitle();
|
|
|
Cryptpad.renamePad(newTitle, undefined, function (err) {
|
|
|
if (err) { cb('ERROR'); } else { cb(); }
|
|
|
});
|
|
|
});
|
|
|
sframeChan.on('EV_SET_TAB_TITLE', function (newTabTitle) {
|
|
|
currentTabTitle = newTabTitle;
|
|
|
setDocumentTitle();
|
|
|
});
|
|
|
|
|
|
|
|
|
sframeChan.on('Q_SETTINGS_SET_DISPLAY_NAME', function (newName, cb) {
|
|
|
Cryptpad.setDisplayName(newName, function (err) {
|
|
|
if (err) {
|
|
|
console.log("Couldn't set username");
|
|
|
console.error(err);
|
|
|
cb('ERROR');
|
|
|
return;
|
|
|
}
|
|
|
Cryptpad.changeDisplayName(newName, true);
|
|
|
cb();
|
|
|
});
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_LOGOUT', function (data, cb) {
|
|
|
Cryptpad.logout(cb);
|
|
|
});
|
|
|
|
|
|
sframeChan.on('EV_NOTIFY', function () {
|
|
|
Cryptpad.notify();
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_SET_LOGIN_REDIRECT', function (data, cb) {
|
|
|
sessionStorage.redirectTo = window.location.href;
|
|
|
cb();
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_GET_PIN_LIMIT_STATUS', function (data, cb) {
|
|
|
Cryptpad.isOverPinLimit(function (e, overLimit, limits) {
|
|
|
cb({
|
|
|
error: e,
|
|
|
overLimit: overLimit,
|
|
|
limits: limits
|
|
|
});
|
|
|
});
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_MOVE_TO_TRASH', function (data, cb) {
|
|
|
Cryptpad.moveToTrash(cb);
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_SAVE_AS_TEMPLATE', function (data, cb) {
|
|
|
Cryptpad.saveAsTemplate(Cryptget.put, data, cb);
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_SEND_FRIEND_REQUEST', function (netfluxId, cb) {
|
|
|
Cryptpad.inviteFromUserlist(Cryptpad, netfluxId);
|
|
|
cb();
|
|
|
});
|
|
|
Cryptpad.onFriendRequest = function (confirmText, cb) {
|
|
|
sframeChan.query('Q_INCOMING_FRIEND_REQUEST', confirmText, function (err, data) {
|
|
|
cb(data);
|
|
|
});
|
|
|
};
|
|
|
Cryptpad.onFriendComplete = function (data) {
|
|
|
sframeChan.event('EV_FRIEND_REQUEST', data);
|
|
|
};
|
|
|
|
|
|
sframeChan.on('Q_GET_FULL_HISTORY', function (data, cb) {
|
|
|
var network = Cryptpad.getNetwork();
|
|
|
var hkn = network.historyKeeper;
|
|
|
var crypto = Crypto.createEncryptor(secret.keys);
|
|
|
// Get the history messages and send them to the iframe
|
|
|
var parse = function (msg) {
|
|
|
try {
|
|
|
return JSON.parse(msg);
|
|
|
} catch (e) {
|
|
|
return null;
|
|
|
}
|
|
|
};
|
|
|
var msgs = [];
|
|
|
var onMsg = function (msg) {
|
|
|
var parsed = parse(msg);
|
|
|
if (parsed[0] === 'FULL_HISTORY_END') {
|
|
|
cb(msgs);
|
|
|
return;
|
|
|
}
|
|
|
if (parsed[0] !== 'FULL_HISTORY') { return; }
|
|
|
if (parsed[1] && parsed[1].validateKey) { // First message
|
|
|
return;
|
|
|
}
|
|
|
msg = parsed[1][4];
|
|
|
if (msg) {
|
|
|
msg = msg.replace(/^cp\|/, '');
|
|
|
var decryptedMsg = crypto.decrypt(msg, secret.keys.validateKey);
|
|
|
msgs.push(decryptedMsg);
|
|
|
}
|
|
|
};
|
|
|
network.on('message', onMsg);
|
|
|
network.sendto(hkn, JSON.stringify(['GET_FULL_HISTORY', secret.channel, secret.keys.validateKey]));
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_GET_PAD_ATTRIBUTE', function (data, cb) {
|
|
|
Cryptpad.getPadAttribute(data.key, function (e, data) {
|
|
|
cb({
|
|
|
error: e,
|
|
|
data: data
|
|
|
});
|
|
|
});
|
|
|
});
|
|
|
sframeChan.on('Q_SET_PAD_ATTRIBUTE', function (data, cb) {
|
|
|
Cryptpad.setPadAttribute(data.key, data.value, function (e) {
|
|
|
cb({error:e});
|
|
|
});
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_GET_ATTRIBUTE', function (data, cb) {
|
|
|
Cryptpad.getAttribute(data.key, function (e, data) {
|
|
|
cb({
|
|
|
error: e,
|
|
|
data: data
|
|
|
});
|
|
|
});
|
|
|
});
|
|
|
sframeChan.on('Q_SET_ATTRIBUTE', function (data, cb) {
|
|
|
Cryptpad.setAttribute(data.key, data.value, function (e) {
|
|
|
cb({error:e});
|
|
|
});
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_SESSIONSTORAGE_PUT', function (data, cb) {
|
|
|
sessionStorage[data.key] = data.value;
|
|
|
cb();
|
|
|
});
|
|
|
|
|
|
|
|
|
// Present mode URL
|
|
|
sframeChan.on('Q_PRESENT_URL_GET_VALUE', function (data, cb) {
|
|
|
var parsed = Cryptpad.parsePadUrl(window.location.href);
|
|
|
cb(parsed.hashData && parsed.hashData.present);
|
|
|
});
|
|
|
sframeChan.on('EV_PRESENT_URL_SET_VALUE', function (data) {
|
|
|
var parsed = Cryptpad.parsePadUrl(window.location.href);
|
|
|
window.location.href = parsed.getUrl({
|
|
|
embed: parsed.hashData.embed,
|
|
|
present: data
|
|
|
});
|
|
|
});
|
|
|
|
|
|
|
|
|
// File upload
|
|
|
var onFileUpload = function (sframeChan, data, cb) {
|
|
|
var sendEvent = function (data) {
|
|
|
sframeChan.event("EV_FILE_UPLOAD_STATE", data);
|
|
|
};
|
|
|
var updateProgress = function (progressValue) {
|
|
|
sendEvent({
|
|
|
progress: progressValue
|
|
|
});
|
|
|
};
|
|
|
var onComplete = function (href) {
|
|
|
sendEvent({
|
|
|
complete: true,
|
|
|
href: href
|
|
|
});
|
|
|
};
|
|
|
var onError = function (e) {
|
|
|
sendEvent({
|
|
|
error: e
|
|
|
});
|
|
|
};
|
|
|
var onPending = function (cb) {
|
|
|
sframeChan.query('Q_CANCEL_PENDING_FILE_UPLOAD', null, function (err, data) {
|
|
|
if (data) {
|
|
|
cb();
|
|
|
}
|
|
|
});
|
|
|
};
|
|
|
data.blob = Crypto.Nacl.util.decodeBase64(data.blob);
|
|
|
Cryptpad.uploadFileSecure(data, data.noStore, Cryptpad, updateProgress, onComplete, onError, onPending);
|
|
|
cb();
|
|
|
};
|
|
|
sframeChan.on('Q_UPLOAD_FILE', function (data, cb) {
|
|
|
onFileUpload(sframeChan, data, cb);
|
|
|
});
|
|
|
|
|
|
// File picker
|
|
|
var FP = {};
|
|
|
var initFilePicker = function (types) {
|
|
|
var config = {};
|
|
|
config.onFilePicked = function (data) {
|
|
|
sframeChan.event('EV_FILE_PICKED', data);
|
|
|
};
|
|
|
config.onClose = function () {
|
|
|
FP.$iframe.hide();
|
|
|
};
|
|
|
config.onFileUpload = onFileUpload;
|
|
|
config.types = types;
|
|
|
if (!FP.$iframe) {
|
|
|
FP.$iframe = $('<iframe>', {id: 'sbox-filePicker-iframe'}).appendTo($('body'));
|
|
|
FP.picker = FilePicker.create(config);
|
|
|
} else {
|
|
|
FP.$iframe.show();
|
|
|
FP.picker.refresh(types);
|
|
|
}
|
|
|
FP.$iframe.focus();
|
|
|
};
|
|
|
sframeChan.on('EV_FILE_PICKER_OPEN', function (data) {
|
|
|
initFilePicker(data);
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_TEMPLATE_USE', function (href, cb) {
|
|
|
Cryptpad.useTemplate(href, Cryptget, cb);
|
|
|
});
|
|
|
sframeChan.on('Q_TEMPLATE_EXIST', function (type, cb) {
|
|
|
var hasTemplate = Cryptpad.listTemplates(type).length > 0;
|
|
|
cb(hasTemplate);
|
|
|
});
|
|
|
|
|
|
sframeChan.on('EV_GOTO_URL', function (url) {
|
|
|
if (url) {
|
|
|
window.location.href = url;
|
|
|
} else {
|
|
|
window.location.reload();
|
|
|
}
|
|
|
});
|
|
|
|
|
|
sframeChan.on('EV_OPEN_URL', function (url) {
|
|
|
if (url) {
|
|
|
window.open(url);
|
|
|
}
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_TAGS_GET', function (data, cb) {
|
|
|
Cryptpad.getPadTags(null, function (err, data) {
|
|
|
cb({
|
|
|
error: err,
|
|
|
data: data
|
|
|
});
|
|
|
});
|
|
|
});
|
|
|
|
|
|
sframeChan.on('EV_TAGS_SET', function (data) {
|
|
|
console.log(data);
|
|
|
Cryptpad.resetTags(null, data);
|
|
|
});
|
|
|
|
|
|
sframeChan.on('Q_PIN_GET_USAGE', function (data, cb) {
|
|
|
Cryptpad.isOverPinLimit(function (err, overLimit, data) {
|
|
|
cb({
|
|
|
error: err,
|
|
|
data: data
|
|
|
});
|
|
|
});
|
|
|
});
|
|
|
|
|
|
if (cfg.addRpc) {
|
|
|
cfg.addRpc(sframeChan, Cryptpad);
|
|
|
}
|
|
|
|
|
|
sframeChan.ready();
|
|
|
|
|
|
Cryptpad.reportAppUsage();
|
|
|
|
|
|
if (!realtime) { return; }
|
|
|
|
|
|
CpNfOuter.start({
|
|
|
sframeChan: sframeChan,
|
|
|
channel: secret.channel,
|
|
|
network: cfg.newNetwork || Cryptpad.getNetwork(),
|
|
|
validateKey: secret.keys.validateKey || undefined,
|
|
|
readOnly: readOnly,
|
|
|
crypto: Crypto.createEncryptor(secret.keys),
|
|
|
onConnect: function (wc) {
|
|
|
if (window.location.hash && window.location.hash !== '#') {
|
|
|
window.location = parsed.getUrl({
|
|
|
present: parsed.hashData.present,
|
|
|
embed: parsed.hashData.embed
|
|
|
});
|
|
|
return;
|
|
|
}
|
|
|
if (readOnly || cfg.noHash) { return; }
|
|
|
Cryptpad.replaceHash(Cryptpad.getEditHashFromKeys(wc.id, secret.keys));
|
|
|
}
|
|
|
});
|
|
|
});
|
|
|
};
|
|
|
|
|
|
return common;
|
|
|
});
|
|
|
|