From 491f2edfe7350d6bbafc87c861ba54ad744cd7b1 Mon Sep 17 00:00:00 2001 From: orangeentropy Date: Thu, 28 Nov 2024 23:07:36 +0100 Subject: [PATCH] added Majestic Bank and bumped version numbers of relevant packages --- .../CurrencyMetadata.js | 10 + .../EventListeners.js | 2 +- .../EventListenersMajesticBank.js | 309 ++++++++++++++ .../UtilityFunctions.js | 67 ++- packages/mymonero-exchange-helper/index.js | 36 +- .../initialiseExchange.js | 80 +++- .../mymonero-exchange-helper/package.json | 8 +- .../mymonero-exchange-majesticbank/index.js | 382 ++++++++++++++++++ .../package.json | 35 ++ packages/mymonero-exchange/index.js | 52 ++- packages/mymonero-exchange/package.json | 5 +- packages/mymonero-page-templates/index.esm.js | 1 + packages/mymonero-page-templates/package.json | 9 +- .../ExchangeNavigationController.js | 2 + .../Elements/ChangenowFixedRateView.js | 13 +- .../Exchange/Elements/ExchangeLandingPage.js | 9 + .../Elements/MajesticBankFloatingRateView.js | 184 +++++++++ packages/mymonero-page-templates/src/index.js | 1 + packages/mymonero-web-components/package.json | 5 +- .../src/Assets/ProviderCardImages.js | 7 +- .../src/Components/Shared/ProviderCard.js | 18 +- 21 files changed, 1176 insertions(+), 59 deletions(-) create mode 100644 packages/mymonero-exchange-helper/EventListenersMajesticBank.js create mode 100644 packages/mymonero-exchange-majesticbank/index.js create mode 100644 packages/mymonero-exchange-majesticbank/package.json create mode 100644 packages/mymonero-page-templates/src/Exchange/Elements/MajesticBankFloatingRateView.js diff --git a/packages/mymonero-exchange-helper/CurrencyMetadata.js b/packages/mymonero-exchange-helper/CurrencyMetadata.js index dc44fb7c..8a683e68 100644 --- a/packages/mymonero-exchange-helper/CurrencyMetadata.js +++ b/packages/mymonero-exchange-helper/CurrencyMetadata.js @@ -39,6 +39,16 @@ currencyMetadata = { symbol: "DOT", precision: 16 }, + "FIRO": { + name: "Firo", + symbol: "FIRO", + precision: 8 + }, + "WOW": { + name: "Wownero", + symbol: "WOW", + precision: 12 + } } module.exports = currencyMetadata \ No newline at end of file diff --git a/packages/mymonero-exchange-helper/EventListeners.js b/packages/mymonero-exchange-helper/EventListeners.js index b7862495..8a1dfbbe 100644 --- a/packages/mymonero-exchange-helper/EventListeners.js +++ b/packages/mymonero-exchange-helper/EventListeners.js @@ -264,7 +264,7 @@ updateMinimumInputValue = function(event, exchangeElements) { // console.log("updateMinimum"); // console.log(event); // console.log(exchangeElements); - Utils.getMinimalExchangeAmount("XMR", exchangeElements.outCurrencyTickerCodeDiv.value).then(response => { + Utils.getMinimalExchangeAmount("mymonero", "XMR", exchangeElements.outCurrencyTickerCodeDiv.value).then(response => { //exchangeElements.minimumFeeText.innerText = response.minAmount + " XMR minimum (excluding tx fee)" // use this line when we switch to polling ChangeNow exchangeElements.minimumFeeText.innerText = response.data.in_min + " XMR minimum (excluding tx fee)" }).catch(error => { diff --git a/packages/mymonero-exchange-helper/EventListenersMajesticBank.js b/packages/mymonero-exchange-helper/EventListenersMajesticBank.js new file mode 100644 index 00000000..37548a9a --- /dev/null +++ b/packages/mymonero-exchange-helper/EventListenersMajesticBank.js @@ -0,0 +1,309 @@ +const Utils = require('./UtilityFunctions.js'); +const CurrencyMetadata = require("./CurrencyMetadata"); +const ExchangeLibrary = require('@mymonero/mymonero-exchange-majesticbank') +const handleOfferError = require('./ErrorHelper') +const exchangeFunctions = new ExchangeLibrary() +const validationMessages = document.getElementById('validation-messages'); +//const addressValidation = document.getElementById('address-messages'); +const serverValidation = document.getElementById('server-messages') +const orderBtn = document.getElementById("order-button"); +const loaderPage = document.getElementById('loader'); + +let exchangeXmrDiv = document.getElementById('exchange-xmr'); +let backBtn = document.getElementsByClassName('nav-button-left-container')[0]; +let inCurrencyInput = document.getElementById('inCurrencyValue'); +let outCurrencyInput = document.getElementById('outCurrencyValue'); +let currencyInputTimer; + +clearCurrencies = function() { + document.getElementById("inCurrencyValue").value = ""; + document.getElementById("outCurrencyValue").value = ""; +} + +outAddressInputListener = function(exchangeElements, currencyTickerCode, address) { + + return new Promise((resolve, reject) => { + exchangeElements.serverValidation.innerText = ""; + try { + exchangeElements.serverValidation.innerHTML = '' + if (exchangeElements.currencyInputTimer !== undefined) { + clearTimeout(exchangeElements.currencyInputTimer) + } + exchangeElements.getAddressValidationLoader.style.display = "block"; + exchangeElements.getAddressValidationLoaderText.style.display = "block"; + exchangeElements.validationMessages.innerHTML = '' + exchangeElements.serverValidation.innerHTML = '' + exchangeElements.addressInputTimer = setTimeout(() => { + Utils.validateOutAddress(currencyTickerCode, address).then(response => { + // successful response in following format: {"isActivated":null,"result":true,"message":null} + let element = document.createElement("div"); + element.classList.add('message-label'); + if (response.result == true) { + element.innerHTML = "✔ Validated address successfully"; + exchangeElements.getAddressValidationLoaderText.innerHTML = "
Validated address successfully
"; + exchangeElements.getAddressValidationLoaderContainer.style.display = "none"; + resolve(); + } else if (response.result == false) { + // failed response in following format: {"isActivated":null,"result":false,"message":"Invalid checksum"} + element.innerText = "Your address is invalid"; + exchangeElements.getAddressValidationLoaderText.innerHTML = " The address you've specified is not valid"; + exchangeElements.getAddressValidationLoaderContainer.style.display = "none"; + resolve(); + } else { + console.log(error); + element.innerText = "An unexpected error occurred: " + response.message.toString(); + exchangeElements.getAddressValidationLoaderContainer.style.display = "none"; + reject(error); + } + }).catch(error => { + // 4xx or 5xx errors of some sort + let errorStr = "An unexpected error has occurred: " + error.message; + if (typeof(error.response) !== "undefined") { + if (error.response.data.message !== null) { + errorStr += "
" + error.response.data.message; + } + } + exchangeElements.getAddressValidationLoaderText.innerHTML = errorStr; + exchangeElements.getAddressValidationLoaderContainer.style.display = "none"; + reject(error); + }) + }, 1500) + } catch (error) { + // console.log("outBalanceChecks error route"); + reject(error); + } + }) +} + +inCurrencyGetOfferMajesticBank = function(inCurrencyDiv, outCurrencyDiv, inAmount, exchangeElements) { + return new Promise((resolve, reject) => { + + exchangeFunctions.getOfferWithInAmount(inCurrencyDiv.value, outCurrencyDiv.value, inAmount) + .then((response) => { + resolve(response) + }).catch((error) => { + // console.log("Rejecting with error to bubble up to inCurrencyBalanceChecks") + reject(error); + }) + }) +} + +outCurrencyGetOfferMajesticBank = function(inCurrencyDiv, outCurrencyDiv, inAmount, exchangeElements) { + return new Promise((resolve, reject) => { + + exchangeFunctions.getOfferWithOutAmount(inCurrencyDiv.value, outCurrencyDiv.value, inAmount) + .then((response) => { + resolve(response) + }).catch((error) => { + // console.log("Rejecting with error to bubble up to outCurrencyBalanceChecks") + reject(error); + }) + }) +} + +walletSelectorClickListener = function(event, exchangeElements) { + let walletElement = document.getElementById('wallet-options'); + let selectedWallet = document.getElementById('selected-wallet'); + walletElement.classList.add('active'); + if (event.srcElement.parentElement.className.includes("optionCell")) { + let dataAttributes = event.srcElement.parentElement.dataset; + selectedWallet.dataset.walletlabel = dataAttributes.walletlabel; + selectedWallet.dataset.walletbalance = dataAttributes.walletbalance; + selectedWallet.dataset.swatch = dataAttributes.swatch; + selectedWallet.dataset.walletselected = true; + selectedWallet.dataset.walletoffset = dataAttributes.walletoffset; + selectedWallet.dataset.walletpublicaddress = dataAttributes.walletpublicaddress; + let walletLabel = document.getElementById('selected-wallet-label'); + let walletBalance = document.getElementById('selected-wallet-balance'); + let walletIcon = document.getElementById('selected-wallet-icon'); + walletElement.classList.remove('active'); + walletIcon.style.backgroundImage = `url('../../../assets/img/wallet-${dataAttributes.swatch}@3x.png'`; + walletLabel.innerText = dataAttributes.walletlabel; + walletBalance.innerText = dataAttributes.walletbalance + " XMR"; + let walletSelector = document.getElementById('wallet-selector'); + walletSelector.dataset.walletchosen = true; + clearCurrencies(); + } else { + // console.log("Didn't update wallet dataset"); + } + if (event.srcElement.parentElement.className.includes("selectionDisplayCellView")) { + walletElement.classList.add('active'); + } + if (event.srcElement == 'div.hoverable-cell.utility.selectionDisplayCellView') { + + } +} + +inBalanceChecks = function (exchangeElements, exchangeFunctions) { + return new Promise((resolve, reject) => { + try { + exchangeElements.serverValidation.innerHTML = '' + //let inAmountToReceive = exchangeElements.BTCToReceive + const inBalance = parseFloat(exchangeElements.inCurrencyValue.value) + const in_amount = inBalance.toFixed(12) + exchangeElements.outCurrencyValue.value = '' + if (exchangeElements.currencyInputTimer !== undefined) { + clearTimeout(exchangeElements.currencyInputTimer) + } + + exchangeElements.validationMessages.innerHTML = '' + exchangeElements.serverValidation.innerHTML = '' + exchangeElements.currencyInputTimer = setTimeout(() => { + let inCurrencyDiv = document.getElementById("inCurrencySelectList"); + let outCurrencyDiv = document.getElementById("outCurrencySelectList"); + let inCurrencyValue = document.getElementById("inCurrencyValue").value; + inCurrencyGetOfferMajesticBank(inCurrencyDiv, outCurrencyDiv, inCurrencyValue, exchangeElements).then((response) => { + // successfully retrieved an offer + exchangeElements.outCurrencyValue.value = response.out_amount; + // replace minAmount with estimated total + let txFee = parseFloat(exchangeElements.txFee.dataset.txFee) + let inAmount = parseFloat(response.in_amount) + let estimatedTotal = txFee + inAmount; + exchangeElements.minimumFeeText.innerText = `~ ${estimatedTotal} XMR EST. TOTAL`; + resolve(response); + }).catch((error) => { + + exchangeElements.minimumFeeText.innerText = `${exchangeElements.minimumFeeText.dataset.minimumAmount} XMR minimum (excluding tx fee)` + reject(error); + }) + }, 1500) + } catch (error) { + // console.log("InBalanceChecks error route"); + reject(error); + } + }) +} + +// This would have been through key screening and an offer would've come back +outBalanceChecks = function(exchangeElements) { + return new Promise((resolve, reject) => { + try { + exchangeElements.serverValidation.innerHTML = '' + //let inAmountToReceive = exchangeElements.BTCToReceive + const outBalance = parseFloat(exchangeElements.outCurrencyValue.value) + const out_amount = outBalance.toFixed(12) + exchangeElements.inCurrencyValue.value = '' + if (exchangeElements.currencyInputTimer !== undefined) { + clearTimeout(exchangeElements.currencyInputTimer) + } + + exchangeElements.validationMessages.innerHTML = '' + exchangeElements.serverValidation.innerHTML = '' + exchangeElements.currencyInputTimer = setTimeout(() => { + let inCurrencyDiv = document.getElementById("inCurrencySelectList"); + let outCurrencyDiv = document.getElementById("outCurrencySelectList"); + let outCurrencyValue = document.getElementById("outCurrencyValue").value; + outCurrencyGetOfferMajesticBank(inCurrencyDiv, outCurrencyDiv, outCurrencyValue, exchangeElements).then((response) => { + exchangeElements.inCurrencyValue.value = response.in_amount; + let txFee = parseFloat(exchangeElements.txFee.dataset.txFee) + let inAmount = parseFloat(response.in_amount) + let estimatedTotal = txFee + inAmount; + exchangeElements.minimumFeeText.innerText = `~ ${estimatedTotal} XMR EST. TOTAL`; + + resolve(response); + }).catch((error) => { + // console.log("outBalance promise rejection"); + exchangeElements.minimumFeeText.innerText = `${exchangeElements.minimumFeeText.dataset.minimumAmount} XMR minimum (excluding tx fee)` + reject(error); + }) + }, 1500) + } catch (error) { + // console.log("outBalanceChecks error route"); + reject(error); + } + }) +} + +backButtonClickListener = function() { + let backBtn = document.getElementsByClassName('nav-button-left-container')[0]; + let viewOrderBtn = document.getElementById('view-order'); + orderCreated = false; + document.getElementById("orderStatusPage").classList.add('active'); + backBtn.style.display = "none"; + let orderStatusDiv = document.getElementById("exchangePage"); + loaderPage.classList.remove('active'); + orderStatusDiv.classList.remove('active'); + exchangeXmrDiv.classList.remove('active'); + viewOrderBtn.style.display = "block"; +} + +function clearCurrencyInputValues() { + document.getElementById("inCurrencyValue").value = ""; + document.getElementById("outCurrencyValue").value = ""; +} + +orderButtonClickedListener = function(orderStarted, ExchangeFunctions) { + // 1. Do validation + if (validateOrder) { + // console.log("Order valid"); + } + // console.log(orderStarted); + // console.log(ExchangeFunctions); + +} + +validateOrder = function() { + let validationError = false; + // console.log(validationMessages); + if (orderStarted == true) { + + return false; + } + if (validationMessages.firstChild !== null) { + validationMessages.firstChild.style.color = "#ff0000"; + validationError = true; + return false; + } + if (addressValidation.firstChild !== null) { + addressValidation.firstChild.style.color = "#ff0000"; + validationError = true; + return false; + } + +} + +updateMinimumInputValue = function(event, exchangeElements) { + // console.log("updateMinimum"); + // console.log(event); + // console.log(exchangeElements); + Utils.getMinimalExchangeAmount("majesticbank", "XMR", exchangeElements.outCurrencyTickerCodeDiv.value).then(response => { + //exchangeElements.minimumFeeText.innerText = response.minAmount + " XMR minimum (excluding tx fee)" // use this line when we switch to polling ChangeNow + exchangeElements.minimumFeeText.innerText = response.data.in_min + " XMR minimum (excluding tx fee)" + }).catch(error => { + exchangeElements.minimumFeeText.innerText = "An error was encountered when fetching the minimum: " + error.message; + }) +} + +outCurrencySelectListChangeListener = function(event, exchangeElements) { + + updateMinimumInputValue(event, exchangeElements); + clearCurrencies(); + clearInterval(exchangeElements.currencyInputTimer); + clearInterval(exchangeElements.offerRetrievalIsSlowTimer); + exchangeElements.getOfferLoader.style.display = "none"; + // get elements that show the + // clear timers too... +} + +updateCurrencyLabels = function(event, exchangeElements) { + let coinTickerCode = event.srcElement.value; + let coinName = CurrencyMetadata[coinTickerCode].name.toUpperCase(); + document.getElementById("outCurrencyTickerCode").innerText = coinTickerCode; + document.getElementById("orderStatusPageCurrencyTicker").innerText = coinName; + document.getElementById("outCurrencyCoinName").innerText = coinName; + document.getElementById("outAddress").placeholder = `Destination ${coinTickerCode} Address`; + clearCurrencyInputValues(); +} + +module.exports = { + outAddressInputListener, + //inCurrencyInputKeydownListener, + walletSelectorClickListener, + inBalanceChecks, + orderButtonClickedListener, + updateCurrencyLabels, + validateOrder, + outBalanceChecks, + inCurrencyGetOffer, + outCurrencySelectListChangeListener +}; \ No newline at end of file diff --git a/packages/mymonero-exchange-helper/UtilityFunctions.js b/packages/mymonero-exchange-helper/UtilityFunctions.js index ff947ff6..39b3f12d 100644 --- a/packages/mymonero-exchange-helper/UtilityFunctions.js +++ b/packages/mymonero-exchange-helper/UtilityFunctions.js @@ -1,19 +1,35 @@ const axios = require('axios'); -function getMinimalExchangeAmount(fromCurrency, toCurrency) { +function getMinimalExchangeAmount(exchange_name, fromCurrency, toCurrency) { + + if(exchange_name === "majesticbank") { + const ExchangeFunctionsMajesticBank = require("@mymonero/mymonero-exchange-majesticbank") + this.exchangeFunctions = new ExchangeFunctionsMajesticBank() + + return new Promise((resolve, reject) => { + this.exchangeFunctions.getRatesAndLimits(fromCurrency, toCurrency) + .then((response) => { + // emulating the response from ChangeNow + resolve({"data": {"in_min": response.data["limits"][fromCurrency].min}}) + }).catch((error) => { + reject(error) + }) + }) + } + let self = this; return new Promise((resolve, reject) => { this.apiUrl = "https://api.mymonero.com:443/cx"; let data = { - "in_currency": "XMR", - "out_currency": "BTC" + "in_currency": fromCurrency, + "out_currency": toCurrency } let endpoint = `${this.apiUrl}/get_info`; axios.post(endpoint, data) .then((response) => { self.currentRates = response.data; - self.in_currency = "XMR"; - self.out_currency = "BTC"; + self.in_currency = fromCurrency; + self.out_currency = toCurrency; self.currentRates.minimum_xmr = self.currentRates.in_min; self.currentRates.maximum_xmr = self.currentRates.in_max; resolve(response); @@ -53,6 +69,46 @@ function getMinimalExchangeAmount(fromCurrency, toCurrency) { // } function validateOutAddress(currencyTickerCode, address) { + + // This is the type of response ChangeNow provides + // For simplicity, we will use this response even when checks are done locally + const successfulResponse = { + "isActivated": null, + "result": true, + "message": "Valid address. (Local checks passed)." + } + + const failedResponse = { + "isActivated": null, + "result": false, + "message": "Invalid address. (Failed local checks)." + } + + // We use regex to validate the address locally for currencies unsupported by ChangeNow + if(currencyTickerCode === "WOW") { + return new Promise((resolve, reject) => { + // Wownero addresses are 97 characters long + // start with W followed by 96 base58 characters + regex = /^W[1-9A-HJ-NP-Za-km-z]{96}$/ + if(regex.test(address)){ + resolve(successfulResponse) + } + reject(failedResponse) + }) + } + else if (currencyTickerCode === "FIRO"){ + return new Promise((resolve, reject) => { + // Firo addresses either: + // start with a followed by 33 base58 characters (transparent address) + // or start with firos followed by 94 base58 characters (shielded address) + regex = /^(a[1-9A-HJ-NP-Za-km-z]{33}|firos[1-9A-HJ-NP-Za-km-z]{94})$/ + if(regex.test(address)){ + resolve(successfulResponse) + } + reject(failedResponse) + }) + } + return new Promise((resolve, reject) => { this.apiUrl = "https://api.changenow.io/v2/"; var axios = require('axios'); @@ -126,6 +182,7 @@ function sendFunds (wallet, xmr_amount, xmr_send_address, sweep_wallet, validati // end of functions to check Bitcoin address function renderOrderStatus (order) { + // TODO - MajesticBank - ensure order statuses match this schema /* "btc_amount", diff --git a/packages/mymonero-exchange-helper/index.js b/packages/mymonero-exchange-helper/index.js index c8ea5024..0d63d22f 100644 --- a/packages/mymonero-exchange-helper/index.js +++ b/packages/mymonero-exchange-helper/index.js @@ -1,30 +1,49 @@ const HtmlHelper = require("./HtmlHelper"); const WalletHelper = require("./WalletHelper") const html = require("./HtmlHelper") -const EventListeners = require("./EventListeners") const TimerHelper = require('./TimerHelper'); const CurrencyMetadata = require('./CurrencyMetadata') const monero_amount_format_utils = require('@mymonero/mymonero-money-format') const JSBigInt = require('@mymonero/mymonero-bigint') const ErrorHelper = require("./ErrorHelper") const InitialiseExchange = require("./initialiseExchange") -const ExchangeFunctions = require("@mymonero/mymonero-exchange") -const exchangeFunctions = new ExchangeFunctions(); class ExchangeHelper { // We declare these in this module so that we don't tightly couple currencies to the REST API module - constructor() { + constructor(exchange_name="mymonero") { // Assignment to `this` variable is so that we can invoke these functions using an instance of this class in a public fashion - this.supportedOutCurrencies = ["BTC", "ETH", "LTC"] - this.supportedInCurrencies = ["XMR"]; + if (exchange_name === "mymonero") { + this.exchange_name = "mymonero"; + this.supportedOutCurrencies = ["BTC", "ETH", "LTC"] + this.supportedInCurrencies = ["XMR"]; + + // very very ugly but solves the problem of global variables overriding each other + const ExchangeFunctions = require("@mymonero/mymonero-exchange") + this.exchangeFunctions = new ExchangeFunctions() + + const EventListeners = require("./EventListeners") + this.eventListeners = EventListeners; + console.log("ExchangeHelper: Using MyMonero exchange functions") + } + else if (exchange_name === "majesticbank") { + this.exchange_name = "majesticbank"; + this.supportedOutCurrencies = ["BTC", "LTC", "WOW", "BCH", "FIRO"] + this.supportedInCurrencies = ["XMR"]; + + const ExchangeFunctionsMajesticBank = require("@mymonero/mymonero-exchange-majesticbank") + this.exchangeFunctions = new ExchangeFunctionsMajesticBank() + + const EventListenersMajesticBank = require("./EventListenersMajesticBank") + this.eventListeners = EventListenersMajesticBank + console.log("ExchangeHelper: Using MajesticBank exchange functions") + } this.baseForm = ""; // Fetch form we'll insert into the content view's innerHTML - + this.htmlHelper = new HtmlHelper(); this.baseForm = this.htmlHelper.getBaseForm(); - this.eventListeners = EventListeners; this.timerHelper = TimerHelper; this.currencyMetadata = CurrencyMetadata; this.errorHelper = ErrorHelper; @@ -34,7 +53,6 @@ class ExchangeHelper { this.renderWalletSelector = this.renderWalletSelector; this.initialiseExchangeHelper = InitialiseExchange; this.doInit = this.doInit; - this.exchangeFunctions = exchangeFunctions; this.handleSendFundsResponseCallback = this.handleSendFundsResponseCallback; this.sendFundsValidationStatusCallback = this.sendFundsValidationStatusCallback; diff --git a/packages/mymonero-exchange-helper/initialiseExchange.js b/packages/mymonero-exchange-helper/initialiseExchange.js index d11c40dc..15868b88 100644 --- a/packages/mymonero-exchange-helper/initialiseExchange.js +++ b/packages/mymonero-exchange-helper/initialiseExchange.js @@ -267,7 +267,7 @@ function initialiseExchangeHelper(context, exchangeHelper) { } // Gets the initial minimum value - Utils.getMinimalExchangeAmount("XMR", "BTC").then(response => { + Utils.getMinimalExchangeAmount(exchangeHelper.exchange_name, "XMR", "BTC").then(response => { // let minimumAmount = parseFloat(response.minAmount); let minimumAmount = parseFloat(response.data.in_min); exchangeElements.minimumFeeText.innerText = `${minimumAmount} XMR minimum (excluding tx fee)`; @@ -466,14 +466,23 @@ function initialiseExchangeHelper(context, exchangeHelper) { // We attempt to retrieve enabled currency pairs from the server exchangeHelper.exchangeFunctions.getCurrencyPairs().then((response) => { - - response = {"out_currencies":[ - {"name":"Bitcoin","symbol":"BTC"}, - {"name":"Ether","symbol":"ETH"}, - {"name":"Litecoin","symbol":"LTC"}, - {"name":"Bitcoin Cash","symbol":"BCH"}, - {"name":"Polkadot","symbol":"DOT"} - ]} + + if(exchangeHelper.exchangeFunctions.apiUrl.includes("majesticbank")){ + // Majestic Bank supports different currencies than the other exchanges + // This isn't the ideal way of detecting the exchange + // For Majestic Bank we use the API to get the available currencies + } + else{ + // For all other exchanges we use the default currencies + response = {"out_currencies":[ + {"name":"Bitcoin","symbol":"BTC"}, + {"name":"Ether","symbol":"ETH"}, + {"name":"Litecoin","symbol":"LTC"}, + {"name":"Bitcoin Cash","symbol":"BCH"}, + {"name":"Polkadot","symbol":"DOT"} + ]} + } + let outCurrencySelectList = document.getElementById('outCurrencySelectList') let length = outCurrencySelectList.length; for (let i = length - 1; i >= 0; i--) { @@ -602,6 +611,11 @@ function initialiseExchangeHelper(context, exchangeHelper) { let e = document.getElementById('orderStatusPage'); e = document.getElementById('orderStatusPage'); // backBtn.innerHTML = `
`; + + // Added a semaphore to prevent multiple order status checks from being fired simultaneously + // This is especially important for Majestic Bank servers as they are quite happy to send 429s (Too Many Requests) + let orderStatusCheckSemaphore = false + exchangeElements.orderTimer = setInterval(() => { //exchangeElements.orderStatusPage.classList.add('active') exchangeElements.exchangePageDiv.classList.add('active') @@ -623,14 +637,17 @@ function initialiseExchangeHelper(context, exchangeHelper) { const xmr_dest_address_elem = document.getElementById('in_address') xmr_dest_address_elem.value = response.receiving_subaddress } - - if (orderStatusResponse.status == 'PAID' - || orderStatusResponse.status == 'TIMED_OUT' - || orderStatusResponse.status == 'DONE' - || orderStatusResponse.status == 'FLAGGED_DESTINATION_ADDRESS' - || orderStatusResponse.status == 'PAYMENT_FAILED' - || orderStatusResponse.status == 'REJECTED' - || orderStatusResponse.status == 'EXPIRED') { + + if (orderStatusResponse.status == 'PAID' + || orderStatusResponse.status == 'TIMED_OUT' + || orderStatusResponse.status == 'DONE' + || orderStatusResponse.status == 'FLAGGED_DESTINATION_ADDRESS' + || orderStatusResponse.status == 'PAYMENT_FAILED' + || orderStatusResponse.status == 'REJECTED' + || orderStatusResponse.status == 'EXPIRED' + || orderStatusResponse.status == 'Completed' //used by MajesticBank + || orderStatusResponse.status == 'Not found' //used by MajesticBank + ) { clearInterval(exchangeElements.orderTimer) document.getElementById("exchange-xmr").classList.remove("active"); } @@ -642,15 +659,24 @@ function initialiseExchangeHelper(context, exchangeHelper) { document.getElementById("exchange-xmr").classList.remove("active"); } } - if ((orderStatusResponse.orderTick % 10) == 0) { + if ((orderStatusResponse.orderTick % 10) === 0 && !orderStatusCheckSemaphore) { + + // Set the semaphore to true to prevent multiple order status checks from being fired simultaneously + orderStatusCheckSemaphore = true + + exchangeHelper.exchangeFunctions.getOrderStatus().then(function (response) { + + // Set the semaphore to false to allow the next order status check to be fired + orderStatusCheckSemaphore = false + let elemArr = document.getElementsByClassName('provider-name') if (firstTick == true || elemArr.length > 0) { exchangeHelper.renderOrderStatus(response) elemArr[0].innerHTML = response.provider_name elemArr[1].innerHTML = response.provider_name elemArr[2].innerHTML = response.provider_name - + elemArr = document.getElementsByClassName('outCurrencyTickerCode'); elemArr[0].innerHTML = out_currency; elemArr[1].innerHTML = out_currency; @@ -660,6 +686,22 @@ function initialiseExchangeHelper(context, exchangeHelper) { orderTick++ response.orderTick = orderTick orderStatusResponse = response + }).catch((error) => { + // Failed to get order status + + // Set the semaphore to false to allow the next order status check to be fired + orderStatusCheckSemaphore = false + + // Axios errors have a status property, so we can check for a 429 (Too Many Requests) error + // If we get a 429, we don't want to throw an error, we just want to wait for the next interval + // Majestic Bank servers are quite happy to send 429s + if (error.hasOwnProperty('status') && error.status == 429) { + return + } + else{ + // We shouldn't get other errors that 429s, so if we do something is wrong and we pass throw the error + throw error + } }) } }, 1000) diff --git a/packages/mymonero-exchange-helper/package.json b/packages/mymonero-exchange-helper/package.json index 8c2a89bd..c4673d5d 100644 --- a/packages/mymonero-exchange-helper/package.json +++ b/packages/mymonero-exchange-helper/package.json @@ -1,6 +1,6 @@ { "name": "@mymonero/mymonero-exchange-helper", - "version": "3.0.3", + "version": "3.0.4", "description": "A library of helper functions used for rendering the in-app exchange", "main": "index.js", "scripts": { @@ -8,7 +8,8 @@ }, "dependencies": { "@mymonero/mymonero-bigint": "^3.0.3", - "@mymonero/mymonero-exchange": "^3.0.3", + "@mymonero/mymonero-exchange": "^3.0.4", + "@mymonero/mymonero-exchange-majesticbank": "^3.0.4", "@mymonero/mymonero-money-format": "^3.0.3", "axios": "^0.21.3" }, @@ -42,6 +43,5 @@ "license": "BSD-3-Clause", "publishConfig": { "access": "public" - }, - "gitHead": "8b2fb278e4a5aa84e577c9985fbca332fca4f1b0" + } } diff --git a/packages/mymonero-exchange-majesticbank/index.js b/packages/mymonero-exchange-majesticbank/index.js new file mode 100644 index 00000000..f39261bd --- /dev/null +++ b/packages/mymonero-exchange-majesticbank/index.js @@ -0,0 +1,382 @@ +const axios = require("axios"); +const FormData = require("form-data"); +//const fetch = require("fetch"); +class ExchangeFunctionsMajesticBank { + + constructor() { + this.apiUrl = "https://majesticbank.sc/api"; + this.apiVersion = "v1"; + this.referral_code = "mgzySX"; + + this.offer = {} + this.offer_type = ""; + this.order = {}; + this.orderRefreshTimer = {}; + this.currentRates = {}; + this.orderStatus = {}; + this.exchangeConfiguration = {}; + } + + getApiPath() { + return `${this.apiUrl}/${this.apiVersion}` + } + + initialiseExchangeConfiguration() { + + // not needed for MB + return new Promise((resolve, reject) => { + resolve(); + }); + } + + getOfferWithOutAmount(in_currency, out_currency, out_amount) { + + // Response + // from_currency -> string + // from_amount -> number + // receive_currency -> string + // receive_amount -> number + + const self = this; + let endpoint = `${self.getApiPath()}/calculate`; + let data = new FormData(); + data.append('from_currency', in_currency); + data.append('receive_currency', out_currency); + data.append('receive_amount', out_amount); + self.offer_type = "out_amount"; + + return new Promise((resolve, reject) => { + axios.post(endpoint, data) + .then(function (response) { + console.log('response from MajesticBank getOfferWithOutAmount', response); + + // changenow has offer_id but majesticbank does not + self.offer = { + "in_amount": parseFloat(response.data.from_amount), + "out_amount": parseFloat(response.data.receive_amount), + "expires_at": new Date(new Date().getTime() + 10 * 60000), + } + + resolve(self.offer); + }) + .catch(function (error) { + console.log(error); + reject(error); + }); + }); + } + + getOfferWithInAmount(in_currency, out_currency, in_amount) { + + // Response + // from_currency -> string + // from_amount -> number + // receive_currency -> string + // receive_amount -> number + + const self = this; + let endpoint = `${self.getApiPath()}/calculate`; + let data = new FormData(); + data.append('from_currency', in_currency); + data.append('receive_currency', out_currency); + data.append('from_amount', in_amount); + self.offer_type = "in_amount"; + + return new Promise((resolve, reject) => { + axios.post(endpoint, data) + .then(function (response) { + console.log('response from MajesticBank getOfferWithInAmount', response); + + // changenow has offer_id but majesticbank does not + self.offer = { + "in_amount": response.data.from_amount, + "out_amount": response.data.receive_amount, + "expires_at": new Date(new Date().getTime() + 600 * 60000), + } + + resolve(self.offer); + }) + .catch(function (error) { + console.log(error); + reject(error); + }); + }); + } + + getOffer(in_currency, out_currency, amount, offerType) { + + return new Promise((resolve, reject) => { + if (offerType == "in") { + this.getOfferWithInAmount(in_currency, out_currency, amount).then(response => { + resolve(response); + }).catch(error => { + reject(error); + }); + } else if (offerType == "out") { + this.getOfferWithOutAmount(in_currency, out_currency, amount).then(response => { + resolve(response); + }).catch(error => { + reject(error); + }); + } else { + // TODO (from original ChangeNow code): Handle error a bit more elegantly + let error = new Error("Please ensure you have specified an amount to exchange"); + console.log(error); + reject(error); + } + }) + } + + + + getOrderStatus() { + + const self = this; + + let endpoint = `${self.getApiPath()}/track`; + return new Promise((resolve, reject) => { + let data = new FormData(); + data.append('trx', self.order.order_id); + + axios.post(endpoint, data) + .then(function (response) { + + // Response: + // trx -> string + // status -> string + // from_currency -> string + // from_amount -> number + // receive_currency -> string + // receive_amount -> number + // address -> string + // received -> number + // confirmed -> number + + let resolve_data = { + "order_id": response.data.trx, + "expires_at": self.order.expires_at, + "in_address": response.data.address, + "in_amount": parseFloat(response.data.from_amount), + "out_currency": response.data.receive_currency, + "out_amount": parseFloat(response.data.receive_amount), + "status": response.data.status, + "in_amount_remaining": parseFloat(response.data.from_amount) - parseFloat(response.data.received), + "out_address": self.order.out_address, + "provider_name": "MajesticBank", + "provider_url": "https://majesticbank.sc/track", + "provider_order_id": response.data.trx, + } + + self.orderStatus = resolve_data + resolve(self.orderStatus); + }) + .catch(function (error) { + console.log(error); + reject(error); + }); + }); + } + + getOrderExpiry() { + return this.orderStatus.expires_at; + } + + getTimeRemaining() { + return this.orderStatus.seconds_till_timeout; + } + + createOrder(out_address, refund_address, in_currency = "XMR", out_currency = "BTC") { + + let self = this; + let endpoint = `${self.getApiPath()}/exchange`; + + let data = new FormData() + data.append('from_currency', in_currency); + data.append('receive_currency', out_currency); + + data.append('receive_address', out_address); + data.append('referral_code', self.referral_code); + + if (self.offer_type === "in_amount") { + data.append('from_amount', self.offer.in_amount); + } + else if (self.offer_type === "out_amount") { + return this.pay(out_address, refund_address, in_currency, out_currency); + } + + return new Promise((resolve, reject) => { + try { + axios.post(endpoint, data) + .then(function (response) { + self.order = response; + // Response: + // trx -> string + // from_currency -> string + // from_amount -> number + // receive_currency -> string + // receive_amount -> number + // address -> string + // expiration -> number (minutes) - 600 + + + self.order = { + "order_id": response.data.trx, + "in_currency": response.data.from_currency, + "in_amount": parseFloat(response.data.from_amount), + "out_currency": response.data.receive_currency, + "out_amount": parseFloat(response.data.receive_amount), + "out_address": response.data.address, + "expires_at": new Date(new Date().getTime() + response.data.expiration * 60000), + "seconds_till_timeout": parseInt(response.data.expiration) * 60 + } + + // expires_at + // seconds_till_timeout + resolve(response); + }) + .catch(function (error) { + reject(error); + }); + } catch (error) { + reject(error); + } + }); + } + + pay(out_address, refund_address, in_currency = "XMR", out_currency = "BTC"){ + // creates a fixed / pay order + // only called from createOrder + let self = this; + let endpoint = `${self.getApiPath()}/pay`; + + let data = new FormData() + data.append('from_currency', in_currency); + data.append('receive_currency', out_currency); + + data.append('receive_address', out_address); + data.append('referral_code', self.referral_code); + + if (self.offer_type === "out_amount") { + data.append('receive_amount', self.offer.out_amount); + } + else if (self.offer_type === "in_amount") { + // This should never happen, but in case it does we can handle it + return this.createOrder(out_address, refund_address, in_currency, out_currency); + } + + return new Promise((resolve, reject) => { + try { + axios.post(endpoint, data) + .then(function (response) { + self.order = response; + // Response: (identical to createOrder, except for expiration time) + // trx -> string + // from_currency -> string + // from_amount -> number + // receive_currency -> string + // receive_amount -> number + // address -> string + // expiration -> number (minutes) - 10 + + + self.order = { + "order_id": response.data.trx, + "in_currency": response.data.from_currency, + "in_amount": parseFloat(response.data.from_amount), + "out_currency": response.data.receive_currency, + "out_amount": parseFloat(response.data.receive_amount), + "out_address": response.data.address, + "expires_at": new Date(new Date().getTime() + response.data.expiration * 60000), + "seconds_till_timeout": parseInt(response.data.expiration) * 60 + } + + // expires_at + // seconds_till_timeout + resolve(response); + }) + .catch(function (error) { + reject(error); + }); + } catch (error) { + reject(error); + } + }); + } + + + getRatesAndLimits(in_currency, out_currency) { + + let self = this; + return new Promise((resolve, reject) => { + axios.get(`${this.getApiPath()}/rates`) + .then(response => { + // sample data + // { "BTC-USD": 95873.25, "BTC-XMR": 587.27871363, "BTC-LTC": 1000.43427567, "BTC-WOW": 907408.37140125, "BTC-FIRO": 66530.88804593, "BTC-BCH": 188.6681292, "XMR-USD": 159.985, "XMR-BTC": 0.00163534, "XMR-LTC": 1.66943832, "XMR-WOW": 1514.20472654, "XMR-FIRO": 111.02100037, "XMR-BCH": 0.31483308, "LTC-USD": 93.915, "LTC-BTC": 0.00095998, "LTC-XMR": 0.57528331, "LTC-WOW": 888.87418754, "LTC-FIRO": 65.17196768, "LTC-BCH": 0.18481451, "WOW-USD": 0.103543, "WOW-BTC": 1.06e-6, "WOW-XMR": 0.00063426, "WOW-LTC": 0.00108047, "WOW-FIRO": 0.07185328, "WOW-BCH": 0.00020376, "FIRO-USD": 1.412213, "FIRO-BTC": 1.444e-5, "FIRO-XMR": 0.00865062, "FIRO-LTC": 0.0147364, "FIRO-WOW": 13.36612557, "FIRO-BCH": 0.00277908, "BCH-USD": 497.995, "BCH-BTC": 0.00509042, "BCH-XMR": 3.05050536, "BCH-LTC": 5.19656178, "BCH-WOW": 4713.35676965, "BCH-FIRO": 345.58179255, "limits": { "BTC": { "min": 0.00104304, "max": 2.60760953 }, "XMR": { "min": 0.07500703, "max": 1562.64649811 }, "LTC": { "min": 0.12777512, "max": 2661.98157909 }, "WOW": { "min": 115.89387984, "max": 2414455.82994505 }, "FIRO": { "min": 8.49730175, "max": 177027.11984665 }, "BCH": { "min": 0.02409663, "max": 502.01307242 } } } + self.in_currency = in_currency; + self.out_currency = out_currency; + self.currentRates = response.data; + // access the limits in the data + self.currentRates.minimum_xmr = response.data["limits"][in_currency].min; + self.currentRates.maximum_xmr = response.data["limits"][in_currency].max; + + resolve(response); + }) + .catch(error => { + reject(error); + }); + }); + } + + getCurrencyPairs(in_currency = "XMR") { + + let self = this; + return new Promise((resolve, reject) => { + + let currencies = [ + { + "name": "Bitcoin", + "symbol": "BTC", + "precision": 8 + }, + { + "name": "Monero", + "symbol": "XMR", + "precision": 12 + }, + { + "name": "Litecoin", + "symbol": "LTC", + "precision": 8 + }, + { + "name": "Bitcoin Cash", + "symbol": "BCH", + "precision": 8 + }, + { + "name": "Wownero", + "symbol": "WOW", + "precision": 12 + }, + { + "name": "Firo", + "symbol": "FIRO", + "precision": 8 + } + ] + + // throw out the in_currency + currencies = currencies.filter(currency => currency.symbol !== in_currency); + + self.enabledCurrencies = currencies; + let out_currencies = { + "out_currencies": currencies + } + resolve(out_currencies); + + }); + } + +} + +module.exports = ExchangeFunctionsMajesticBank; diff --git a/packages/mymonero-exchange-majesticbank/package.json b/packages/mymonero-exchange-majesticbank/package.json new file mode 100644 index 00000000..99db333a --- /dev/null +++ b/packages/mymonero-exchange-majesticbank/package.json @@ -0,0 +1,35 @@ +{ + "name": "@mymonero/mymonero-exchange-majesticbank", + "version": "3.0.4", + "description": "This module is used by the desktop and web versions of the MyMonero wallet to facilitate the exchange of XMR into other cryptocurrencies. It can be used to facilitate transactions where a user converts one cryptocurrency into another.", + "main": "index.js", + "scripts": { + "test": "echo \"Error: no test specified\" && exit 1" + }, + "keywords": [ + "cryptocurrency-conversion", + "cryptocurrency-exchange", + "mymonero", + "xmrtobtc", + "monerotobitcoin", + "monero-to-bitcoin" + ], + "author": "MyMonero", + "repository": { + "type": "git", + "url": "git+https://github.com/mymonero/mymonero-utils.git" + }, + "homepage": "https://mymonero.com", + "bugs": { + "url": "https://github.com/mymonero/mymonero-utils/issues", + "email": "info@mymonero.com" + }, + "license": "BSD-3-Clause", + "dependencies": { + "axios": "^1.6.8", + "process": "^0.11.10" + }, + "publishConfig": { + "access": "public" + } +} diff --git a/packages/mymonero-exchange/index.js b/packages/mymonero-exchange/index.js index a4708c1e..7840f46c 100644 --- a/packages/mymonero-exchange/index.js +++ b/packages/mymonero-exchange/index.js @@ -79,9 +79,18 @@ class ExchangeFunctions { self.offer_type = "out_amount"; let endpoint = `${self.apiUrl}/get_offer`; return new Promise((resolve, reject) => { + + //sample response + // { + // "offer_id": "d604-2-101", + // "expires_at": "2024-11-24T20:37:46.328Z", + // "in_amount": "60.75456303", + // "out_amount": "0.1" + // } + axios.post(endpoint, data) .then(function (response) { - console.log('outAmount', response); + console.log('response from MyMonero for getOfferWithOutAmount', response); self.offer = response.data; self.offer.out_amount = out_amount; resolve(self.offer); @@ -104,9 +113,18 @@ class ExchangeFunctions { self.offer_type = "in_amount"; let endpoint = `${self.apiUrl}/get_offer`; return new Promise((resolve, reject) => { + + //sample response + // { + // "offer_id": "d604-1-lZDUeP+JpbSTxZ8gOCHLwR7WL8ETr1Jbkji7NqVlFVXusF4oQ+e/3uL8/2jL2dfoCqjTf2X9NIrBg4QrrB2XHh9XmlLEL6i1IR9B72qDnZ0Dzamr9AlMJDcm4cxfxDA8BzYN8x2n7BjT+dZgUJszBVivKZwpHMaPWAVUso11QIda/0I/o2NsjjRGoolJ/46tVncCOJGi6PhO9H4qPwSHkg==", + // "expires_at": "2024-11-24T20:35:43.541Z", + // "in_amount": "1", + // "out_amount": "0.00162612" + // } + axios.post(endpoint, data) .then(function (response) { - console.log('resp from getOfferwithtinamount', response); + console.log('response from MyMonero for getOfferwithtinamount', response); self.offer = response.data; resolve(self.offer); }) @@ -167,6 +185,24 @@ class ExchangeFunctions { let endpoint = `${self.apiUrl}/order_status`; return new Promise((resolve, reject) => { + + // sample response + // { + // "order_id": "d604-662e2c5f78f8b2", + // "expires_at": "2024-11-24T21:52:15.460Z", + // "in_address": "8B96tmezmMhWVwUdaYe87oY9y7SP3YXND3HmGYZKACyYFJYh9wG4LaVKCzWrpQmUvHYQZxmsUERJaidhtRSRFKMmUQVQ7zm", + // "in_currency": "XMR", + // "in_amount": "0.5", + // "out_currency": "BTC", + // "out_amount": "0.00079765", + // "status": "NEW", + // "in_amount_remaining": "0.5", + // "out_address": "bc1pp83964w2q3nun33gd79gg54n3zfudmfzn6ar4spk8g9dc86mtv3s6me7lq", + // "provider_name": "ChangeNOW", + // "provider_url": "https://support.changenow.io", + // "provider_order_id": "662e2c5f78f8b2" + // } + let data = { "order_id": self.order.data.order_id } @@ -209,6 +245,18 @@ class ExchangeFunctions { delete data.out_amount; } return new Promise((resolve, reject) => { + + // sample response + // { + // "order_id": "d604-662e2c5f78f8b2", + // "expires_at": "2024-11-24T21:52:15.460Z", + // "in_address": "8B96tmezmMhWVwUdaYe87oY9y7SP3YXND3HmGYZKACyYFJYh9wG4LaVKCzWrpQmUvHYQZxmsUERJaidhtRSRFKMmUQVQ7zm", + // "in_currency": "XMR", + // "in_amount": "0.1", + // "out_currency": "BTC", + // "out_amount": "0.00079765" + // } + try { axios.post(endpoint, data) .then(function (response) { diff --git a/packages/mymonero-exchange/package.json b/packages/mymonero-exchange/package.json index d2a8cf24..ee2a3beb 100644 --- a/packages/mymonero-exchange/package.json +++ b/packages/mymonero-exchange/package.json @@ -1,6 +1,6 @@ { "name": "@mymonero/mymonero-exchange", - "version": "3.0.3", + "version": "3.0.4", "description": "This module is used by the desktop and web versions of the MyMonero wallet to facilitate the exchange of XMR into other cryptocurrencies. It can be used to facilitate transactions where a user converts one cryptocurrency into another.", "main": "index.js", "scripts": { @@ -31,6 +31,5 @@ }, "publishConfig": { "access": "public" - }, - "gitHead": "8b2fb278e4a5aa84e577c9985fbca332fca4f1b0" + } } diff --git a/packages/mymonero-page-templates/index.esm.js b/packages/mymonero-page-templates/index.esm.js index f21f7956..23336ff4 100644 --- a/packages/mymonero-page-templates/index.esm.js +++ b/packages/mymonero-page-templates/index.esm.js @@ -2,6 +2,7 @@ require("@mymonero/mymonero-web-components"); /* Require various view elements */ +require("./src/Exchange/Elements/MajesticBankFloatingRateView"); require("./src/Exchange/Elements/ChangenowBuyWithFiatView"); require("./src/Exchange/Elements/ChangenowFixedRateView") require("./src/Exchange/Elements/ChangenowFloatingRateView"); diff --git a/packages/mymonero-page-templates/package.json b/packages/mymonero-page-templates/package.json index b54c0816..68112e2a 100644 --- a/packages/mymonero-page-templates/package.json +++ b/packages/mymonero-page-templates/package.json @@ -1,6 +1,6 @@ { "name": "@mymonero/mymonero-page-templates", - "version": "3.0.3", + "version": "3.0.4", "description": "This npm package contains a number of Lit element templates that MyMonero uses for various page layouts.", "main": "./lib/index", "exports": { @@ -30,8 +30,8 @@ "license": "BSD-3-Clause", "dependencies": { "@mymonero/changenow-exchange-integration": "^3.0.3", - "@mymonero/mymonero-exchange-helper": "^3.0.3", - "@mymonero/mymonero-web-components": "^3.0.3", + "@mymonero/mymonero-exchange-helper": "^3.0.4", + "@mymonero/mymonero-web-components": "^3.0.4", "lit": "*", "lit-html": "^1.4.1", "sweetalert2": "^11.3.0" @@ -44,6 +44,5 @@ }, "publishConfig": { "access": "public" - }, - "gitHead": "8b2fb278e4a5aa84e577c9985fbca332fca4f1b0" + } } diff --git a/packages/mymonero-page-templates/src/Exchange/Controllers/ExchangeNavigationController.js b/packages/mymonero-page-templates/src/Exchange/Controllers/ExchangeNavigationController.js index 9424e0f4..a1f8bd99 100644 --- a/packages/mymonero-page-templates/src/Exchange/Controllers/ExchangeNavigationController.js +++ b/packages/mymonero-page-templates/src/Exchange/Controllers/ExchangeNavigationController.js @@ -5,6 +5,7 @@ const ExchangeNavigationController = (superClass) => class extends superClass { /* class fields & methods to extend superClass with */ navigateToPage(destination) { let routeMap = { + "majesticbankFloatingRateView": "majesticbank-floating-rate-view", "changenowBuyWithFiatView": "changenow-buy-with-fiat-view", "changenowFixedRateView": "changenow-fixed-rate-view", "changenowFloatingRateView": "changenow-floating-rate-view", @@ -92,6 +93,7 @@ const ExchangeNavigationController = (superClass) => class extends superClass { selfNavigate(page) { let routeMap = { + "majesticbankFloatingRateView": "majesticbank-floating-rate-view", "changenowBuyWithFiatView": "changenow-buy-with-fiat-view", "changenowFixedRateView": "changenow-fixed-rate-view", "changenowFloatingRateView": "changenow-floating-rate-view" diff --git a/packages/mymonero-page-templates/src/Exchange/Elements/ChangenowFixedRateView.js b/packages/mymonero-page-templates/src/Exchange/Elements/ChangenowFixedRateView.js index b1939020..fbd1945d 100644 --- a/packages/mymonero-page-templates/src/Exchange/Elements/ChangenowFixedRateView.js +++ b/packages/mymonero-page-templates/src/Exchange/Elements/ChangenowFixedRateView.js @@ -12,8 +12,7 @@ const ExchangeUtils = require("../Utils/ExchangeUtilityFunctions") // const commonComponents_activityIndicators = require('../../MMAppUICommonComponents/activityIndicators.web') const JSBigInt = require('@mymonero/mymonero-bigint').BigInteger // important: grab defined export const monero_amount_format_utils = require('@mymonero/mymonero-money-format') -const ExchangeHelper = require("@mymonero/mymonero-exchange-helper") -let exchangeHelper = new ExchangeHelper(); +const ExchangeHelperMyMonero = require("@mymonero/mymonero-exchange-helper") // NB: because of legacy reasons, we don't want this to render inside a shadow dom. We override createRenderRoot to address this export class ChangenowFixedRateView extends ExchangeNavigationController(LitElement) { @@ -75,7 +74,7 @@ export class ChangenowFixedRateView extends ExchangeNavigationController(LitElem connectedCallback() { super.connectedCallback(); - exchangeHelper.doInit(this.context); + this.exchangeHelper.doInit(this.context); } sendFunds() { @@ -93,7 +92,7 @@ export class ChangenowFixedRateView extends ExchangeNavigationController(LitElem this.context.walletsListController.orderSent = false } - ExchangeUtils.default.sendFunds(this.context.walletsListController.records[selectorOffset], in_amount, send_address, sweep_wallet, exchangeHelper.sendFundsValidationStatusCallback, exchangeHelper.handleSendFundsResponseCallback, this.context) + ExchangeUtils.default.sendFunds(this.context.walletsListController.records[selectorOffset], in_amount, send_address, sweep_wallet, this.exchangeHelper.sendFundsValidationStatusCallback, this.exchangeHelper.handleSendFundsResponseCallback, this.context) } catch (error) { console.log(error) } @@ -101,7 +100,11 @@ export class ChangenowFixedRateView extends ExchangeNavigationController(LitElem constructor() { super(); + // Previously the exchangeHelper was initialized globally. This was causing some issues whenever this file was imported. + // To avoid those issues we are now initializing the exchangeHelper in the constructor. + // There seems to be no reason for the exchangeHelper to be global. this.clickHandler = this.clickHandler; + this.exchangeHelper = new ExchangeHelperMyMonero(); } clickHandler(event) { @@ -109,7 +112,7 @@ export class ChangenowFixedRateView extends ExchangeNavigationController(LitElem } render() { - let exchangeFormTemplate = exchangeHelper.htmlFormTemplate(); + let exchangeFormTemplate = this.exchangeHelper.htmlFormTemplate(); let exchangeFormHtml = exchangeFormTemplate.content.firstElementChild.cloneNode(true); return html` diff --git a/packages/mymonero-page-templates/src/Exchange/Elements/ExchangeLandingPage.js b/packages/mymonero-page-templates/src/Exchange/Elements/ExchangeLandingPage.js index cf097f2e..d5fdbfd1 100644 --- a/packages/mymonero-page-templates/src/Exchange/Elements/ExchangeLandingPage.js +++ b/packages/mymonero-page-templates/src/Exchange/Elements/ExchangeLandingPage.js @@ -36,6 +36,15 @@ export default class ExchangeLandingPage extends ExchangeNavigationController(Li this.clickHandler = this.clickHandler; this.context = {}; this.providerServices = [ + { + service_provider: "majesticbank", + title: "Exchange Monero for other cryptocurrencies (floating rate)", + description: ` + Use a floating rate when you want to take advantage of the best exchange rate. + This exchange provides the best privacy.`, + navigationType: "internalLink", + destination: "majesticbankFloatingRateView" + }, { service_provider: "changenow", title: "Exchange Monero for other cryptocurrencies (fixed rate)", diff --git a/packages/mymonero-page-templates/src/Exchange/Elements/MajesticBankFloatingRateView.js b/packages/mymonero-page-templates/src/Exchange/Elements/MajesticBankFloatingRateView.js new file mode 100644 index 00000000..046efbd7 --- /dev/null +++ b/packages/mymonero-page-templates/src/Exchange/Elements/MajesticBankFloatingRateView.js @@ -0,0 +1,184 @@ + + +import { html, css, LitElement } from 'lit'; +import ExchangeNavigationController from "../Controllers/ExchangeNavigationController"; + +// Legacy imports for fixed rate exchange +const Utils = require("../Utils/ExchangeUtilityFunctions") +const ExchangeUtils = require("../Utils/ExchangeUtilityFunctions") +// const ValidationLibrary = require('wallet-address-validator') +//const View = require('../../Views/View.web') +// const commonComponents_navigationBarButtons = require('../../MMAppUICommonComponents/navigationBarButtons.web') +// const commonComponents_activityIndicators = require('../../MMAppUICommonComponents/activityIndicators.web') +const JSBigInt = require('@mymonero/mymonero-bigint').BigInteger // important: grab defined export +const monero_amount_format_utils = require('@mymonero/mymonero-money-format') +const ExchangeHelperMajesticBank = require("@mymonero/mymonero-exchange-helper") + +// NB: because of legacy reasons, we don't want this to render inside a shadow dom. We override createRenderRoot to address this +export class MajesticBankFloatingRateView extends ExchangeNavigationController(LitElement) { + + static get styles() { + return css` + .submit-button-wrapper { + position: fixed; + top: -45px; + right: 16px; + width: 15%; + min-width: 41px; + height: 41px; + z-index: 12; + } + .submit-button { + z-index: 13; + position: fixed; + right: 16px; + font-weight: bold; + top: -40px; + z-index: 10000; + } + .submit-button, .confirmation-button { + cursor: default; + border-radius: 3px; + height: 24px; + font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Oxygen, Ubuntu, Cantarell, "Open Sans", "Helvetica Neue", sans-serif; + text-align: center; + border: none; + text-decoration: none; + line-height: 24px; + box-sizing: border-box; + width: auto; + padding: 0px 8px; + background-color: rgb(0, 198, 255); + box-shadow: rgb(22 20 22) 0px 0.5px 1px 0px, rgb(255 255 255 / 20%) 0px 0.5px 0px 0px inset; + color: rgb(22, 20, 22); + -webkit-font-smoothing: subpixel-antialiased; + font-size: 12px; + font-weight: bold; + letter-spacing: 0.5px; + float: right; + margin-top: 5px; + -webkit-app-region: no-drag; + } + ` + } + + createRenderRoot() { + return this; + } + + static get properties() { + return { + context: Object, + } + } + + connectedCallback() { + super.connectedCallback(); + this.exchangeHelper.doInit(this.context); + } + + sendFunds() { + const in_amount = document.getElementById('in_amount_remaining').innerHTML + const send_address = document.getElementById('receiving_subaddress').innerHTML + const in_amount_str = '' + in_amount + + const selectedWallet = document.getElementById('selected-wallet') + const selectorOffset = selectedWallet.dataset.walletoffset + const sweep_wallet = false // TODO: Add sweeping functionality + try { + if (this.context.walletsListController.hasOwnProperty('orderSent')) { + console.log('Order already sent previously') + } else { + this.context.walletsListController.orderSent = false + } + + ExchangeUtils.default.sendFunds(this.context.walletsListController.records[selectorOffset], in_amount, send_address, sweep_wallet, this.exchangeHelper.sendFundsValidationStatusCallback, this.exchangeHelper.handleSendFundsResponseCallback, this.context) + } catch (error) { + console.log(error) + } + } + + constructor() { + super(); + this.clickHandler = this.clickHandler; + this.exchangeHelper = new ExchangeHelperMajesticBank("majesticbank"); + } + + clickHandler(event) { + console.log(event); + } + + render() { + let exchangeFormTemplate = this.exchangeHelper.htmlFormTemplate(); + let exchangeFormHtml = exchangeFormTemplate.content.firstElementChild.cloneNode(true); + + return html` +
+
 
+ ${exchangeFormHtml} +
+ +
+ +
+ + + `; + } + +} + +try { + customElements.define('majesticbank-floating-rate-view', MajesticBankFloatingRateView); +} catch (error) { + // already defined +} \ No newline at end of file diff --git a/packages/mymonero-page-templates/src/index.js b/packages/mymonero-page-templates/src/index.js index 21930bcf..9e78a735 100644 --- a/packages/mymonero-page-templates/src/index.js +++ b/packages/mymonero-page-templates/src/index.js @@ -4,6 +4,7 @@ //require("@mymonero/mymonero-web-components"); /* Require various view elements */ +require("./Exchange/Elements/MajesticBankFloatingRateView"); require("./Exchange/Elements/ChangenowBuyWithFiatView"); require("./Exchange/Elements/ChangenowFixedRateView") require("./Exchange/Elements/ChangenowFloatingRateView"); diff --git a/packages/mymonero-web-components/package.json b/packages/mymonero-web-components/package.json index 7c545044..9dcff99f 100644 --- a/packages/mymonero-web-components/package.json +++ b/packages/mymonero-web-components/package.json @@ -1,6 +1,6 @@ { "name": "@mymonero/mymonero-web-components", - "version": "3.0.3", + "version": "3.0.4", "description": "This npm package contains a number of Lit web components.", "main": "./lib/index", "exports": { @@ -42,6 +42,5 @@ "@babel/core": "^7.5.5", "@babel/plugin-transform-runtime": "^7.15.0", "@babel/preset-env": "^7.5.5" - }, - "gitHead": "8b2fb278e4a5aa84e577c9985fbca332fca4f1b0" + } } diff --git a/packages/mymonero-web-components/src/Assets/ProviderCardImages.js b/packages/mymonero-web-components/src/Assets/ProviderCardImages.js index 4a563824..2b7e4f5f 100644 --- a/packages/mymonero-web-components/src/Assets/ProviderCardImages.js +++ b/packages/mymonero-web-components/src/Assets/ProviderCardImages.js @@ -1,6 +1,11 @@ +let MajesticbankLogo = 'url("")' let ChangenowLogo = 'url("");' let GuardarianLogo = 'url();' -let ProviderImages = { ChangenowLogo, GuardarianLogo } +let ProviderImages = { + MajesticbankLogo, + ChangenowLogo, + GuardarianLogo, +} // export let ProviderImages = { ChangenowLogo, GuardarianLogo } export { ProviderImages as default } diff --git a/packages/mymonero-web-components/src/Components/Shared/ProviderCard.js b/packages/mymonero-web-components/src/Components/Shared/ProviderCard.js index f0e39d2b..132d8ca8 100644 --- a/packages/mymonero-web-components/src/Components/Shared/ProviderCard.js +++ b/packages/mymonero-web-components/src/Components/Shared/ProviderCard.js @@ -49,6 +49,13 @@ export class ExchangeServiceProviderCard extends LitElement { left: 6px; top: 16px; } + div.majesticbank-logo { + background-image: url(""); + width: 120px; + background-size: contain; + background-repeat: no-repeat; + background-position: center; + } div.changenow-logo { //background-image: url("/src/assets/img/ChangeNow120x69.jpg"); background-image: url(""); @@ -82,7 +89,7 @@ export class ExchangeServiceProviderCard extends LitElement { max-width: 100% !important; margin-left: 10px !important; } - .changenow-logo, .guardarian-logo { + .majesticbank-logo, .changenow-logo, .guardarian-logo { width: 100% !important; height: 69px !important; background-position: top !important; @@ -182,8 +189,15 @@ export class ExchangeServiceProviderCard extends LitElement { } render() { + // Previously the provider-card had the following attributes: + // @click=${this.handleClickEvent} ontouchstart=${this.handleClickEvent} + // This was in addition to the event listeners in createRenderRoot + // However, the event listeners in createRenderRoot should be sufficient + // Since there was no clear reason for the doubling of event listeners they were removed + // The previous solution had the unfortunate side effect of causing the event to fire twice which created two instances of the same things + // This in turn resulted in duplicate API calls being made later on return html` -
+