2021-03-23 13:16:32 +01:00
|
|
|
import React, { useEffect, useState, useContext } from 'react';
|
|
|
|
import {
|
|
|
|
ActivityIndicator,
|
|
|
|
Alert,
|
|
|
|
Keyboard,
|
|
|
|
KeyboardAvoidingView,
|
|
|
|
Platform,
|
|
|
|
StyleSheet,
|
|
|
|
TextInput,
|
|
|
|
TouchableWithoutFeedback,
|
|
|
|
View,
|
|
|
|
} from 'react-native';
|
2021-04-15 19:46:10 +02:00
|
|
|
import { useRoute, useTheme, useNavigation } from '@react-navigation/native';
|
2021-03-23 13:16:32 +01:00
|
|
|
import ReactNativeHapticFeedback from 'react-native-haptic-feedback';
|
|
|
|
|
2021-04-15 19:46:10 +02:00
|
|
|
import AOPP from '../../class/aopp';
|
2021-03-23 13:16:32 +01:00
|
|
|
import { BlueDoneAndDismissKeyboardInputAccessory, BlueFormLabel, BlueSpacing10, BlueSpacing20, SafeBlueArea } from '../../BlueComponents';
|
|
|
|
import navigationStyle from '../../components/navigationStyle';
|
|
|
|
import { FContainer, FButton } from '../../components/FloatButtons';
|
|
|
|
import { BlueStorageContext } from '../../blue_modules/storage-context';
|
|
|
|
import loc from '../../loc';
|
2021-04-15 19:46:10 +02:00
|
|
|
import confirm from '../../helpers/confirm';
|
2021-07-29 22:01:31 +02:00
|
|
|
import { Icon } from 'react-native-elements';
|
|
|
|
import Share from 'react-native-share';
|
2021-03-23 13:16:32 +01:00
|
|
|
|
|
|
|
const SignVerify = () => {
|
|
|
|
const { colors } = useTheme();
|
|
|
|
const { wallets, sleep } = useContext(BlueStorageContext);
|
|
|
|
const { params } = useRoute();
|
2021-04-15 19:46:10 +02:00
|
|
|
const navigation = useNavigation();
|
2021-03-23 13:16:32 +01:00
|
|
|
const [isKeyboardVisible, setIsKeyboardVisible] = useState(false);
|
2021-04-15 19:46:10 +02:00
|
|
|
const [address, setAddress] = useState(params.address ?? '');
|
|
|
|
const [message, setMessage] = useState(params.message ?? '');
|
2021-03-23 13:16:32 +01:00
|
|
|
const [signature, setSignature] = useState('');
|
|
|
|
const [loading, setLoading] = useState(false);
|
|
|
|
const [messageHasFocus, setMessageHasFocus] = useState(false);
|
2021-07-29 22:01:31 +02:00
|
|
|
const [isShareVisible, setIsShareVisible] = useState(false);
|
2021-03-23 13:16:32 +01:00
|
|
|
|
|
|
|
const wallet = wallets.find(w => w.getID() === params.walletID);
|
|
|
|
const isToolbarVisibleForAndroid = Platform.OS === 'android' && messageHasFocus && isKeyboardVisible;
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
Keyboard.addListener(Platform.OS === 'ios' ? 'keyboardWillShow' : 'keyboardDidShow', () => setIsKeyboardVisible(true));
|
|
|
|
Keyboard.addListener(Platform.OS === 'ios' ? 'keyboardWillHide' : 'keyboardDidHide', () => setIsKeyboardVisible(false));
|
|
|
|
return () => {
|
|
|
|
Keyboard.removeListener(Platform.OS === 'ios' ? 'keyboardWillHide' : 'keyboardDidHide');
|
|
|
|
Keyboard.removeListener(Platform.OS === 'ios' ? 'keyboardWillShow' : 'keyboardDidShow');
|
|
|
|
};
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
const stylesHooks = StyleSheet.create({
|
|
|
|
root: {
|
|
|
|
backgroundColor: colors.elevated,
|
|
|
|
},
|
|
|
|
text: {
|
|
|
|
borderColor: colors.formBorder,
|
|
|
|
borderBottomColor: colors.formBorder,
|
|
|
|
backgroundColor: colors.inputBackgroundColor,
|
|
|
|
color: colors.foregroundColor,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2021-07-29 22:01:31 +02:00
|
|
|
const handleShare = () => {
|
|
|
|
const baseUri = 'https://bluewallet.github.io/VerifySignature';
|
|
|
|
const uri = `${baseUri}?a=${address}&m=${encodeURIComponent(message)}&s=${encodeURIComponent(signature)}`;
|
|
|
|
Share.open({ message: uri }).catch(error => console.log(error));
|
|
|
|
};
|
|
|
|
|
2021-03-23 13:16:32 +01:00
|
|
|
const handleSign = async () => {
|
|
|
|
setLoading(true);
|
|
|
|
await sleep(10); // wait for loading indicator to appear
|
2021-04-15 19:46:10 +02:00
|
|
|
let newSignature;
|
|
|
|
const useSegwit = Boolean(params.aoppURI);
|
2021-03-23 13:16:32 +01:00
|
|
|
try {
|
2021-04-15 19:46:10 +02:00
|
|
|
newSignature = wallet.signMessage(message, address, useSegwit);
|
2021-03-23 13:16:32 +01:00
|
|
|
setSignature(newSignature);
|
2021-07-29 22:01:31 +02:00
|
|
|
setIsShareVisible(true);
|
2021-03-23 13:16:32 +01:00
|
|
|
} catch (e) {
|
|
|
|
ReactNativeHapticFeedback.trigger('notificationError', { ignoreAndroidSystemSettings: false });
|
|
|
|
Alert.alert(loc.errors.error, e.message);
|
|
|
|
}
|
2021-04-15 19:46:10 +02:00
|
|
|
|
|
|
|
if (!params.aoppURI) return setLoading(false);
|
|
|
|
|
|
|
|
let aopp;
|
|
|
|
try {
|
|
|
|
aopp = new AOPP(params.aoppURI);
|
|
|
|
} catch (e) {
|
|
|
|
ReactNativeHapticFeedback.trigger('notificationError', { ignoreAndroidSystemSettings: false });
|
|
|
|
Alert.alert(loc.errors.error, e.message);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
|
|
|
!(await confirm(
|
|
|
|
loc.addresses.sign_aopp_title,
|
|
|
|
loc.formatString(loc.addresses.sign_aopp_confirm, { hostname: aopp.callbackHostname }),
|
|
|
|
))
|
|
|
|
)
|
|
|
|
return setLoading(false);
|
|
|
|
|
|
|
|
try {
|
|
|
|
await aopp.send({ address, signature: newSignature });
|
|
|
|
Alert.alert(loc._.success, loc.aopp.send_success);
|
|
|
|
navigation.dangerouslyGetParent().pop();
|
|
|
|
} catch (e) {
|
|
|
|
ReactNativeHapticFeedback.trigger('notificationError', { ignoreAndroidSystemSettings: false });
|
|
|
|
Alert.alert(loc.errors.error, loc.aopp.send_error);
|
|
|
|
}
|
|
|
|
|
2021-03-23 13:16:32 +01:00
|
|
|
setLoading(false);
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleVerify = async () => {
|
|
|
|
setLoading(true);
|
|
|
|
await sleep(10); // wait for loading indicator to appear
|
|
|
|
try {
|
|
|
|
const res = wallet.verifyMessage(message, address, signature);
|
|
|
|
Alert.alert(
|
|
|
|
res ? loc._.success : loc.errors.error,
|
|
|
|
res ? loc.addresses.sign_signature_correct : loc.addresses.sign_signature_incorrect,
|
|
|
|
);
|
|
|
|
if (res) {
|
|
|
|
ReactNativeHapticFeedback.trigger('notificationSuccess', { ignoreAndroidSystemSettings: false });
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
ReactNativeHapticFeedback.trigger('notificationError', { ignoreAndroidSystemSettings: false });
|
|
|
|
Alert.alert(loc.errors.error, e.message);
|
|
|
|
}
|
|
|
|
setLoading(false);
|
|
|
|
};
|
|
|
|
|
|
|
|
if (loading)
|
|
|
|
return (
|
2021-03-25 01:27:51 +01:00
|
|
|
<View style={[stylesHooks.root, styles.loading]}>
|
2021-03-23 13:16:32 +01:00
|
|
|
<ActivityIndicator />
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
|
|
|
|
return (
|
2021-04-19 12:24:04 +02:00
|
|
|
<SafeBlueArea style={[styles.root, stylesHooks.root]}>
|
2021-03-23 13:16:32 +01:00
|
|
|
<TouchableWithoutFeedback onPress={Keyboard.dismiss} accessible={false}>
|
|
|
|
<KeyboardAvoidingView style={[styles.root, stylesHooks.root]}>
|
|
|
|
{!isKeyboardVisible && (
|
|
|
|
<>
|
|
|
|
<BlueSpacing20 />
|
|
|
|
<BlueFormLabel>{loc.addresses.sign_help}</BlueFormLabel>
|
|
|
|
<BlueSpacing20 />
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
|
|
|
|
<TextInput
|
|
|
|
multiline
|
|
|
|
textAlignVertical="top"
|
|
|
|
blurOnSubmit
|
|
|
|
placeholder={loc.addresses.sign_placeholder_address}
|
|
|
|
placeholderTextColor="#81868e"
|
|
|
|
value={address}
|
|
|
|
onChangeText={t => setAddress(t.replace('\n', ''))}
|
|
|
|
testID="Signature"
|
|
|
|
style={[styles.text, stylesHooks.text]}
|
|
|
|
autoCorrect={false}
|
|
|
|
autoCapitalize="none"
|
|
|
|
spellCheck={false}
|
|
|
|
editable={!loading}
|
|
|
|
/>
|
|
|
|
<BlueSpacing10 />
|
|
|
|
|
|
|
|
<TextInput
|
|
|
|
multiline
|
|
|
|
textAlignVertical="top"
|
|
|
|
blurOnSubmit
|
|
|
|
placeholder={loc.addresses.sign_placeholder_signature}
|
|
|
|
placeholderTextColor="#81868e"
|
|
|
|
value={signature}
|
|
|
|
onChangeText={t => setSignature(t.replace('\n', ''))}
|
|
|
|
testID="Signature"
|
|
|
|
style={[styles.text, stylesHooks.text]}
|
|
|
|
autoCorrect={false}
|
|
|
|
autoCapitalize="none"
|
|
|
|
spellCheck={false}
|
|
|
|
editable={!loading}
|
|
|
|
/>
|
|
|
|
<BlueSpacing10 />
|
|
|
|
|
|
|
|
<TextInput
|
|
|
|
multiline
|
|
|
|
placeholder={loc.addresses.sign_placeholder_message}
|
|
|
|
placeholderTextColor="#81868e"
|
|
|
|
value={message}
|
|
|
|
onChangeText={setMessage}
|
|
|
|
testID="Message"
|
|
|
|
inputAccessoryViewID={BlueDoneAndDismissKeyboardInputAccessory.InputAccessoryViewID}
|
|
|
|
style={[styles.flex, styles.text, styles.textMessage, stylesHooks.text]}
|
|
|
|
autoCorrect={false}
|
|
|
|
autoCapitalize="none"
|
|
|
|
spellCheck={false}
|
|
|
|
editable={!loading}
|
|
|
|
onFocus={() => setMessageHasFocus(true)}
|
|
|
|
onBlur={() => setMessageHasFocus(false)}
|
|
|
|
/>
|
|
|
|
<BlueSpacing10 />
|
|
|
|
|
2021-07-29 22:01:31 +02:00
|
|
|
{isShareVisible && !isKeyboardVisible && (
|
|
|
|
<>
|
|
|
|
<FContainer inline>
|
|
|
|
<FButton
|
|
|
|
onPress={handleShare}
|
|
|
|
text={loc.multisig.share}
|
|
|
|
icon={
|
|
|
|
<View style={styles.buttonsIcon}>
|
|
|
|
<Icon name="external-link" size={16} type="font-awesome" color={colors.buttonAlternativeTextColor} />
|
|
|
|
</View>
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
</FContainer>
|
|
|
|
<BlueSpacing10 />
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
|
2021-03-23 13:16:32 +01:00
|
|
|
{!isKeyboardVisible && (
|
|
|
|
<>
|
|
|
|
<FContainer inline>
|
2021-04-15 19:46:10 +02:00
|
|
|
<FButton
|
|
|
|
onPress={handleSign}
|
|
|
|
text={params.aoppURI ? loc.addresses.sign_sign_submit : loc.addresses.sign_sign}
|
|
|
|
disabled={loading}
|
|
|
|
/>
|
2021-03-23 13:16:32 +01:00
|
|
|
<FButton onPress={handleVerify} text={loc.addresses.sign_verify} disabled={loading} />
|
|
|
|
</FContainer>
|
|
|
|
<BlueSpacing10 />
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
|
|
|
|
{Platform.select({
|
|
|
|
ios: (
|
|
|
|
<BlueDoneAndDismissKeyboardInputAccessory
|
|
|
|
onClearTapped={() => setMessage('')}
|
|
|
|
onPasteTapped={text => {
|
|
|
|
setMessage(text);
|
|
|
|
Keyboard.dismiss();
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
),
|
|
|
|
android: isToolbarVisibleForAndroid && (
|
|
|
|
<BlueDoneAndDismissKeyboardInputAccessory
|
|
|
|
onClearTapped={() => {
|
|
|
|
setMessage('');
|
|
|
|
Keyboard.dismiss();
|
|
|
|
}}
|
|
|
|
onPasteTapped={text => {
|
|
|
|
setMessage(text);
|
|
|
|
Keyboard.dismiss();
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
),
|
|
|
|
})}
|
|
|
|
</KeyboardAvoidingView>
|
|
|
|
</TouchableWithoutFeedback>
|
|
|
|
</SafeBlueArea>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2021-03-25 01:27:51 +01:00
|
|
|
SignVerify.navigationOptions = navigationStyle({ closeButton: true, headerLeft: null }, opts => ({
|
|
|
|
...opts,
|
|
|
|
title: loc.addresses.sign_title,
|
|
|
|
}));
|
2021-03-23 13:16:32 +01:00
|
|
|
|
|
|
|
export default SignVerify;
|
|
|
|
|
|
|
|
const styles = StyleSheet.create({
|
|
|
|
root: {
|
|
|
|
flex: 1,
|
|
|
|
},
|
|
|
|
text: {
|
|
|
|
paddingHorizontal: 8,
|
|
|
|
paddingVertical: 8,
|
|
|
|
marginTop: 5,
|
|
|
|
marginHorizontal: 20,
|
|
|
|
borderWidth: 1,
|
|
|
|
borderBottomWidth: 0.5,
|
|
|
|
borderRadius: 4,
|
|
|
|
textAlignVertical: 'top',
|
|
|
|
},
|
|
|
|
textMessage: {
|
|
|
|
minHeight: 50,
|
|
|
|
},
|
|
|
|
flex: {
|
|
|
|
flex: 1,
|
|
|
|
},
|
|
|
|
loading: {
|
|
|
|
position: 'absolute',
|
|
|
|
top: 0,
|
|
|
|
left: 0,
|
|
|
|
right: 0,
|
|
|
|
bottom: 0,
|
|
|
|
justifyContent: 'center',
|
|
|
|
alignItems: 'center',
|
|
|
|
},
|
|
|
|
});
|