- Stuff
This commit is contained in:
17
app/components/Profile/EditProfile.container.js
Normal file
17
app/components/Profile/EditProfile.container.js
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
import { connect } from 'react-redux';
|
||||||
|
|
||||||
|
import { commonProfileStateToProps } from './Profile.container.js';
|
||||||
|
import EditProfile from './EditProfile.js';
|
||||||
|
|
||||||
|
const matchStateToProps = (dispatch) => {
|
||||||
|
const commonProps = commonProfileStateToProps(state);
|
||||||
|
const profile = getProfile(state);
|
||||||
|
|
||||||
|
return {
|
||||||
|
...commonProps,
|
||||||
|
firstName: profile.get('firstName'),
|
||||||
|
lastName: profile.get('lastName'),
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export default connect(matchStateToProps, null)(EditProfile);
|
||||||
178
app/components/Profile/EditProfile.js
Normal file
178
app/components/Profile/EditProfile.js
Normal file
@@ -0,0 +1,178 @@
|
|||||||
|
import React, { Component } from 'react';
|
||||||
|
import { Text, TextInput, View } from 'react-native';
|
||||||
|
import { Avatar } from 'react-native-elements';
|
||||||
|
|
||||||
|
import { getEmailAvailability, getNomAvailability } from '../api/profile.js';
|
||||||
|
|
||||||
|
import styles from './Profile.styles.js';
|
||||||
|
|
||||||
|
const STRINGS = {
|
||||||
|
CANCEL: 'Cancel',
|
||||||
|
NOM_EXPLANATION: 'Selecting a nom de bid allows you to bid anonymously - or not. By default, we\'ll use your first initial and last name.',
|
||||||
|
SAVE_PROFILE: 'Save profile',
|
||||||
|
};
|
||||||
|
|
||||||
|
export default class EditProfile extends Component {
|
||||||
|
|
||||||
|
static get propTypes() {
|
||||||
|
return {
|
||||||
|
addresses: PropTypes.array,
|
||||||
|
avatar: PropTypes.string,
|
||||||
|
cancelEditAction: PropTypes.func.isRequired,
|
||||||
|
email: PropTypes.string,
|
||||||
|
firstName: PropTypes.string,
|
||||||
|
initials: PropTypes.string,
|
||||||
|
lastName: PropTypes.string,
|
||||||
|
nomDeBid: PropTypes.string,
|
||||||
|
phones: PropTypes.array,
|
||||||
|
saveProfileAction: PropTypes.func.isRequired,
|
||||||
|
saveProfileLabel: PropTypes.string,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
static get defaultProps() {
|
||||||
|
return {
|
||||||
|
addresses: null,
|
||||||
|
avatar: null,
|
||||||
|
email: null,
|
||||||
|
firstName: null,
|
||||||
|
initials: null,
|
||||||
|
lastName: null,
|
||||||
|
nomDeBid: null,
|
||||||
|
phones: null,
|
||||||
|
saveProfileLabel: STRINGS.SAVE_PROFILE,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
super(props);
|
||||||
|
|
||||||
|
this.state = {
|
||||||
|
addresses: this.props.addresses,
|
||||||
|
avatar: this.props.avatar,
|
||||||
|
email: this.props.email,
|
||||||
|
firstName: this.props.firstName,
|
||||||
|
lastName: this.props.lastName,
|
||||||
|
invalidEmail: false,
|
||||||
|
invalidNomDeBid: false,
|
||||||
|
nomDeBid: this.props.nomDeBid,
|
||||||
|
password: this.props.password,
|
||||||
|
phones: this.props.phones,
|
||||||
|
};
|
||||||
|
|
||||||
|
this.handleCancel = this.handleCancel.bind(this);
|
||||||
|
this.handleSubmit = this.handleSubmit.bind(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
_validateEmail() {
|
||||||
|
getEmailAvailability(this.state.email, (result) => this.setState('invalidEmail', !result.available));
|
||||||
|
}
|
||||||
|
|
||||||
|
_validateNomDeBid() {
|
||||||
|
getNomAvailability(this.state.nomDeBid, (result) => this.setState('invalidNomDeBid', !result.available));
|
||||||
|
}
|
||||||
|
|
||||||
|
getProfileFromState() {
|
||||||
|
return {
|
||||||
|
addresses: this.state.addresses,
|
||||||
|
avatar: this.state.avatar,
|
||||||
|
email: this.state.email,
|
||||||
|
firstName: this.state.firstName,
|
||||||
|
lastName: this.state.lastName,
|
||||||
|
nomDeBid: this.state.nomDeBid,
|
||||||
|
password: this.state.password,
|
||||||
|
phones: this.state.phones,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
handleCancel() {
|
||||||
|
this.props.cancelEditAction();
|
||||||
|
}
|
||||||
|
|
||||||
|
handleSubmit() {
|
||||||
|
if (!this.isFormComplete()) {
|
||||||
|
console.error('Incomplete form... how did the button become enabled?');
|
||||||
|
alert('Please complete all of the required fields. They have bold labels.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.props.saveProfileAction(this.getProfileFromState());
|
||||||
|
}
|
||||||
|
|
||||||
|
isFormComplete() {
|
||||||
|
return !this.state.invalidEmail && !this.state.invalidNomDeBid &&
|
||||||
|
!!this.state.firstName && !!this.state.lastName && !!this.state.email &&
|
||||||
|
!!this.state.nomDeBid && !!this.state.phones.length && !!this.state.password;
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { avatar, firstName, lastName } = this.state;
|
||||||
|
const avatarTitle = !avatar && firstName && lastName
|
||||||
|
? `${firstName.substring(0,1)}${lastName.substring(0,1)}`
|
||||||
|
: null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<View style={styles.profileFormWrap}>
|
||||||
|
<View style={styles.avatarWrap}>
|
||||||
|
{avatar !== null ? (
|
||||||
|
<Avatar source={{ uri: this.state.avatar }} showEditButton />
|
||||||
|
) : (
|
||||||
|
<Avatar title={this.props.initials} showEditButton />
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
<View style={styles.nameWrap}>
|
||||||
|
<TextInput
|
||||||
|
onChange={(text) => this.setState({ firstName: text })}
|
||||||
|
placeholder="first name"
|
||||||
|
style={[styles.textInput, styles.requiredInput]}
|
||||||
|
value={this.state.firstName}
|
||||||
|
/>
|
||||||
|
<TextInput
|
||||||
|
onChange={(text) => this.setState({ lastName: text })}
|
||||||
|
placeholder="last name"
|
||||||
|
style={[styles.textInput, styles.requiredInput]}
|
||||||
|
value={this.state.lastName}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
<View style={[styles.emailWrap, styles.requiredWrap]}>
|
||||||
|
<TextInput
|
||||||
|
keyboardType="email-address"
|
||||||
|
onChangeText={(text) => this.setState({ email: text })}
|
||||||
|
onEndEditing={(text) => this._validateEmail(text)}
|
||||||
|
placeholder="email address"
|
||||||
|
style={[styles.textInput, styles.requiredInput]}
|
||||||
|
value={this.state.email}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
<View style={[styles.nomWrap, styles.requiredWrap]}>
|
||||||
|
<Text style={styles.hintText}>{STRINGS.NOM_EXPLANATION}</Text>
|
||||||
|
<TextInput
|
||||||
|
onChangeText={(text) => this.setState({ nomDeBid: text })}
|
||||||
|
onEndEditing={(text) => this._validateEmail(text)}
|
||||||
|
placeholder="nom de bid"
|
||||||
|
style={[styles.textInput, styles.requiredInput]}
|
||||||
|
value={this.state.nomDeBid}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
<View style={styles.phonesWrap}>
|
||||||
|
<Text style={[styles.groupLabel, styles.requiredLabel]}>Numbers</Text>
|
||||||
|
{phones.length > 0 && (
|
||||||
|
// LIST PHONES
|
||||||
|
)}
|
||||||
|
<Button title="Add number" onPress={() => false}/>
|
||||||
|
</View>
|
||||||
|
<View style={styles.addressesWrap}>
|
||||||
|
<Text style={styles.groupLabel}>Addresses</Text>
|
||||||
|
{addresses.length > 0 && (
|
||||||
|
// LIST ADDRESSES
|
||||||
|
)}
|
||||||
|
<Button title="Add address" onPress={() => false}/>
|
||||||
|
</View>
|
||||||
|
<View style={styles.register}>
|
||||||
|
<Button title={this.props.saveProfileLabel} onPress={this.handleSubmit} />
|
||||||
|
<Button title={STRINGS.CANCEL} onPress={this.handleCancel} />
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
20
app/components/Profile/Profile.container.js
Normal file
20
app/components/Profile/Profile.container.js
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
import { connect } from 'react-redux';
|
||||||
|
|
||||||
|
import { getProfile } from '../selectors/profile.js';
|
||||||
|
|
||||||
|
import Profile from './Profile.js';
|
||||||
|
|
||||||
|
export const commonProfileStateToProps = (state) => {
|
||||||
|
const profile = getProfile(state);
|
||||||
|
|
||||||
|
return {
|
||||||
|
addresses: profile.get('addresses').toArray(),
|
||||||
|
avatar: profile.get('avatar'),
|
||||||
|
email: profile.get('email'),
|
||||||
|
initials: profile.get('initials'),
|
||||||
|
nomDeBid: profile.get('nomDeBid'),
|
||||||
|
phones: profile.get('phones'),
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export default connect(commonProfileStateToProps, null)(Profile);
|
||||||
50
app/components/Profile/Profile.js
Normal file
50
app/components/Profile/Profile.js
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
import React, { useState } from 'react';
|
||||||
|
import { View } from 'react-native';
|
||||||
|
|
||||||
|
import EditProfile from './EditProfile.container.js';
|
||||||
|
import ViewProfile from './ViewProfile.container.js';
|
||||||
|
|
||||||
|
export default function Profile({
|
||||||
|
cancelEditAction,
|
||||||
|
isInEditMode,
|
||||||
|
saveProfileAction,
|
||||||
|
saveProfileLabel,
|
||||||
|
}) {
|
||||||
|
const [editMode, setEditMode] = useState(isInEditMode);
|
||||||
|
|
||||||
|
const _cancelEditAction = () => {
|
||||||
|
setEditMode(false);
|
||||||
|
cancelEditAction();
|
||||||
|
};
|
||||||
|
|
||||||
|
const _saveProfileAction = (profile) => {
|
||||||
|
setEditMode(false);
|
||||||
|
saveProfileAction(profile);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<View>
|
||||||
|
{editMode ? (
|
||||||
|
<EditProfile
|
||||||
|
cancelEditAction={_cancelEditAction}
|
||||||
|
saveProfileAction={_saveProfileAction}
|
||||||
|
saveProfileLabel={saveProfileLabel}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<ViewProfile editProfileAction={() => setEditMode(true)} />
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Profile.propTypes = {
|
||||||
|
cancelEditAction: PropTypes.func.isRequired,
|
||||||
|
isInEditMode: PropTypes.bool,
|
||||||
|
saveProfileAction: PropTypes.func.isRequired,
|
||||||
|
saveProfileLabel: PropTypes.string,
|
||||||
|
};
|
||||||
|
|
||||||
|
Profile.defaultProps = {
|
||||||
|
isInEditMode: false,
|
||||||
|
saveProfileLabel: null,
|
||||||
|
};
|
||||||
0
app/components/Profile/Profile.styles.js
Normal file
0
app/components/Profile/Profile.styles.js
Normal file
17
app/components/Profile/ViewProfile.container.js
Normal file
17
app/components/Profile/ViewProfile.container.js
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
import { connect } from 'react-redux';
|
||||||
|
|
||||||
|
import { commonProfileStateToProps } from './Profile.container.js';
|
||||||
|
import ViewProfile from './ViewProfile.js';
|
||||||
|
|
||||||
|
const matchStateToProps = (dispatch) => {
|
||||||
|
const commonProps = commonProfileStateToProps(state);
|
||||||
|
const profile = getProfile(state);
|
||||||
|
|
||||||
|
return {
|
||||||
|
...commonProps,
|
||||||
|
fullName: profile.get('fullName'),
|
||||||
|
generatedNomDeBid: propfile.get('generatedNomDeBid'),
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export default connect(matchStateToProps, null)(ViewProfile);
|
||||||
102
app/components/Profile/ViewProfile.js
Normal file
102
app/components/Profile/ViewProfile.js
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
import React, { useState } from 'react';
|
||||||
|
import { Text, TextInput, View } from 'react-native';
|
||||||
|
import { Avatar } from 'react-native-elements';
|
||||||
|
|
||||||
|
import { getEmailAvailability, getNomAvailability } from '../api/profile.js';
|
||||||
|
|
||||||
|
import styles from './Profile.styles.js';
|
||||||
|
|
||||||
|
const STRINGS = {
|
||||||
|
EDIT: 'Edit profile',
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function ViewProfile({
|
||||||
|
addresses,
|
||||||
|
avatar,
|
||||||
|
editProfileAction,
|
||||||
|
email,
|
||||||
|
fullName,
|
||||||
|
generatedNomDeBid,
|
||||||
|
initials,
|
||||||
|
nomDeBid,
|
||||||
|
phones,
|
||||||
|
}) {
|
||||||
|
const addressesCount = addresses.length;
|
||||||
|
const phonesCount = phones.length;
|
||||||
|
|
||||||
|
const [isEditingNom, setEditNom] = useState(false);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<View style={styles.profileFormWrap}>
|
||||||
|
<View style={styles.avatarWrap}>
|
||||||
|
{avatar !== null ? (
|
||||||
|
<Avatar source={{ uri: avatar }} />
|
||||||
|
) : (
|
||||||
|
<Avatar title={initials} />
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
<View style={styles.nameWrap}>
|
||||||
|
<Text style={styles.fullName}>{fullName}</Text>
|
||||||
|
<View style={styles.nomWrap}>
|
||||||
|
{isEditingNom ? (
|
||||||
|
<EditNomDeBid
|
||||||
|
) : (
|
||||||
|
<Text style={styles.nom}>{nomDeBid}</Text>
|
||||||
|
{generatedNomDeBid && (
|
||||||
|
<Button
|
||||||
|
title="Set bidding alias"
|
||||||
|
onPress={() => setEditNom(true)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
<View style={styles.emailWrap}>
|
||||||
|
<Text style={styles.label}>email</Text>
|
||||||
|
<text style={styles.value}>{email}</Text>
|
||||||
|
</View>
|
||||||
|
<View style={styles.phonesWrap}>
|
||||||
|
<Text style={styles.label}>numbers</Text>
|
||||||
|
<Text style={styles.value}>{`${phonesCount} saved`}
|
||||||
|
</View>
|
||||||
|
<View style={styles.addressesWrap}>
|
||||||
|
<Text style={styles.label}>addresses</Text>
|
||||||
|
<Text style={styles.value}>{`${addressesCount} saved`}
|
||||||
|
</View>
|
||||||
|
{editProfileAction !== null && (
|
||||||
|
<View style={styles.register}>
|
||||||
|
<Button title={STRINGS.EDIT} onPress={editProfile} />
|
||||||
|
</View>
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
ViewProfile.propTypes = {
|
||||||
|
addresses: PropTypes.array,
|
||||||
|
avatar: PropTypes.string,
|
||||||
|
editProfileAction: PropTypes.func,
|
||||||
|
email: PropTypes.string,
|
||||||
|
fullName: PropTypes.string,
|
||||||
|
generatedNomDeBid: PropTypes.bool,
|
||||||
|
initials: PropTypes.string,
|
||||||
|
nomDeBid: PropTypes.string,
|
||||||
|
phones: PropTypes.array,
|
||||||
|
};
|
||||||
|
|
||||||
|
ViewProfile.defaultProps = {
|
||||||
|
addresses: [],
|
||||||
|
avatar: null,
|
||||||
|
editProfileAction: null,
|
||||||
|
email: null,
|
||||||
|
fullName: null,
|
||||||
|
generatedNomDeBid: false,
|
||||||
|
initials: null,
|
||||||
|
nomDeBid: null,
|
||||||
|
phones: [],
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -28,6 +28,17 @@ export default class Profile extends Record({
|
|||||||
return `${this.firstName} ${this.lastName}`;
|
return `${this.firstName} ${this.lastName}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get initials() {
|
||||||
|
const firstInitial = this.firstName ? this.firstName.substring(0,1) : null;
|
||||||
|
const lastInitial = this.firstName ? this.firstName.substring(0,1) : null;
|
||||||
|
|
||||||
|
if (!firstInitial && !lastInitial) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return `${firstInitial || ''}${lastInitial || ''}`;
|
||||||
|
}
|
||||||
|
|
||||||
get isRegisteredAccount() {
|
get isRegisteredAccount() {
|
||||||
return (
|
return (
|
||||||
this.hasLinkedApple ||
|
this.hasLinkedApple ||
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
|
|
||||||
import { fetchProfile, updateProfile } from '../actions/profile.js';
|
import { fetchProfile, updateProfile } from '../actions/profile.js';
|
||||||
import { getNomDeBid, getProfile, isAllowedToBid } from '../selectors/profile.js';
|
import { getProfile, isAllowedToBid } from '../selectors/profile.js';
|
||||||
|
|
||||||
import Profile from './Profile.js';
|
import Profile from './Profile.js';
|
||||||
|
|
||||||
@@ -13,13 +13,8 @@ const matchStateToProps = (state) => {
|
|||||||
hasLinkedFacebook: profile.get('hasLinkedFacebook'),
|
hasLinkedFacebook: profile.get('hasLinkedFacebook'),
|
||||||
hasLinkedGoogle: profile.get('hasLinkedGoogle'),
|
hasLinkedGoogle: profile.get('hasLinkedGoogle'),
|
||||||
hasLocalAccount: profile.get('hasLocalAccount'),
|
hasLocalAccount: profile.get('hasLocalAccount'),
|
||||||
hasRegisteredAcccount: profile.get('hasRegisteredAcccount'),
|
|
||||||
id: profile.get('id'),
|
|
||||||
isAllowedToBid: isAllowedToBid(state),
|
isAllowedToBid: isAllowedToBid(state),
|
||||||
isVerified: profile.get('isVerified'),
|
isVerified: profile.get('isVerified'),
|
||||||
lastName: profile.get('lastName'),
|
|
||||||
nomDeBid: getNomDeBid(state),
|
|
||||||
organizationIdentifier: profile.get('organizationIdentifier'),
|
|
||||||
paymentToken: profile.get('paymentToken'),
|
paymentToken: profile.get('paymentToken'),
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,26 +1,87 @@
|
|||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
import { StyleSheet, Text, View } from 'react-native';
|
import { Text, View } from 'react-native';
|
||||||
|
|
||||||
|
import ProfileUtility from '../components/Profile/Profile.container.js';
|
||||||
|
|
||||||
|
import styles from './Profile.styles.js';
|
||||||
|
|
||||||
export default class Profile extends Component {
|
export default class Profile extends Component {
|
||||||
|
|
||||||
|
static get propTypes() {
|
||||||
|
return {
|
||||||
|
hasLinkedApple: PropTypes.bool,
|
||||||
|
hasLinkedFacebook: PropTypes.bool,
|
||||||
|
hasLinkedGoogle: PropTypes.bool,
|
||||||
|
hasLocalAccount: PropTypes.bool,
|
||||||
|
isAllowedToBid: PropTypes.bool,
|
||||||
|
isVerified: PropTypes.bool,
|
||||||
|
paymentToken: PropTypes.string,
|
||||||
|
updateProfile: PropTypes.func.isRequired,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
static get defaultProps() {
|
||||||
|
return {
|
||||||
|
hasLinkedApple: false,
|
||||||
|
hasLinkedFacebook: false,
|
||||||
|
hasLinkedGoogle: false,
|
||||||
|
hasLocalAccount: false,
|
||||||
|
isAllowedToBid: false,
|
||||||
|
isVerified: false,
|
||||||
|
paymentToken: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
|
const {
|
||||||
|
hasLinkedApple,
|
||||||
|
hasLinkedFacebook,
|
||||||
|
hasLinkedGoogle,
|
||||||
|
hasLocalAccount,
|
||||||
|
isAllowedToBid,
|
||||||
|
isVerified,
|
||||||
|
paymentToken,
|
||||||
|
updateProfile,
|
||||||
|
} = this.props;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={styles.container}>
|
<View style={styles.container}>
|
||||||
<Text style={styles.title}>Profile</Text>
|
{!isVerified && (
|
||||||
|
<View style={styles.alertBar}>
|
||||||
|
<Text style={styles.alert}>
|
||||||
|
{`Your acount has not been verified, please check your email.`}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<ProfileUtility
|
||||||
|
cancelEditAction={() => false}
|
||||||
|
saveProfileAction={updateProfile}
|
||||||
|
saveProfileLabel="Update profile"
|
||||||
|
/>
|
||||||
|
|
||||||
|
{!isAllowedToBid ? (
|
||||||
|
/* ADD PAYMENT METHOD */
|
||||||
|
) : (
|
||||||
|
/* SHOW/EDIT PAYMENT METHOD */
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!hasLocalAccount && (
|
||||||
|
/* CREATE LOCAL ACCOUNT PASSWORD CTA */
|
||||||
|
)}
|
||||||
|
|
||||||
|
{hasLinkedApple && (
|
||||||
|
/* APPLE LINK/UNLINK */
|
||||||
|
)}
|
||||||
|
|
||||||
|
{hasLinkedFacebook && (
|
||||||
|
/* FACEBOOK LINK/UNLINK */
|
||||||
|
)}
|
||||||
|
|
||||||
|
{hasLinkedGoogle && (
|
||||||
|
/* GOOGLE LINK/UNLINK */
|
||||||
|
)}
|
||||||
</View>
|
</View>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const styles = StyleSheet.create({
|
|
||||||
container: {
|
|
||||||
flex: 1,
|
|
||||||
justifyContent: 'center',
|
|
||||||
alignItems: 'center',
|
|
||||||
backgroundColor: '#F5FCFF',
|
|
||||||
},
|
|
||||||
title: {
|
|
||||||
fontSize: 20,
|
|
||||||
textAlign: 'center',
|
|
||||||
margin: 10,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|||||||
15
app/screens/Profile.styles.js
Normal file
15
app/screens/Profile.styles.js
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
import { StyleSheet } from 'react-native';
|
||||||
|
|
||||||
|
export default StyleSheet.create({
|
||||||
|
container: {
|
||||||
|
flex: 1,
|
||||||
|
justifyContent: 'center',
|
||||||
|
alignItems: 'center',
|
||||||
|
backgroundColor: '#F5FCFF',
|
||||||
|
},
|
||||||
|
title: {
|
||||||
|
fontSize: 20,
|
||||||
|
textAlign: 'center',
|
||||||
|
margin: 10,
|
||||||
|
},
|
||||||
|
});
|
||||||
@@ -1,132 +1,32 @@
|
|||||||
import React, { Component } from 'react';
|
import React from 'react';
|
||||||
import { Text, View } from 'react-native';
|
import { Text, View } from 'react-native';
|
||||||
|
|
||||||
|
import EditProfile from '../components/Profile/EditProfile.container.js';
|
||||||
|
|
||||||
import styles from './Register.styles.js';
|
import styles from './Register.styles.js';
|
||||||
|
|
||||||
const STRINGS = {
|
export default function Register({ doRegistration, navigation }) {
|
||||||
NOM_EXPLANATION: 'Selecting a nom de bid allows you to bid anonymously - or not. By default, we\'ll use your first initial and last name.',
|
|
||||||
SUBMIT_REGISTRATION: 'Register',
|
|
||||||
};
|
|
||||||
|
|
||||||
export default class Register extends Component {
|
const _doRegistration = (profile) => {
|
||||||
static get propTypes() {
|
if (!profile) {
|
||||||
return {
|
|
||||||
checkEmail: PropTypes.func.isRequired,
|
|
||||||
checkNomDeBid: PropTypes.func.isRequired,
|
|
||||||
doRegistration: PropTypes.func.isRequired,
|
|
||||||
// invalidEmail: PropTypes.bool.isRequired,
|
|
||||||
// invalidNomDeBid: PropTypes.bool.isRequired,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
constructor() {
|
|
||||||
super(props);
|
|
||||||
|
|
||||||
this.state = {
|
|
||||||
addresses: [],
|
|
||||||
avatar: null,
|
|
||||||
email: null,
|
|
||||||
firstName: null,
|
|
||||||
lastName: null,
|
|
||||||
nomDeBid: null,
|
|
||||||
invalidEmail: false,
|
|
||||||
invalidNomDeBid: false,
|
|
||||||
password: null,
|
|
||||||
phones: [],
|
|
||||||
};
|
|
||||||
|
|
||||||
this._doRegistration = this._doRegistration.bind(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
_doRegistration() {
|
|
||||||
if (!this.isFormComplete()) {
|
|
||||||
console.error('Incomplete form... how did the button become enabled?');
|
|
||||||
alert('Please complete all of the required fields. They have bold labels.');
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.props.doRegistration(this.getUserRegistration());
|
doRegistration(profile);
|
||||||
}
|
|
||||||
|
|
||||||
_validateEmail() {
|
|
||||||
this.props.checkEmail(this.state.email, (valid) => this.setState('invalidEmail', valid));
|
|
||||||
}
|
|
||||||
|
|
||||||
_validateNomDeBid() {
|
|
||||||
this.props.checkNomDeBid(this.state.nomDeBid, (valid) =>
|
|
||||||
this.setState('invalidNomDeBid', valid),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
getUserRegistration() {
|
|
||||||
return {
|
|
||||||
addresses: this.state.addresses,
|
|
||||||
avatar: this.state.avatar,
|
|
||||||
email: this.state.email,
|
|
||||||
firstName: this.state.firstName,
|
|
||||||
lastName: this.state.lastName,
|
|
||||||
nomDeBid: this.state.nomDeBid,
|
|
||||||
password: this.state.password,
|
|
||||||
phones: this.state.phones,
|
|
||||||
};
|
};
|
||||||
}
|
|
||||||
|
|
||||||
isFormComplete() {
|
|
||||||
return (
|
|
||||||
!this.state.invalidEmail &&
|
|
||||||
!this.state.invalidNomDeBid &&
|
|
||||||
!!this.state.firstName &&
|
|
||||||
!!this.state.lastName &&
|
|
||||||
!!this.state.email &&
|
|
||||||
!!this.state.nomDeBid &&
|
|
||||||
!!this.state.phones.length &&
|
|
||||||
!!this.state.password
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
render() {
|
|
||||||
return (
|
return (
|
||||||
<View style={styles.container}>
|
<View style={styles.container}>
|
||||||
<Text style={styles.title}>Register</Text>
|
<Text style={styles.heading}>Register</Text>
|
||||||
<View style={styles.nameWrap}>
|
<EditProfile
|
||||||
<TextInput
|
cancelEditProfile={() => navigation.goBack()}
|
||||||
onChange={(text) => this.setState({ firstName: text })}
|
saveProfileAction={doRegistration}
|
||||||
placeholder="first name"
|
saveProfileLabel="Register"
|
||||||
style={styles.textInput}
|
|
||||||
value={this.state.firstName}
|
|
||||||
/>
|
/>
|
||||||
<TextInput
|
|
||||||
onChange={(text) => this.setState({ lastName: text })}
|
|
||||||
placeholder="last name"
|
|
||||||
style={styles.textInput}
|
|
||||||
value={this.state.lastName}
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
<View style={styles.emailWrap}>
|
|
||||||
<TextInput
|
|
||||||
keyboardType="email-address"
|
|
||||||
onChangeText={(text) => _updateState('username', text)}
|
|
||||||
onEndEditing={(text) => this._validateEmail(text)}
|
|
||||||
placeholder="email address"
|
|
||||||
style={styles.textInput}
|
|
||||||
value={this.state.email}
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
<View style={styles.nomWrap}>
|
|
||||||
<Text style={styles.hintText}>{STRINGS.NOM_EXPLANATION}</Text>
|
|
||||||
<TextInput
|
|
||||||
keyboardType="email-address"
|
|
||||||
onChangeText={(text) => _updateState('username', text)}
|
|
||||||
onEndEditing={(text) => this._validateEmail(text)}
|
|
||||||
placeholder="email address"
|
|
||||||
style={styles.textInput}
|
|
||||||
value={this.state.email}
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
<View style={styles.register}>
|
|
||||||
<Button title={STRINGS.SUBMIT_REGISTRATION} onPress={this._doRegistration} />
|
|
||||||
</View>
|
|
||||||
</View>
|
</View>
|
||||||
);
|
);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Register.propTypes = {
|
||||||
|
doRegistration: PropTypes.func.isRequired,
|
||||||
|
};
|
||||||
|
|||||||
@@ -1,12 +1,9 @@
|
|||||||
import { StyleSheet } from 'react-native';
|
import { StyleSheet } from 'react-native';
|
||||||
|
|
||||||
export default (styles = StyleSheet.create({
|
export default StyleSheet.create({
|
||||||
container: {
|
container: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
flexDirection: 'column',
|
flexDirection: 'column',
|
||||||
},
|
},
|
||||||
title: {},
|
heading: {},
|
||||||
localLogin: {},
|
});
|
||||||
services: {},
|
|
||||||
register: {},
|
|
||||||
}));
|
|
||||||
|
|||||||
@@ -1,24 +1,12 @@
|
|||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
import { Text, View } from 'react-native';
|
import { Text, View } from 'react-native';
|
||||||
import { Button } from 'react-native-elements';
|
|
||||||
|
|
||||||
import FacebookLogin from '../components/Login/FacebookLogin.container.js';
|
import FacebookLogin from '../components/Login/FacebookLogin.container.js';
|
||||||
import LocalLogin from '../components/Login/LocalLogin.container.js';
|
import LocalLogin from '../components/Login/LocalLogin.container.js';
|
||||||
|
|
||||||
import styles from './SignInOrRegister.styles.js';
|
import styles from './SignInOrRegister.styles.js';
|
||||||
|
|
||||||
export default class SignInOrRegister extends Component {
|
export default function SignInOrRegister({ navigation }) {
|
||||||
constructor() {
|
|
||||||
super(props);
|
|
||||||
|
|
||||||
this._doRegistration = this._doRegistration.bind(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
_doRegistration() {
|
|
||||||
this.props.navigation.navigate('Register');
|
|
||||||
}
|
|
||||||
|
|
||||||
render() {
|
|
||||||
return (
|
return (
|
||||||
<View style={styles.container}>
|
<View style={styles.container}>
|
||||||
<Text style={styles.title}>Sign In or Register</Text>
|
<Text style={styles.title}>Sign In or Register</Text>
|
||||||
@@ -29,9 +17,8 @@ export default class SignInOrRegister extends Component {
|
|||||||
<FacebookLogin />
|
<FacebookLogin />
|
||||||
</View>
|
</View>
|
||||||
<View style={styles.register}>
|
<View style={styles.register}>
|
||||||
<Button title="Signup with Email" onPress={this._doRegistration} />
|
<Button title="Signup with Email" onPress={() => navigation.navigate('Register')} />
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
);
|
);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user