copay/js/services/controllerUtils.js

348 lines
10 KiB
JavaScript
Raw Normal View History

2014-04-17 07:46:49 -07:00
'use strict';
2014-06-12 13:42:26 -07:00
var bitcore = require('bitcore');
2014-04-17 07:46:49 -07:00
2014-06-03 13:42:36 -07:00
angular.module('copayApp.services')
2014-06-30 14:41:17 -07:00
.factory('controllerUtils', function($rootScope, $sce, $location, notification, $timeout, Socket, video, uriHandler) {
2014-04-23 17:20:44 -07:00
var root = {};
root.getVideoMutedStatus = function(copayer) {
2014-07-07 15:38:45 -07:00
if (!$rootScope.videoInfo) return;
2014-05-07 15:04:36 -07:00
var vi = $rootScope.videoInfo[copayer]
if (!vi) {
return;
}
return vi.muted;
};
2014-04-23 17:20:44 -07:00
root.logout = function() {
2014-06-24 08:57:15 -07:00
Socket.removeAllListeners();
2014-04-23 17:20:44 -07:00
$rootScope.wallet = null;
delete $rootScope['wallet'];
2014-04-28 07:56:27 -07:00
video.close();
// Clear rootScope
for (var i in $rootScope) {
2014-05-20 08:30:20 -07:00
if (i.charAt(0) != '$') {
delete $rootScope[i];
}
}
$location.path('/');
2014-04-23 17:20:44 -07:00
};
2014-04-23 17:20:44 -07:00
root.onError = function(scope) {
if (scope) scope.loading = false;
root.logout();
}
2014-04-23 14:07:20 -07:00
root.onErrorDigest = function(scope, msg) {
2014-04-23 17:20:44 -07:00
root.onError(scope);
2014-06-26 13:17:24 -07:00
if (msg) {
notification.error('Error', msg);
}
2014-04-23 17:20:44 -07:00
$rootScope.$digest();
2014-06-12 07:03:24 -07:00
};
root.installStartupHandlers = function(wallet, $scope) {
wallet.on('connectionError', function() {
2014-06-26 13:17:24 -07:00
var message = "Looks like you are already connected to this wallet, please logout and try importing it again.";
notification.error('PeerJS Error', message);
2014-06-12 07:03:24 -07:00
root.onErrorDigest($scope);
});
2014-07-07 10:18:11 -07:00
wallet.on('serverError', function(m) {
var message = m || 'The PeerJS server is not responding, please try again';
$location.path('addresses');
root.onErrorDigest($scope, message);
2014-06-12 07:03:24 -07:00
});
wallet.on('ready', function() {
$scope.loading = false;
});
};
2014-06-24 08:57:15 -07:00
root.setupRootVariables = function() {
2014-06-30 14:41:17 -07:00
uriHandler.register();
2014-06-24 08:57:15 -07:00
$rootScope.unitName = config.unitName;
$rootScope.txAlertCount = 0;
$rootScope.insightError = 0;
$rootScope.isCollapsed = true;
$rootScope.$watch('txAlertCount', function(txAlertCount) {
if (txAlertCount && txAlertCount > 0) {
2014-07-08 04:58:24 -07:00
2014-06-24 08:57:15 -07:00
notification.info('New Transaction', ($rootScope.txAlertCount == 1) ? 'You have a pending transaction proposal' : 'You have ' + $rootScope.txAlertCount + ' pending transaction proposals', txAlertCount);
}
});
2014-07-08 04:58:24 -07:00
$rootScope.$watch('receivedFund', function(receivedFund) {
if (receivedFund) {
var currentAddr;
for (var i = 0; i < $rootScope.addrInfos.length; i++) {
var addrinfo = $rootScope.addrInfos[i];
if (addrinfo.address.toString() == receivedFund[1] && !addrinfo.isChange) {
currentAddr = addrinfo.address.toString();
break;
}
}
if (currentAddr) {
//var beep = new Audio('sound/transaction.mp3');
notification.funds('Received fund', currentAddr, receivedFund);
//beep.play();
}
}
});
2014-06-24 08:57:15 -07:00
};
2014-06-12 07:03:24 -07:00
root.startNetwork = function(w, $scope) {
2014-06-24 08:57:15 -07:00
Socket.removeAllListeners();
2014-06-12 07:03:24 -07:00
2014-06-24 08:57:15 -07:00
root.setupRootVariables();
2014-06-12 07:03:24 -07:00
root.installStartupHandlers(w, $scope);
2014-06-24 08:57:15 -07:00
root.setSocketHandlers();
2014-04-24 19:13:55 -07:00
2014-04-23 17:20:44 -07:00
var handlePeerVideo = function(err, peerID, url) {
if (err) {
2014-05-07 15:04:36 -07:00
delete $rootScope.videoInfo[peerID];
2014-04-24 14:24:03 -07:00
return;
2014-04-23 17:20:44 -07:00
}
2014-05-07 15:04:36 -07:00
$rootScope.videoInfo[peerID] = {
url: encodeURI(url),
muted: peerID === w.network.peerId
};
2014-04-30 15:50:13 -07:00
$rootScope.$digest();
2014-04-23 17:20:44 -07:00
};
2014-06-18 09:01:50 -07:00
notification.enableHtml5Mode(); // for chrome: if support, enable it
2014-04-23 17:20:44 -07:00
w.on('badMessage', function(peerId) {
2014-06-26 13:17:24 -07:00
notification.error('Error', 'Received wrong message from peer ' + peerId);
2014-04-23 17:20:44 -07:00
});
w.on('ready', function(myPeerID) {
2014-04-23 17:20:44 -07:00
$rootScope.wallet = w;
if ($rootScope.pendingPayment) {
$location.path('send');
} else {
$location.path('addresses');
}
if (!config.disableVideo)
video.setOwnPeer(myPeerID, w, handlePeerVideo);
2014-04-23 17:20:44 -07:00
});
2014-05-16 14:48:17 -07:00
w.on('publicKeyRingUpdated', function(dontDigest) {
2014-06-24 08:57:15 -07:00
root.setSocketHandlers();
if (!dontDigest) {
$rootScope.$digest();
}
2014-04-23 17:20:44 -07:00
});
w.on('txProposalsUpdated', function(dontDigest) {
root.updateTxs({
onlyPending: true
});
2014-06-12 13:42:26 -07:00
// give sometime to the tx to propagate.
$timeout(function() {
root.updateBalance(function() {
2014-06-12 13:42:26 -07:00
if (!dontDigest) {
$rootScope.$digest();
}
});
}, 3000);
});
2014-06-24 08:57:15 -07:00
w.on('txProposalEvent', function(e) {
switch (e.type) {
case 'signed':
var user = w.publicKeyRing.nicknameForCopayer(e.cId);
2014-06-18 09:01:50 -07:00
notification.info('Transaction Update', 'A transaction was signed by ' + user);
break;
case 'rejected':
var user = w.publicKeyRing.nicknameForCopayer(e.cId);
2014-06-18 09:01:50 -07:00
notification.info('Transaction Update', 'A transaction was rejected by ' + user);
break;
}
});
2014-06-18 16:18:13 -07:00
w.on('addressBookUpdated', function(dontDigest) {
if (!dontDigest) {
$rootScope.$digest();
}
});
w.on('connectionError', function(msg) {
root.onErrorDigest(null, msg);
});
2014-05-09 10:35:57 -07:00
w.on('connect', function(peerID) {
if (peerID && !config.disableVideo) {
2014-05-09 10:35:57 -07:00
video.callPeer(peerID, handlePeerVideo);
}
$rootScope.$digest();
2014-04-23 17:20:44 -07:00
});
2014-05-09 10:44:05 -07:00
w.on('disconnect', function(peerID) {
$rootScope.$digest();
});
2014-04-23 17:20:44 -07:00
w.on('close', root.onErrorDigest);
w.netStart();
};
root.updateAddressList = function() {
var w = $rootScope.wallet;
2014-06-25 07:12:38 -07:00
if (w)
$rootScope.addrInfos = w.getAddressesInfo();
};
2014-05-15 13:43:41 -07:00
root.updateBalance = function(cb) {
2014-04-23 17:20:44 -07:00
var w = $rootScope.wallet;
2014-06-05 08:18:54 -07:00
if (!w) return root.onErrorDigest();
$rootScope.balanceByAddr = {};
$rootScope.updatingBalance = true;
2014-06-13 15:45:00 -07:00
w.getBalance(function(err, balanceSat, balanceByAddrSat, safeBalanceSat) {
if (err) {
console.error('Error: ' + err.message); //TODO
2014-06-05 08:18:54 -07:00
root._setCommError();
return null;
} else {
2014-06-05 08:18:54 -07:00
root._clearCommError();
}
var satToUnit = 1 / config.unitToSatoshi;
var COIN = bitcore.util.COIN;
$rootScope.totalBalance = balanceSat * satToUnit;
$rootScope.totalBalanceBTC = (balanceSat / COIN);
$rootScope.availableBalance = safeBalanceSat * satToUnit;
$rootScope.availableBalanceBTC = (safeBalanceSat / COIN);
var balanceByAddr = {};
for (var ii in balanceByAddrSat) {
balanceByAddr[ii] = balanceByAddrSat[ii] * satToUnit;
}
2014-06-12 13:42:26 -07:00
$rootScope.balanceByAddr = balanceByAddr;
root.updateAddressList();
$rootScope.updatingBalance = false;
return cb ? cb() : null;
2014-04-21 09:16:15 -07:00
});
2014-04-23 17:20:44 -07:00
};
2014-04-17 07:46:49 -07:00
root.updateTxs = function(opts) {
var w = $rootScope.wallet;
if (!w) return;
opts = opts || {};
var satToUnit = 1 / config.unitToSatoshi;
var myCopayerId = w.getMyCopayerId();
var pendingForUs = 0;
var inT = w.getTxProposals().sort(function(t1, t2) {
return t2.createdTs - t1.createdTs
});
var txs = [];
inT.forEach(function(i, index) {
if (opts.skip && (index < opts.skip[0] || index >= opts.skip[1])) {
return txs.push(null);
}
if (myCopayerId != i.creator && !i.finallyRejected && !i.sentTs && !i.rejectedByUs && !i.signedByUs) {
pendingForUs++;
}
if (!i.finallyRejected && !i.sentTs) {
i.isPending = 1;
}
if (!opts.onlyPending || i.isPending) {
var tx = i.builder.build();
var outs = [];
tx.outs.forEach(function(o) {
var addr = bitcore.Address.fromScriptPubKey(o.getScript(), config.networkName)[0].toString();
if (!w.addressIsOwn(addr, {
excludeMain: true
})) {
outs.push({
address: addr,
value: bitcore.util.valueToBigInt(o.getValue()) * satToUnit,
});
}
});
// extra fields
i.outs = outs;
i.fee = i.builder.feeSat * satToUnit;
i.missingSignatures = tx.countInputMissingSignatures(0);
txs.push(i);
}
});
$rootScope.txs = txs; //.some(function(i) {return i.isPending; } );
if ($rootScope.pendingTxCount < pendingForUs) {
$rootScope.txAlertCount = pendingForUs;
}
$rootScope.pendingTxCount = pendingForUs;
};
2014-06-30 08:00:14 -07:00
$rootScope.$watch('insightError', function(status) {
if (status) {
if (status === -1) {
notification.success('Networking restored', 'Connection to Insight re-established');
} else if (!isNaN(status)) {
notification.error('Networking problem', 'Connection to Insight lost, reconnecting (attempt number ' + status + ')');
}
}
});
root._setCommError = function(e) {
if ($rootScope.insightError < 0)
$rootScope.insightError = 0;
2014-06-05 08:18:54 -07:00
$rootScope.insightError++;
};
root._clearCommError = function(e) {
if ($rootScope.insightError > 0)
$rootScope.insightError = -1;
2014-06-05 10:55:19 -07:00
else
$rootScope.insightError = 0;
};
2014-04-23 17:20:44 -07:00
root.setSocketHandlers = function() {
2014-06-25 07:12:38 -07:00
root.updateAddressList();
if (!Socket.sysEventsSet) {
Socket.sysOn('error', root._setCommError);
Socket.sysOn('reconnect_error', root._setCommError);
Socket.sysOn('reconnect_failed', root._setCommError);
Socket.sysOn('connect', root._clearCommError);
Socket.sysOn('reconnect', root._clearCommError);
Socket.sysEventsSet = true;
}
2014-04-30 10:28:33 -07:00
if (!$rootScope.wallet) return;
2014-04-18 15:08:01 -07:00
var currentAddrs = Socket.getListeners();
2014-06-24 08:57:15 -07:00
var allAddrs = $rootScope.addrInfos;
var newAddrs = [];
2014-06-24 08:57:15 -07:00
for (var i in allAddrs) {
var a = allAddrs[i];
if (!currentAddrs[a.addressStr])
newAddrs.push(a);
}
for (var i = 0; i < newAddrs.length; i++) {
2014-06-24 08:57:15 -07:00
Socket.emit('subscribe', newAddrs[i].addressStr);
2014-04-23 17:20:44 -07:00
}
2014-06-24 08:57:15 -07:00
newAddrs.forEach(function(a) {
Socket.on(a.addressStr, function(txid) {
if (!a.isChange)
2014-06-25 07:12:38 -07:00
notification.funds('Funds received!', a.addressStr);
2014-06-24 08:57:15 -07:00
root.updateBalance(function() {
$rootScope.$digest();
});
2014-04-23 17:20:44 -07:00
});
});
2014-06-23 10:57:23 -07:00
if (!$rootScope.wallet.spendUnconfirmed && !Socket.isListeningBlocks()) {
Socket.emit('subscribe', 'inv');
Socket.on('block', function(block) {
root.updateBalance(function() {
2014-06-24 08:57:15 -07:00
$rootScope.$digest();
2014-06-23 10:57:23 -07:00
});
});
}
2014-04-23 17:20:44 -07:00
};
return root;
});