You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
259 lines
6.2 KiB
259 lines
6.2 KiB
import { createSelector } from 'reselect'
|
|
import bitcoin from 'bitcoinjs-lib'
|
|
|
|
import isEmpty from 'lodash/isEmpty'
|
|
|
|
import { setFormType } from './form'
|
|
import { tickerSelectors } from './ticker'
|
|
import { infoSelectors } from './info'
|
|
import { btc, bech32 } from '../utils'
|
|
|
|
// Initial State
|
|
const initialState = {
|
|
amount: '',
|
|
payInput: '',
|
|
|
|
invoice: {
|
|
payreq: '',
|
|
r_hash: '',
|
|
amount: '0',
|
|
description: '',
|
|
destination: ''
|
|
},
|
|
|
|
showCurrencyFilters: false,
|
|
|
|
showErrors: {
|
|
amount: false,
|
|
payInput: false
|
|
}
|
|
}
|
|
|
|
// Constants
|
|
// ------------------------------------
|
|
export const SET_PAY_AMOUNT = 'SET_PAY_AMOUNT'
|
|
export const SET_PAY_INPUT = 'SET_PAY_INPUT'
|
|
export const SET_PAY_INVOICE = 'SET_PAY_INVOICE'
|
|
|
|
export const SET_PAY_CURRENCY_FILTERS = 'SET_PAY_CURRENCY_FILTERS'
|
|
|
|
export const UPDATE_PAY_ERRORS = 'UPDATE_PAY_ERRORS'
|
|
|
|
export const RESET_FORM = 'RESET_FORM'
|
|
|
|
// ------------------------------------
|
|
// Actions
|
|
// ------------------------------------
|
|
export function setPayAmount(amount) {
|
|
return {
|
|
type: SET_PAY_AMOUNT,
|
|
amount
|
|
}
|
|
}
|
|
|
|
export function setPayInput(payInput) {
|
|
return {
|
|
type: SET_PAY_INPUT,
|
|
payInput
|
|
}
|
|
}
|
|
|
|
export function setPayInvoice(invoice) {
|
|
return {
|
|
type: SET_PAY_INVOICE,
|
|
invoice
|
|
}
|
|
}
|
|
|
|
export function setCurrencyFilters(showCurrencyFilters) {
|
|
return {
|
|
type: SET_PAY_CURRENCY_FILTERS,
|
|
showCurrencyFilters
|
|
}
|
|
}
|
|
|
|
export function updatePayErrors(errorsObject) {
|
|
return {
|
|
type: UPDATE_PAY_ERRORS,
|
|
errorsObject
|
|
}
|
|
}
|
|
|
|
export const lightningPaymentUri = (event, { payreq }) => dispatch => {
|
|
// Open pay form
|
|
dispatch(setFormType('PAY_FORM'))
|
|
// Set payreq
|
|
dispatch(setPayInput(payreq))
|
|
}
|
|
|
|
export function resetPayForm() {
|
|
return {
|
|
type: RESET_FORM
|
|
}
|
|
}
|
|
|
|
// ------------------------------------
|
|
// Action Handlers
|
|
// ------------------------------------
|
|
const ACTION_HANDLERS = {
|
|
[SET_PAY_AMOUNT]: (state, { amount }) => ({ ...state, amount, showErrors: Object.assign(state.showErrors, { amount: false }) }),
|
|
[SET_PAY_INPUT]: (state, { payInput }) => ({ ...state, payInput, showErrors: Object.assign(state.showErrors, { payInput: false }) }),
|
|
[SET_PAY_INVOICE]: (state, { invoice }) => ({ ...state, invoice, showErrors: Object.assign(state.showErrors, { amount: false }) }),
|
|
[SET_PAY_CURRENCY_FILTERS]: (state, { showCurrencyFilters }) => ({ ...state, showCurrencyFilters }),
|
|
|
|
[UPDATE_PAY_ERRORS]: (state, { errorsObject }) => ({ ...state, showErrors: Object.assign(state.showErrors, errorsObject) }),
|
|
|
|
[RESET_FORM]: () => initialState
|
|
}
|
|
|
|
// ------------------------------------
|
|
// Selector
|
|
// ------------------------------------
|
|
const payFormSelectors = {}
|
|
const payAmountSelector = state => state.payform.amount
|
|
const payInputSelector = state => state.payform.payInput
|
|
const payInvoiceSelector = state => state.payform.invoice
|
|
|
|
// transaction
|
|
const sendingTransactionSelector = state => state.transaction.sendingTransaction
|
|
|
|
// payment
|
|
const sendingPaymentSelector = state => state.payment.sendingPayment
|
|
|
|
// ticker
|
|
const currencySelector = state => state.ticker.currency
|
|
|
|
payFormSelectors.isOnchain = createSelector(payInputSelector, infoSelectors.networkSelector, (input, network) => {
|
|
try {
|
|
bitcoin.address.toOutputScript(input, network.bitcoinJsNetwork)
|
|
return true
|
|
} catch (e) {
|
|
return false
|
|
}
|
|
})
|
|
|
|
payFormSelectors.isLn = createSelector(payInputSelector, input => {
|
|
if (!input.startsWith('ln')) {
|
|
return false
|
|
}
|
|
|
|
try {
|
|
bech32.decode(input)
|
|
return true
|
|
} catch (e) {
|
|
return false
|
|
}
|
|
})
|
|
|
|
payFormSelectors.currentAmount = createSelector(
|
|
payFormSelectors.isLn,
|
|
payAmountSelector,
|
|
payInvoiceSelector,
|
|
currencySelector,
|
|
(isLn, amount, invoice, currency) => {
|
|
if (isLn) {
|
|
switch (currency) {
|
|
case 'btc':
|
|
return btc.satoshisToBtc(invoice.num_satoshis || 0)
|
|
case 'bits':
|
|
return btc.satoshisToBits(invoice.num_satoshis || 0)
|
|
case 'sats':
|
|
return invoice.num_satoshis
|
|
default:
|
|
return invoice.num_satoshis
|
|
}
|
|
}
|
|
|
|
return amount
|
|
}
|
|
)
|
|
|
|
payFormSelectors.usdAmount = createSelector(
|
|
payFormSelectors.isLn,
|
|
payAmountSelector,
|
|
payInvoiceSelector,
|
|
currencySelector,
|
|
tickerSelectors.currentTicker,
|
|
(isLn, amount, invoice, currency, ticker) => {
|
|
if (!ticker || !ticker.price_usd) {
|
|
return false
|
|
}
|
|
|
|
if (isLn) {
|
|
return btc.satoshisToUsd(invoice.num_satoshis || 0, ticker.price_usd)
|
|
}
|
|
|
|
return btc.convert(currency, 'usd', amount, ticker.price_usd)
|
|
}
|
|
)
|
|
|
|
payFormSelectors.payInputMin = createSelector(currencySelector, currency => {
|
|
switch (currency) {
|
|
case 'btc':
|
|
return '0.00000001'
|
|
case 'bits':
|
|
return '0.01'
|
|
case 'sats':
|
|
return '1'
|
|
default:
|
|
return '0'
|
|
}
|
|
})
|
|
|
|
payFormSelectors.inputCaption = createSelector(
|
|
payFormSelectors.isOnchain,
|
|
payFormSelectors.isLn,
|
|
payFormSelectors.currentAmount,
|
|
currencySelector,
|
|
(isOnchain, isLn, amount, currency) => {
|
|
if (!isOnchain && !isLn) {
|
|
return ''
|
|
}
|
|
|
|
if (isOnchain) {
|
|
return `You're about to send ${amount} ${currency.toUpperCase()} on-chain which should take around 10 minutes`
|
|
}
|
|
|
|
if (isLn) {
|
|
return `You're about to send ${amount} ${currency.toUpperCase()} over the Lightning Network which will be instant`
|
|
}
|
|
|
|
return ''
|
|
}
|
|
)
|
|
|
|
payFormSelectors.showPayLoadingScreen = createSelector(
|
|
sendingTransactionSelector,
|
|
sendingPaymentSelector,
|
|
(sendingTransaction, sendingPayment) => sendingTransaction || sendingPayment
|
|
)
|
|
|
|
payFormSelectors.payFormIsValid = createSelector(payFormSelectors.isOnchain, payFormSelectors.isLn, payAmountSelector, (isOnchain, isLn, amount) => {
|
|
const errors = {}
|
|
|
|
if (!isLn && amount <= 0) {
|
|
errors.amount = 'Amount must be more than 0'
|
|
}
|
|
|
|
if (!isOnchain && !isLn) {
|
|
errors.payInput = 'Must be a valid BTC address or Lightning Network request'
|
|
}
|
|
|
|
return {
|
|
errors,
|
|
amountIsValid: isEmpty(errors.amount),
|
|
payInputIsValid: isEmpty(errors.payInput),
|
|
isValid: isEmpty(errors)
|
|
}
|
|
})
|
|
|
|
export { payFormSelectors }
|
|
|
|
// ------------------------------------
|
|
// Reducer
|
|
// ------------------------------------
|
|
export default function payFormReducer(state = initialState, action) {
|
|
const handler = ACTION_HANDLERS[action.type]
|
|
|
|
return handler ? handler(state, action) : state
|
|
}
|
|
|