import { ipcRenderer } from 'electron' import { showNotification } from '../notifications' import { btc } from '../utils' import { newAddress } from './address' import { fetchBalance } from './balance' import { setFormType } from './form' import { resetPayForm } from './payform' import { setError } from './error' // ------------------------------------ // 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' // ------------------------------------ // 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 => dispatch({ type: RECEIVE_TRANSACTIONS, transactions }) 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) // Fetch new balance dispatch(fetchBalance()) // 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) => { // Fetch new balance dispatch(fetchBalance()) dispatch({ type: ADD_TRANSACTION, transaction }) // HTML 5 desktop notification for the new transaction const notifTitle = transaction.amount > 0 ? 'On-chain Transaction Received!' : 'On-chain Transaction Sent!' const notifBody = transaction.amount > 0 ? 'Lucky you, you just received a new on-chain transaction. I\'m jealous.' : 'Hate to see \'em go but love to watch \'em leave. Your on-chain transaction successfully sent.' // eslint-disable-line max-len showNotification(notifTitle, notifBody) // Generate a new address dispatch(newAddress('p2pkh')) } // ------------------------------------ // 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 }) => ( // add the transaction only if we are not already aware of it state.transactions.find(tx => (tx.tx_hash === transaction.tx_hash)) ? state : { ...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 }