|
|
|
import { createSelector } from 'reselect'
|
|
|
|
import { btc } from 'lib/utils'
|
|
|
|
import { tickerSelectors } from './ticker'
|
|
|
|
|
|
|
|
// Initial State
|
|
|
|
const initialState = {
|
|
|
|
// this determines whether or not the network side bar is in search state for a peer or not
|
|
|
|
isOpen: false,
|
|
|
|
// this determines what form (manual or submit) the user currently has open
|
|
|
|
// if this is not null the ChannelForm component will be open
|
|
|
|
formType: null,
|
|
|
|
|
|
|
|
searchQuery: '',
|
|
|
|
manualSearchQuery: '',
|
|
|
|
contactCapacity: 0,
|
|
|
|
node: {},
|
|
|
|
showErrors: {
|
|
|
|
manualInput: false
|
|
|
|
},
|
|
|
|
|
|
|
|
manualFormOpen: false,
|
|
|
|
submitChannelFormOpen: false,
|
|
|
|
|
|
|
|
showCurrencyFilters: false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Constants
|
|
|
|
// ------------------------------------
|
|
|
|
export const OPEN_CONTACTS_FORM = 'OPEN_CONTACTS_FORM'
|
|
|
|
export const CLOSE_CONTACTS_FORM = 'CLOSE_CONTACTS_FORM'
|
|
|
|
|
|
|
|
export const OPEN_CHANNEL_FORM_FORM = 'OPEN_CHANNEL_FORM_FORM'
|
|
|
|
export const CLOSE_CHANNEL_FORM_FORM = 'CLOSE_CHANNEL_FORM_FORM'
|
|
|
|
|
|
|
|
export const SET_CHANNEL_FORM_TYPE = 'SET_CHANNEL_FORM_TYPE'
|
|
|
|
|
|
|
|
export const OPEN_MANUAL_FORM = 'OPEN_MANUAL_FORM'
|
|
|
|
export const CLOSE_MANUAL_FORM = 'CLOSE_MANUAL_FORM'
|
|
|
|
|
|
|
|
export const OPEN_SUBMIT_CHANNEL_FORM = 'OPEN_SUBMIT_CHANNEL_FORM'
|
|
|
|
export const CLOSE_SUBMIT_CHANNEL_FORM = 'CLOSE_SUBMIT_CHANNEL_FORM'
|
|
|
|
|
|
|
|
export const SET_NODE = 'SET_NODE'
|
|
|
|
|
|
|
|
export const UPDATE_CONTACT_FORM_SEARCH_QUERY = 'UPDATE_CONTACT_FORM_SEARCH_QUERY'
|
|
|
|
|
|
|
|
export const UPDATE_CONTACT_CAPACITY = 'UPDATE_CONTACT_CAPACITY'
|
|
|
|
|
|
|
|
export const UPDATE_MANUAL_FORM_ERRORS = 'UPDATE_MANUAL_FORM_ERRORS'
|
|
|
|
|
|
|
|
export const UPDATE_MANUAL_FORM_SEARCH_QUERY = 'UPDATE_MANUAL_FORM_SEARCH_QUERY'
|
|
|
|
|
|
|
|
export const SET_CONTACTS_CURRENCY_FILTERS = 'SET_CONTACTS_CURRENCY_FILTERS'
|
|
|
|
|
|
|
|
// ------------------------------------
|
|
|
|
// Actions
|
|
|
|
// ------------------------------------
|
|
|
|
export function openContactsForm() {
|
|
|
|
return {
|
|
|
|
type: OPEN_CONTACTS_FORM
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function closeContactsForm() {
|
|
|
|
return {
|
|
|
|
type: CLOSE_CONTACTS_FORM
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function openChannelForm() {
|
|
|
|
return {
|
|
|
|
type: OPEN_CONTACTS_FORM
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function closeChannelForm() {
|
|
|
|
return {
|
|
|
|
type: CLOSE_CONTACTS_FORM
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function setChannelFormType(formType) {
|
|
|
|
return {
|
|
|
|
type: SET_CHANNEL_FORM_TYPE,
|
|
|
|
formType
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function openManualForm() {
|
|
|
|
return {
|
|
|
|
type: OPEN_MANUAL_FORM
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function closeManualForm() {
|
|
|
|
return {
|
|
|
|
type: CLOSE_MANUAL_FORM
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function openSubmitChannelForm() {
|
|
|
|
return {
|
|
|
|
type: OPEN_SUBMIT_CHANNEL_FORM
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function closeSubmitChannelForm() {
|
|
|
|
return {
|
|
|
|
type: CLOSE_SUBMIT_CHANNEL_FORM
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function updateContactFormSearchQuery(searchQuery) {
|
|
|
|
return {
|
|
|
|
type: UPDATE_CONTACT_FORM_SEARCH_QUERY,
|
|
|
|
searchQuery
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function updateManualFormSearchQuery(manualSearchQuery) {
|
|
|
|
return {
|
|
|
|
type: UPDATE_MANUAL_FORM_SEARCH_QUERY,
|
|
|
|
manualSearchQuery
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function updateContactCapacity(contactCapacity) {
|
|
|
|
return {
|
|
|
|
type: UPDATE_CONTACT_CAPACITY,
|
|
|
|
contactCapacity
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function setNode(node) {
|
|
|
|
return {
|
|
|
|
type: SET_NODE,
|
|
|
|
node
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function updateManualFormErrors(errorsObject) {
|
|
|
|
return {
|
|
|
|
type: UPDATE_MANUAL_FORM_ERRORS,
|
|
|
|
errorsObject
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function setContactsCurrencyFilters(showCurrencyFilters) {
|
|
|
|
return {
|
|
|
|
type: SET_CONTACTS_CURRENCY_FILTERS,
|
|
|
|
showCurrencyFilters
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------
|
|
|
|
// Action Handlers
|
|
|
|
// ------------------------------------
|
|
|
|
const ACTION_HANDLERS = {
|
|
|
|
[OPEN_CONTACTS_FORM]: state => ({ ...state, isOpen: true }),
|
|
|
|
[CLOSE_CONTACTS_FORM]: state => ({ ...state, isOpen: false }),
|
|
|
|
|
|
|
|
[SET_CHANNEL_FORM_TYPE]: (state, { formType }) => ({ ...state, formType }),
|
|
|
|
|
|
|
|
[OPEN_MANUAL_FORM]: state => ({ ...state, manualFormOpen: true }),
|
|
|
|
[CLOSE_MANUAL_FORM]: state => ({ ...state, manualFormOpen: false }),
|
|
|
|
|
|
|
|
[OPEN_SUBMIT_CHANNEL_FORM]: state => ({ ...state, submitChannelFormOpen: true }),
|
|
|
|
[CLOSE_SUBMIT_CHANNEL_FORM]: state => ({ ...state, submitChannelFormOpen: false }),
|
|
|
|
|
|
|
|
[UPDATE_CONTACT_FORM_SEARCH_QUERY]: (state, { searchQuery }) => ({ ...state, searchQuery }),
|
|
|
|
|
|
|
|
[UPDATE_MANUAL_FORM_SEARCH_QUERY]: (state, { searchQuery }) => ({ ...state, searchQuery }),
|
|
|
|
|
|
|
|
[UPDATE_CONTACT_CAPACITY]: (state, { contactCapacity }) => ({ ...state, contactCapacity }),
|
|
|
|
|
|
|
|
[SET_NODE]: (state, { node }) => ({ ...state, node }),
|
|
|
|
|
|
|
|
[UPDATE_MANUAL_FORM_ERRORS]: (state, { errorsObject }) => ({
|
|
|
|
...state,
|
|
|
|
showErrors: Object.assign(state.showErrors, errorsObject)
|
|
|
|
}),
|
|
|
|
|
|
|
|
[UPDATE_MANUAL_FORM_SEARCH_QUERY]: (state, { manualSearchQuery }) => ({
|
|
|
|
...state,
|
|
|
|
manualSearchQuery
|
|
|
|
}),
|
|
|
|
|
|
|
|
[SET_CONTACTS_CURRENCY_FILTERS]: (state, { showCurrencyFilters }) => ({
|
|
|
|
...state,
|
|
|
|
showCurrencyFilters
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------
|
|
|
|
// Selector
|
|
|
|
// ------------------------------------
|
|
|
|
const contactFormSelectors = {}
|
|
|
|
const networkNodesSelector = state => state.network.nodes
|
|
|
|
const searchQuerySelector = state => state.contactsform.searchQuery
|
|
|
|
const manualSearchQuerySelector = state => state.contactsform.manualSearchQuery
|
|
|
|
const contactCapacitySelector = state => state.contactsform.contactCapacity
|
|
|
|
const currencySelector = state => state.ticker.currency
|
|
|
|
const fiatTickerSelector = state => state.ticker.fiatTicker
|
|
|
|
const nodeSelector = state => state.contactsform.node
|
|
|
|
const channelsSelector = state => state.channels.channels
|
|
|
|
|
|
|
|
const contactable = node => node.addresses.length > 0
|
|
|
|
|
|
|
|
// comparator to sort the contacts list with contactable contacts first
|
|
|
|
const contactableFirst = (a, b) => {
|
|
|
|
if (contactable(a) && !contactable(b)) {
|
|
|
|
return -1
|
|
|
|
} else if (!contactable(a) && contactable(b)) {
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
contactFormSelectors.filteredNetworkNodes = createSelector(
|
|
|
|
networkNodesSelector,
|
|
|
|
searchQuerySelector,
|
|
|
|
(nodes, searchQuery) => {
|
|
|
|
// If there is no search query default to showing the first 20 nodes from the nodes array
|
|
|
|
// (performance hit to render the entire thing by default)
|
|
|
|
// if (!searchQuery.length) { return nodes.sort(contactableFirst).slice(0, 20) }
|
|
|
|
|
|
|
|
// return an empty array if there is no search query
|
|
|
|
if (!searchQuery.length) {
|
|
|
|
return []
|
|
|
|
}
|
|
|
|
|
|
|
|
// if there is an '@' in the search query we are assuming they are using the format pubkey@host
|
|
|
|
// we can ignore the '@' and the host and just grab the pubkey for our search
|
|
|
|
const query = searchQuery.includes('@') ? searchQuery.split('@')[0] : searchQuery
|
|
|
|
|
|
|
|
// list of the nodes
|
|
|
|
const list = nodes
|
|
|
|
.filter(node => node.alias.includes(query) || node.pub_key.includes(query))
|
|
|
|
.sort(contactableFirst)
|
|
|
|
|
|
|
|
// if we don't limit the nodes returned then we take a huge performance hit
|
|
|
|
// rendering thousands of nodes potentially, so we just render 20 for the time being
|
|
|
|
return list.slice(0, 20)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
contactFormSelectors.showManualForm = createSelector(
|
|
|
|
searchQuerySelector,
|
|
|
|
contactFormSelectors.filteredNetworkNodes,
|
|
|
|
(searchQuery, filteredNetworkNodes) => {
|
|
|
|
if (!searchQuery.length) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
const connectableNodes = filteredNetworkNodes.filter(node => node.addresses.length > 0)
|
|
|
|
|
|
|
|
if (!filteredNetworkNodes.length || !connectableNodes.length) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
contactFormSelectors.manualFormIsValid = createSelector(manualSearchQuerySelector, input => {
|
|
|
|
const errors = {}
|
|
|
|
if (!input.length || !input.includes('@')) {
|
|
|
|
errors.manualInput = 'Invalid format'
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
errors,
|
|
|
|
isValid: Object.keys(errors).length === 0
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
contactFormSelectors.contactFormFiatAmount = createSelector(
|
|
|
|
contactCapacitySelector,
|
|
|
|
currencySelector,
|
|
|
|
tickerSelectors.currentTicker,
|
|
|
|
fiatTickerSelector,
|
|
|
|
(amount, currency, currentTicker, fiatTicker) => {
|
|
|
|
if (!currentTicker || !currentTicker[fiatTicker].last) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return btc.convert(currency, 'fiat', amount, currentTicker[fiatTicker].last)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
// compose warning info when a channel is being created with a node that
|
|
|
|
// already has one or more active channels open
|
|
|
|
contactFormSelectors.dupeChanInfo = createSelector(
|
|
|
|
channelsSelector,
|
|
|
|
nodeSelector,
|
|
|
|
networkNodesSelector,
|
|
|
|
currencySelector,
|
|
|
|
tickerSelectors.currencyName,
|
|
|
|
(activeChannels, newNode, allNodes, currency, currencyName) => {
|
|
|
|
const chans = activeChannels.filter(
|
|
|
|
chan => chan.active && chan.remote_pubkey === newNode.pub_key
|
|
|
|
)
|
|
|
|
|
|
|
|
if (!chans.length) {
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
|
|
|
|
const node = allNodes.filter(node => node.pub_key === newNode.pub_key)[0]
|
|
|
|
// use the alias unless its the first 20 chars of the pub_key
|
|
|
|
const alias =
|
|
|
|
node && node.alias !== node.pub_key.substring(0, node.alias.length) ? node.alias : null
|
|
|
|
|
|
|
|
const totalSats = chans.reduce((agg, chan) => agg + parseInt(chan.capacity, 10), 0)
|
|
|
|
const capacity = parseFloat(btc.convert('sats', currency, totalSats))
|
|
|
|
|
|
|
|
return {
|
|
|
|
alias,
|
|
|
|
activeChannels: chans.length,
|
|
|
|
capacity,
|
|
|
|
currencyName
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
export { contactFormSelectors }
|
|
|
|
|
|
|
|
// ------------------------------------
|
|
|
|
// Reducer
|
|
|
|
// ------------------------------------
|
|
|
|
export default function contactFormReducer(state = initialState, action) {
|
|
|
|
const handler = ACTION_HANDLERS[action.type]
|
|
|
|
|
|
|
|
return handler ? handler(state, action) : state
|
|
|
|
}
|