|
|
|
import { createSelector } from 'reselect'
|
|
|
|
import { ipcRenderer } from 'electron'
|
|
|
|
import { push } from 'react-router-redux'
|
|
|
|
import Store from 'electron-store'
|
|
|
|
|
|
|
|
import { showNotification } from 'lib/utils/notifications'
|
|
|
|
import { btc } from 'lib/utils'
|
|
|
|
|
|
|
|
import { showActivityModal } from './activity'
|
|
|
|
import { fetchBalance } from './balance'
|
|
|
|
import { setFormType } from './form'
|
|
|
|
import { setPayInvoice } from './payform'
|
|
|
|
import { resetRequestForm } from './requestform'
|
|
|
|
import { setError } from './error'
|
|
|
|
|
|
|
|
// ------------------------------------
|
|
|
|
// 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'
|
|
|
|
|
|
|
|
// ------------------------------------
|
|
|
|
// Helpers
|
|
|
|
// ------------------------------------
|
|
|
|
|
|
|
|
// Decorate invoice object with custom/computed properties.
|
|
|
|
const decorateInvoice = invoice => {
|
|
|
|
invoice.finalAmount = invoice.value
|
|
|
|
if (invoice.amt_paid) {
|
|
|
|
invoice.finalAmount = btc.millisatoshisToSatoshis(invoice.amt_paid)
|
|
|
|
}
|
|
|
|
return 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 })
|
|
|
|
invoices.forEach(decorateInvoice)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Send IPC event for creating an invoice
|
|
|
|
export const createInvoice = (amount, memo, currency) => dispatch => {
|
|
|
|
// backend needs value in satoshis no matter what currency we are using
|
|
|
|
const value = btc.convert(currency, 'sats', amount)
|
|
|
|
|
|
|
|
dispatch(sendInvoice())
|
|
|
|
|
|
|
|
// Grab the activeConnection type from our local store. If the active connection type is local (light clients using
|
|
|
|
// neutrino) we will have to flag private as true when creating this invoice. All light cliets open private channels
|
|
|
|
// (both manual and autopilot ones). In order for these clients to receive money through these channels the invoices
|
|
|
|
// need to come with routing hints for private channels
|
|
|
|
const store = new Store({ name: 'settings' })
|
|
|
|
const { type } = store.get('activeConnection', {})
|
|
|
|
|
|
|
|
ipcRenderer.send('lnd', {
|
|
|
|
msg: 'createInvoice',
|
|
|
|
data: { value, memo, private: type === 'local' }
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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))
|
|
|
|
|
|
|
|
decorateInvoice(invoice)
|
|
|
|
|
|
|
|
// Add new invoice to invoices list
|
|
|
|
dispatch({ type: INVOICE_SUCCESSFUL, invoice })
|
|
|
|
|
|
|
|
// Reset the payment form
|
|
|
|
dispatch(resetRequestForm())
|
|
|
|
|
|
|
|
// Transition to wallet route
|
|
|
|
dispatch(push('/'))
|
|
|
|
|
|
|
|
// Set invoice modal to newly created invoice
|
|
|
|
dispatch(showActivityModal('INVOICE', invoice.payment_request))
|
|
|
|
}
|
|
|
|
|
|
|
|
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())
|
|
|
|
|
|
|
|
decorateInvoice(invoice)
|
|
|
|
|
|
|
|
if (invoice.settled) {
|
|
|
|
// 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))
|
|
|
|
)
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|