You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

314 lines
8.3 KiB

import { createSelector } from 'reselect'
// ------------------------------------
// Initial State
// ------------------------------------
const initialState = {
filterPulldown: false,
filter: { key: 'ALL_ACTIVITY', name: 'All Activity' },
filters: [
{ key: 'ALL_ACTIVITY', name: 'all' },
{ key: 'SENT_ACTIVITY', name: 'sent' },
{ key: 'REQUESTED_ACTIVITY', name: 'requested' },
{ key: 'PENDING_ACTIVITY', name: 'pending' }
],
modal: {
itemType: null,
itemId: null,
showCurrencyFilters: false
},
searchActive: false,
searchText: '',
showExpiredRequests: false
}
// ------------------------------------
// Constants
// ------------------------------------
export const SHOW_ACTIVITY_MODAL = 'SHOW_ACTIVITY_MODAL'
export const HIDE_ACTIVITY_MODAL = 'HIDE_ACTIVITY_MODAL'
export const CHANGE_FILTER = 'CHANGE_FILTER'
export const TOGGLE_PULLDOWN = 'TOGGLE_PULLDOWN'
export const TOGGLE_EXPIRED_REQUESTS = 'TOGGLE_EXPIRED_REQUESTS'
export const SET_ACTIVITY_MODAL_CURRENCY_FILTERS = 'SET_ACTIVITY_MODAL_CURRENCY_FILTERS'
export const UPDATE_SEARCH_ACTIVE = 'UPDATE_SEARCH_ACTIVE'
export const UPDATE_SEARCH_TEXT = 'UPDATE_SEARCH_TEXT'
// ------------------------------------
// Actions
// ------------------------------------
export function showActivityModal(itemType, itemId) {
return {
type: SHOW_ACTIVITY_MODAL,
itemType,
itemId
}
}
export function hideActivityModal() {
return {
type: HIDE_ACTIVITY_MODAL
}
}
export function changeFilter(filter) {
return {
type: CHANGE_FILTER,
filter
}
}
export function toggleFilterPulldown() {
return {
type: TOGGLE_PULLDOWN
}
}
export function updateSearchActive(searchActive) {
return {
type: UPDATE_SEARCH_ACTIVE,
searchActive
}
}
export function updateSearchText(searchText) {
return {
type: UPDATE_SEARCH_TEXT,
searchText
}
}
export function setActivityModalCurrencyFilters(showCurrencyFilters) {
return {
type: SET_ACTIVITY_MODAL_CURRENCY_FILTERS,
showCurrencyFilters
}
}
export function toggleExpiredRequests() {
return {
type: TOGGLE_EXPIRED_REQUESTS
}
}
// ------------------------------------
// Action Handlers
// ------------------------------------
const ACTION_HANDLERS = {
[SHOW_ACTIVITY_MODAL]: (state, { itemType, itemId }) => ({
...state,
modal: { itemType, itemId }
}),
[HIDE_ACTIVITY_MODAL]: state => ({ ...state, modal: { itemType: null, itemId: null } }),
[CHANGE_FILTER]: (state, { filter }) => ({ ...state, filter, filterPulldown: false }),
[TOGGLE_PULLDOWN]: state => ({ ...state, filterPulldown: !state.filterPulldown }),
[TOGGLE_EXPIRED_REQUESTS]: state => ({
...state,
showExpiredRequests: !state.showExpiredRequests
}),
[SET_ACTIVITY_MODAL_CURRENCY_FILTERS]: (state, { showCurrencyFilters }) => ({
...state,
modal: {
itemType: state.modal.itemType,
itemId: state.modal.itemId,
showCurrencyFilters
}
}),
[UPDATE_SEARCH_ACTIVE]: (state, { searchActive }) => ({ ...state, searchActive }),
[UPDATE_SEARCH_TEXT]: (state, { searchText }) => ({ ...state, searchText })
}
// ------------------------------------
// Selectors
// ------------------------------------
const activitySelectors = {}
const filtersSelector = state => state.activity.filters
const filterSelector = state => state.activity.filter
const searchSelector = state => state.activity.searchText
const showExpiredSelector = state => state.activity.showExpiredRequests
const paymentsSelector = state => state.payment.payments
const invoicesSelector = state => state.invoice.invoices
const transactionsSelector = state => state.transaction.transactions
const modalItemTypeSelector = state => state.activity.modal.itemType
const modalItemIdSelector = state => state.activity.modal.itemId
const invoiceExpired = invoice => {
const expiresAt = parseInt(invoice.creation_date, 10) + parseInt(invoice.expiry, 10)
return expiresAt < Date.now() / 1000
}
activitySelectors.activityModalItem = createSelector(
paymentsSelector,
invoicesSelector,
transactionsSelector,
modalItemTypeSelector,
modalItemIdSelector,
(payments, invoices, transactions, itemType, itemId) => {
switch (itemType) {
case 'INVOICE':
return invoices.find(invoice => invoice.payment_request === itemId)
case 'TRANSACTION':
return transactions.find(transaction => transaction.tx_hash === itemId)
case 'PAYMENT':
return payments.find(payment => payment.payment_hash === itemId)
default:
return null
}
}
)
// helper function that returns invoice, payment or transaction timestamp
function returnTimestamp(transaction) {
// if on-chain txn
if (Object.prototype.hasOwnProperty.call(transaction, 'time_stamp')) {
return transaction.time_stamp
}
// if invoice that has been paid
if (transaction.settled) {
return transaction.settle_date
}
// if invoice that has not been paid or an LN payment
return transaction.creation_date
}
// getMonth() returns the month in 0 index (0 for Jan), so we create an arr of the
// string representation we want for the UI
const months = [
'Jan',
'Feb',
'Mar',
'April',
'May',
'June',
'July',
'Aug',
'Sep',
'Oct',
'Nov',
'Dec'
]
// groups the data by day
function groupData(data) {
return data.reduce((arr, el) => {
const d = new Date(returnTimestamp(el) * 1000)
const date = d.getDate()
const title = `${months[d.getMonth()]} ${date}, ${d.getFullYear()}`
if (!arr[title]) {
arr[title] = []
}
arr[title].push({ el })
// sort the activity within a day new -> old
arr[title].sort((a, b) => returnTimestamp(b.el) - returnTimestamp(a.el))
return arr
}, {})
}
// takes the result of groupData and returns an array
function groupArray(data) {
return Object.keys(data).map(title => ({ title, activity: data[title] }))
}
// sorts data form new to old according to the timestamp
function sortNewToOld(data) {
return data.sort((a, b) => new Date(b.title).getTime() - new Date(a.title).getTime())
}
// take in a dataset and return an array grouped by day
function groupAll(data) {
const groups = groupData(data)
const groupArrays = groupArray(groups)
return sortNewToOld(groupArrays)
}
const allActivity = createSelector(
searchSelector,
paymentsSelector,
invoicesSelector,
transactionsSelector,
showExpiredSelector,
(searchText, payments, invoices, transactions, showExpired) => {
const filteredInvoices = invoices.filter(
invoice => showExpired || invoice.settled || !invoiceExpired(invoice)
)
const searchedArr = [...payments, ...filteredInvoices, ...transactions].filter(tx => {
if (
(tx.tx_hash && tx.tx_hash.includes(searchText)) ||
(tx.payment_hash && tx.payment_hash.includes(searchText)) ||
(tx.payment_request && tx.payment_request.includes(searchText))
) {
return true
}
return false
})
if (!searchedArr.length) {
return []
}
return groupAll(searchedArr)
}
)
const invoiceActivity = createSelector(
invoicesSelector,
showExpiredSelector,
(invoices, showExpired) =>
groupAll(invoices.filter(invoice => showExpired || invoice.settled || !invoiceExpired(invoice)))
)
const sentActivity = createSelector(
transactionsSelector,
paymentsSelector,
(transactions, payments) =>
groupAll([...transactions.filter(transaction => !transaction.received), ...payments])
)
const pendingActivity = createSelector(invoicesSelector, invoices =>
groupAll(invoices.filter(invoice => !invoice.settled && !invoiceExpired(invoice)))
)
const FILTERS = {
ALL_ACTIVITY: allActivity,
SENT_ACTIVITY: sentActivity,
REQUESTED_ACTIVITY: invoiceActivity,
PENDING_ACTIVITY: pendingActivity
}
activitySelectors.currentActivity = createSelector(filterSelector, filter => FILTERS[filter.key])
activitySelectors.nonActiveFilters = createSelector(
filtersSelector,
filterSelector,
(filters, filter) => filters.filter(f => f.key !== filter.key)
)
activitySelectors.showExpiredToggle = createSelector(
filterSelector,
filter => filter.key === 'REQUESTED_ACTIVITY' || filter.key === 'ALL_ACTIVITY'
)
export { activitySelectors }
// ------------------------------------
// Reducer
// ------------------------------------
export default function activityReducer(state = initialState, action) {
const handler = ACTION_HANDLERS[action.type]
return handler ? handler(state, action) : state
}