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: { modalType: null, modalProps: {}, showCurrencyFilters: false }, searchText: '' } // ------------------------------------ // 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 SET_ACTIVITY_MODAL_CURRENCY_FILTERS = 'SET_ACTIVITY_MODAL_CURRENCY_FILTERS' export const UPDATE_SEARCH_TEXT = 'UPDATE_SEARCH_TEXT' // ------------------------------------ // Actions // ------------------------------------ export function showActivityModal(modalType, modalProps) { return { type: SHOW_ACTIVITY_MODAL, modalType, modalProps } } 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 updateSearchText(searchText) { return { type: UPDATE_SEARCH_TEXT, searchText } } export function setActivityModalCurrencyFilters(showCurrencyFilters) { return { type: SET_ACTIVITY_MODAL_CURRENCY_FILTERS, showCurrencyFilters } } // ------------------------------------ // Action Handlers // ------------------------------------ const ACTION_HANDLERS = { [SHOW_ACTIVITY_MODAL]: (state, { modalType, modalProps }) => ({ ...state, modal: { modalType, modalProps } }), [HIDE_ACTIVITY_MODAL]: state => ({ ...state, modal: { modalType: null, modalProps: {} } }), [CHANGE_FILTER]: (state, { filter }) => ({ ...state, filter, filterPulldown: false }), [TOGGLE_PULLDOWN]: state => ({ ...state, filterPulldown: !state.filterPulldown }), [SET_ACTIVITY_MODAL_CURRENCY_FILTERS]: (state, { showCurrencyFilters }) => ( { ...state, modal: { modalType: state.modal.modalType, modalProps: state.modal.modalProps, showCurrencyFilters } } ), [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 paymentsSelector = state => state.payment.payments const invoicesSelector = state => state.invoice.invoices const transactionsSelector = state => state.transaction.transactions const channelsSelector = state => state.channels.channels const invoiceExpired = (invoice) => { const expiresAt = (parseInt(invoice.creation_date, 10) + parseInt(invoice.expiry, 10)) return expiresAt < (Date.now() / 1000) } const allActivity = createSelector( searchSelector, paymentsSelector, invoicesSelector, transactionsSelector, (searchText, payments, invoices, transactions) => { const searchedArr = [...payments, ...invoices, ...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 }) // return searchedArr.sort((a, b) => { // // this will return the correct timestamp to use when sorting (time_stamp, creation_date, or settle_date) // 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 // } // const aTimestamp = returnTimestamp(a) // const bTimestamp = returnTimestamp(b) // // console.log('aTimestamp: ', aTimestamp) // // console.log('bTimestamp: ', bTimestamp) // console.log('date: ', new Date(aTimestamp * 1000)) // return bTimestamp - aTimestamp // }) if (!searchedArr.length) { return [] } const groups = searchedArr.reduce((groups, el) => { 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 } const months = ['Jan', 'Feb', 'Mar', 'April', 'May', 'June', 'July', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'] const d = new Date(returnTimestamp(el) * 1000) const date = d.getDate() const title = `${months[d.getMonth()]} ${date}, ${d.getFullYear()}` if (!groups[title]) { groups[title] = [] } groups[title].push({ el }) return groups }, {}) const groupArrays = Object.keys(groups).map((title) => { return { title, activity: groups[title] } }) console.log('groupArrays: ', groupArrays) return groupArrays.sort((a, b) => new Date(b.title).getTime() - new Date(a.title).getTime()) } ) const invoiceActivity = createSelector( invoicesSelector, invoices => invoices ) const sentActivity = createSelector( transactionsSelector, paymentsSelector, (transactions, payments) => { const sentTransactions = transactions.filter(transaction => transaction.amount < 0) return [...sentTransactions, ...payments].sort((a, b) => { const aTimestamp = Object.prototype.hasOwnProperty.call(a, 'time_stamp') ? a.time_stamp : a.creation_date const bTimestamp = Object.prototype.hasOwnProperty.call(b, 'time_stamp') ? b.time_stamp : b.creation_date return bTimestamp - aTimestamp }) } ) const pendingActivity = createSelector( invoicesSelector, invoices => 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) ) export { activitySelectors } // ------------------------------------ // Reducer // ------------------------------------ export default function activityReducer(state = initialState, action) { const handler = ACTION_HANDLERS[action.type] return handler ? handler(state, action) : state }