BlueWallet/screen/wallets/add.js
2020-11-05 16:58:12 +01:00

409 lines
13 KiB
JavaScript

/* global alert */
import React, { useState, useEffect, useContext } from 'react';
import {
Text,
ScrollView,
ActivityIndicator,
Keyboard,
KeyboardAvoidingView,
Platform,
View,
StatusBar,
TextInput,
StyleSheet,
} from 'react-native';
import AsyncStorage from '@react-native-community/async-storage';
import {
BlueTextHooks,
BlueListItem,
LightningButton,
BitcoinButton,
VaultButton,
BlueFormLabel,
BlueButtonHook,
BlueNavigationStyle,
BlueButtonLinkHook,
BlueSpacing20,
} from '../../BlueComponents';
import { HDSegwitBech32Wallet, SegwitP2SHWallet, HDSegwitP2SHWallet, LightningCustodianWallet, AppStorage } from '../../class';
import ReactNativeHapticFeedback from 'react-native-haptic-feedback';
import { useTheme, useNavigation } from '@react-navigation/native';
import { Chain } from '../../models/bitcoinUnits';
import loc from '../../loc';
import { BlueStorageContext } from '../../blue_modules/storage-context';
const A = require('../../blue_modules/analytics');
const ButtonSelected = Object.freeze({
ONCHAIN: Chain.ONCHAIN,
OFFCHAIN: Chain.OFFCHAIN,
VAULT: 'VAULT',
});
const WalletsAdd = () => {
const { colors } = useTheme();
const { addWallet, saveToDisk, setNewWalletAdded, isAdancedModeEnabled } = useContext(BlueStorageContext);
const [isLoading, setIsLoading] = useState(true);
const [walletBaseURI, setWalletBaseURI] = useState();
const [selectedIndex, setSelectedIndex] = useState(0);
const [label, setLabel] = useState('');
const [isAdvancedOptionsEnabled, setIsAdvancedOptionsEnabled] = useState(false);
const [selectedWalletType, setSelectedWalletType] = useState(false);
const [backdoorPressedTimes, setBackdoorPressedTimes] = useState(0);
const { navigate, goBack } = useNavigation();
const [entropy, setEntropy] = useState();
const [entropyButtonText, setEntropyButtonText] = useState(loc.wallets.add_entropy_provide);
const stylesHook = {
advancedText: {
color: colors.feeText,
},
label: {
borderColor: colors.formBorder,
borderBottomColor: colors.formBorder,
backgroundColor: colors.inputBackgroundColor,
},
noPadding: {
backgroundColor: colors.elevated,
},
root: {
backgroundColor: colors.elevated,
},
lndUri: {
borderColor: colors.formBorder,
borderBottomColor: colors.formBorder,
backgroundColor: colors.inputBackgroundColor,
},
};
useEffect(() => {
AsyncStorage.getItem(AppStorage.LNDHUB)
.then(setWalletBaseURI)
.catch(() => setWalletBaseURI(''));
isAdancedModeEnabled()
.then(setIsAdvancedOptionsEnabled)
.finally(() => setIsLoading(false));
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [isAdvancedOptionsEnabled]);
const entropyGenerated = newEntropy => {
let entropyTitle;
if (!newEntropy) {
entropyTitle = loc.wallets.add_entropy_provide;
} else if (newEntropy.length < 32) {
entropyTitle = loc.formatString(loc.wallets.add_entropy_remain, {
gen: newEntropy.length,
rem: 32 - newEntropy.length,
});
} else {
entropyTitle = loc.formatString(loc.wallets.add_entropy_generated, {
gen: newEntropy.length,
});
}
setEntropy(newEntropy);
setEntropyButtonText(entropyTitle);
};
const createWallet = async () => {
setIsLoading(true);
let w;
if (selectedWalletType === ButtonSelected.OFFCHAIN) {
createLightningWallet(w);
} else if (selectedWalletType === ButtonSelected.ONCHAIN) {
if (selectedIndex === 2) {
// zero index radio - HD segwit
w = new HDSegwitP2SHWallet();
w.setLabel(label || loc.wallets.details_title);
} else if (selectedIndex === 1) {
// btc was selected
// index 1 radio - segwit single address
w = new SegwitP2SHWallet();
w.setLabel(label || loc.wallets.details_title);
} else {
// btc was selected
// index 2 radio - hd bip84
w = new HDSegwitBech32Wallet();
w.setLabel(label || loc.wallets.details_title);
}
if (selectedWalletType === ButtonSelected.ONCHAIN) {
if (entropy) {
try {
await w.generateFromEntropy(entropy);
} catch (e) {
console.log(e.toString());
alert(e.toString());
goBack();
return;
}
} else {
await w.generate();
}
addWallet(w);
await saveToDisk();
setNewWalletAdded(true);
A(A.ENUM.CREATED_WALLET);
ReactNativeHapticFeedback.trigger('notificationSuccess', { ignoreAndroidSystemSettings: false });
if (w.type === HDSegwitP2SHWallet.type || w.type === HDSegwitBech32Wallet.type) {
navigate('PleaseBackup', {
walletID: w.getID(),
});
} else {
goBack();
}
}
} else if (selectedWalletType === ButtonSelected.VAULT) {
setIsLoading(false);
navigate('WalletsAddMultisig');
}
};
const createLightningWallet = async wallet => {
wallet = new LightningCustodianWallet();
wallet.setLabel(label || loc.wallets.details_title);
try {
const lndhub = walletBaseURI && walletBaseURI.trim().length > 0 ? walletBaseURI : LightningCustodianWallet.defaultBaseUri;
if (lndhub) {
const isValidNodeAddress = await LightningCustodianWallet.isValidNodeAddress(lndhub);
if (isValidNodeAddress) {
wallet.setBaseURI(lndhub);
wallet.init();
} else {
throw new Error('The provided node address is not valid LNDHub node.');
}
}
await wallet.createAccount();
await wallet.authorize();
} catch (Err) {
setIsLoading(false);
console.warn('lnd create failure', Err);
return alert(Err);
// giving app, not adding anything
}
A(A.ENUM.CREATED_LIGHTNING_WALLET);
await wallet.generate();
addWallet(wallet);
await saveToDisk();
setNewWalletAdded(true);
A(A.ENUM.CREATED_WALLET);
ReactNativeHapticFeedback.trigger('notificationSuccess', { ignoreAndroidSystemSettings: false });
navigate('PleaseBackupLNDHub', {
wallet,
});
};
const navigateToEntropy = () => {
navigate('ProvideEntropy', { onGenerated: entropyGenerated });
};
const navigateToImportWallet = () => {
navigate('ImportWallet');
};
const handleOnVaultButtonPressed = () => {
setSelectedWalletType(ButtonSelected.VAULT);
};
const handleOnBitcoinButtonPressed = () => {
Keyboard.dismiss();
setSelectedWalletType(ButtonSelected.ONCHAIN);
setBackdoorPressedTimes(backdoorPressedTimes + 1);
};
const handleOnLightningButtonPressed = () => {
Keyboard.dismiss();
setSelectedWalletType(ButtonSelected.OFFCHAIN);
};
return (
<ScrollView style={stylesHook.root}>
<StatusBar barStyle="light-content" />
<BlueSpacing20 />
<KeyboardAvoidingView enabled behavior={Platform.OS === 'ios' ? 'padding' : null} keyboardVerticalOffset={62}>
<BlueFormLabel>{loc.wallets.add_wallet_name}</BlueFormLabel>
<View style={[styles.label, stylesHook.label]}>
<TextInput
testID="WalletNameInput"
value={label}
placeholderTextColor="#81868e"
placeholder="my first wallet"
onChangeText={setLabel}
style={styles.textInputCommon}
editable={!isLoading}
underlineColorAndroid="transparent"
/>
</View>
<BlueFormLabel>{loc.wallets.add_wallet_type}</BlueFormLabel>
<View style={styles.buttons}>
<BitcoinButton
testID="ActivateBitcoinButton"
active={selectedWalletType === ButtonSelected.ONCHAIN}
onPress={handleOnBitcoinButtonPressed}
style={styles.button}
/>
<LightningButton
active={selectedWalletType === ButtonSelected.OFFCHAIN}
onPress={handleOnLightningButtonPressed}
style={styles.button}
/>
{backdoorPressedTimes >= 15 && (
<VaultButton active={selectedWalletType === ButtonSelected.VAULT} onPress={handleOnVaultButtonPressed} style={styles.button} />
)}
</View>
<View style={styles.advanced}>
{(() => {
if (selectedWalletType === ButtonSelected.ONCHAIN && isAdvancedOptionsEnabled) {
return (
<View>
<BlueSpacing20 />
<Text style={[styles.advancedText, stylesHook.advancedText]}>{loc.settings.advanced_options}</Text>
<BlueListItem
containerStyle={[styles.noPadding, stylesHook.noPadding]}
bottomDivider={false}
onPress={() => setSelectedIndex(0)}
title={HDSegwitBech32Wallet.typeReadable}
checkmark={selectedIndex === 0}
/>
<BlueListItem
containerStyle={[styles.noPadding, stylesHook.noPadding]}
bottomDivider={false}
onPress={() => setSelectedIndex(1)}
title={SegwitP2SHWallet.typeReadable}
checkmark={selectedIndex === 1}
/>
<BlueListItem
containerStyle={[styles.noPadding, stylesHook.noPadding]}
bottomDivider={false}
onPress={() => setSelectedIndex(2)}
title={HDSegwitP2SHWallet.typeReadable}
checkmark={selectedIndex === 2}
/>
</View>
);
} else if (selectedWalletType === ButtonSelected.OFFCHAIN && isAdvancedOptionsEnabled) {
return (
<>
<BlueSpacing20 />
<Text style={[styles.advancedText, stylesHook.advancedText]}>{loc.settings.advanced_options}</Text>
<BlueSpacing20 />
<BlueTextHooks>Connect to your LNDHub</BlueTextHooks>
<View style={[styles.lndUri, stylesHook.lndUri]}>
<TextInput
value={walletBaseURI}
onChangeText={setWalletBaseURI}
onSubmitEditing={Keyboard.dismiss}
placeholder="your node address"
clearButtonMode="while-editing"
autoCapitalize="none"
textContentType="URL"
autoCorrect={false}
placeholderTextColor="#81868e"
style={styles.textInputCommon}
editable={!isLoading}
underlineColorAndroid="transparent"
/>
</View>
</>
);
}
})()}
<BlueSpacing20 />
<View style={styles.createButton}>
{!isLoading ? (
<BlueButtonHook testID="Create" title={loc.wallets.add_create} disabled={!selectedWalletType} onPress={createWallet} />
) : (
<ActivityIndicator />
)}
</View>
{!isLoading && (
<BlueButtonLinkHook
testID="ImportWallet"
style={styles.import}
title={loc.wallets.add_import_wallet}
onPress={navigateToImportWallet}
/>
)}
{isAdvancedOptionsEnabled && !isLoading && (
<BlueButtonLinkHook style={styles.import} title={entropyButtonText} onPress={navigateToEntropy} />
)}
</View>
</KeyboardAvoidingView>
</ScrollView>
);
};
WalletsAdd.navigationOptions = ({ navigation }) => ({
...BlueNavigationStyle(navigation, true),
headerTitle: loc.wallets.add_title,
headerLeft: null,
});
const styles = StyleSheet.create({
createButton: {
alignItems: 'center',
flex: 1,
justifyContent: 'center',
},
loading: {
flex: 1,
paddingTop: 20,
},
label: {
flexDirection: 'row',
borderWidth: 1,
borderBottomWidth: 0.5,
minHeight: 44,
height: 44,
marginHorizontal: 20,
alignItems: 'center',
marginVertical: 16,
borderRadius: 4,
},
textInputCommon: {
flex: 1,
marginHorizontal: 8,
color: '#81868e',
},
buttons: {
flexDirection: 'column',
marginHorizontal: 20,
marginTop: 16,
borderWidth: 0,
minHeight: 100,
},
button: {
width: '100%',
height: 'auto',
},
advanced: {
marginHorizontal: 20,
},
advancedText: {
fontWeight: '500',
},
lndUri: {
flexDirection: 'row',
borderWidth: 1,
borderBottomWidth: 0.5,
minHeight: 44,
height: 44,
alignItems: 'center',
marginVertical: 16,
borderRadius: 4,
},
import: {
marginBottom: 0,
marginTop: 24,
},
noPadding: {
paddingHorizontal: 0,
},
typeMargin: {
marginTop: 8,
},
});
export default WalletsAdd;