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 { 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, targetConf, satPerByte }) => 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, currency, target_conf: targetConf, sat_per_byte: satPerByte } }) // 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) } 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 }