BlueWallet/screen/wallets/transactions.js

870 lines
27 KiB
JavaScript
Raw Normal View History

/* global alert */
import React, { useEffect, useState, useCallback, useContext, useRef } from 'react';
2019-09-19 01:26:28 +02:00
import {
ActivityIndicator,
2020-01-01 04:31:04 +01:00
Alert,
2020-11-17 09:43:38 +01:00
Dimensions,
FlatList,
InteractionManager,
2020-11-17 09:43:38 +01:00
Keyboard,
KeyboardAvoidingView,
Linking,
2020-09-09 19:51:49 +02:00
PixelRatio,
2020-11-17 09:43:38 +01:00
Platform,
ScrollView,
StatusBar,
StyleSheet,
Text,
findNodeHandle,
2020-11-17 09:43:38 +01:00
TouchableOpacity,
View,
2019-09-19 01:26:28 +02:00
} from 'react-native';
2020-12-10 18:40:14 +01:00
import { launchImageLibrary } from 'react-native-image-picker';
2020-08-21 00:50:38 +02:00
import Clipboard from '@react-native-community/clipboard';
2020-11-17 09:43:38 +01:00
import { Icon } from 'react-native-elements';
import { useRoute, useNavigation, useTheme, useFocusEffect } from '@react-navigation/native';
2021-02-04 05:54:24 +01:00
import { Chain } from '../../models/bitcoinUnits';
2020-10-02 11:57:25 +02:00
import { BlueTransactionListItem, BlueWalletNavigationHeader, BlueAlertWalletExportReminder, BlueListItem } from '../../BlueComponents';
import WalletGradient from '../../class/wallet-gradient';
2020-12-25 17:09:53 +01:00
import navigationStyle from '../../components/navigationStyle';
2020-12-11 18:18:10 +01:00
import { LightningCustodianWallet, MultisigHDWallet, WatchOnlyWallet } from '../../class';
2021-01-22 17:34:47 +01:00
import HandoffComponent from '../../components/handoff';
import ActionSheet from '../ActionSheet';
2020-07-20 15:38:46 +02:00
import loc from '../../loc';
2020-09-09 19:51:49 +02:00
import { FContainer, FButton } from '../../components/FloatButtons';
2020-11-17 09:43:38 +01:00
import BottomModal from '../../components/BottomModal';
2020-09-05 02:47:34 +02:00
import BuyBitcoin from './buyBitcoin';
import { BlueStorageContext } from '../../blue_modules/storage-context';
2021-02-04 05:54:24 +01:00
import { isCatalyst, isMacCatalina } from '../../blue_modules/environment';
const fs = require('../../blue_modules/fs');
const BlueElectrum = require('../../blue_modules/BlueElectrum');
const LocalQRCode = require('@remobile/react-native-qrcode-local-image');
2020-09-07 19:46:37 +02:00
const buttonFontSize =
PixelRatio.roundToNearestPixel(Dimensions.get('window').width / 26) > 22
? 22
: PixelRatio.roundToNearestPixel(Dimensions.get('window').width / 26);
const WalletTransactions = () => {
2021-01-28 01:30:42 +01:00
const { wallets, saveToDisk, setSelectedWallet, walletTransactionUpdateStatus } = useContext(BlueStorageContext);
const [isLoading, setIsLoading] = useState(false);
const [isManageFundsModalVisible, setIsManageFundsModalVisible] = useState(false);
const { walletID, name } = useRoute().params;
const wallet = wallets.find(w => w.getID() === walletID);
2021-02-04 05:54:24 +01:00
const [itemPriceUnit, setItemPriceUnit] = useState(wallet.getPreferredBalanceUnit());
const [dataSource, setDataSource] = useState(wallet.getTransactions(15));
const [timeElapsed, setTimeElapsed] = useState(0);
const [limit, setLimit] = useState(15);
const [pageSize, setPageSize] = useState(20);
const { setParams, setOptions, navigate } = useNavigation();
const { colors } = useTheme();
const walletActionButtonsRef = useRef();
const stylesHook = StyleSheet.create({
advancedTransactionOptionsModalContent: {
backgroundColor: colors.elevated,
},
listHeaderText: {
color: colors.foregroundColor,
},
marketplaceButton1: {
backgroundColor: colors.lightButton,
},
marketplaceButton2: {
backgroundColor: colors.lightButton,
},
marketpalceText1: {
color: colors.cta2,
},
marketpalceText2: {
color: colors.cta2,
},
list: {
backgroundColor: colors.background,
},
});
/**
* Simple wrapper for `wallet.getTransactions()`, where `wallet` is current wallet.
* Sorts. Provides limiting.
*
* @param limit {Integer} How many txs return, starting from the earliest. Default: all of them.
* @returns {Array}
*/
const getTransactionsSliced = (limit = Infinity) => {
let txs = wallet.getTransactions();
for (const tx of txs) {
tx.sort_ts = +new Date(tx.received);
}
txs = txs.sort(function (a, b) {
return b.sort_ts - a.sort_ts;
});
return txs.slice(0, limit);
};
useEffect(() => {
2020-09-14 12:49:08 +02:00
const interval = setInterval(() => setTimeElapsed(prev => prev + 1), 60000);
return () => {
clearInterval(interval);
};
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);
2021-01-28 01:30:42 +01:00
useEffect(() => {
setOptions({ headerTitle: walletTransactionUpdateStatus === walletID ? loc.transactions.updating : '' });
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [walletTransactionUpdateStatus]);
useEffect(() => {
setIsLoading(true);
setLimit(15);
setPageSize(20);
setTimeElapsed(0);
setItemPriceUnit(wallet.getPreferredBalanceUnit());
setIsLoading(false);
setSelectedWallet(wallet.getID());
setDataSource(wallet.getTransactions(15));
setOptions({
headerStyle: {
backgroundColor: WalletGradient.headerColorFor(wallet.type),
borderBottomWidth: 0,
elevation: 0,
// shadowRadius: 0,
shadowOffset: { height: 0, width: 0 },
},
});
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [wallets, wallet, walletID]);
useEffect(() => {
const newWallet = wallets.find(w => w.getID() === walletID);
if (newWallet) {
setParams({ walletID, isLoading: false });
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [walletID]);
// if balance of the wallet positive and there are no transactions, then
// it'a freshly impoted wallet and we need to refresh transactions
useEffect(() => {
if (dataSource.length === 0 && wallet.getBalance() > 0) {
refreshTransactions();
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);
// if description of transaction has been changed we want to show new one
useFocusEffect(
useCallback(() => {
setTimeElapsed(prev => prev + 1);
}, []),
);
const isLightning = () => {
const w = wallet;
2019-12-25 21:53:53 +01:00
if (w && w.chain === Chain.OFFCHAIN) {
return true;
}
return false;
};
/**
* Forcefully fetches TXs and balance for wallet
*/
const refreshTransactions = async () => {
if (isLoading) return;
setIsLoading(true);
let noErr = true;
let smthChanged = false;
try {
// await BlueElectrum.ping();
await BlueElectrum.waitTillConnected();
/** @type {LegacyWallet} */
const balanceStart = +new Date();
const oldBalance = wallet.getBalance();
await wallet.fetchBalance();
if (oldBalance !== wallet.getBalance()) smthChanged = true;
const balanceEnd = +new Date();
console.log(wallet.getLabel(), 'fetch balance took', (balanceEnd - balanceStart) / 1000, 'sec');
const start = +new Date();
const oldTxLen = wallet.getTransactions().length;
await wallet.fetchTransactions();
if (wallet.fetchPendingTransactions) {
await wallet.fetchPendingTransactions();
}
if (wallet.fetchUserInvoices) {
await wallet.fetchUserInvoices();
}
if (oldTxLen !== wallet.getTransactions().length) smthChanged = true;
const end = +new Date();
console.log(wallet.getLabel(), 'fetch tx took', (end - start) / 1000, 'sec');
} catch (err) {
noErr = false;
alert(err.message);
setIsLoading(false);
setTimeElapsed(prev => prev + 1);
}
if (noErr && smthChanged) {
console.log('saving to disk');
await saveToDisk(); // caching
// setDataSource([...getTransactionsSliced(limit)]);
}
setIsLoading(false);
setTimeElapsed(prev => prev + 1);
};
const _keyExtractor = (_item, index) => index.toString();
const renderListFooterComponent = () => {
// if not all txs rendered - display indicator
return (getTransactionsSliced(Infinity).length > limit && <ActivityIndicator style={styles.activityIndicator} />) || <View />;
};
const renderListHeaderComponent = () => {
const style = {};
if (!isCatalyst) {
// we need this button for testing
style.opacity = 0;
style.height = 1;
style.width = 1;
} else if (isLoading) {
style.opacity = 0.5;
} else {
style.opacity = 1.0;
}
return (
<View style={styles.flex}>
<View style={styles.listHeader}>
{/*
2020-04-29 17:40:55 +02:00
Current logic - Onchain:
- Shows buy button on middle when empty
- Show buy button on top when not empty
- Shows Marketplace button on details screen, open in browser (iOS)
- Shows Marketplace button on details screen, open in in-app (android)
Current logic - Offchain:
- Shows Lapp Browser empty (iOS)
- Shows Lapp Browser with marketplace (android)
- Shows Marketplace button to open in browser (iOS)
The idea is to avoid showing on iOS an appstore/market style app that goes against the TOS.
2020-05-27 13:12:17 +02:00
*/}
{wallet.getTransactions().length > 0 && wallet.type !== LightningCustodianWallet.type && renderSellFiat()}
{wallet.type === LightningCustodianWallet.type && renderMarketplaceButton()}
{wallet.type === LightningCustodianWallet.type && Platform.OS === 'ios' && renderLappBrowserButton()}
2019-12-25 21:53:53 +01:00
</View>
<View style={[styles.listHeaderTextRow, stylesHook.listHeaderTextRow]}>
<Text style={[styles.listHeaderText, stylesHook.listHeaderText]}>{loc.transactions.list_title}</Text>
<TouchableOpacity testID="refreshTransactions" style={style} onPress={refreshTransactions} disabled={isLoading}>
<Icon name="refresh" type="font-awesome" color={colors.feeText} />
</TouchableOpacity>
2020-08-21 00:50:38 +02:00
</View>
</View>
);
};
2019-09-29 22:01:27 +02:00
2020-11-11 17:59:03 +01:00
const hideManageFundsModal = () => {
Keyboard.dismiss();
setIsManageFundsModalVisible(false);
};
const renderManageFundsModal = () => {
2019-09-29 22:01:27 +02:00
return (
2020-11-17 09:43:38 +01:00
<BottomModal isVisible={isManageFundsModalVisible} onClose={hideManageFundsModal}>
2021-02-25 02:56:06 +01:00
<KeyboardAvoidingView enabled={!Platform.isPad} behavior={Platform.OS === 'ios' ? 'position' : null}>
<View style={[styles.advancedTransactionOptionsModalContent, stylesHook.advancedTransactionOptionsModalContent]}>
2020-09-22 03:53:28 +02:00
<BlueListItem
2019-09-29 22:01:27 +02:00
hideChevron
component={TouchableOpacity}
2020-09-05 02:49:32 +02:00
onPress={() => {
const availableWallets = [...wallets.filter(item => item.chain === Chain.ONCHAIN && item.allowSend())];
if (availableWallets.length === 0) {
2020-07-20 15:38:46 +02:00
alert(loc.lnd.refill_create);
2019-09-29 22:01:27 +02:00
} else {
setIsManageFundsModalVisible(false);
navigate('SelectWallet', { onWalletSelect, chainType: Chain.ONCHAIN });
2019-09-29 22:01:27 +02:00
}
}}
2019-10-01 00:13:22 +02:00
title={loc.lnd.refill}
2019-09-29 22:01:27 +02:00
/>
2020-09-22 03:53:28 +02:00
<BlueListItem
2019-09-29 22:01:27 +02:00
hideChevron
component={TouchableOpacity}
2020-09-05 02:49:32 +02:00
onPress={() => {
setIsManageFundsModalVisible(false);
navigate('ReceiveDetailsRoot', {
screen: 'ReceiveDetails',
params: {
walletID: wallet.getID(),
},
});
2019-09-29 22:01:27 +02:00
}}
2020-07-20 15:38:46 +02:00
title={loc.lnd.refill_external}
2019-09-29 22:01:27 +02:00
/>
2020-09-22 03:53:28 +02:00
<BlueListItem
hideChevron
component={TouchableOpacity}
2020-09-05 02:49:32 +02:00
onPress={() => {
setIsManageFundsModalVisible(false);
setTimeout(() => navigateToBuyBitcoin(), 500);
}}
2020-07-20 15:38:46 +02:00
title={loc.lnd.refill_card}
/>
2020-09-22 03:53:28 +02:00
<BlueListItem
2020-06-16 17:08:25 +02:00
title={loc.lnd.exchange}
2019-09-29 22:01:27 +02:00
hideChevron
component={TouchableOpacity}
2020-09-05 02:49:32 +02:00
onPress={() => {
setIsManageFundsModalVisible(false);
2019-09-29 22:01:27 +02:00
Linking.openURL('https://zigzag.io/?utm_source=integration&utm_medium=bluewallet&utm_campaign=withdrawLink');
}}
/>
</View>
</KeyboardAvoidingView>
2020-11-17 09:43:38 +01:00
</BottomModal>
2019-09-29 22:01:27 +02:00
);
};
const navigateToBuyBitcoin = () => {
BuyBitcoin.navigate(wallet);
};
const renderMarketplaceButton = () => {
2019-10-17 03:51:22 +02:00
return Platform.select({
android: (
<TouchableOpacity
onPress={() => {
if (wallet.type === LightningCustodianWallet.type) {
navigate('LappBrowserRoot', {
screen: 'LappBrowser',
params: { fromSecret: wallet.getSecret(), fromWallet: wallet },
});
2020-04-29 17:40:55 +02:00
} else {
navigate('Marketplace', { fromWallet: wallet });
2020-04-29 17:40:55 +02:00
}
2019-10-17 03:51:22 +02:00
}}
style={[styles.marketplaceButton1, stylesHook.marketplaceButton1]}
2019-10-17 03:51:22 +02:00
>
2020-12-25 19:46:06 +01:00
<Text style={[styles.marketpalceText1, stylesHook.marketpalceText1]}>{loc.wallets.list_marketplace}</Text>
2019-10-17 03:51:22 +02:00
</TouchableOpacity>
),
ios:
wallet.getBalance() > 0 ? (
2019-10-17 03:51:22 +02:00
<TouchableOpacity
onPress={async () => {
2020-10-26 00:22:45 +01:00
Linking.openURL('https://bluewallet.io/marketplace/');
2019-10-17 03:51:22 +02:00
}}
style={[styles.marketplaceButton1, stylesHook.marketplaceButton1]}
2019-10-17 03:51:22 +02:00
>
2019-10-17 04:08:09 +02:00
<Icon name="external-link" size={18} type="font-awesome" color="#9aa0aa" />
2020-12-25 19:46:06 +01:00
<Text style={[styles.marketpalceText2, stylesHook.marketpalceText2]}>{loc.wallets.list_marketplace}</Text>
2019-10-17 03:51:22 +02:00
</TouchableOpacity>
) : null,
});
};
const renderLappBrowserButton = () => {
2019-10-21 13:52:38 +02:00
return (
<TouchableOpacity
onPress={() => {
navigate('LappBrowserRoot', {
screen: 'LappBrowser',
params: {
fromSecret: wallet.getSecret(),
fromWallet: wallet,
url: 'https://duckduckgo.com',
},
2019-10-21 13:52:38 +02:00
});
}}
style={[styles.marketplaceButton2, stylesHook.marketplaceButton2]}
2019-10-21 13:52:38 +02:00
>
2020-12-21 19:27:49 +01:00
<Text style={[styles.marketpalceText1, stylesHook.marketpalceText1]}>{loc.wallets.list_ln_browser}</Text>
2019-10-21 13:52:38 +02:00
</TouchableOpacity>
);
};
const renderSellFiat = () => {
return (
<TouchableOpacity onPress={navigateToBuyBitcoin} style={[styles.marketplaceButton2, stylesHook.marketplaceButton2]}>
<Text style={[styles.marketpalceText1, stylesHook.marketpalceText1]}>{loc.wallets.list_tap_here_to_buy}</Text>
</TouchableOpacity>
);
};
const onWalletSelect = async selectedWallet => {
if (selectedWallet) {
navigate('WalletTransactions', {
walletType: wallet.type,
walletID: wallet.getID(),
key: `WalletTransactions-${wallet.getID()}`,
});
/** @type {LightningCustodianWallet} */
let toAddress = false;
if (wallet.refill_addressess.length > 0) {
toAddress = wallet.refill_addressess[0];
} else {
try {
await wallet.fetchBtcAddress();
toAddress = wallet.refill_addressess[0];
} catch (Err) {
return alert(Err.message);
}
2019-09-29 22:01:27 +02:00
}
navigate('SendDetailsRoot', {
2020-05-27 13:12:17 +02:00
screen: 'SendDetails',
params: {
memo: loc.lnd.refill_lnd_balance,
address: toAddress,
fromWallet: selectedWallet,
2020-05-27 13:12:17 +02:00
},
2019-09-29 22:01:27 +02:00
});
}
};
const navigateToSendScreen = () => {
navigate('SendDetailsRoot', {
2020-05-27 13:12:17 +02:00
screen: 'SendDetails',
params: {
fromWallet: wallet,
2020-05-27 13:12:17 +02:00
},
2020-01-01 04:31:04 +01:00
});
};
const renderItem = item => <BlueTransactionListItem item={item.item} itemPriceUnit={itemPriceUnit} timeElapsed={timeElapsed} />;
const onBarCodeRead = ret => {
if (!isLoading) {
setIsLoading(true);
const params = {
walletID: wallet.getID(),
uri: ret.data ? ret.data : ret,
fromWallet: wallet,
};
if (wallet.chain === Chain.ONCHAIN) {
navigate('SendDetailsRoot', { screen: 'SendDetails', params });
} else {
navigate('ScanLndInvoiceRoot', { screen: 'ScanLndInvoice', params });
}
}
setIsLoading(false);
};
const choosePhoto = () => {
2020-12-10 18:40:14 +01:00
launchImageLibrary(
{
title: null,
mediaType: 'photo',
takePhotoButtonTitle: null,
},
response => {
if (response.uri) {
2020-12-11 00:26:45 +01:00
const uri = Platform.OS === 'ios' ? response.uri.toString().replace('file://', '') : response.uri;
LocalQRCode.decode(uri, (error, result) => {
if (!error) {
onBarCodeRead({ data: result });
} else {
2020-07-20 15:38:46 +02:00
alert(loc.send.qr_error_no_qrcode);
}
});
}
},
);
};
2020-11-17 02:50:55 +01:00
const copyFromClipboard = async () => {
onBarCodeRead({ data: await Clipboard.getString() });
};
2020-09-09 19:51:49 +02:00
const sendButtonPress = () => {
if (wallet.chain === Chain.OFFCHAIN) {
navigate('ScanLndInvoiceRoot', { screen: 'ScanLndInvoice', params: { walletID: wallet.getID() } });
2020-09-07 19:46:37 +02:00
} else {
if (wallet.type === WatchOnlyWallet.type && wallet.isHd() && wallet.getSecret().startsWith('zpub')) {
if (wallet.useWithHardwareWalletEnabled()) {
2020-09-09 19:51:49 +02:00
navigateToSendScreen();
2020-09-07 19:46:37 +02:00
} else {
Alert.alert(
loc.wallets.details_title,
2021-02-18 14:37:43 +01:00
loc.transactions.enable_offline_signing,
2020-09-07 19:46:37 +02:00
[
{
text: loc._.ok,
2020-09-09 19:51:49 +02:00
onPress: async () => {
wallet.setUseWithHardwareWalletEnabled(true);
await saveToDisk();
2020-09-09 19:51:49 +02:00
navigateToSendScreen();
2020-09-07 19:46:37 +02:00
},
style: 'default',
},
2020-09-09 19:51:49 +02:00
2020-09-07 19:46:37 +02:00
{ text: loc._.cancel, onPress: () => {}, style: 'cancel' },
],
{ cancelable: false },
);
}
} else {
2020-09-09 19:51:49 +02:00
navigateToSendScreen();
2020-09-07 19:46:37 +02:00
}
}
};
const sendButtonLongPress = async () => {
2021-02-04 05:54:24 +01:00
if (isMacCatalina) {
fs.showActionSheet({ anchor: walletActionButtonsRef.current }).then(onBarCodeRead);
} else {
const isClipboardEmpty = (await Clipboard.getString()).replace(' ', '').length === 0;
if (Platform.OS === 'ios') {
const options = [loc._.cancel, loc.wallets.list_long_choose, loc.wallets.list_long_scan];
if (!isClipboardEmpty) {
options.push(loc.wallets.list_long_clipboard);
}
ActionSheet.showActionSheetWithOptions(
{ options, cancelButtonIndex: 0, anchor: findNodeHandle(walletActionButtonsRef.current) },
buttonIndex => {
if (buttonIndex === 1) {
choosePhoto();
} else if (buttonIndex === 2) {
navigate('ScanQRCodeRoot', {
screen: 'ScanQRCode',
params: {
launchedBy: name,
onBarScanned: onBarCodeRead,
showFileImportButton: false,
},
});
} else if (buttonIndex === 3) {
copyFromClipboard();
}
},
);
} else if (Platform.OS === 'android') {
const buttons = [
{
text: loc._.cancel,
onPress: () => {},
style: 'cancel',
},
{
text: loc.wallets.list_long_choose,
onPress: choosePhoto,
},
{
text: loc.wallets.list_long_scan,
onPress: () =>
navigate('ScanQRCodeRoot', {
screen: 'ScanQRCode',
params: {
launchedBy: name,
onBarScanned: onBarCodeRead,
showFileImportButton: false,
},
}),
},
];
if (!isClipboardEmpty) {
buttons.push({
text: loc.wallets.list_long_clipboard,
onPress: copyFromClipboard,
});
}
ActionSheet.showActionSheetWithOptions({
title: '',
message: '',
buttons,
});
}
}
};
2020-12-11 18:18:10 +01:00
const navigateToViewEditCosigners = () => {
2020-12-12 01:27:43 +01:00
navigate('ViewEditMultisigCosignersRoot', {
screen: 'ViewEditMultisigCosigners',
params: {
walletId: wallet.getID(),
2020-12-12 01:27:43 +01:00
},
2020-12-11 18:18:10 +01:00
});
};
return (
<View style={styles.flex}>
<StatusBar barStyle="light-content" backgroundColor={WalletGradient.headerColorFor(wallet.type)} />
{wallet.chain === Chain.ONCHAIN && wallet.type !== MultisigHDWallet.type && (
2021-01-19 04:40:11 +01:00
<HandoffComponent
title={`Bitcoin Wallet ${wallet.getLabel()}`}
2020-11-17 02:50:16 +01:00
type="io.bluewallet.bluewallet"
url={`https://blockpath.com/search/addr?q=${wallet.getXpub()}`}
/>
)}
<BlueWalletNavigationHeader
wallet={wallet}
onWalletUnitChange={passedWallet =>
InteractionManager.runAfterInteractions(async () => {
setItemPriceUnit(passedWallet.getPreferredBalanceUnit());
saveToDisk();
})
}
onManageFundsPressed={() => {
if (wallet.type === MultisigHDWallet.type) {
2020-12-11 18:18:10 +01:00
navigateToViewEditCosigners();
} else if (wallet.type === LightningCustodianWallet.type) {
if (wallet.getUserHasSavedExport()) {
2020-12-11 18:18:10 +01:00
setIsManageFundsModalVisible(true);
} else {
BlueAlertWalletExportReminder({
onSuccess: async () => {
wallet.setUserHasSavedExport(true);
2020-12-11 18:18:10 +01:00
await saveToDisk();
setIsManageFundsModalVisible(true);
},
onFailure: () =>
navigate('WalletExportRoot', {
screen: 'WalletExport',
params: {
walletID: wallet.getID(),
2020-12-11 18:18:10 +01:00
},
}),
});
}
2019-08-04 08:42:05 +02:00
}
}}
/>
<View style={[styles.list, stylesHook.list]}>
<FlatList
ListHeaderComponent={renderListHeaderComponent}
onEndReachedThreshold={0.3}
onEndReached={async () => {
2020-11-17 02:50:55 +01:00
// pagination in works. in this block we will add more txs to FlatList
// so as user scrolls closer to bottom it will render mode transactions
if (getTransactionsSliced(Infinity).length < limit) {
// all list rendered. nop
return;
}
setDataSource(getTransactionsSliced(limit + pageSize));
setLimit(prev => prev + pageSize);
setPageSize(prev => prev * 2);
}}
ListFooterComponent={renderListFooterComponent}
ListEmptyComponent={
<ScrollView style={styles.flex} contentContainerStyle={styles.scrollViewContent}>
<Text numberOfLines={0} style={styles.emptyTxs}>
{(isLightning() && loc.wallets.list_empty_txs1_lightning) || loc.wallets.list_empty_txs1}
</Text>
{isLightning() && <Text style={styles.emptyTxsLightning}>{loc.wallets.list_empty_txs2_lightning}</Text>}
{!isLightning() && (
<TouchableOpacity onPress={navigateToBuyBitcoin} style={styles.buyBitcoin}>
2020-11-03 14:18:46 +01:00
<Text testID="NoTxBuyBitcoin" style={styles.buyBitcoinText}>
{loc.wallets.list_tap_here_to_buy}
</Text>
</TouchableOpacity>
)}
</ScrollView>
}
onRefresh={refreshTransactions}
refreshing={isLoading}
data={dataSource}
2021-02-05 23:12:06 +01:00
extraData={[timeElapsed, dataSource, wallets]}
keyExtractor={_keyExtractor}
renderItem={renderItem}
contentInset={{ top: 0, left: 0, bottom: 90, right: 0 }}
2019-08-04 08:42:05 +02:00
/>
{renderManageFundsModal()}
</View>
<FContainer ref={walletActionButtonsRef}>
{wallet.allowReceive() && (
2020-09-09 19:51:49 +02:00
<FButton
testID="ReceiveButton"
2020-09-09 19:51:49 +02:00
text={loc.receive.header}
onPress={() => {
if (wallet.chain === Chain.OFFCHAIN) {
navigate('LNDCreateInvoiceRoot', { screen: 'LNDCreateInvoice', params: { walletID: wallet.getID() } });
2020-09-09 19:51:49 +02:00
} else {
navigate('ReceiveDetailsRoot', { screen: 'ReceiveDetails', params: { walletID: wallet.getID() } });
2020-09-09 19:51:49 +02:00
}
}}
2020-09-09 19:51:49 +02:00
icon={
<View style={styles.receiveIcon}>
<Icon name="arrow-down" size={buttonFontSize} type="font-awesome" color={colors.buttonAlternativeTextColor} />
2020-09-09 19:51:49 +02:00
</View>
}
/>
2020-09-09 19:51:49 +02:00
)}
{(wallet.allowSend() || (wallet.type === WatchOnlyWallet.type && wallet.isHd() && wallet.getSecret().startsWith('zpub'))) && (
2020-09-09 19:51:49 +02:00
<FButton
onLongPress={sendButtonLongPress}
onPress={sendButtonPress}
text={loc.send.header}
testID="SendButton"
icon={
<View style={styles.sendIcon}>
<Icon name="arrow-down" size={buttonFontSize} type="font-awesome" color={colors.buttonAlternativeTextColor} />
2020-09-09 19:51:49 +02:00
</View>
}
/>
)}
</FContainer>
</View>
);
};
2020-07-15 19:32:59 +02:00
export default WalletTransactions;
2020-12-25 17:09:53 +01:00
WalletTransactions.navigationOptions = navigationStyle({}, (options, { theme, navigation, route }) => {
2020-07-15 19:32:59 +02:00
return {
headerRight: () => (
<TouchableOpacity
disabled={route.params.isLoading === true}
style={styles.walletDetails}
onPress={() =>
navigation.navigate('WalletDetails', {
walletID: route.params.walletID,
2020-07-15 19:32:59 +02:00
})
}
>
<Icon name="kebab-horizontal" type="octicon" size={22} color="#FFFFFF" />
</TouchableOpacity>
),
2020-12-25 17:09:53 +01:00
title: '',
2020-07-15 19:32:59 +02:00
headerStyle: {
backgroundColor: WalletGradient.headerColorFor(route.params.walletType),
2020-07-15 19:32:59 +02:00
borderBottomWidth: 0,
elevation: 0,
// shadowRadius: 0,
shadowOffset: { height: 0, width: 0 },
},
headerTintColor: '#FFFFFF',
headerBackTitleVisible: false,
};
2020-12-25 17:09:53 +01:00
});
const styles = StyleSheet.create({
flex: {
flex: 1,
},
scrollViewContent: {
flex: 1,
justifyContent: 'center',
paddingHorizontal: 16,
paddingBottom: 40,
},
modalContent: {
backgroundColor: '#FFFFFF',
padding: 22,
justifyContent: 'center',
alignItems: 'center',
borderTopLeftRadius: 16,
borderTopRightRadius: 16,
borderColor: 'rgba(0, 0, 0, 0.1)',
minHeight: 200,
height: 200,
},
advancedTransactionOptionsModalContent: {
padding: 22,
borderTopLeftRadius: 16,
borderTopRightRadius: 16,
borderColor: 'rgba(0, 0, 0, 0.1)',
minHeight: 130,
},
walletDetails: {
marginHorizontal: 16,
minWidth: 150,
justifyContent: 'center',
alignItems: 'flex-end',
},
activityIndicator: {
marginVertical: 20,
},
listHeader: {
marginLeft: 16,
marginRight: 16,
marginVertical: 16,
flex: 1,
flexDirection: 'row',
justifyContent: 'space-around',
},
listHeaderTextRow: {
flex: 1,
marginHorizontal: 16,
flexDirection: 'row',
justifyContent: 'space-between',
},
listHeaderText: {
marginTop: 8,
marginBottom: 8,
fontWeight: 'bold',
fontSize: 24,
},
marketplaceButton1: {
borderRadius: 9,
minHeight: 49,
paddingHorizontal: 8,
justifyContent: 'center',
alignItems: 'center',
flexDirection: 'row',
alignSelf: 'auto',
flexGrow: 1,
marginHorizontal: 4,
},
marketplaceButton2: {
borderRadius: 9,
minHeight: 49,
paddingHorizontal: 8,
justifyContent: 'center',
alignItems: 'center',
flexDirection: 'row',
alignSelf: 'auto',
flexGrow: 1,
marginHorizontal: 4,
},
marketpalceText1: {
fontSize: 18,
},
marketpalceText2: {
fontSize: 18,
marginHorizontal: 8,
},
list: {
flex: 1,
},
emptyTxs: {
fontSize: 18,
color: '#9aa0aa',
textAlign: 'center',
marginVertical: 16,
},
emptyTxsLightning: {
fontSize: 18,
color: '#9aa0aa',
textAlign: 'center',
fontWeight: '600',
},
buyBitcoin: {
backgroundColor: '#007AFF',
minWidth: 260,
borderRadius: 8,
alignSelf: 'center',
paddingVertical: 14,
paddingHorizontal: 32,
},
buyBitcoinText: {
fontSize: 15,
color: '#fff',
textAlign: 'center',
fontWeight: '600',
},
sendIcon: {
transform: [{ rotate: '225deg' }],
},
receiveIcon: {
transform: [{ rotate: '-45deg' }],
},
});