import { withRouter } from 'react-router-dom' import { connect } from 'react-redux' import get from 'lodash.get' import { btc } from 'lib/utils' import { fetchTicker, setCurrency, tickerSelectors } from 'reducers/ticker' import { closeWalletModal } from 'reducers/address' import { fetchInfo } from 'reducers/info' import { setFormType } from 'reducers/form' import { setPayAmount, setPayInput, setCurrencyFilters, updatePayErrors, payFormSelectors } from 'reducers/payform' import { setRequestAmount, setRequestMemo, setRequestCurrencyFilters, requestFormSelectors } from 'reducers/requestform' import { sendCoins } from 'reducers/transaction' import { payInvoice } from 'reducers/payment' import { createInvoice, fetchInvoice } from 'reducers/invoice' import { lndSelectors } from 'reducers/lnd' import { fetchChannels, fetchSuggestedNodes, openChannel, closeChannel, channelsSelectors, currentChannels, toggleFilterPulldown, changeFilter, updateChannelSearchQuery, setSelectedChannel } from 'reducers/channels' import { openContactsForm, closeContactsForm, setChannelFormType, openManualForm, closeManualForm, openSubmitChannelForm, closeSubmitChannelForm, updateContactFormSearchQuery, updateManualFormSearchQuery, updateContactCapacity, setNode, contactFormSelectors, updateManualFormErrors, setContactsCurrencyFilters } from 'reducers/contactsform' import { fetchBalance } from 'reducers/balance' import { fetchDescribeNetwork } from 'reducers/network' import { clearError } from 'reducers/error' import { hideActivityModal, setActivityModalCurrencyFilters, activitySelectors } from 'reducers/activity' import App from 'components/App' const mapDispatchToProps = { fetchTicker, setCurrency, closeWalletModal, fetchInfo, setFormType, setPayAmount, setPayInput, setCurrencyFilters, updatePayErrors, setRequestAmount, setRequestMemo, setRequestCurrencyFilters, sendCoins, payInvoice, createInvoice, fetchInvoice, clearError, fetchBalance, fetchChannels, fetchSuggestedNodes, openChannel, closeChannel, toggleFilterPulldown, changeFilter, updateChannelSearchQuery, setSelectedChannel, openContactsForm, closeContactsForm, openSubmitChannelForm, closeSubmitChannelForm, openManualForm, closeManualForm, updateContactFormSearchQuery, updateManualFormSearchQuery, updateContactCapacity, setNode, contactFormSelectors, updateManualFormErrors, setContactsCurrencyFilters, setChannelFormType, fetchDescribeNetwork, hideActivityModal, setActivityModalCurrencyFilters } const mapStateToProps = state => ({ activity: state.activity, lnd: state.lnd, ticker: state.ticker, address: state.address, info: state.info, payment: state.payment, transaction: state.transaction, peers: state.peers, channels: state.channels, contactsform: state.contactsform, balance: state.balance, form: state.form, payform: state.payform, requestform: state.requestform, invoice: state.invoice, error: state.error, network: state.network, settings: state.settings, activityModalItem: activitySelectors.activityModalItem(state), currentTicker: tickerSelectors.currentTicker(state), currentCurrencyFilters: tickerSelectors.currentCurrencyFilters(state), currencyName: tickerSelectors.currencyName(state), isOnchain: payFormSelectors.isOnchain(state), isLn: payFormSelectors.isLn(state), currentAmount: payFormSelectors.currentAmount(state), fiatAmount: payFormSelectors.fiatAmount(state), inputCaption: payFormSelectors.inputCaption(state), showPayLoadingScreen: payFormSelectors.showPayLoadingScreen(state), payFormIsValid: payFormSelectors.payFormIsValid(state), payInputMin: payFormSelectors.payInputMin(state), requestFiatAmount: requestFormSelectors.fiatAmount(state), syncPercentage: lndSelectors.syncPercentage(state), filteredNetworkNodes: contactFormSelectors.filteredNetworkNodes(state), showManualForm: contactFormSelectors.showManualForm(state), manualFormIsValid: contactFormSelectors.manualFormIsValid(state), contactFormFiatAmount: contactFormSelectors.contactFormFiatAmount(state), dupeChanInfo: contactFormSelectors.dupeChanInfo(state), currentChannels: currentChannels(state), activeChannelPubkeys: channelsSelectors.activeChannelPubkeys(state), nonActiveChannelPubkeys: channelsSelectors.nonActiveChannelPubkeys(state), pendingOpenChannelPubkeys: channelsSelectors.pendingOpenChannelPubkeys(state), nonActiveFilters: channelsSelectors.nonActiveFilters(state), channelNodes: channelsSelectors.channelNodes(state) }) const mergeProps = (stateProps, dispatchProps, ownProps) => { const payFormProps = { payform: stateProps.payform, currency: stateProps.ticker.currency, crypto: stateProps.ticker.crypto, nodes: stateProps.network.nodes, ticker: stateProps.ticker, isOnchain: stateProps.isOnchain, isLn: stateProps.isLn, currentAmount: stateProps.currentAmount, fiatAmount: stateProps.fiatAmount, inputCaption: stateProps.inputCaption, showPayLoadingScreen: stateProps.showPayLoadingScreen, payFormIsValid: stateProps.payFormIsValid, payInputMin: stateProps.payInputMin, currentCurrencyFilters: stateProps.currentCurrencyFilters, currencyName: stateProps.currencyName, setPayAmount: dispatchProps.setPayAmount, setPayInput: dispatchProps.setPayInput, setCurrencyFilters: dispatchProps.setCurrencyFilters, fetchInvoice: dispatchProps.fetchInvoice, setCurrency: dispatchProps.setCurrency, onPayAmountBlur: () => { // If the amount is now valid and showErrors was on, turn it off if (stateProps.payFormIsValid.amountIsValid && stateProps.payform.showErrors.amount) { dispatchProps.updatePayErrors({ amount: false }) } // If the amount is not valid and showErrors was off, turn it on if (!stateProps.payFormIsValid.amountIsValid && !stateProps.payform.showErrors.amount) { dispatchProps.updatePayErrors({ amount: true }) } }, onPayInputBlur: () => { // If the input is now valid and showErrors was on, turn it off if (stateProps.payFormIsValid.payInputIsValid && stateProps.payform.showErrors.payInput) { dispatchProps.updatePayErrors({ payInput: false }) } // If the input is not valid and showErrors was off, turn it on if (!stateProps.payFormIsValid.payInputIsValid && !stateProps.payform.showErrors.payInput) { dispatchProps.updatePayErrors({ payInput: true }) } }, onPaySubmit: () => { if (!stateProps.payFormIsValid.isValid) { dispatchProps.updatePayErrors({ amount: Object.prototype.hasOwnProperty.call(stateProps.payFormIsValid.errors, 'amount'), payInput: Object.prototype.hasOwnProperty.call( stateProps.payFormIsValid.errors, 'payInput' ) }) return } if (stateProps.isOnchain) { dispatchProps.sendCoins({ value: stateProps.payform.amount, addr: stateProps.payform.payInput, currency: stateProps.ticker.currency }) } if (stateProps.isLn) { dispatchProps.payInvoice(stateProps.payform.payInput) } } } const requestFormProps = { requestform: stateProps.requestform, ticker: stateProps.ticker, currentCurrencyFilters: stateProps.currentCurrencyFilters, showCurrencyFilters: stateProps.showCurrencyFilters, currencyName: stateProps.currencyName, requestFiatAmount: stateProps.requestFiatAmount, setRequestAmount: dispatchProps.setRequestAmount, setRequestMemo: dispatchProps.setRequestMemo, setCurrency: dispatchProps.setCurrency, setRequestCurrencyFilters: dispatchProps.setRequestCurrencyFilters, onRequestSubmit: () => dispatchProps.createInvoice( stateProps.requestform.amount, stateProps.requestform.memo, stateProps.ticker.currency ) } const formProps = formType => { if (!formType) { return {} } if (formType === 'PAY_FORM') { return payFormProps } if (formType === 'REQUEST_FORM') { return requestFormProps } return {} } const networkTabProps = { currentChannels: stateProps.currentChannels, channels: stateProps.channels, balance: stateProps.balance, currentTicker: stateProps.currentTicker, contactsform: stateProps.contactsform, nodes: stateProps.network.nodes, nonActiveFilters: stateProps.nonActiveFilters, ticker: stateProps.ticker, network: stateProps.info.network, currencyName: stateProps.currencyName, fetchChannels: dispatchProps.fetchChannels, openContactsForm: dispatchProps.openContactsForm, contactFormSelectors: dispatchProps.contactFormSelectors, updateManualFormError: dispatchProps.updateManualFormErrors, toggleFilterPulldown: dispatchProps.toggleFilterPulldown, changeFilter: dispatchProps.changeFilter, updateChannelSearchQuery: dispatchProps.updateChannelSearchQuery, setSelectedChannel: dispatchProps.setSelectedChannel, closeChannel: dispatchProps.closeChannel, suggestedNodesProps: { suggestedNodesLoading: stateProps.channels.suggestedNodesLoading, suggestedNodes: stateProps.info.data.testnet ? stateProps.channels.suggestedNodes.testnet : stateProps.channels.suggestedNodes.mainnet, setNode: dispatchProps.setNode, openSubmitChannelForm: () => dispatchProps.setChannelFormType('SUBMIT_CHANNEL_FORM') } } const contactsFormProps = { closeContactsForm: dispatchProps.closeContactsForm, openSubmitChannelForm: () => dispatchProps.setChannelFormType('SUBMIT_CHANNEL_FORM'), updateContactFormSearchQuery: dispatchProps.updateContactFormSearchQuery, updateManualFormSearchQuery: dispatchProps.updateManualFormSearchQuery, updateContactCapacity: dispatchProps.updateContactCapacity, setNode: dispatchProps.setNode, openChannel: dispatchProps.openChannel, updateManualFormErrors: dispatchProps.updateManualFormErrors, openManualForm: () => dispatchProps.setChannelFormType('MANUAL_FORM'), contactsform: stateProps.contactsform, filteredNetworkNodes: stateProps.filteredNetworkNodes, loadingChannelPubkeys: stateProps.channels.loadingChannelPubkeys, showManualForm: stateProps.showManualForm, manualFormIsValid: stateProps.manualFormIsValid, activeChannelPubkeys: stateProps.activeChannelPubkeys, nonActiveChannelPubkeys: stateProps.nonActiveChannelPubkeys, pendingOpenChannelPubkeys: stateProps.pendingOpenChannelPubkeys } const activityModalProps = { itemType: stateProps.activity.modal.itemType, itemId: stateProps.activity.modal.itemId, item: stateProps.activityModalItem, ticker: stateProps.ticker, currentTicker: stateProps.currentTicker, network: stateProps.info.network, hideActivityModal: dispatchProps.hideActivityModal, toggleCurrencyProps: { currentCurrencyFilters: stateProps.currentCurrencyFilters, currencyName: stateProps.currencyName, showCurrencyFilters: stateProps.activity.modal.showCurrencyFilters, setActivityModalCurrencyFilters: dispatchProps.setActivityModalCurrencyFilters, setCurrencyFilters: dispatchProps.setCurrencyFilters, onCurrencyFilterClick: currency => { dispatchProps.setCurrency(currency) dispatchProps.setActivityModalCurrencyFilters(false) } } } const receiveModalProps = { isOpen: stateProps.address.walletModal, network: stateProps.info.network, pubkey: get(stateProps.info, 'data.uris[0]') || get(stateProps.info, 'data.identity_pubkey'), address: stateProps.address.address, alias: stateProps.info.data.alias, closeReceiveModal: dispatchProps.closeWalletModal } const submitChannelFormProps = { submitChannelFormOpen: stateProps.contactsform.submitChannelFormOpen, node: stateProps.contactsform.node, contactCapacity: stateProps.contactsform.contactCapacity, fiatTicker: stateProps.ticker.fiatTicker, dupeChanInfo: stateProps.dupeChanInfo, updateContactCapacity: dispatchProps.updateContactCapacity, closeChannelForm: () => dispatchProps.setChannelFormType(null), closeContactsForm: dispatchProps.closeContactsForm, openChannel: dispatchProps.openChannel, ticker: stateProps.ticker, toggleCurrencyProps: { currentCurrencyFilters: stateProps.currentCurrencyFilters, currencyName: stateProps.currencyName, showCurrencyFilters: stateProps.contactsform.showCurrencyFilters, contactFormFiatAmount: stateProps.contactFormFiatAmount, setContactsCurrencyFilters: dispatchProps.setContactsCurrencyFilters, setCurrencyFilters: dispatchProps.setCurrencyFilters, onCurrencyFilterClick: currency => { dispatchProps.updateContactCapacity( btc.convert(stateProps.ticker.currency, currency, stateProps.contactsform.contactCapacity) ) dispatchProps.setCurrency(currency) dispatchProps.setContactsCurrencyFilters(false) } } } const connectManuallyProps = { closeManualForm: dispatchProps.closeManualForm, updateManualFormSearchQuery: dispatchProps.updateManualFormSearchQuery, updateManualFormErrors: dispatchProps.updateManualFormErrors, setNode: dispatchProps.setNode, openSubmitChannelForm: () => dispatchProps.setChannelFormType('SUBMIT_CHANNEL_FORM'), manualFormOpen: stateProps.contactsform.manualFormOpen, manualSearchQuery: stateProps.contactsform.manualSearchQuery, manualFormIsValid: stateProps.manualFormIsValid, showErrors: stateProps.contactsform.showErrors } const calcChannelFormProps = formType => { if (formType === 'MANUAL_FORM') { return connectManuallyProps } if (formType === 'SUBMIT_CHANNEL_FORM') { return submitChannelFormProps } return {} } const channelFormProps = { formType: stateProps.contactsform.formType, formProps: calcChannelFormProps(stateProps.contactsform.formType), closeForm: () => dispatchProps.setChannelFormType(null) } return { ...stateProps, ...dispatchProps, ...ownProps, // props for the network sidebar networkTabProps, // props for the contacts form contactsFormProps, // props for the receive modal receiveModalProps, // props for the activity modals activityModalProps, // props for the form to open a channel submitChannelFormProps, // props for the form to connect manually to a peer connectManuallyProps, // props for the channel form wrapper channelFormProps, // Props to pass to the pay form formProps: formProps(stateProps.form.formType), // action to close form closeForm: () => dispatchProps.setFormType(null) } } export default withRouter( connect( mapStateToProps, mapDispatchToProps, mergeProps )(App) )