diff --git a/src/js/controllers/amazon.js b/src/js/controllers/amazon.js index 6bc4a4108..9e1787f57 100644 --- a/src/js/controllers/amazon.js +++ b/src/js/controllers/amazon.js @@ -1,7 +1,7 @@ 'use strict'; angular.module('copayApp.controllers').controller('amazonController', - function($scope, $timeout, $ionicModal, $log, lodash, bwcError, amazonService, platformInfo, externalLinkService, popupService) { + function($scope, $timeout, $ionicModal, $log, lodash, amazonService, platformInfo, externalLinkService, popupService, gettextCatalog) { $scope.network = amazonService.getEnvironment(); @@ -19,6 +19,14 @@ angular.module('copayApp.controllers').controller('amazonController', $timeout(function() { $scope.$digest(); }); + if ($scope.cardClaimCode) { + var card = lodash.find($scope.giftCards, { claimCode: $scope.cardClaimCode }); + if (lodash.isEmpty(card)) { + popupService.showAlert(gettextCatalog.getString('Error'), gettextCatalog.getString('Card not found')); + return; + } + $scope.openCardModal(card); + } }); $scope.updatePendingGiftCards(); }; @@ -26,12 +34,16 @@ angular.module('copayApp.controllers').controller('amazonController', $scope.updatePendingGiftCards = lodash.debounce(function() { amazonService.getPendingGiftCards(function(err, gcds) { + $timeout(function() { + $scope.giftCards = gcds; + $scope.$digest(); + }); lodash.forEach(gcds, function(dataFromStorage) { if (dataFromStorage.status == 'PENDING') { $log.debug("creating gift card"); amazonService.createGiftCard(dataFromStorage, function(err, giftCard) { if (err) { - popupService.showAlert(gettextCatalog.getString('Error'), bwcError.msg(err)); + popupService.showAlert(gettextCatalog.getString('Error'), err); return; } if (giftCard.status != 'PENDING') { @@ -84,6 +96,7 @@ angular.module('copayApp.controllers').controller('amazonController', }; $scope.$on("$ionicView.beforeEnter", function(event, data) { + $scope.cardClaimCode = data.stateParams.cardClaimCode; initAmazon(); }); }); diff --git a/src/js/controllers/amount.js b/src/js/controllers/amount.js index 73c53515d..4d83a4bd7 100644 --- a/src/js/controllers/amount.js +++ b/src/js/controllers/amount.js @@ -1,6 +1,6 @@ 'use strict'; -angular.module('copayApp.controllers').controller('amountController', function($rootScope, $scope, $filter, $timeout, $ionicScrollDelegate, gettextCatalog, platformInfo, lodash, configService, rateService, $stateParams, $window, $state, $log, txFormatService, ongoingProcess, bitpayCardService, popupService, bwcError, payproService) { +angular.module('copayApp.controllers').controller('amountController', function($rootScope, $scope, $filter, $timeout, $ionicScrollDelegate, gettextCatalog, platformInfo, lodash, configService, rateService, $stateParams, $window, $state, $log, txFormatService, ongoingProcess, bitpayCardService, popupService, bwcError, payproService, amazonService, profileService) { var unitToSatoshi; var satToUnit; @@ -16,17 +16,18 @@ angular.module('copayApp.controllers').controller('amountController', function($ $scope.$on("$ionicView.beforeEnter", function(event, data) { + $scope.isGiftCard = data.stateParams.isGiftCard; $scope.isWallet = data.stateParams.isWallet; $scope.cardId = data.stateParams.cardId; $scope.toAddress = data.stateParams.toAddress; $scope.toName = data.stateParams.toName; $scope.toEmail = data.stateParams.toEmail; - $scope.showAlternativeAmount = !!$scope.cardId; + $scope.showAlternativeAmount = !!$scope.cardId || !!$scope.isGiftCard; $scope.toColor = data.stateParams.toColor; $scope.customAmount = data.stateParams.customAmount; - if (!$scope.cardId && !data.stateParams.toAddress) { + if (!$scope.cardId && !$scope.isGiftCard && !data.stateParams.toAddress) { $log.error('Bad params at amount') throw ('bad params'); } @@ -201,6 +202,7 @@ angular.module('copayApp.controllers').controller('amountController', function($ amount: amountUSD, currency: 'USD' }; + ongoingProcess.set('Preparing transaction...', true); $timeout(function() { @@ -240,6 +242,62 @@ angular.module('copayApp.controllers').controller('amountController', function($ }); }); + } else if ($scope.isGiftCard) { + ongoingProcess.set('Preparing transaction...', true); + // Get first wallet as UUID + var uuid; + try { + uuid = profileService.getWallets({ + onlyComplete: true, + network: 'livenet', + })[0].id; + } catch(err) { + popupService.showAlert(gettextCatalog.getString('Error'), gettextCatalog.getString('No wallet found!')); + return; + }; + var amountUSD = $scope.showAlternativeAmount ? _amount : $filter('formatFiatAmount')(toFiat(_amount)); + var dataSrc = { + currency: 'USD', + amount: amountUSD, + uuid: uuid + }; + + amazonService.createBitPayInvoice(dataSrc, function(err, dataInvoice) { + if (err) { + ongoingProcess.set('Preparing transaction...', false); + popupService.showAlert(gettextCatalog.getString('Error'), bwcError.msg(err)); + return; + } + + amazonService.getBitPayInvoice(dataInvoice.invoiceId, function(err, invoice) { + if (err) { + popupService.showAlert(gettextCatalog.getString('Error'), bwcError.msg(err)); + return; + } + + var payProUrl = invoice.paymentUrls.BIP73; + + payproService.getPayProDetails(payProUrl, function(err, payProDetails) { + ongoingProcess.set('Preparing transaction...', false); + if (err) { + popupService.showAlert(gettextCatalog.getString('Error'), bwcError.msg(err)); + return; + } + var stateParams = { + giftCardAmountUSD: amountUSD, + giftCardAccessKey: dataInvoice.accessKey, + giftCardInvoiceTime: invoice.invoiceTime, + giftCardUUID: dataSrc.uuid, + toAmount: payProDetails.amount, + toAddress: payProDetails.toAddress, + description: payProDetails.memo, + paypro: payProDetails + }; + + $state.transitionTo('tabs.giftcards.amazon.confirm', stateParams); + }, true); + }); + }); } else { var amount = $scope.showAlternativeAmount ? fromFiat(_amount) : _amount; if ($scope.customAmount) { diff --git a/src/js/controllers/buyAmazon.js b/src/js/controllers/buyAmazon.js deleted file mode 100644 index ca2aede91..000000000 --- a/src/js/controllers/buyAmazon.js +++ /dev/null @@ -1,224 +0,0 @@ -'use strict'; - -angular.module('copayApp.controllers').controller('buyAmazonController', - function($scope, $log, $timeout, $state, lodash, profileService, bwcError, gettextCatalog, configService, walletService, amazonService, ongoingProcess, platformInfo, externalLinkService, popupService) { - - var self = this; - var network = amazonService.getEnvironment(); - var wallet; - - $scope.$on('Wallet/Changed', function(event, w) { - if (lodash.isEmpty(w)) { - $log.debug('No wallet provided'); - return; - } - wallet = w; - $log.debug('Wallet changed: ' + w.name); - }); - - $scope.openExternalLink = function(url, optIn, title, message, okText, cancelText) { - externalLinkService.open(url, optIn, title, message, okText, cancelText); - }; - - this.confirm = function() { - var message = gettextCatalog.getString('Amazon.com Gift Card purchase for ${{amount}} USD', { - amount: $scope.formData.fiat - }); - var ok = gettextCatalog.getString('Buy'); - popupService.showConfirm(null, message, ok, null, function(res) { - if (res) self.createTx(); - }); - }; - - this.createTx = function() { - self.errorInfo = null; - - if (lodash.isEmpty(wallet)) return; - - if (!wallet.canSign() && !wallet.isPrivKeyExternal()) { - $log.info('No signing proposal: No private key'); - popupService.showAlert(gettextCatalog.getString('Error'), bwcError.msg('MISSING_PRIVATE_KEY')); - return; - } - - var dataSrc = { - currency: 'USD', - amount: $scope.formData.fiat, - uuid: wallet.id - }; - var outputs = []; - var config = configService.getSync(); - var configWallet = config.wallet; - var walletSettings = configWallet.settings; - - - ongoingProcess.set('Processing Transaction...', true); - $timeout(function() { - amazonService.createBitPayInvoice(dataSrc, function(err, dataInvoice) { - if (err) { - ongoingProcess.set('Processing Transaction...', false); - popupService.showAlert(gettextCatalog.getString('Error'), bwcError.msg(err)); - return; - } - - amazonService.getBitPayInvoice(dataInvoice.invoiceId, function(err, invoice) { - if (err) { - ongoingProcess.set('Processing Transaction...', false); - popupService.showAlert(gettextCatalog.getString('Error'), bwcError.msg(err)); - return; - } - - $log.debug('Fetch PayPro Request...', invoice.paymentUrls.BIP73); - - wallet.fetchPayPro({ - payProUrl: invoice.paymentUrls.BIP73, - }, function(err, paypro) { - - if (err) { - ongoingProcess.set('Processing Transaction...', false); - $log.warn('Could not fetch payment request:', err); - var msg = err.toString(); - if (msg.match('HTTP')) { - msg = gettextCatalog.getString('Could not fetch payment information'); - } - popupService.showAlert(gettextCatalog.getString('Error'), msg); - return; - } - - if (!paypro.verified) { - ongoingProcess.set('Processing Transaction...', false); - $log.warn('Failed to verify payment protocol signatures'); - popupService.showAlert(gettextCatalog.getString('Error'), gettextCatalog.getString('Payment Protocol Invalid')); - $timeout(function() { - $scope.$digest(); - }); - return; - } - - var address, comment, amount, url; - - address = paypro.toAddress; - amount = paypro.amount; - url = paypro.url; - comment = 'Amazon.com Gift Card'; - - outputs.push({ - 'toAddress': address, - 'amount': amount, - 'message': comment - }); - - var txp = { - toAddress: address, - amount: amount, - outputs: outputs, - message: comment, - payProUrl: url, - excludeUnconfirmedUtxos: configWallet.spendUnconfirmed ? false : true, - feeLevel: walletSettings.feeLevel || 'normal' - }; - - walletService.createTx(wallet, txp, function(err, createdTxp) { - ongoingProcess.set('Processing Transaction...', false); - if (err) { - popupService.showAlert(gettextCatalog.getString('Error'), bwcError.msg(err)); - return; - } - walletService.publishAndSign(wallet, createdTxp, function(err, tx) { - if (err) { - ongoingProcess.set('Processing Transaction...', false); - popupService.showAlert(gettextCatalog.getString('Error'), bwcError.msg(err)); - walletService.removeTx(wallet, tx, function(err) { - if (err) $log.debug(err); - }); - $timeout(function() { - $scope.$digest(); - }); - return; - } - var count = 0; - ongoingProcess.set('Processing Transaction...', true); - - dataSrc.accessKey = dataInvoice.accessKey; - dataSrc.invoiceId = invoice.id; - dataSrc.invoiceUrl = invoice.url; - dataSrc.invoiceTime = invoice.invoiceTime; - - self.debounceCreate(count, dataSrc); - }); - }); - }); - }); - }); - }, 100); - }; - - self.debounceCreate = lodash.throttle(function(count, dataSrc) { - self.debounceCreateGiftCard(count, dataSrc); - }, 8000, { - 'leading': true - }); - - self.debounceCreateGiftCard = function(count, dataSrc) { - - amazonService.createGiftCard(dataSrc, function(err, giftCard) { - $log.debug("creating gift card " + count); - if (err) { - giftCard = {}; - giftCard.status = 'FAILURE'; - ongoingProcess.set('Processing Transaction...', false); - popupService.showAlert(gettextCatalog.getString('Error'), bwcError.msg(err)); - self.errorInfo = dataSrc; - $timeout(function() { - $scope.$digest(); - }); - } - - if (giftCard.status == 'PENDING' && count < 3) { - $log.debug("pending gift card not available yet"); - self.debounceCreate(count + 1, dataSrc, dataSrc); - return; - } - - var now = moment().unix() * 1000; - - var newData = giftCard; - newData['invoiceId'] = dataSrc.invoiceId; - newData['accessKey'] = dataSrc.accessKey; - newData['invoiceUrl'] = dataSrc.invoiceUrl; - newData['amount'] = dataSrc.amount; - newData['date'] = dataSrc.invoiceTime || now; - newData['uuid'] = dataSrc.uuid; - - if (newData.status == 'expired') { - amazonService.savePendingGiftCard(newData, { - remove: true - }, function(err) { - return; - }); - } - - amazonService.savePendingGiftCard(newData, null, function(err) { - ongoingProcess.set('Processing Transaction...', false); - $log.debug("Saving new gift card with status: " + newData.status); - - self.giftCard = newData; - if (newData.status == 'PENDING') $state.transitionTo('tabs.giftcards.amazon'); - $timeout(function() { - $scope.$digest(); - }); - }); - }); - }; - - $scope.$on("$ionicView.enter", function(event, data) { - $scope.formData = { - fiat: null - }; - $scope.wallets = profileService.getWallets({ - network: network, - onlyComplete: true - }); - }); - - }); diff --git a/src/js/controllers/confirm.js b/src/js/controllers/confirm.js index 3c8ac591d..4919263d8 100644 --- a/src/js/controllers/confirm.js +++ b/src/js/controllers/confirm.js @@ -1,13 +1,25 @@ 'use strict'; -angular.module('copayApp.controllers').controller('confirmController', function($rootScope, $scope, $interval, $filter, $timeout, $ionicScrollDelegate, gettextCatalog, walletService, platformInfo, lodash, configService, rateService, $stateParams, $window, $state, $log, profileService, bitcore, gettext, txFormatService, ongoingProcess, $ionicModal, popupService, $ionicHistory, $ionicConfig, payproService) { +angular.module('copayApp.controllers').controller('confirmController', function($rootScope, $scope, $interval, $filter, $timeout, $ionicScrollDelegate, gettextCatalog, walletService, platformInfo, lodash, configService, rateService, $stateParams, $window, $state, $log, profileService, bitcore, gettext, txFormatService, ongoingProcess, $ionicModal, popupService, $ionicHistory, $ionicConfig, payproService, amazonService) { var cachedTxp = {}; var isChromeApp = platformInfo.isChromeApp; var countDown = null; + var giftCardAmountUSD; + var giftCardAccessKey; + var giftCardInvoiceTime; + var giftCardUUID; $scope.isCordova = platformInfo.isCordova; $ionicConfig.views.swipeBackEnabled(false); $scope.$on("$ionicView.beforeEnter", function(event, data) { + + // Amazon.com Gift Card parameters + $scope.isGiftCard = data.stateParams.isGiftCard; + giftCardAmountUSD = data.stateParams.giftCardAmountUSD; + giftCardAccessKey = data.stateParams.giftCardAccessKey; + giftCardInvoiceTime = data.stateParams.giftCardInvoiceTime; + giftCardUUID = data.stateParams.giftCardUUID; + $scope.isWallet = data.stateParams.isWallet; $scope.cardId = data.stateParams.cardId; $scope.toAmount = data.stateParams.toAmount; @@ -49,6 +61,7 @@ angular.module('copayApp.controllers').controller('confirmController', function( var wallets = profileService.getWallets({ onlyComplete: true, network: networkName, + n: $scope.isGiftCard ? true : false }); if (!wallets || !wallets.length) { @@ -373,6 +386,7 @@ angular.module('copayApp.controllers').controller('confirmController', function( $scope.onSuccessConfirm = function() { var previousView = $ionicHistory.viewHistory().backView && $ionicHistory.viewHistory().backView.stateName; var fromBitPayCard = previousView.match(/tabs.bitpayCard/) ? true : false; + var fromAmazon = previousView.match(/tabs.giftcards.amazon/) ? true : false; $ionicHistory.nextViewOptions({ disableAnimate: true @@ -386,6 +400,17 @@ angular.module('copayApp.controllers').controller('confirmController', function( id: $stateParams.cardId }); }, 100); + } else if (fromAmazon) { + $ionicHistory.nextViewOptions({ + disableAnimate: true, + historyRoot: true + }); + $ionicHistory.clearHistory(); + $state.go('tabs.home').then(function() { + $state.transitionTo('tabs.giftcards.amazon', { + cardClaimCode: $scope.amazonGiftCard ? $scope.amazonGiftCard.claimCode : null + }); + }); } else { $state.go('tabs.send'); } @@ -394,6 +419,71 @@ angular.module('copayApp.controllers').controller('confirmController', function( function publishAndSign(wallet, txp, onSendStatusChange) { walletService.publishAndSign(wallet, txp, function(err, txp) { if (err) return setSendError(err); + + var previousView = $ionicHistory.viewHistory().backView && $ionicHistory.viewHistory().backView.stateName; + var fromAmazon = previousView.match(/tabs.giftcards.amazon/) ? true : false; + if (fromAmazon) { + var count = 0; + var invoiceId = JSON.parse($scope.paypro.merchant_data).invoiceId; + var dataSrc = { + currency: 'USD', + amount: giftCardAmountUSD, + uuid: giftCardUUID, + accessKey: giftCardAccessKey, + invoiceId: invoiceId, + invoiceUrl: $scope.paypro.url, + invoiceTime: giftCardInvoiceTime + }; + debounceCreate(count, dataSrc, onSendStatusChange); + } }, onSendStatusChange); } + + var debounceCreate = lodash.throttle(function(count, dataSrc) { + debounceCreateGiftCard(count, dataSrc); + }, 8000, { + 'leading': true + }); + + var debounceCreateGiftCard = function(count, dataSrc, onSendStatusChange) { + + amazonService.createGiftCard(dataSrc, function(err, giftCard) { + $log.debug("creating gift card " + count); + if (err) { + giftCard = {}; + giftCard.status = 'FAILURE'; + popupService.showAlert(gettextCatalog.getString('Error'), err); + } + + if (giftCard.status == 'PENDING' && count < 3) { + $log.debug("pending gift card not available yet"); + debounceCreate(count + 1, dataSrc); + return; + } + + var now = moment().unix() * 1000; + + var newData = giftCard; + newData['invoiceId'] = dataSrc.invoiceId; + newData['accessKey'] = dataSrc.accessKey; + newData['invoiceUrl'] = dataSrc.invoiceUrl; + newData['amount'] = dataSrc.amount; + newData['date'] = dataSrc.invoiceTime || now; + newData['uuid'] = dataSrc.uuid; + + if (newData.status == 'expired') { + amazonService.savePendingGiftCard(newData, { + remove: true + }, function(err) { + $log.error(err); + return; + }); + } + + amazonService.savePendingGiftCard(newData, null, function(err) { + $log.debug("Saving new gift card with status: " + newData.status); + $scope.amazonGiftCard = newData; + }); + }); + }; }); diff --git a/src/js/routes.js b/src/js/routes.js index 6aa16fa48..d86d0c8ab 100644 --- a/src/js/routes.js +++ b/src/js/routes.js @@ -965,16 +965,36 @@ angular.module('copayApp').config(function(historicLogProvider, $provide, $logPr controller: 'amazonController', templateUrl: 'views/amazon.html' } + }, + params: { + cardClaimCode: null } }) - .state('tabs.giftcards.amazon.buy', { - url: '/buy', + .state('tabs.giftcards.amazon.amount', { + url: '/amount', views: { 'tab-home@tabs': { - controller: 'buyAmazonController', - controllerAs: 'buy', - templateUrl: 'views/buyAmazon.html' + controller: 'amountController', + templateUrl: 'views/amount.html' } + }, + params: { + isGiftCard: true, + toName: 'Amazon.com Gift Card' + } + }) + .state('tabs.giftcards.amazon.confirm', { + url: '/confirm/:toAmount/:toAddress/:description/:giftCardAmountUSD/:giftCardAccessKey/:giftCardInvoiceTime/:giftCardUUID', + views: { + 'tab-home@tabs': { + controller: 'confirmController', + templateUrl: 'views/confirm.html' + } + }, + params: { + isGiftCard: true, + toName: 'Amazon.com Gift Card', + paypro: null } }) diff --git a/src/js/services/bitpayCardService.js b/src/js/services/bitpayCardService.js index 1d6d52983..656078fd7 100644 --- a/src/js/services/bitpayCardService.js +++ b/src/js/services/bitpayCardService.js @@ -136,7 +136,7 @@ angular.module('copayApp.services').factory('bitpayCardService', function($http, }; root.bitAuthPair = function(obj, cb) { - var deviceName = 'Unknow device'; + var deviceName = 'Unknown device'; if (platformInfo.isNW) { deviceName = require('os').platform(); } else if (platformInfo.isCordova) { diff --git a/src/sass/views/amount.scss b/src/sass/views/amount.scss index 0933f08ad..7c4c1681e 100644 --- a/src/sass/views/amount.scss +++ b/src/sass/views/amount.scss @@ -11,6 +11,9 @@ .icon-bitpay-card { background-image: url("../img/icon-bitpay.svg"); } + .icon-amazon { + background-image: url("../img/icon-amazon.svg"); + } @media(max-width: 480px) { .bitcoin-address { .icon { diff --git a/src/sass/views/confirm.scss b/src/sass/views/confirm.scss index 4f6cbc0af..2372290ce 100644 --- a/src/sass/views/confirm.scss +++ b/src/sass/views/confirm.scss @@ -1,5 +1,8 @@ #view-confirm { @extend .deflash-blue; + .icon-amazon { + background-image: url("../img/icon-amazon.svg"); + } .tx-details-content > .scroll { padding-bottom: .25rem; } diff --git a/www/views/amazon.html b/www/views/amazon.html index 8e17d5559..444046cda 100644 --- a/www/views/amazon.html +++ b/www/views/amazon.html @@ -18,7 +18,7 @@
- @@ -30,7 +30,8 @@
-
-
Amount
+
+
+ Amount + (Purchase Amount is limited to USD 500 per day) +
diff --git a/www/views/buyAmazon.html b/www/views/buyAmazon.html deleted file mode 100644 index 69ac4b790..000000000 --- a/www/views/buyAmazon.html +++ /dev/null @@ -1,154 +0,0 @@ - - - - - Buy - - - - -
- - - -
- Amazon.com Gift Card -
- Use your Amazon.com Gift Card* to shop from a huge selection of books, electronics, music, movies, software, apparel, toys, and more. -
-
- -
- -
- -
- - - -
- -
- Purchase Amount is limited to USD 500 per day -
-
-
-
- -
-
-

Gift card could not be created

-
- There was an error when trying to create the Amazon.com Gift Card. Status: {{buy.giftCard.status}} -
-
- - This is a temporary/recoverable system failure that can be - resolved retrying the request from your list of cards - - - This failure could not be recoverable. Request your refund from your list of cards - - -
-
-
-
- Thank you for participating in the BitPay offer. It is our pleasure to send - you this Amazon.com Gift Card* that can be redeemed towards millions of items at - www.amazon.com. - You may want to print this screen for easy reference later you will need the gift card claim code below. -
- -
- Amazon.com Gift Cards -
- Gift Card Amount: - - {{buy.giftCard.amount | currency : '$ ' : 2 }} - -
-
- Claim code: {{buy.giftCard.claimCode}} -
-
- -
- -
-
- To redeem your gift card, follow these steps: - -
    -
  1. 1. Visit www.amazon.com/gc -
  2. 2. Click Apply to Account and enter the Claim Code when prompted. -
  3. 3. Gift card funds will be applied automatically to eligible orders during the checkout process. -
  4. 4. You must pay for any remaining balance on your order with another payment method. -
- -

- Your gift card claim code may also be entered when prompted during checkout. To redeem your gift card using - the Amazon.com 1-Click® service, first add the gift card funds to Your Account. -

- -

- If you have questions about redeeming your gift card, please visit - www.amazon.com/gc-redeem. - If you have questions regarding the BitPay Introductory offer, please contact BitPay. -

- -
-
-
- -
- * Amazon.com is not a sponsor of this promotion. - Except as required by law, Amazon.com - Gift Cards ("GCs") cannot be transferred for value or redeemed for cash. GCs may be used only for purchases of - eligible goods at Amazon.com or certain of its - affiliated websites. For complete terms and conditions, see - www.amazon.com/gc-legal. - GCs are issued by ACI Gift Cards, Inc., a Washington corporation. All Amazon ®, ™ & © are IP - of Amazon.com, Inc. or its affiliates. - No expiration date or service fees. -
- - - diff --git a/www/views/confirm.html b/www/views/confirm.html index 500a79671..85d9006d4 100644 --- a/www/views/confirm.html +++ b/www/views/confirm.html @@ -28,8 +28,11 @@
To - + + +
+