Commit b1a61a55 authored by Gijs Hendriksen's avatar Gijs Hendriksen
Browse files

Retrieve display name and avatar from website

parent 7201d033
......@@ -127,6 +127,7 @@ android {
}
dependencies {
compile project(':react-native-vector-icons')
compile fileTree(dir: "libs", include: ["*.jar"])
compile "com.android.support:appcompat-v7:23.0.1"
compile "com.facebook.react:react-native:+" // From node_modules
......
......@@ -3,13 +3,19 @@ import * as types from './actionTypes';
const USERNAMEKEY = '@MyStore:username';
const TOKENKEY = '@MyStore:token';
const DISPLAYNAMEKEY = '@MyStore:displayName';
const PHOTOKEY = '@MyStore:photo';
export function loginSuccess(username, token) {
const defaultAvatar = 'http://localhost:8000/static/members/images/default-avatar.jpg';
export function loginSuccess(username, token, displayName, photo) {
return {
type: types.LOGINSUCCESS,
loginState: 'loggedIn',
username,
token,
displayName,
photo,
};
}
......@@ -27,7 +33,41 @@ export function loginFailure() {
};
}
export function login(username, password) {
async function getUserInfo(token) {
const data = {
method: 'GET',
headers: {
Accept: 'application/json',
'Content-Type': 'application/json',
Authorization: `Token ${token}`,
},
};
fetch('http://localhost:8000/api/members/info/', data)
.then(
(response) => {
console.log(response);
return response.json();
})
.then(
(responseJson) => {
const avatar = responseJson.photo === null ? defaultAvatar : responseJson.photo;
return {
photo: avatar,
displayName: responseJson.display_name,
};
},
)
.catch(
() => ({
photo: defaultAvatar,
displayName: 'Naamloos',
}),
);
}
export function login(user, pass) {
return (dispatch) => {
dispatch(loginProgress());
const data = {
......@@ -37,8 +77,8 @@ export function login(username, password) {
'Content-Type': 'application/json',
},
body: JSON.stringify({
username,
password,
user,
pass,
}),
};
return fetch('http://localhost:8000/api/token-auth/', data)
......@@ -47,9 +87,22 @@ export function login(username, password) {
.then(
(responseJson) => {
if (responseJson.token) {
return AsyncStorage.multiSet([[USERNAMEKEY, username], [TOKENKEY, responseJson.token]])
const token = responseJson.token;
const username = responseJson.username;
getUserInfo(token)
.then(
() => dispatch(loginSuccess(username, responseJson.token)),
(userInfo) => {
console.log(userInfo);
AsyncStorage.multiSet([
[USERNAMEKEY, username],
[TOKENKEY, token],
[DISPLAYNAMEKEY, userInfo.displayName],
[PHOTOKEY, userInfo.photo],
]);
dispatch(loginSuccess(username, token, userInfo.displayName, userInfo.photo));
},
);
}
return dispatch(loginFailure());
......
......@@ -14,6 +14,9 @@ const store = createStoreWithMiddleware(reducer);
const USERNAMEKEY = '@MyStore:username';
const TOKENKEY = '@MyStore:token';
const DISPLAYNAMEKEY = '@MyStore:displayName';
const PHOTOKEY = '@MyStore:photo';
const pairsToObject = (obj, pair) => {
const obj2 = { ...obj };
obj2[pair[0]] = pair[1];
......@@ -22,15 +25,17 @@ const pairsToObject = (obj, pair) => {
class Main extends Component {
componentDidMount() {
AsyncStorage.multiGet([USERNAMEKEY, TOKENKEY])
AsyncStorage.multiGet([USERNAMEKEY, TOKENKEY, DISPLAYNAMEKEY, PHOTOKEY])
.then(
(result) => {
const values = result.reduce(pairsToObject, {});
const username = values[USERNAMEKEY];
const token = values[TOKENKEY];
const displayName = values[DISPLAYNAMEKEY];
const photo = values[DISPLAYNAMEKEY];
if (username !== null && token !== null) {
store.dispatch(loginSuccess(username, token));
store.dispatch(loginSuccess(username, token, displayName, photo));
}
});
}
......
......@@ -26,7 +26,7 @@ const Calendar = (props) => {
<ListView
dataSource={dataSource}
renderRow={rowData => rowData}
enableEmptySections={true}
enableEmptySections
/>
</View>
);
......
import React, { Component } from 'react';
import { Text, View, Image, Dimensions } from 'react-native';
import React from 'react';
import { Text, View, Image, TouchableHighlight } from 'react-native';
import { connect } from 'react-redux';
import Icon from 'react-native-vector-icons/MaterialIcons';
import styles, { colors } from '../style';
import { colors } from '../style';
import styles from './style/sidebar';
import * as actions from '../actions/navigation';
import * as loginActions from '../actions/login';
const Sidebar = props => {
const b64Image = '';
const background = require('../img/huygens.jpg');
const Sidebar = (props) => {
const buttons = [
{
onPress: () => props.navigate('welcome'),
......@@ -21,80 +21,83 @@ const Sidebar = props => {
},
{
onPress: () => props.navigate('eventList'),
iconName: 'calendar',
iconName: 'event',
text: 'Agenda',
style: {},
scene: 'eventList',
},
{
onPress: props.logout,
iconName: 'lock open',
iconName: 'lock',
text: 'Uitloggen',
style: {
borderTopColor: colors.lightGray,
borderTopWidth: 1,
},
scene: '',
scene: 'logout',
},
];
const width = Dimensions.get('window').width * 0.8;
const height = width / 640 * 426;
return (
<View
style={styles.sidebar}
>
<Image
source={require('../img/huygens.jpg')}
style={{width: width, height: height, justifyContent: 'space-around', paddingLeft: 10}}
resizeMode='contain'
<TouchableHighlight
onPress={() => props.navigate('profile')}
style={styles.headerButton}
>
<Image
source={{uri: b64Image}}
style={{width: 100, height: 100, borderRadius: 50, marginBottom: 10}}
resizeMode='cover'
/>
<Text style={{color: colors.white, fontSize: 24}}>Gijs Hendriksen</Text>
</Image>
{buttons.map((object, i) => (
<Icon.Button
onPress={object.onPress}
name={object.iconName}
borderRadius={0}
backgroundColor={colors.white}
color={props.currentScene === object.scene ?
colors.magenta : colors.textColour}
size={28}
iconStyle={{
marginRight: 30,
width: 28,
textAlign: 'center',
}}
textStyle={{fontSize: 28}}
style={[{padding: 20}, object.style]}
key={i}
source={background}
style={styles.headerImage}
resizeMode="cover"
>
{object.text}
</Icon.Button>
))}
<Image
source={{ uri: props.photo }}
style={styles.profileImage}
resizeMode="cover"
/>
<Text style={styles.nameField}>{props.displayName}</Text>
</Image>
</TouchableHighlight>
<View style={styles.buttonList}>
{buttons.map(button => (
<Icon.Button
onPress={button.onPress}
name={button.iconName}
borderRadius={0}
backgroundColor={colors.white}
color={props.currentScene === button.scene ?
colors.magenta : colors.textColour}
size={28}
iconStyle={styles.buttonIcon}
style={[styles.buttonText, button.style]}
key={button.scene}
>
{button.text}
</Icon.Button>
))}
</View>
</View>
)
);
};
Sidebar.propTypes = {
currentScene: React.PropTypes.string.isRequired,
displayName: React.PropTypes.string.isRequired,
photo: React.PropTypes.string.isRequired,
navigate: React.PropTypes.func.isRequired,
logout: React.PropTypes.func.isRequired,
};
const mapStateToProps = state => ({
currentScene: state.navigation.currentScene,
displayName: state.session.displayName,
photo: state.session.photo,
});
const mapDispatchToProps = dispatch => ({
navigate: scene => dispatch(actions.navigate(scene)),
logout: () => dispatch(loginActions.logout())
logout: () => dispatch(loginActions.logout()),
});
export default connect(mapStateToProps, mapDispatchToProps)(Sidebar);
import React from 'react';
import { Platform, StyleSheet, Text, View, StatusBar, TouchableOpacity } from 'react-native';
import { Text, View, StatusBar, TouchableOpacity } from 'react-native';
import { connect } from 'react-redux';
import Drawer from 'react-native-drawer';
import Icon from 'react-native-vector-icons/FontAwesome';
import Icon from 'react-native-vector-icons/MaterialIcons';
import Login from './Login';
import Welcome from './Welcome';
import Sidebar from './Sidebar';
......@@ -10,11 +10,9 @@ import Event from './Event';
import Calendar from './Calendar';
import * as actions from '../actions/navigation';
import styles from './style/navigator';
import { colors } from '../style';
const STATUSBAR_HEIGHT = Platform.OS === 'ios' ? 20 : 0;
const APPBAR_HEIGHT = Platform.OS === 'ios' ? 44 : 56;
const sceneToComponent = (scene) => {
switch (scene) {
case 'welcome':
......@@ -41,33 +39,6 @@ const sceneToTitle = (scene) => {
}
};
const styles = StyleSheet.create({
container: {
flex: 1,
},
statusBar: {
height: STATUSBAR_HEIGHT,
backgroundColor: colors.darkMagenta,
},
appBar: {
backgroundColor: colors.magenta,
height: APPBAR_HEIGHT,
justifyContent: 'flex-start',
alignItems: 'center',
flexWrap: 'wrap',
flexDirection: 'row',
},
title: {
color: colors.white,
fontSize: 24,
},
icon: {
paddingLeft: 20,
paddingRight: 50,
color: colors.white,
},
});
const ReduxNavigator = (props) => {
const { currentScene, loggedIn, drawerOpen, updateDrawer } = props;
if (loggedIn) {
......@@ -82,7 +53,7 @@ const ReduxNavigator = (props) => {
mainOverlay: {
backgroundColor: colors.black,
opacity: 0,
}
},
}}
tweenHandler={ratio => ({ mainOverlay: { opacity: ratio * 0.75 } })}
open={drawerOpen}
......@@ -98,7 +69,7 @@ const ReduxNavigator = (props) => {
onPress={props.isFirstScene ? () => props.updateDrawer(!props.drawerOpen) : props.back}
>
<Icon
name={props.isFirstScene ? 'bars' : 'arrow-left'}
name={props.isFirstScene ? 'menu' : 'arrow-back'}
onClick={props.back}
style={styles.icon}
size={20}
......@@ -109,7 +80,13 @@ const ReduxNavigator = (props) => {
{sceneToComponent(currentScene)}
</Drawer>);
}
return <Login />;
return (
<View>
<View style={styles.statusBar}>
<StatusBar backgroundColor={colors.darkMagenta} barStyle='light-content' />
</View>
<Login />
</View>);
};
ReduxNavigator.propTypes = {
......@@ -125,7 +102,7 @@ const mapStateToProps = state => ({
currentScene: state.navigation.currentScene,
loggedIn: state.navigation.loggedIn,
drawerOpen: state.navigation.drawerOpen,
isFirstScene: state.navigation.previousScenes.length == 0,
isFirstScene: state.navigation.previousScenes.length === 0,
});
const mapDispatchToProps = dispatch => ({
......
import { Platform, StyleSheet } from 'react-native';
import { colors } from '../../style';
const STATUSBAR_HEIGHT = Platform.OS === 'ios' ? 20 : 0;
const APPBAR_HEIGHT = Platform.OS === 'ios' ? 44 : 56;
const styles = StyleSheet.create({
statusBar: {
height: STATUSBAR_HEIGHT,
backgroundColor: colors.darkMagenta,
},
appBar: {
backgroundColor: colors.magenta,
height: APPBAR_HEIGHT,
justifyContent: 'flex-start',
alignItems: 'center',
flexWrap: 'wrap',
flexDirection: 'row',
},
title: {
color: colors.white,
fontSize: 24,
},
icon: {
paddingLeft: 20,
paddingRight: 50,
color: colors.white,
},
});
export default styles;
import { StyleSheet } from 'react-native';
import { colors } from '../../style';
const styles = StyleSheet.create({
sidebar: {
backgroundColor: colors.white,
flex: 1,
alignItems: 'stretch',
},
headerButton: {
flex: 1,
},
headerImage: {
flex: 1,
width: null,
height: null,
justifyContent: 'space-around',
paddingLeft: 10,
},
profileImage: {
width: 100,
height: 100,
borderRadius: 50,
},
nameField: {
color: colors.white,
fontSize: 24,
},
buttonList: {
flex: 3,
},
buttonIcon: {
marginRight: 30,
width: 28,
textAlign: 'center',
},
buttonText: {
padding: 20,
},
});
export default styles;
......@@ -4,12 +4,21 @@ const initialState = {
loginState: '',
token: '',
username: '',
displayName: '',
photo: '',
};
export default function session(state = initialState, action = {}) {
switch (action.type) {
case types.LOGINSUCCESS:
return { ...state, loginState: 'success', username: action.username, token: action.token };
return {
...state,
loginState: 'success',
username: action.username,
token: action.token,
displayName: action.displayName,
photo: action.photo,
};
case types.LOGINFAILURE:
return { ...state, loginState: 'failure' };
case types.LOGINPROGRESS:
......
import { StyleSheet } from 'react-native';
export const colors = {
magenta: '#E62272',
darkMagenta: '#C2185B',
......@@ -8,11 +6,3 @@ export const colors = {
lightGray: '#BBBBBB',
textColour: '#313131',
};
const styles = StyleSheet.create({
sidebar: {
backgroundColor: colors.white,
},
});
export default styles;
Markdown is supported
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment