'use strict'; angular.module('copayApp.controllers').controller('txpDetailsController', function($scope, $rootScope, $timeout, $interval, $ionicModal, $log, ongoingProcess, platformInfo, $ionicScrollDelegate, txFormatService, fingerprintService, bwcError, gettextCatalog, lodash, walletService, popupService, $state, $ionicHistory) { var self = $scope.self; var tx = $scope.tx; var copayers = $scope.copayers; var isGlidera = $scope.isGlidera; var GLIDERA_LOCK_TIME = 6 * 60 * 60; var now = Math.floor(Date.now() / 1000); var countDown; $scope.init = function() { $scope.loading = null; $scope.isCordova = platformInfo.isCordova; $scope.copayerId = $scope.wallet.credentials.copayerId; $scope.isShared = $scope.wallet.credentials.n > 1; $scope.canSign = $scope.wallet.canSign() || $scope.wallet.isPrivKeyExternal(); $scope.color = $scope.wallet.color; $scope.data = {}; $scope.hasClick = platformInfo.hasClick; initActionList(); checkPaypro(); } function initActionList() { $scope.actionList = []; if (!$scope.isShared) return; var actionDescriptions = { created: gettextCatalog.getString('Proposal Created'), accept: gettextCatalog.getString('Accepted'), reject: gettextCatalog.getString('Rejected'), broadcasted: gettextCatalog.getString('Broadcasted'), }; $scope.actionList.push({ type: 'created', time: tx.createdOn, description: actionDescriptions['created'], by: tx.creatorName }); lodash.each(tx.actions, function(action) { $scope.actionList.push({ type: action.type, time: action.createdOn, description: actionDescriptions[action.type], by: action.copayerName }); }); $scope.actionList.reverse(); }; $scope.$on('accepted', function(event) { $scope.sign(); }); // ToDo: use tx.customData instead of tx.message if (tx.message === 'Glidera transaction' && isGlidera) { tx.isGlidera = true; if (tx.canBeRemoved) { tx.canBeRemoved = (Date.now() / 1000 - (tx.ts || tx.createdOn)) > GLIDERA_LOCK_TIME; } } var setSendError = function(msg) { $scope.sendStatus = ''; var error = msg || gettextCatalog.getString('Could not send payment'); popupService.showAlert(gettextCatalog.getString('Error'), error); } $scope.sign = function(onSendStatusChange) { $scope.loading = true; walletService.publishAndSign($scope.wallet, $scope.tx, function(err, txp) { $scope.$emit('UpdateTx'); if (err) return setSendError(err); success(); }, onSendStatusChange); }; function setError(err, prefix) { $scope.loading = false; popupService.showAlert(gettextCatalog.getString('Error'), bwcError.msg(err, prefix)); }; $scope.reject = function(txp) { $scope.loading = true; walletService.reject($scope.wallet, $scope.tx, function(err, txpr) { if (err) return setError(err, gettextCatalog.getString('Could not reject payment')); $scope.close(); }); }; $scope.remove = function() { $scope.loading = true; $timeout(function() { ongoingProcess.set('removeTx', true); walletService.removeTx($scope.wallet, $scope.tx, function(err) { ongoingProcess.set('removeTx', false); // Hacky: request tries to parse an empty response if (err && !(err.message && err.message.match(/Unexpected/))) { $scope.$emit('UpdateTx'); return setError(err, gettextCatalog.getString('Could not delete payment proposal')); } $scope.close(); }); }, 10); }; $scope.broadcast = function(txp) { $scope.loading = true; $timeout(function() { ongoingProcess.set('broadcastTx', true); walletService.broadcastTx($scope.wallet, $scope.tx, function(err, txpb) { ongoingProcess.set('broadcastTx', false); if (err) { return setError(err, gettextCatalog.getString('Could not broadcast payment')); } $scope.close(); }); }, 10); }; $scope.getShortNetworkName = function() { return $scope.wallet.credentials.networkName.substring(0, 4); }; function checkPaypro() { if (tx.payProUrl && !platformInfo.isChromeApp) { $scope.wallet.fetchPayPro({ payProUrl: tx.payProUrl, }, function(err, paypro) { if (err) return; tx.paypro = paypro; paymentTimeControl(tx.paypro.expires); $timeout(function() { $ionicScrollDelegate.resize(); }, 10); }); } }; function paymentTimeControl(expirationTime) { $scope.paymentExpired = false; setExpirationTime(); countDown = $interval(function() { setExpirationTime(); }, 1000); function setExpirationTime() { var now = Math.floor(Date.now() / 1000); if (now > expirationTime) { $scope.paymentExpired = true; if (countDown) $interval.cancel(countDown); return; } var totalSecs = expirationTime - now; var m = Math.floor(totalSecs / 60); var s = totalSecs % 60; $scope.expires = ('0' + m).slice(-2) + ":" + ('0' + s).slice(-2); }; }; lodash.each(['TxProposalRejectedBy', 'TxProposalAcceptedBy', 'transactionProposalRemoved', 'TxProposalRemoved', 'NewOutgoingTx', 'UpdateTx'], function(eventName) { $rootScope.$on(eventName, function() { $scope.wallet.getTx($scope.tx.id, function(err, tx) { if (err) { if (err.message && err.message == 'TX_NOT_FOUND' && (eventName == 'transactionProposalRemoved' || eventName == 'TxProposalRemoved')) { $scope.tx.removed = true; $scope.tx.canBeRemoved = false; $scope.tx.pendingForUs = false; $scope.$apply(); } return; } var action = lodash.find(tx.actions, { copayerId: $scope.wallet.credentials.copayerId }); $scope.tx = txFormatService.processTx(tx); if (!action && tx.status == 'pending') $scope.tx.pendingForUs = true; $scope.updateCopayerList(); $scope.$apply(); }); }); }); $scope.updateCopayerList = function() { lodash.map($scope.copayers, function(cp) { lodash.each($scope.tx.actions, function(ac) { if (cp.id == ac.copayerId) { cp.action = ac.type; } }); }); }; function statusChangeHandler(processName, showName, isOn) { $log.debug('statusChangeHandler: ', processName, showName, isOn); if (showName) { $scope.sendStatus = showName; } } function success() { $scope.sendStatus = 'success'; $scope.$digest(); } $scope.statusChangeHandler = statusChangeHandler; $scope.onConfirm = function() { $scope.sign(statusChangeHandler); }; $scope.onSuccessConfirm = function() { $ionicHistory.nextViewOptions({ disableAnimate: true }); $scope.close(); }; $scope.close = function() { $scope.loading = null; $scope.txpDetailsModal.hide(); }; });