2016-11-02 20:42:55 +00:00
|
|
|
import React, { Component } from 'react';
|
|
|
|
import {
|
|
|
|
KeyboardAvoidingView,
|
|
|
|
StyleSheet,
|
|
|
|
Text,
|
|
|
|
TextInput,
|
|
|
|
TouchableHighlight,
|
|
|
|
View,
|
|
|
|
} from 'react-native';
|
|
|
|
|
2017-01-10 21:27:47 +00:00
|
|
|
import * as Keychain from 'react-native-keychain';
|
2016-11-02 20:42:55 +00:00
|
|
|
|
|
|
|
export default class KeychainExample extends Component {
|
|
|
|
state = {
|
|
|
|
username: '',
|
|
|
|
password: '',
|
|
|
|
status: '',
|
|
|
|
};
|
|
|
|
|
|
|
|
save() {
|
|
|
|
Keychain
|
|
|
|
.setGenericPassword(this.state.username, this.state.password)
|
|
|
|
.then(() => {
|
2017-02-10 09:59:39 +00:00
|
|
|
this.setState({ status: 'Credentials saved!' });
|
2016-11-02 20:42:55 +00:00
|
|
|
})
|
|
|
|
.catch((err) => {
|
2017-02-10 09:59:39 +00:00
|
|
|
this.setState({ status: 'Could not save credentials, ' + err });
|
2016-11-02 20:42:55 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
load() {
|
|
|
|
Keychain
|
|
|
|
.getGenericPassword()
|
|
|
|
.then((credentials) => {
|
2017-02-10 09:59:39 +00:00
|
|
|
if (credentials) {
|
|
|
|
this.setState({ ...credentials, status: 'Credentials loaded!' });
|
|
|
|
} else {
|
|
|
|
this.setState({ status: 'No credentials stored.' });
|
|
|
|
}
|
2016-11-02 20:42:55 +00:00
|
|
|
})
|
|
|
|
.catch((err) => {
|
2017-02-10 09:59:39 +00:00
|
|
|
this.setState({ status: 'Could not load credentials. ' + err });
|
2016-11-02 20:42:55 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
reset() {
|
|
|
|
Keychain
|
|
|
|
.resetGenericPassword()
|
|
|
|
.then(() => {
|
2017-02-10 09:59:39 +00:00
|
|
|
this.setState({ status: 'Credentials Reset!', username: '', password: '' });
|
2016-11-02 20:42:55 +00:00
|
|
|
})
|
|
|
|
.catch((err) => {
|
2017-02-10 09:59:39 +00:00
|
|
|
this.setState({ status: 'Could not reset credentials, ' + err });
|
2016-11-02 20:42:55 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
return (
|
|
|
|
<KeyboardAvoidingView behavior="padding" style={styles.container}>
|
|
|
|
<View style={styles.content}>
|
|
|
|
<Text style={styles.title}>
|
|
|
|
Keychain Example
|
|
|
|
</Text>
|
|
|
|
<View style={styles.field}>
|
|
|
|
<Text style={styles.label}>Username</Text>
|
|
|
|
<TextInput
|
|
|
|
style={styles.input}
|
|
|
|
autoFocus={true}
|
|
|
|
autoCapitalize="none"
|
|
|
|
value={this.state.username}
|
|
|
|
onChange={(event) => this.setState({ username: event.nativeEvent.text })}
|
|
|
|
/>
|
|
|
|
</View>
|
|
|
|
<View style={styles.field}>
|
|
|
|
<Text style={styles.label}>Password</Text>
|
|
|
|
<TextInput
|
|
|
|
style={styles.input}
|
|
|
|
password={true}
|
|
|
|
autoCapitalize="none"
|
|
|
|
value={this.state.password}
|
|
|
|
onChange={(event) => this.setState({ password: event.nativeEvent.text })}
|
|
|
|
/>
|
|
|
|
</View>
|
|
|
|
{!!this.state.status && (<Text style={styles.status}>{this.state.status}</Text>)}
|
|
|
|
<View style={styles.buttons}>
|
|
|
|
<TouchableHighlight onPress={() => this.save()} style={styles.button}>
|
|
|
|
<View style={styles.save}>
|
|
|
|
<Text style={styles.buttonText}>Save</Text>
|
|
|
|
</View>
|
|
|
|
</TouchableHighlight>
|
|
|
|
<TouchableHighlight onPress={() => this.load()} style={styles.button}>
|
|
|
|
<View style={styles.load}>
|
|
|
|
<Text style={styles.buttonText}>Load</Text>
|
|
|
|
</View>
|
|
|
|
</TouchableHighlight>
|
|
|
|
<TouchableHighlight onPress={() => this.reset()} style={styles.button}>
|
|
|
|
<View style={styles.reset}>
|
|
|
|
<Text style={styles.buttonText}>Reset</Text>
|
|
|
|
</View>
|
|
|
|
</TouchableHighlight>
|
|
|
|
</View>
|
|
|
|
</View>
|
|
|
|
</KeyboardAvoidingView>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const styles = StyleSheet.create({
|
|
|
|
container: {
|
|
|
|
flex: 1,
|
|
|
|
alignItems: 'center',
|
|
|
|
justifyContent: 'center',
|
|
|
|
backgroundColor: '#F5FCFF'
|
|
|
|
},
|
|
|
|
content: {
|
|
|
|
width: 250,
|
|
|
|
},
|
|
|
|
title: {
|
|
|
|
fontSize: 28,
|
|
|
|
fontWeight: '200',
|
|
|
|
textAlign: 'center',
|
|
|
|
marginBottom: 20,
|
|
|
|
},
|
|
|
|
field: {
|
|
|
|
marginVertical: 5,
|
|
|
|
},
|
|
|
|
label: {
|
|
|
|
fontWeight: '500',
|
|
|
|
fontSize: 15,
|
|
|
|
marginBottom: 5,
|
|
|
|
},
|
|
|
|
input: {
|
|
|
|
borderWidth: StyleSheet.hairlineWidth,
|
|
|
|
borderColor: '#ccc',
|
|
|
|
backgroundColor: 'white',
|
|
|
|
height: 32,
|
|
|
|
fontSize: 14,
|
|
|
|
padding: 8,
|
|
|
|
},
|
|
|
|
status: {
|
|
|
|
color: '#333',
|
|
|
|
fontSize: 12,
|
|
|
|
marginTop: 15,
|
|
|
|
},
|
|
|
|
buttons: {
|
|
|
|
flexDirection: 'row',
|
|
|
|
justifyContent: 'space-between',
|
|
|
|
marginTop: 20,
|
|
|
|
},
|
|
|
|
button: {
|
|
|
|
borderRadius: 3,
|
|
|
|
overflow: 'hidden',
|
|
|
|
},
|
|
|
|
save: {
|
|
|
|
backgroundColor: '#0c0',
|
|
|
|
},
|
|
|
|
load: {
|
|
|
|
backgroundColor: '#333',
|
|
|
|
},
|
|
|
|
reset: {
|
|
|
|
backgroundColor: '#c00',
|
|
|
|
},
|
|
|
|
buttonText: {
|
|
|
|
color: 'white',
|
|
|
|
fontSize: 14,
|
|
|
|
paddingHorizontal: 16,
|
|
|
|
paddingVertical: 8,
|
|
|
|
}
|
|
|
|
});
|