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
}