378 lines
10 KiB
JavaScript
378 lines
10 KiB
JavaScript
'use strict';
|
||
|
||
angular.module('copayApp.controllers').controller('walletDetailsController', function($scope, $rootScope, $interval, $timeout, $filter, $log, $ionicModal, $ionicPopover, $state, $stateParams, $ionicHistory, profileService, lodash, configService, gettextCatalog, platformInfo, walletService, txpModalService, externalLinkService, popupService, addressbookService, storageService, $ionicScrollDelegate, $window) {
|
||
|
||
var HISTORY_SHOW_LIMIT = 10;
|
||
var currentTxHistoryPage = 0;
|
||
var listeners = [];
|
||
$scope.txps = [];
|
||
$scope.completeTxHistory = [];
|
||
$scope.openTxpModal = txpModalService.open;
|
||
$scope.isCordova = platformInfo.isCordova;
|
||
$scope.isAndroid = platformInfo.isAndroid;
|
||
$scope.isIOS = platformInfo.isIOS;
|
||
|
||
$scope.amountIsCollapsible = !$scope.isAndroid;
|
||
|
||
$scope.openExternalLink = function(url, target) {
|
||
externalLinkService.open(url, target);
|
||
};
|
||
|
||
var setPendingTxps = function(txps) {
|
||
|
||
/* Uncomment to test multiple outputs */
|
||
|
||
// var txp = {
|
||
// message: 'test multi-output',
|
||
// fee: 1000,
|
||
// createdOn: new Date() / 1000,
|
||
// outputs: [],
|
||
// wallet: $scope.wallet
|
||
// };
|
||
//
|
||
// function addOutput(n) {
|
||
// txp.outputs.push({
|
||
// amount: 600,
|
||
// toAddress: '2N8bhEwbKtMvR2jqMRcTCQqzHP6zXGToXcK',
|
||
// message: 'output #' + (Number(n) + 1)
|
||
// });
|
||
// };
|
||
// lodash.times(15, addOutput);
|
||
// txps.push(txp);
|
||
|
||
if (!txps) {
|
||
$scope.txps = [];
|
||
return;
|
||
}
|
||
$scope.txps = lodash.sortBy(txps, 'createdOn').reverse();
|
||
};
|
||
|
||
var updateStatus = function(force) {
|
||
$scope.updatingStatus = true;
|
||
$scope.updateStatusError = null;
|
||
$scope.walletNotRegistered = false;
|
||
|
||
walletService.getStatus($scope.wallet, {
|
||
force: !!force,
|
||
}, function(err, status) {
|
||
$scope.updatingStatus = false;
|
||
if (err) {
|
||
if (err === 'WALLET_NOT_REGISTERED') {
|
||
$scope.walletNotRegistered = true;
|
||
} else {
|
||
$scope.updateStatusError = err;
|
||
}
|
||
$scope.status = null;
|
||
} else {
|
||
setPendingTxps(status.pendingTxps);
|
||
$scope.status = status;
|
||
}
|
||
$timeout(function() {
|
||
$scope.$apply();
|
||
});
|
||
|
||
});
|
||
};
|
||
|
||
$scope.openSearchModal = function() {
|
||
$scope.color = $scope.wallet.color;
|
||
|
||
$ionicModal.fromTemplateUrl('views/modals/search.html', {
|
||
scope: $scope,
|
||
focusFirstInput: true
|
||
}).then(function(modal) {
|
||
$scope.searchModal = modal;
|
||
$scope.searchModal.show();
|
||
});
|
||
|
||
$scope.close = function() {
|
||
$scope.searchModal.hide();
|
||
};
|
||
|
||
$scope.openTx = function(tx) {
|
||
$ionicHistory.nextViewOptions({
|
||
disableAnimate: true
|
||
});
|
||
$scope.searchModal.hide();
|
||
$scope.openTxModal(tx);
|
||
};
|
||
};
|
||
|
||
$scope.openTxModal = function(btx) {
|
||
$scope.btx = lodash.cloneDeep(btx);
|
||
$scope.walletId = $scope.wallet.id;
|
||
$state.transitionTo('tabs.wallet.tx-details', {
|
||
txid: $scope.btx.txid,
|
||
walletId: $scope.walletId
|
||
});
|
||
};
|
||
|
||
$scope.recreate = function() {
|
||
walletService.recreate($scope.wallet, function(err) {
|
||
if (err) return;
|
||
$timeout(function() {
|
||
walletService.startScan($scope.wallet, function() {
|
||
$scope.updateAll();
|
||
$scope.$apply();
|
||
});
|
||
});
|
||
});
|
||
};
|
||
|
||
var updateTxHistory = function(cb) {
|
||
if (!cb) cb = function() {};
|
||
if ($scope.updatingTxHistory) return;
|
||
|
||
$scope.updatingTxHistory = true;
|
||
$scope.updateTxHistoryError = false;
|
||
$scope.updatingTxHistoryProgress = 0;
|
||
|
||
var progressFn = function(txs, newTxs) {
|
||
$scope.updatingTxHistoryProgress = newTxs;
|
||
$scope.completeTxHistory = txs;
|
||
$scope.showHistory();
|
||
$timeout(function() {
|
||
$scope.$apply();
|
||
});
|
||
};
|
||
|
||
$timeout(function() {
|
||
walletService.getTxHistory($scope.wallet, {
|
||
progressFn: progressFn,
|
||
}, function(err, txHistory) {
|
||
$scope.updatingTxHistory = false;
|
||
if (err) {
|
||
$scope.txHistory = null;
|
||
$scope.updateTxHistoryError = true;
|
||
return;
|
||
}
|
||
$scope.completeTxHistory = txHistory;
|
||
$scope.showHistory();
|
||
$scope.$apply();
|
||
return cb();
|
||
});
|
||
});
|
||
};
|
||
|
||
$scope.showHistory = function() {
|
||
if ($scope.completeTxHistory) {
|
||
$scope.txHistory = $scope.completeTxHistory.slice(0, (currentTxHistoryPage + 1) * HISTORY_SHOW_LIMIT);
|
||
$scope.txHistoryShowMore = $scope.completeTxHistory.length > $scope.txHistory.length;
|
||
}
|
||
};
|
||
|
||
$scope.getDate = function(txCreated) {
|
||
var date = new Date(txCreated * 1000);
|
||
return date;
|
||
};
|
||
|
||
$scope.isFirstInGroup = function(index) {
|
||
if (index === 0) {
|
||
return true;
|
||
}
|
||
var curTx = $scope.txHistory[index];
|
||
var prevTx = $scope.txHistory[index - 1];
|
||
return !createdDuringSameMonth(curTx, prevTx);
|
||
};
|
||
|
||
$scope.isLastInGroup = function(index) {
|
||
if (index === $scope.txHistory.length - 1) {
|
||
return true;
|
||
}
|
||
return $scope.isFirstInGroup(index + 1);
|
||
};
|
||
|
||
function createdDuringSameMonth(tx1, tx2) {
|
||
var date1 = new Date(tx1.time * 1000);
|
||
var date2 = new Date(tx2.time * 1000);
|
||
return getMonthYear(date1) === getMonthYear(date2);
|
||
}
|
||
|
||
$scope.createdWithinPastDay = function(time) {
|
||
var now = new Date();
|
||
var date = new Date(time * 1000);
|
||
return (now.getTime() - date.getTime()) < (1000 * 60 * 60 * 24);
|
||
};
|
||
|
||
$scope.isDateInCurrentMonth = function(date) {
|
||
var now = new Date();
|
||
return getMonthYear(now) === getMonthYear(date);
|
||
};
|
||
|
||
function getMonthYear(date) {
|
||
return date.getMonth() + date.getFullYear();
|
||
}
|
||
|
||
$scope.isUnconfirmed = function(tx) {
|
||
return !tx.confirmations || tx.confirmations === 0;
|
||
};
|
||
|
||
$scope.showMore = function() {
|
||
$timeout(function() {
|
||
currentTxHistoryPage++;
|
||
$scope.showHistory();
|
||
$scope.$broadcast('scroll.infiniteScrollComplete');
|
||
}, 100);
|
||
};
|
||
|
||
$scope.onRefresh = function() {
|
||
$timeout(function() {
|
||
$scope.$broadcast('scroll.refreshComplete');
|
||
}, 300);
|
||
$scope.updateAll(true);
|
||
};
|
||
|
||
$scope.updateAll = function(force, cb) {
|
||
updateStatus(force);
|
||
updateTxHistory(cb);
|
||
};
|
||
|
||
$scope.hideToggle = function() {
|
||
profileService.toggleHideBalanceFlag($scope.wallet.credentials.walletId, function(err) {
|
||
if (err) $log.error(err);
|
||
});
|
||
};
|
||
|
||
var prevPos;
|
||
var screenInactive = true;
|
||
|
||
function getScrollPosition() {
|
||
var scrollPosition = $ionicScrollDelegate.getScrollPosition();
|
||
if (!scrollPosition || screenInactive) {
|
||
$window.requestAnimationFrame(function() {
|
||
getScrollPosition();
|
||
});
|
||
return;
|
||
}
|
||
var pos = scrollPosition.top;
|
||
if (pos === prevPos) {
|
||
$window.requestAnimationFrame(function() {
|
||
getScrollPosition();
|
||
});
|
||
return;
|
||
}
|
||
prevPos = pos;
|
||
var amountHeight = 180 - pos;
|
||
if (amountHeight < 80) {
|
||
amountHeight = 80;
|
||
}
|
||
var contentMargin = amountHeight;
|
||
if (contentMargin > 180) {
|
||
contentMargin = 180;
|
||
}
|
||
|
||
var amountScale = (amountHeight / 180);
|
||
if (amountScale < 0.5) {
|
||
amountScale = 0.5;
|
||
}
|
||
if (amountScale > 1.1) {
|
||
amountScale = 1.1;
|
||
}
|
||
|
||
var s = amountScale;
|
||
|
||
$scope.altAmountOpacity = (amountHeight - 100) / 80;
|
||
$window.requestAnimationFrame(function() {
|
||
$scope.amountHeight = amountHeight + 'px';
|
||
$scope.contentMargin = contentMargin + 'px';
|
||
$scope.amountScale = 'scale3d(' + s + ',' + s + ',' + s + ')';
|
||
$scope.$digest();
|
||
getScrollPosition();
|
||
});
|
||
}
|
||
|
||
var scrollWatcherInitialized;
|
||
|
||
$scope.$on("$ionicView.enter", function(event, data) {
|
||
setAndroidStatusBarColor();
|
||
$timeout(function() {
|
||
screenInactive = false;
|
||
}, 200);
|
||
if (scrollWatcherInitialized || !$scope.amountIsCollapsible) {
|
||
return;
|
||
}
|
||
scrollWatcherInitialized = true;
|
||
$timeout(function() {
|
||
getScrollPosition();
|
||
}, 100);
|
||
});
|
||
|
||
$scope.$on("$ionicView.beforeEnter", function(event, data) {
|
||
$scope.walletId = data.stateParams.walletId;
|
||
$scope.wallet = profileService.getWallet($scope.walletId);
|
||
$scope.requiresMultipleSignatures = $scope.wallet.credentials.m > 1;
|
||
|
||
addressbookService.list(function(err, ab) {
|
||
if (err) $log.error(err);
|
||
$scope.addressbook = ab || {};
|
||
});
|
||
|
||
$scope.updateAll();
|
||
|
||
listeners = [
|
||
$rootScope.$on('bwsEvent', function(e, walletId) {
|
||
if (walletId == $scope.wallet.id)
|
||
updateStatus();
|
||
}),
|
||
$rootScope.$on('Local/TxAction', function(e, walletId) {
|
||
if (walletId == $scope.wallet.id)
|
||
updateStatus();
|
||
}),
|
||
];
|
||
});
|
||
|
||
$scope.$on("$ionicView.beforeLeave", function(event, data) {
|
||
if($window.StatusBar) {
|
||
$window.StatusBar.backgroundColorByHexString('#1e3186');
|
||
}
|
||
});
|
||
|
||
$scope.$on("$ionicView.leave", function(event, data) {
|
||
screenInactive = true;
|
||
lodash.each(listeners, function(x) {
|
||
x();
|
||
});
|
||
});
|
||
|
||
function setAndroidStatusBarColor() {
|
||
var SUBTRACT_AMOUNT = 15;
|
||
if(!$scope.isAndroid) {
|
||
return;
|
||
}
|
||
var rgb = hexToRgb($scope.wallet.color);
|
||
var keys = Object.keys(rgb);
|
||
keys.forEach(function(k) {
|
||
if(rgb[k] - SUBTRACT_AMOUNT < 0) {
|
||
rgb[k] = 0;
|
||
} else {
|
||
rgb[k] -= SUBTRACT_AMOUNT;
|
||
}
|
||
});
|
||
var statusBarColorHexString = rgbToHex(rgb.r, rgb.g, rgb.b);
|
||
$window.StatusBar.backgroundColorByHexString(statusBarColorHexString);
|
||
}
|
||
|
||
function hexToRgb(hex) {
|
||
// Expand shorthand form (e.g. "03F") to full form (e.g. "0033FF")
|
||
var shorthandRegex = /^#?([a-f\d])([a-f\d])([a-f\d])$/i;
|
||
hex = hex.replace(shorthandRegex, function(m, r, g, b) {
|
||
return r + r + g + g + b + b;
|
||
});
|
||
|
||
var result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
|
||
return result ? {
|
||
r: parseInt(result[1], 16),
|
||
g: parseInt(result[2], 16),
|
||
b: parseInt(result[3], 16)
|
||
} : null;
|
||
}
|
||
|
||
function componentToHex(c) {
|
||
var hex = c.toString(16);
|
||
return hex.length == 1 ? "0" + hex : hex;
|
||
}
|
||
|
||
function rgbToHex(r, g, b) {
|
||
return "#" + componentToHex(r) + componentToHex(g) + componentToHex(b);
|
||
}
|
||
});
|