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.

203 lines
6.1 KiB

import { ipcRenderer } from 'electron'
import { showNotification } from 'lib/utils/notifications'
import { btc } from 'lib/utils'
import { newAddress } from './address'
import { fetchBalance } from './balance'
import { setFormType } from './form'
import { resetPayForm } from './payform'
import { setError } from './error'
import { fetchChannels } from './channels'
// ------------------------------------
// Constants
// ------------------------------------
export const GET_TRANSACTIONS = 'GET_TRANSACTIONS'
export const RECEIVE_TRANSACTIONS = 'RECEIVE_TRANSACTIONS'
export const SEND_TRANSACTION = 'SEND_TRANSACTION'
export const TRANSACTION_SUCCESSFULL = 'TRANSACTION_SUCCESSFULL'
export const TRANSACTION_FAILED = 'TRANSACTION_FAILED'
export const ADD_TRANSACTION = 'ADD_TRANSACTION'
export const SHOW_SUCCESS_TRANSACTION_SCREEN = 'SHOW_SUCCESS_TRANSACTION_SCREEN'
export const HIDE_SUCCESS_TRANSACTION_SCREEN = 'HIDE_SUCCESS_TRANSACTION_SCREEN'
// ------------------------------------
// Helpers
// ------------------------------------
// Decorate transaction object with custom/computed properties.
const decorateTransaction = transaction => {
transaction.received = transaction.amount > 0
return transaction
}
// ------------------------------------
// Actions
// ------------------------------------
export function getTransactions() {
return {
type: GET_TRANSACTIONS
}
}
export function sendTransaction() {
return {
type: SEND_TRANSACTION
}
}
export function showSuccessTransactionScreen(txid) {
return {
type: SHOW_SUCCESS_TRANSACTION_SCREEN,
txid
}
}
export function hideSuccessTransactionScreen() {
return {
type: HIDE_SUCCESS_TRANSACTION_SCREEN
}
}
// Send IPC event for payments
export const fetchTransactions = () => dispatch => {
dispatch(getTransactions())
ipcRenderer.send('lnd', { msg: 'transactions' })
}
// Receive IPC event for payments
export const receiveTransactions = (event, { transactions }) => (dispatch, getState) => {
dispatch({ type: RECEIVE_TRANSACTIONS, transactions })
// If our current wallet address has been used, generate a new one.
const state = getState()
const currentAddress = state.address.address
let usedAddresses = []
transactions.forEach(transaction => {
decorateTransaction(transaction)
usedAddresses = usedAddresses.concat(transaction.dest_addresses)
})
if (usedAddresses.includes(currentAddress)) {
dispatch(newAddress('np2wkh'))
}
// fetch new balance
dispatch(fetchBalance())
}
export const sendCoins = ({ value, addr, currency }) => dispatch => {
// backend needs amount in satoshis no matter what currency we are using
const amount = btc.convert(currency, 'sats', value)
// submit the transaction to LND
dispatch(sendTransaction())
ipcRenderer.send('lnd', { msg: 'sendCoins', data: { amount, addr } })
// Close the form modal once the payment was sent to LND
// we will do the loading/success UX on the main page
// so we aren't blocking the user
dispatch(setFormType(null))
}
// Receive IPC event for successful payment
// TODO: Add payment to state, not a total re-fetch
export const transactionSuccessful = (event, { txid }) => dispatch => {
// Get the new list of transactions (TODO dont do an entire new fetch)
dispatch(fetchTransactions())
// Show successful payment state
dispatch({ type: TRANSACTION_SUCCESSFULL })
// Show successful tx state for 5 seconds
dispatch(showSuccessTransactionScreen(txid))
setTimeout(() => dispatch(hideSuccessTransactionScreen()), 5000)
// Reset the payment form
dispatch(resetPayForm())
}
export const transactionError = (event, { error }) => dispatch => {
dispatch({ type: TRANSACTION_FAILED })
dispatch(setError(error))
}
// Listener for when a new transaction is pushed from the subscriber
export const newTransaction = (event, { transaction }) => (dispatch, getState) => {
// add the transaction only if we are not already aware of it
const state = getState()
if (
!state.transaction ||
!state.transaction.transactions ||
!state.transaction.transactions.find(tx => tx.tx_hash === transaction.tx_hash)
) {
decorateTransaction(transaction)
dispatch({ type: ADD_TRANSACTION, transaction })
// fetch updated channels
dispatch(fetchChannels())
// fetch new balance
dispatch(fetchBalance())
// HTML 5 desktop notification for the new transaction
if (transaction.received) {
showNotification(
'On-chain Transaction Received!',
"Lucky you, you just received a new on-chain transaction. I'm jealous."
)
dispatch(newAddress('np2wkh')) // Generate a new address
} else {
showNotification(
'On-chain Transaction Sent!',
"Hate to see 'em go but love to watch 'em leave. Your on-chain transaction successfully sent."
)
}
}
}
// ------------------------------------
// Action Handlers
// ------------------------------------
const ACTION_HANDLERS = {
[GET_TRANSACTIONS]: state => ({ ...state, transactionLoading: true }),
[SEND_TRANSACTION]: state => ({ ...state, sendingTransaction: true }),
[RECEIVE_TRANSACTIONS]: (state, { transactions }) => ({
...state,
transactionLoading: false,
transactions
}),
[TRANSACTION_SUCCESSFULL]: state => ({ ...state, sendingTransaction: false }),
[TRANSACTION_FAILED]: state => ({ ...state, sendingTransaction: false }),
[ADD_TRANSACTION]: (state, { transaction }) => ({
...state,
transactions: [transaction, ...state.transactions]
}),
[SHOW_SUCCESS_TRANSACTION_SCREEN]: (state, { txid }) => ({
...state,
successTransactionScreen: { show: true, txid }
}),
[HIDE_SUCCESS_TRANSACTION_SCREEN]: state => ({
...state,
successTransactionScreen: { show: false, txid: '' }
})
}
// ------------------------------------
// Reducer
// ------------------------------------
const initialState = {
sendingTransaction: false,
transactionLoading: false,
transactions: [],
successTransactionScreen: {
show: false,
txid: ''
}
}
export default function transactionReducer(state = initialState, action) {
const handler = ACTION_HANDLERS[action.type]
return handler ? handler(state, action) : state
}