import 'react-native-gesture-handler'; // should be on top import React from 'react'; import { Linking, DeviceEventEmitter, AppState, StyleSheet, KeyboardAvoidingView, Platform, View } from 'react-native'; import Clipboard from '@react-native-community/clipboard'; import Modal from 'react-native-modal'; import { NavigationContainer, CommonActions } from '@react-navigation/native'; import { SafeAreaProvider } from 'react-native-safe-area-context'; import Navigation from './Navigation'; import NavigationService from './NavigationService'; import { BlueTextCentered, BlueButton } from './BlueComponents'; import ReactNativeHapticFeedback from 'react-native-haptic-feedback'; import { Chain } from './models/bitcoinUnits'; import QuickActions from 'react-native-quick-actions'; import * as Sentry from '@sentry/react-native'; import OnAppLaunch from './class/on-app-launch'; import DeeplinkSchemaMatch from './class/deeplink-schema-match'; import BitcoinBIP70TransactionDecode from './bip70/bip70'; const A = require('./analytics'); if (process.env.NODE_ENV !== 'development') { Sentry.init({ dsn: 'https://23377936131848ca8003448a893cb622@sentry.io/1295736', }); } const bitcoinModalString = 'Bitcoin address'; const lightningModalString = 'Lightning Invoice'; const loc = require('./loc'); const BlueApp = require('./BlueApp'); export default class App extends React.Component { navigation = null; state = { appState: AppState.currentState, isClipboardContentModalVisible: false, clipboardContentModalAddressType: bitcoinModalString, clipboardContent: '', }; componentDidMount() { Linking.addEventListener('url', this.handleOpenURL); AppState.addEventListener('change', this._handleAppStateChange); QuickActions.popInitialAction().then(this.popInitialAction); DeviceEventEmitter.addListener('quickActionShortcut', this.walletQuickActions); this._handleAppStateChange(undefined); } popInitialAction = async data => { if (data) { const wallet = BlueApp.getWallets().find(wallet => wallet.getID() === data.userInfo.url.split('wallet/')[1]); this.navigation.dispatch( CommonActions.navigate({ name: 'WalletTransactions', key: `WalletTransactions-${wallet.getID()}`, params: { wallet, }, }), ); } else { const url = await Linking.getInitialURL(); if (url) { if (DeeplinkSchemaMatch.hasSchema(url)) { this.handleOpenURL({ url }); } } else { const isViewAllWalletsEnabled = await OnAppLaunch.isViewAllWalletsEnabled(); if (!isViewAllWalletsEnabled) { const selectedDefaultWallet = await OnAppLaunch.getSelectedDefaultWallet(); const wallet = BlueApp.getWallets().find(wallet => wallet.getID() === selectedDefaultWallet.getID()); if (wallet) { this.navigation.dispatch( CommonActions.navigate({ name: 'WalletTransactions', key: `WalletTransactions-${wallet.getID()}`, params: { wallet, }, }), ); } } } } }; walletQuickActions = data => { const wallet = BlueApp.getWallets().find(wallet => wallet.getID() === data.userInfo.url.split('wallet/')[1]); this.navigation.dispatch( CommonActions.navigate({ name: 'WalletTransactions', key: `WalletTransactions-${wallet.getID()}`, params: { wallet, }, }), ); }; componentWillUnmount() { Linking.removeEventListener('url', this.handleOpenURL); AppState.removeEventListener('change', this._handleAppStateChange); } _handleAppStateChange = async nextAppState => { if (BlueApp.getWallets().length > 0) { if ((this.state.appState.match(/background/) && nextAppState) === 'active' || nextAppState === undefined) { setTimeout(() => A(A.ENUM.APP_UNSUSPENDED), 2000); const clipboard = await Clipboard.getString(); const isAddressFromStoredWallet = BlueApp.getWallets().some(wallet => { if (wallet.chain === Chain.ONCHAIN) { // checking address validity is faster than unwrapping hierarchy only to compare it to garbage return wallet.isAddressValid && wallet.isAddressValid(clipboard) && wallet.weOwnAddress(clipboard); } else { return wallet.isInvoiceGeneratedByWallet(clipboard) || wallet.weOwnAddress(clipboard); } }); const isBitcoinAddress = DeeplinkSchemaMatch.isBitcoinAddress(clipboard) || BitcoinBIP70TransactionDecode.matchesPaymentURL(clipboard); const isLightningInvoice = DeeplinkSchemaMatch.isLightningInvoice(clipboard); const isLNURL = DeeplinkSchemaMatch.isLnUrl(clipboard); const isBothBitcoinAndLightning = DeeplinkSchemaMatch.isBothBitcoinAndLightning(clipboard); if ( !isAddressFromStoredWallet && this.state.clipboardContent !== clipboard && (isBitcoinAddress || isLightningInvoice || isLNURL || isBothBitcoinAndLightning) ) { if (isBitcoinAddress) { this.setState({ clipboardContentModalAddressType: bitcoinModalString }); } else if (isLightningInvoice || isLNURL) { this.setState({ clipboardContentModalAddressType: lightningModalString }); } else if (isBothBitcoinAndLightning) { this.setState({ clipboardContentModalAddressType: bitcoinModalString }); } this.setState({ isClipboardContentModalVisible: true }); } this.setState({ clipboardContent: clipboard }); } if (nextAppState) { this.setState({ appState: nextAppState }); } } }; isBothBitcoinAndLightningWalletSelect = wallet => { const clipboardContent = this.state.clipboardContent; if (wallet.chain === Chain.ONCHAIN) { this.navigation && this.navigation.dispatch( CommonActions.navigate({ name: 'SendDetails', params: { uri: clipboardContent.bitcoin, fromWallet: wallet, }, }), ); } else if (wallet.chain === Chain.OFFCHAIN) { this.navigation && this.navigation.dispatch( CommonActions.navigate({ name: 'ScanLndInvoice', params: { uri: clipboardContent.lndInvoice, fromSecret: wallet.getSecret(), }, }), ); } }; handleOpenURL = event => { DeeplinkSchemaMatch.navigationRouteFor(event, value => this.navigation && this.navigation.navigate(...value)); }; renderClipboardContentModal = () => { return ( ReactNativeHapticFeedback.trigger('impactLight', { ignoreAndroidSystemSettings: false })} isVisible={this.state.isClipboardContentModalVisible} style={styles.bottomModal} onBackdropPress={() => { this.setState({ isClipboardContentModalVisible: false }); }} > You have a {this.state.clipboardContentModalAddressType} on your clipboard. Would you like to use it for a transaction? this.setState({ isClipboardContentModalVisible: false })} /> { this.setState({ isClipboardContentModalVisible: false }, async () => { const clipboard = await Clipboard.getString(); setTimeout(() => this.handleOpenURL({ url: clipboard }), 100); }); }} /> ); }; render() { return ( { this.navigation = nav; NavigationService.setTopLevelNavigator(nav); }} > {this.renderClipboardContentModal()} ); } } const styles = StyleSheet.create({ root: { flex: 1, }, space: { marginHorizontal: 8, }, modalContent: { backgroundColor: '#FFFFFF', padding: 22, justifyContent: 'center', alignItems: 'center', borderTopLeftRadius: 16, borderTopRightRadius: 16, borderColor: 'rgba(0, 0, 0, 0.1)', minHeight: 200, height: 200, }, bottomModal: { justifyContent: 'flex-end', margin: 0, }, modelContentButtonLayout: { flexDirection: 'row', margin: 16, justifyContent: 'space-between', alignItems: 'flex-end', }, });