import { createSelector } from 'reselect' import { ipcRenderer } from 'electron' import { fetchBalance } from './balance' import { setFormType } from './form' import { setPayInvoice } from './payform' import { resetRequestForm } from './requestform' import { setError } from './error' import { showNotification } from '../notifications' import { btc, usd } from '../utils' // ------------------------------------ // Constants // ------------------------------------ export const SEARCH_INVOICES = 'SEARCH_INVOICES' export const SET_INVOICE = 'SET_INVOICE' export const GET_INVOICE = 'GET_INVOICE' export const RECEIVE_INVOICE = 'RECEIVE_INVOICE' export const RECEIVE_FORM_INVOICE = 'RECEIVE_FORM_INVOICE' export const GET_INVOICES = 'GET_INVOICES' export const RECEIVE_INVOICES = 'RECEIVE_INVOICES' export const SEND_INVOICE = 'SEND_INVOICE' export const INVOICE_SUCCESSFUL = 'INVOICE_SUCCESSFUL' export const INVOICE_FAILED = 'INVOICE_FAILED' export const UPDATE_INVOICE = 'UPDATE_INVOICE' // ------------------------------------ // Actions // ------------------------------------ export function searchInvoices(invoicesSearchText) { return { type: SEARCH_INVOICES, invoicesSearchText } } export function setInvoice(invoice) { return { type: SET_INVOICE, invoice } } export function getInvoice() { return { type: GET_INVOICE } } export function receiveInvoice(invoice) { return { type: RECEIVE_INVOICE, invoice } } export function getInvoices() { return { type: GET_INVOICES } } export function sendInvoice() { return { type: SEND_INVOICE } } // Send IPC event for a specific invoice export const fetchInvoice = payreq => (dispatch) => { dispatch(getInvoice()) ipcRenderer.send('lnd', { msg: 'invoice', data: { payreq } }) } // Receive IPC event for form invoice export const receiveFormInvoice = (event, invoice) => (dispatch) => { dispatch(setPayInvoice(invoice)) dispatch({ type: RECEIVE_FORM_INVOICE }) } // Send IPC event for invoices export const fetchInvoices = () => (dispatch) => { dispatch(getInvoices()) ipcRenderer.send('lnd', { msg: 'invoices' }) } // Receive IPC event for invoices export const receiveInvoices = (event, { invoices }) => dispatch => dispatch({ type: RECEIVE_INVOICES, invoices }) // Send IPC event for creating an invoice export const createInvoice = (amount, memo, currency, rate) => (dispatch) => { const value = currency === 'btc' ? btc.btcToSatoshis(amount) : btc.btcToSatoshis(usd.usdToBtc(amount, rate)) dispatch(sendInvoice()) ipcRenderer.send('lnd', { msg: 'createInvoice', data: { value, memo } }) } // Receive IPC event for newly created invoice export const createdInvoice = (event, invoice) => (dispatch) => { // Close the form modal once the payment was succesful dispatch(setFormType(null)) // Add new invoice to invoices list dispatch({ type: INVOICE_SUCCESSFUL, invoice }) // Reset the payment form dispatch(resetRequestForm()) } export const invoiceFailed = (event, { error }) => dispatch => { dispatch({ type: INVOICE_FAILED }) dispatch(setError(error)) } // Listen for invoice updates pushed from backend from subscribeToInvoices export const invoiceUpdate = (event, { invoice }) => (dispatch) => { dispatch({ type: UPDATE_INVOICE, invoice }) // Fetch new balance dispatch(fetchBalance()) // HTML 5 desktop notification for the invoice update const notifTitle = 'You\'ve been Zapped' const notifBody = 'Congrats, someone just paid an invoice of yours' showNotification(notifTitle, notifBody) } // ------------------------------------ // Action Handlers // ------------------------------------ const ACTION_HANDLERS = { [SEARCH_INVOICES]: (state, { invoicesSearchText }) => ({ ...state, invoicesSearchText }), [SET_INVOICE]: (state, { invoice }) => ({ ...state, invoice }), [GET_INVOICE]: state => ({ ...state, invoiceLoading: true }), [RECEIVE_INVOICE]: (state, { invoice }) => ({ ...state, invoiceLoading: false, invoice }), [RECEIVE_FORM_INVOICE]: state => ({ ...state, invoiceLoading: false }), [GET_INVOICES]: state => ({ ...state, invoiceLoading: true }), [RECEIVE_INVOICES]: (state, { invoices }) => ({ ...state, invoiceLoading: false, invoices }), [SEND_INVOICE]: state => ({ ...state, invoiceLoading: true }), [INVOICE_SUCCESSFUL]: (state, { invoice }) => ( { ...state, invoiceLoading: false, invoices: [invoice, ...state.invoices] } ), [INVOICE_FAILED]: state => ({ ...state, invoiceLoading: false, data: null }), [UPDATE_INVOICE]: (state, action) => { const updatedInvoices = state.invoices.map((invoice) => { if (invoice.r_hash.toString('hex') !== action.invoice.r_hash.toString('hex')) { return invoice } return { ...invoice, ...action.invoice } }) return { ...state, invoices: updatedInvoices } } } const invoiceSelectors = {} const invoiceSelector = state => state.invoice.invoice const invoicesSelector = state => state.invoice.invoices const invoicesSearchTextSelector = state => state.invoice.invoicesSearchText invoiceSelectors.invoiceModalOpen = createSelector( invoiceSelector, invoice => (!!invoice) ) invoiceSelectors.invoices = createSelector( invoicesSelector, invoicesSearchTextSelector, (invoices, invoicesSearchText) => invoices.filter(invoice => invoice.memo.includes(invoicesSearchText)) ) export { invoiceSelectors } // ------------------------------------ // Reducer // ------------------------------------ const initialState = { invoiceLoading: false, invoices: [], invoice: null, invoicesSearchText: '', data: {}, formInvoice: { payreq: '', r_hash: '', amount: '0' } } export default function invoiceReducer(state = initialState, action) { const handler = ACTION_HANDLERS[action.type] return handler ? handler(state, action) : state }