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.
 
 
 

279 lines
6.3 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
}