BlueWallet/screen/lnd/lndCreateInvoice.js

273 lines
9.3 KiB
JavaScript
Raw Normal View History

/* global alert */
2018-12-25 17:34:51 +01:00
import React, { Component } from 'react';
2019-08-04 21:34:17 +02:00
import {
Dimensions,
ActivityIndicator,
View,
TextInput,
KeyboardAvoidingView,
Keyboard,
TouchableWithoutFeedback,
TouchableOpacity,
Text,
} from 'react-native';
2019-08-03 14:27:09 +02:00
import { BlueNavigationStyle, BlueButton, BlueBitcoinAmount, BlueDismissKeyboardInputAccessory } from '../../BlueComponents';
2019-09-03 22:44:57 +02:00
import { LightningCustodianWallet } from '../../class/lightning-custodian-wallet';
2018-12-25 17:34:51 +01:00
import PropTypes from 'prop-types';
import bech32 from 'bech32';
2018-12-25 17:34:51 +01:00
import { BitcoinUnit } from '../../models/bitcoinUnits';
2019-08-03 14:27:09 +02:00
import NavigationService from '../../NavigationService';
2018-12-25 17:34:51 +01:00
import ReactNativeHapticFeedback from 'react-native-haptic-feedback';
2019-08-03 14:27:09 +02:00
let BlueApp = require('../../BlueApp');
let EV = require('../../events');
2018-12-25 17:34:51 +01:00
let loc = require('../../loc');
2019-08-03 14:27:09 +02:00
const { width } = Dimensions.get('window');
2018-12-25 17:34:51 +01:00
export default class LNDCreateInvoice extends Component {
static navigationOptions = ({ navigation }) => ({
...BlueNavigationStyle(navigation, true),
title: loc.receive.header,
});
constructor(props) {
super(props);
2019-09-03 22:44:57 +02:00
let fromWallet;
if (props.navigation.state.params.fromWallet) fromWallet = props.navigation.getParam('fromWallet');
2019-09-03 22:44:57 +02:00
2018-12-25 17:34:51 +01:00
// fallback to first wallet if it exists
2019-09-03 22:44:57 +02:00
if (!fromWallet) {
const lightningWallets = BlueApp.getWallets().filter(item => item.type === LightningCustodianWallet.type);
if (lightningWallets.length > 0) {
fromWallet = lightningWallets[0];
console.warn('warning: using ln wallet index 0');
}
}
2018-12-25 17:34:51 +01:00
this.state = {
fromWallet,
amount: '',
description: '',
2018-12-25 17:34:51 +01:00
isLoading: false,
lnurl: '',
lnurlParams: null,
2018-12-25 17:34:51 +01:00
};
}
2019-10-06 13:40:21 +02:00
componentDidMount() {
2019-09-03 22:44:57 +02:00
if (this.props.navigation.state.params.uri) {
this.processLnurl(this.props.navigation.getParam('uri'));
}
}
2018-12-25 17:34:51 +01:00
async createInvoice() {
this.setState({ isLoading: true }, async () => {
try {
const invoiceRequest = await this.state.fromWallet.addInvoice(this.state.amount, this.state.description);
EV(EV.enum.TRANSACTIONS_COUNT_CHANGED);
2019-05-03 14:36:11 +02:00
ReactNativeHapticFeedback.trigger('notificationSuccess', { ignoreAndroidSystemSettings: false });
// send to lnurl-withdraw callback url if that exists
if (this.state.lnurlParams) {
2019-08-04 21:34:17 +02:00
let { callback, k1 } = this.state.lnurlParams;
let callbackUrl = callback + (callback.indexOf('?') !== -1 ? '&' : '?') + 'k1=' + k1 + '&pr=' + invoiceRequest;
2019-08-04 21:34:17 +02:00
let resp = await fetch(callbackUrl, { method: 'GET' });
if (resp.status >= 300) {
let text = await resp.text();
throw new Error(text);
}
let reply = await resp.json();
if (reply.status === 'ERROR') {
throw new Error('Reply from server: ' + reply.reason);
}
}
2019-08-24 23:11:47 +02:00
await BlueApp.saveToDisk();
2018-12-25 17:34:51 +01:00
this.props.navigation.navigate('LNDViewInvoice', {
invoice: invoiceRequest,
fromWallet: this.state.fromWallet,
isModal: true,
2018-12-25 17:34:51 +01:00
});
} catch (Err) {
ReactNativeHapticFeedback.trigger('notificationError', { ignoreAndroidSystemSettings: false });
this.setState({ isLoading: false });
alert(Err.message);
}
});
}
processLnurl = data => {
this.setState({ isLoading: true }, async () => {
if (!this.state.fromWallet) {
2019-05-03 14:36:11 +02:00
ReactNativeHapticFeedback.trigger('notificationError', { ignoreAndroidSystemSettings: false });
alert('Before paying a Lightning invoice, you must first add a Lightning wallet.');
return this.props.navigation.goBack();
}
// handling fallback lnurl
let ind = data.indexOf('lightning=');
if (ind !== -1) {
data = data.substring(ind + 10).split('&')[0];
}
data = data.replace('LIGHTNING:', '').replace('lightning:', '');
console.log(data);
// decoding the lnurl
let decoded = bech32.decode(data, 1500);
let url = Buffer.from(bech32.fromWords(decoded.words)).toString();
// calling the url
try {
2019-08-04 21:34:17 +02:00
let resp = await fetch(url, { method: 'GET' });
if (resp.status >= 300) {
2019-08-04 21:34:17 +02:00
throw new Error('Bad response from server');
}
let reply = await resp.json();
if (reply.status === 'ERROR') {
throw new Error('Reply from server: ' + reply.reason);
}
if (reply.tag !== 'withdrawRequest') {
throw new Error('Unsupported lnurl');
}
// setting the invoice creating screen with the parameters
this.setState({
isLoading: false,
lnurlParams: {
k1: reply.k1,
callback: reply.callback,
fixed: reply.minWithdrawable === reply.maxWithdrawable,
2019-08-03 14:27:09 +02:00
min: (reply.minWithdrawable || 0) / 1000,
2019-08-04 21:34:17 +02:00
max: reply.maxWithdrawable / 1000,
},
amount: (reply.maxWithdrawable / 1000).toString(),
description: reply.defaultDescription,
});
} catch (Err) {
Keyboard.dismiss();
this.setState({ isLoading: false });
ReactNativeHapticFeedback.trigger('notificationError', { ignoreAndroidSystemSettings: false });
alert(Err.message);
2018-12-25 17:34:51 +01:00
}
});
2019-08-04 21:34:17 +02:00
};
2018-12-25 17:34:51 +01:00
renderCreateButton = () => {
return (
<View style={{ marginHorizontal: 56, marginVertical: 16, minHeight: 45, alignContent: 'center', backgroundColor: '#FFFFFF' }}>
2018-12-25 17:34:51 +01:00
{this.state.isLoading ? (
<ActivityIndicator />
) : (
2019-01-24 20:31:30 +01:00
<BlueButton disabled={!this.state.amount > 0} onPress={() => this.createInvoice()} title={loc.send.details.create} />
2018-12-25 17:34:51 +01:00
)}
</View>
);
};
renderScanClickable = () => {
return (
<View style={{ marginHorizontal: 0, marginVertical: 16, minHeight: 25, alignContent: 'center' }}>
2019-08-03 14:27:09 +02:00
<TouchableOpacity
2019-08-04 21:34:17 +02:00
onPress={() => NavigationService.navigate('ScanQrAddress', { onBarScanned: this.processLnurl })}
2019-08-03 14:27:09 +02:00
style={{
flex: 1,
flexDirection: 'row',
minWidth: width,
justifyContent: 'center',
2019-08-04 21:34:17 +02:00
alignItems: 'center',
2019-08-03 14:27:09 +02:00
}}
>
2019-08-04 21:34:17 +02:00
<Text style={{ color: BlueApp.settings.buttonTextColor, textAlign: 'center' }}>{loc.receive.scan_lnurl}</Text>
2019-08-03 14:27:09 +02:00
</TouchableOpacity>
</View>
);
};
2018-12-25 17:34:51 +01:00
render() {
if (!this.state.fromWallet) {
return (
<View style={{ flex: 1, paddingTop: 20 }}>
<Text>System error: Source wallet not found (this should never happen)</Text>
</View>
);
}
return (
<TouchableWithoutFeedback onPress={Keyboard.dismiss} accessible={false}>
<View style={{ flex: 1, justifyContent: 'space-between' }}>
<View style={{ flex: 1, backgroundColor: '#FFFFFF' }}>
<KeyboardAvoidingView behavior="position">
<BlueBitcoinAmount
isLoading={this.state.isLoading}
amount={this.state.amount}
onChangeText={text => {
if (this.state.lnurlParams) {
// in this case we prevent the user from changing the amount to < min or > max
2019-08-04 21:34:17 +02:00
let { min, max } = this.state.lnurlParams;
let nextAmount = parseInt(text);
if (nextAmount < min) {
2019-08-04 21:34:17 +02:00
text = min.toString();
} else if (nextAmount > max) {
2019-08-04 21:34:17 +02:00
text = max.toString();
}
}
this.setState({ amount: text });
}}
disabled={this.state.isLoading || (this.state.lnurlParams && this.state.lnurlParams.fixed)}
unit={BitcoinUnit.SATS}
inputAccessoryViewID={BlueDismissKeyboardInputAccessory.InputAccessoryViewID}
/>
2018-12-25 17:34:51 +01:00
<View
style={{
flexDirection: 'row',
borderColor: '#d2d2d2',
borderBottomColor: '#d2d2d2',
borderWidth: 1.0,
borderBottomWidth: 0.5,
backgroundColor: '#f5f5f5',
minHeight: 44,
height: 44,
marginHorizontal: 20,
alignItems: 'center',
marginVertical: 8,
borderRadius: 4,
}}
>
<TextInput
onChangeText={text => this.setState({ description: text })}
placeholder={loc.receive.details.label}
value={this.state.description}
2018-12-25 17:34:51 +01:00
numberOfLines={1}
style={{ flex: 1, marginHorizontal: 8, minHeight: 33 }}
editable={!this.state.isLoading}
onSubmitEditing={Keyboard.dismiss}
inputAccessoryViewID={BlueDismissKeyboardInputAccessory.InputAccessoryViewID}
2018-12-25 17:34:51 +01:00
/>
</View>
<BlueDismissKeyboardInputAccessory />
2018-12-25 17:34:51 +01:00
{this.renderCreateButton()}
</KeyboardAvoidingView>
</View>
{this.state.lnurlParams ? null : this.renderScanClickable()}
2018-12-25 17:34:51 +01:00
</View>
</TouchableWithoutFeedback>
);
}
}
LNDCreateInvoice.propTypes = {
navigation: PropTypes.shape({
goBack: PropTypes.func,
2018-12-25 17:34:51 +01:00
navigate: PropTypes.func,
getParam: PropTypes.func,
state: PropTypes.shape({
params: PropTypes.shape({
uri: PropTypes.string,
fromWallet: PropTypes.string,
}),
}),
2018-12-25 17:34:51 +01:00
}),
};