import { createSelector } from 'reselect';
import * as TRANSACTIONS from 'constants/transaction_types';
import { PAGE_SIZE, LATEST_PAGE_SIZE } from 'constants/transaction_list';
import { selectClaimIdsByUri } from 'redux/selectors/claims';
import parseData from 'util/parse-data';
export const selectState = (state) => state.wallet || {};

export const selectWalletState = selectState;

export const selectWalletIsEncrypted = createSelector(selectState, (state) => state.walletIsEncrypted);

export const selectWalletEncryptPending = createSelector(selectState, (state) => state.walletEncryptPending);

export const selectWalletEncryptSucceeded = createSelector(selectState, (state) => state.walletEncryptSucceded);

export const selectPendingSupportTransactions = createSelector(
  selectState,
  (state) => state.pendingSupportTransactions
);

export const selectPendingOtherTransactions = createSelector(selectState, (state) => state.pendingTxos);

export const selectAbandonClaimSupportError = createSelector(selectState, (state) => state.abandonClaimSupportError);

export const makeSelectPendingAmountByUri = (uri) =>
  createSelector(selectClaimIdsByUri, selectPendingSupportTransactions, (claimIdsByUri, pendingSupports) => {
    const uriEntry = Object.entries(claimIdsByUri).find(([u, cid]) => u === uri);
    const claimId = uriEntry && uriEntry[1];
    const pendingSupport = claimId && pendingSupports[claimId];
    return pendingSupport ? pendingSupport.effective : undefined;
  });

export const selectWalletEncryptResult = createSelector(selectState, (state) => state.walletEncryptResult);

export const selectWalletDecryptPending = createSelector(selectState, (state) => state.walletDecryptPending);

export const selectWalletDecryptSucceeded = createSelector(selectState, (state) => state.walletDecryptSucceded);

export const selectWalletDecryptResult = createSelector(selectState, (state) => state.walletDecryptResult);

export const selectWalletUnlockPending = createSelector(selectState, (state) => state.walletUnlockPending);

export const selectWalletUnlockSucceeded = createSelector(selectState, (state) => state.walletUnlockSucceded);

export const selectWalletUnlockResult = createSelector(selectState, (state) => state.walletUnlockResult);

export const selectWalletLockPending = createSelector(selectState, (state) => state.walletLockPending);

export const selectWalletLockSucceeded = createSelector(selectState, (state) => state.walletLockSucceded);

export const selectWalletLockResult = createSelector(selectState, (state) => state.walletLockResult);

export const selectBalance = createSelector(selectState, (state) => state.balance);

export const selectTotalBalance = createSelector(selectState, (state) => state.totalBalance);

export const selectReservedBalance = createSelector(selectState, (state) => state.reservedBalance);

export const selectClaimsBalance = createSelector(selectState, (state) => state.claimsBalance);

export const selectSupportsBalance = createSelector(selectState, (state) => state.supportsBalance);

export const selectTipsBalance = createSelector(selectState, (state) => state.tipsBalance);

export const selectTransactionsById = createSelector(selectState, (state) => state.transactions || {});

export const selectSupportsByOutpoint = createSelector(selectState, (state) => state.supports || {});

export const selectTotalSupports = createSelector(selectSupportsByOutpoint, (byOutpoint) => {
  let total = parseFloat('0.0');

  Object.values(byOutpoint).forEach((support) => {
    const { amount } = support;
    total = amount ? total + parseFloat(amount) : total;
  });

  return total;
});

export const selectTransactionItems = createSelector(selectTransactionsById, (byId) => {
  const items = [];

  Object.keys(byId).forEach((txid) => {
    const tx = byId[txid];

    // ignore dust/fees
    // it is fee only txn if all infos are also empty
    if (
      Math.abs(tx.value) === Math.abs(tx.fee) &&
      tx.claim_info.length === 0 &&
      tx.support_info.length === 0 &&
      tx.update_info.length === 0 &&
      tx.abandon_info.length === 0
    ) {
      return;
    }

    const append = [];

    append.push(
      ...tx.claim_info.map((item) =>
        Object.assign({}, tx, item, {
          type: item.claim_name[0] === '@' ? TRANSACTIONS.CHANNEL : TRANSACTIONS.PUBLISH,
        })
      )
    );
    append.push(
      ...tx.support_info.map((item) =>
        Object.assign({}, tx, item, {
          type: !item.is_tip ? TRANSACTIONS.SUPPORT : TRANSACTIONS.TIP,
        })
      )
    );
    append.push(...tx.update_info.map((item) => Object.assign({}, tx, item, { type: TRANSACTIONS.UPDATE })));
    append.push(...tx.abandon_info.map((item) => Object.assign({}, tx, item, { type: TRANSACTIONS.ABANDON })));

    if (!append.length) {
      append.push(
        Object.assign({}, tx, {
          type: tx.value < 0 ? TRANSACTIONS.SPEND : TRANSACTIONS.RECEIVE,
        })
      );
    }

    items.push(
      ...append.map((item) => {
        // value on transaction, amount on outpoint
        // amount is always positive, but should match sign of value
        const balanceDelta = parseFloat(item.balance_delta);
        const value = parseFloat(item.value);
        const amount = balanceDelta || value;
        const fee = parseFloat(tx.fee);

        return {
          txid,
          timestamp: tx.timestamp,
          date: tx.timestamp ? new Date(Number(tx.timestamp) * 1000) : null,
          amount,
          fee,
          claim_id: item.claim_id,
          claim_name: item.claim_name,
          type: item.type || TRANSACTIONS.SPEND,
          nout: item.nout,
          confirmations: tx.confirmations,
        };
      })
    );
  });

  return items.sort((tx1, tx2) => {
    if (!tx1.timestamp && !tx2.timestamp) {
      return 0;
    } else if (!tx1.timestamp && tx2.timestamp) {
      return -1;
    } else if (tx1.timestamp && !tx2.timestamp) {
      return 1;
    }

    return tx2.timestamp - tx1.timestamp;
  });
});

export const selectRecentTransactions = createSelector(selectTransactionItems, (transactions) => {
  const threshold = new Date();
  threshold.setDate(threshold.getDate() - 7);
  return transactions.filter((transaction) => {
    if (!transaction.date) {
      return true; // pending transaction
    }

    return transaction.date > threshold;
  });
});

export const selectHasTransactions = createSelector(
  selectTransactionItems,
  (transactions) => transactions && transactions.length > 0
);

export const selectIsFetchingTransactions = createSelector(selectState, (state) => state.fetchingTransactions);

/**
 * CSV of 'selectTransactionItems'.
 */
export const selectTransactionsFile = createSelector(selectTransactionItems, (transactions) => {
  if (!transactions || transactions.length === 0) {
    // No data.
    return undefined;
  }

  const parsed = parseData(transactions, 'csv');
  if (!parsed) {
    // Invalid data, or failed to parse.
    return null;
  }

  return parsed;
});

export const selectIsSendingSupport = createSelector(selectState, (state) => state.sendingSupport);

export const selectReceiveAddress = createSelector(selectState, (state) => state.receiveAddress);

export const selectGettingNewAddress = createSelector(selectState, (state) => state.gettingNewAddress);

export const selectDraftTransaction = createSelector(selectState, (state) => state.draftTransaction || {});

export const selectDraftTransactionAmount = createSelector(selectDraftTransaction, (draft) => draft.amount);

export const selectDraftTransactionAddress = createSelector(selectDraftTransaction, (draft) => draft.address);

export const selectDraftTransactionError = createSelector(selectDraftTransaction, (draft) => draft.error);

export const selectBlocks = createSelector(selectState, (state) => state.blocks);

export const selectCurrentHeight = createSelector(selectState, (state) => state.latestBlock);

export const selectTransactionListFilter = createSelector(selectState, (state) => state.transactionListFilter || '');

export const selectFilteredTransactions = createSelector(
  selectTransactionItems,
  selectTransactionListFilter,
  (transactions, filter) => {
    return transactions.filter((transaction) => {
      return filter === TRANSACTIONS.ALL || filter === transaction.type;
    });
  }
);

export const selectTxoPageParams = createSelector(selectState, (state) => state.txoFetchParams);

export const selectTxoPage = createSelector(selectState, (state) => (state.txoPage && state.txoPage.items) || []);

export const selectTxoPageNumber = createSelector(selectState, (state) => (state.txoPage && state.txoPage.page) || 1);

export const selectTxoItemCount = createSelector(
  selectState,
  (state) => (state.txoPage && state.txoPage.total_items) || 1
);

export const selectFetchingTxosError = createSelector(selectState, (state) => state.fetchingTxosError);

export const selectIsFetchingTxos = createSelector(selectState, (state) => state.fetchingTxos);

export const makeSelectFilteredTransactionsForPage = (page = 1) =>
  createSelector(selectFilteredTransactions, (filteredTransactions) => {
    const start = (Number(page) - 1) * Number(PAGE_SIZE);
    const end = Number(page) * Number(PAGE_SIZE);
    return filteredTransactions && filteredTransactions.length ? filteredTransactions.slice(start, end) : [];
  });

export const makeSelectLatestTransactions = createSelector(selectTransactionItems, (transactions) => {
  return transactions && transactions.length ? transactions.slice(0, LATEST_PAGE_SIZE) : [];
});

export const selectFilteredTransactionCount = createSelector(
  selectFilteredTransactions,
  (filteredTransactions) => filteredTransactions.length
);

export const selectIsWalletReconnecting = createSelector(selectState, (state) => state.walletReconnecting);

export const selectWalletRollbackToDefault = createSelector(selectState, (state) => state.walletRollbackToDefault);

export const selectWalletConnectingToDefault = createSelector(
  selectState,
  (state) => state.walletReconnectingToDefault
);

export const selectIsFetchingUtxoCounts = createSelector(selectState, (state) => state.fetchingUtxoCounts);

export const selectIsConsolidatingUtxos = createSelector(selectState, (state) => state.consolidatingUtxos);

export const selectIsMassClaimingTips = createSelector(selectState, (state) => state.massClaimingTips);

export const selectPendingConsolidateTxid = createSelector(selectState, (state) => state.pendingConsolidateTxid);

export const selectPendingMassClaimTxid = createSelector(selectState, (state) => state.pendingMassClaimTxid);

export const selectUtxoCounts = createSelector(selectState, (state) => state.utxoCounts);