Upload profile image (#29)
* add editable button profile picture * add editable button cover picture * upload profile photos (save them locally) * rollback pbxproj changes * rollback podfile checksum (for git only) * move edit photos onto edit profile modal * adjust edit icon and image cropping size * added temporary (react state) image * added IMAGES_ENABLED flag * minor lint fix * save local photos on edit profile upload (wip) * save profile photos on profile view state (wip) * remove unecessary computed * save photo in state before pushing it to viewmodel * refactor profile pictures's state * remove unnecessary isMe prop * removing old comments * tweak icon size & position * A few styling tweaks and a fix to mobx state management Co-authored-by: Paul Frazee <pfrazee@gmail.com>
This commit is contained in:
		
							parent
							
								
									4cc90b8ac9
								
							
						
					
					
						commit
						84a60592a8
					
				
					 5 changed files with 269 additions and 16 deletions
				
			
		|  | @ -43,6 +43,10 @@ export class ProfileViewModel { | ||||||
|   postsCount: number = 0 |   postsCount: number = 0 | ||||||
|   myState = new ProfileViewMyStateModel() |   myState = new ProfileViewMyStateModel() | ||||||
| 
 | 
 | ||||||
|  |   // TODO TEMP data to be implemented in the protocol
 | ||||||
|  |   userAvatar: string | null = null | ||||||
|  |   userBanner: string | null = null | ||||||
|  | 
 | ||||||
|   // added data
 |   // added data
 | ||||||
|   descriptionEntities?: Entity[] |   descriptionEntities?: Entity[] | ||||||
| 
 | 
 | ||||||
|  | @ -115,7 +119,15 @@ export class ProfileViewModel { | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   async updateProfile(fn: (existing?: Profile.Record) => Profile.Record) { |   async updateProfile( | ||||||
|  |     fn: (existing?: Profile.Record) => Profile.Record, | ||||||
|  |     userAvatar: string | null, // TODO TEMP
 | ||||||
|  |     userBanner: string | null, // TODO TEMP
 | ||||||
|  |   ) { | ||||||
|  |     // TODO TEMP add userBanner & userAvatar in the protocol when suported
 | ||||||
|  |     this.userAvatar = userAvatar | ||||||
|  |     this.userBanner = userBanner | ||||||
|  | 
 | ||||||
|     await apilib.updateProfile(this.rootStore, this.did, fn) |     await apilib.updateProfile(this.rootStore, this.did, fn) | ||||||
|     await this.refresh() |     await this.refresh() | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -13,8 +13,10 @@ import { | ||||||
|   MAX_DESCRIPTION, |   MAX_DESCRIPTION, | ||||||
| } from '../../../lib/strings' | } from '../../../lib/strings' | ||||||
| import * as Profile from '../../../third-party/api/src/client/types/app/bsky/actor/profile' | import * as Profile from '../../../third-party/api/src/client/types/app/bsky/actor/profile' | ||||||
|  | import {UserBanner} from '../util/UserBanner' | ||||||
|  | import {UserAvatar} from '../util/UserAvatar' | ||||||
| 
 | 
 | ||||||
| export const snapPoints = ['60%'] | export const snapPoints = ['80%'] | ||||||
| 
 | 
 | ||||||
| export function Component({ | export function Component({ | ||||||
|   profileView, |   profileView, | ||||||
|  | @ -31,6 +33,12 @@ export function Component({ | ||||||
|   const [description, setDescription] = useState<string>( |   const [description, setDescription] = useState<string>( | ||||||
|     profileView.description || '', |     profileView.description || '', | ||||||
|   ) |   ) | ||||||
|  |   const [userBanner, setUserBanner] = useState<string | null>( | ||||||
|  |     profileView.userBanner, | ||||||
|  |   ) | ||||||
|  |   const [userAvatar, setUserAvatar] = useState<string | null>( | ||||||
|  |     profileView.userAvatar, | ||||||
|  |   ) | ||||||
|   const onPressCancel = () => { |   const onPressCancel = () => { | ||||||
|     store.shell.closeModal() |     store.shell.closeModal() | ||||||
|   } |   } | ||||||
|  | @ -51,6 +59,8 @@ export function Component({ | ||||||
|             description, |             description, | ||||||
|           } |           } | ||||||
|         }, |         }, | ||||||
|  |         userAvatar, // TEMP
 | ||||||
|  |         userBanner, // TEMP
 | ||||||
|       ) |       ) | ||||||
|       Toast.show('Profile updated') |       Toast.show('Profile updated') | ||||||
|       onUpdate?.() |       onUpdate?.() | ||||||
|  | @ -67,12 +77,28 @@ export function Component({ | ||||||
|     <View style={s.flex1}> |     <View style={s.flex1}> | ||||||
|       <BottomSheetScrollView style={styles.inner}> |       <BottomSheetScrollView style={styles.inner}> | ||||||
|         <Text style={styles.title}>Edit my profile</Text> |         <Text style={styles.title}>Edit my profile</Text> | ||||||
|  |         <View style={styles.photos}> | ||||||
|  |           <UserBanner | ||||||
|  |             userBanner={userBanner} | ||||||
|  |             setUserBanner={setUserBanner} | ||||||
|  |             handle={profileView.handle} | ||||||
|  |           /> | ||||||
|  |           <View style={styles.avi}> | ||||||
|  |             <UserAvatar | ||||||
|  |               size={80} | ||||||
|  |               userAvatar={userAvatar} | ||||||
|  |               handle={profileView.handle} | ||||||
|  |               setUserAvatar={setUserAvatar} | ||||||
|  |               displayName={profileView.displayName} | ||||||
|  |             /> | ||||||
|  |           </View> | ||||||
|  |         </View> | ||||||
|         {error !== '' && ( |         {error !== '' && ( | ||||||
|           <View style={s.mb10}> |           <View style={s.mb10}> | ||||||
|             <ErrorMessage message={error} /> |             <ErrorMessage message={error} /> | ||||||
|           </View> |           </View> | ||||||
|         )} |         )} | ||||||
|         <View style={styles.group}> |         <View> | ||||||
|           <Text style={styles.label}>Display Name</Text> |           <Text style={styles.label}>Display Name</Text> | ||||||
|           <BottomSheetTextInput |           <BottomSheetTextInput | ||||||
|             style={styles.textInput} |             style={styles.textInput} | ||||||
|  | @ -81,7 +107,7 @@ export function Component({ | ||||||
|             onChangeText={v => setDisplayName(enforceLen(v, MAX_DISPLAY_NAME))} |             onChangeText={v => setDisplayName(enforceLen(v, MAX_DISPLAY_NAME))} | ||||||
|           /> |           /> | ||||||
|         </View> |         </View> | ||||||
|         <View style={styles.group}> |         <View style={s.pb10}> | ||||||
|           <Text style={styles.label}>Description</Text> |           <Text style={styles.label}>Description</Text> | ||||||
|           <BottomSheetTextInput |           <BottomSheetTextInput | ||||||
|             style={[styles.textArea]} |             style={[styles.textArea]} | ||||||
|  | @ -120,13 +146,11 @@ const styles = StyleSheet.create({ | ||||||
|     fontSize: 24, |     fontSize: 24, | ||||||
|     marginBottom: 18, |     marginBottom: 18, | ||||||
|   }, |   }, | ||||||
|   group: { |  | ||||||
|     marginBottom: 10, |  | ||||||
|   }, |  | ||||||
|   label: { |   label: { | ||||||
|     fontWeight: 'bold', |     fontWeight: 'bold', | ||||||
|     paddingHorizontal: 4, |     paddingHorizontal: 4, | ||||||
|     paddingBottom: 4, |     paddingBottom: 4, | ||||||
|  |     marginTop: 20, | ||||||
|   }, |   }, | ||||||
|   textInput: { |   textInput: { | ||||||
|     borderWidth: 1, |     borderWidth: 1, | ||||||
|  | @ -155,4 +179,19 @@ const styles = StyleSheet.create({ | ||||||
|     padding: 10, |     padding: 10, | ||||||
|     marginBottom: 10, |     marginBottom: 10, | ||||||
|   }, |   }, | ||||||
|  |   avi: { | ||||||
|  |     position: 'absolute', | ||||||
|  |     top: 80, | ||||||
|  |     left: 10, | ||||||
|  |     width: 84, | ||||||
|  |     height: 84, | ||||||
|  |     borderWidth: 2, | ||||||
|  |     borderRadius: 42, | ||||||
|  |     borderColor: colors.white, | ||||||
|  |     backgroundColor: colors.white, | ||||||
|  |   }, | ||||||
|  |   photos: { | ||||||
|  |     marginBottom: 36, | ||||||
|  |     marginHorizontal: -14, | ||||||
|  |   }, | ||||||
| }) | }) | ||||||
|  |  | ||||||
|  | @ -152,12 +152,13 @@ export const ProfileHeader = observer(function ProfileHeader({ | ||||||
|   } |   } | ||||||
|   return ( |   return ( | ||||||
|     <View style={styles.outer}> |     <View style={styles.outer}> | ||||||
|       <UserBanner handle={view.handle} /> |       <UserBanner handle={view.handle} userBanner={view.userBanner} /> | ||||||
|       <View style={styles.avi}> |       <View style={styles.avi}> | ||||||
|         <UserAvatar |         <UserAvatar | ||||||
|           size={80} |           size={80} | ||||||
|           displayName={view.displayName} |  | ||||||
|           handle={view.handle} |           handle={view.handle} | ||||||
|  |           displayName={view.displayName} | ||||||
|  |           userAvatar={view.userAvatar} | ||||||
|         /> |         /> | ||||||
|       </View> |       </View> | ||||||
|       <View style={styles.content}> |       <View style={styles.content}> | ||||||
|  |  | ||||||
|  | @ -1,19 +1,74 @@ | ||||||
| import React from 'react' | import React, {useCallback} from 'react' | ||||||
|  | import {StyleSheet, View, TouchableOpacity, Alert, Image} from 'react-native' | ||||||
| import Svg, {Circle, Text, Defs, LinearGradient, Stop} from 'react-native-svg' | import Svg, {Circle, Text, Defs, LinearGradient, Stop} from 'react-native-svg' | ||||||
|  | import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' | ||||||
|  | import { | ||||||
|  |   openCamera, | ||||||
|  |   openCropper, | ||||||
|  |   openPicker, | ||||||
|  | } from 'react-native-image-crop-picker' | ||||||
| import {getGradient} from '../../lib/asset-gen' | import {getGradient} from '../../lib/asset-gen' | ||||||
|  | import {colors} from '../../lib/styles' | ||||||
|  | import {IMAGES_ENABLED} from '../../../build-flags' | ||||||
| 
 | 
 | ||||||
| export function UserAvatar({ | export function UserAvatar({ | ||||||
|   size, |   size, | ||||||
|   displayName, |  | ||||||
|   handle, |   handle, | ||||||
|  |   userAvatar, | ||||||
|  |   displayName, | ||||||
|  |   setUserAvatar, | ||||||
| }: { | }: { | ||||||
|   size: number |   size: number | ||||||
|   displayName: string | undefined |  | ||||||
|   handle: string |   handle: string | ||||||
|  |   displayName: string | undefined | ||||||
|  |   userAvatar: string | null | ||||||
|  |   setUserAvatar?: React.Dispatch<React.SetStateAction<string | null>> | ||||||
| }) { | }) { | ||||||
|   const initials = getInitials(displayName || handle) |   const initials = getInitials(displayName || handle) | ||||||
|   const gradient = getGradient(handle) |   const gradient = getGradient(handle) | ||||||
|   return ( | 
 | ||||||
|  |   const handleEditAvatar = useCallback(() => { | ||||||
|  |     Alert.alert('Select upload method', '', [ | ||||||
|  |       { | ||||||
|  |         text: 'Take a new photo', | ||||||
|  |         onPress: () => { | ||||||
|  |           openCamera({ | ||||||
|  |             mediaType: 'photo', | ||||||
|  |             cropping: true, | ||||||
|  |             width: 80, | ||||||
|  |             height: 80, | ||||||
|  |             cropperCircleOverlay: true, | ||||||
|  |           }).then(item => { | ||||||
|  |             if (setUserAvatar != null) { | ||||||
|  |               setUserAvatar(item.path) | ||||||
|  |             } | ||||||
|  |           }) | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         text: 'Select from gallery', | ||||||
|  |         onPress: () => { | ||||||
|  |           openPicker({ | ||||||
|  |             mediaType: 'photo', | ||||||
|  |           }).then(async item => { | ||||||
|  |             await openCropper({ | ||||||
|  |               mediaType: 'photo', | ||||||
|  |               path: item.path, | ||||||
|  |               width: 80, | ||||||
|  |               height: 80, | ||||||
|  |               cropperCircleOverlay: true, | ||||||
|  |             }).then(croppedItem => { | ||||||
|  |               if (setUserAvatar != null) { | ||||||
|  |                 setUserAvatar(croppedItem.path) | ||||||
|  |               } | ||||||
|  |             }) | ||||||
|  |           }) | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|  |     ]) | ||||||
|  |   }, [setUserAvatar]) | ||||||
|  | 
 | ||||||
|  |   const renderSvg = (size: number, initials: string) => ( | ||||||
|     <Svg width={size} height={size} viewBox="0 0 100 100"> |     <Svg width={size} height={size} viewBox="0 0 100 100"> | ||||||
|       <Defs> |       <Defs> | ||||||
|         <LinearGradient id="grad" x1="0" y1="0" x2="1" y2="1"> |         <LinearGradient id="grad" x1="0" y1="0" x2="1" y2="1"> | ||||||
|  | @ -33,6 +88,32 @@ export function UserAvatar({ | ||||||
|       </Text> |       </Text> | ||||||
|     </Svg> |     </Svg> | ||||||
|   ) |   ) | ||||||
|  | 
 | ||||||
|  |   // setUserAvatar is only passed as prop on the EditProfile component
 | ||||||
|  |   return setUserAvatar != null && IMAGES_ENABLED ? ( | ||||||
|  |     <TouchableOpacity onPress={handleEditAvatar}> | ||||||
|  |       {userAvatar != null ? ( | ||||||
|  |         <Image style={styles.avatarImage} source={{uri: userAvatar}} /> | ||||||
|  |       ) : ( | ||||||
|  |         renderSvg(size, initials) | ||||||
|  |       )} | ||||||
|  |       <View style={styles.editButtonContainer}> | ||||||
|  |         <FontAwesomeIcon | ||||||
|  |           icon="camera" | ||||||
|  |           size={12} | ||||||
|  |           style={{color: colors.white}} | ||||||
|  |         /> | ||||||
|  |       </View> | ||||||
|  |     </TouchableOpacity> | ||||||
|  |   ) : userAvatar != null ? ( | ||||||
|  |     <Image | ||||||
|  |       style={styles.avatarImage} | ||||||
|  |       resizeMode="stretch" | ||||||
|  |       source={{uri: userAvatar}} | ||||||
|  |     /> | ||||||
|  |   ) : ( | ||||||
|  |     renderSvg(size, initials) | ||||||
|  |   ) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function getInitials(str: string): string { | function getInitials(str: string): string { | ||||||
|  | @ -50,3 +131,22 @@ function getInitials(str: string): string { | ||||||
|   } |   } | ||||||
|   return 'X' |   return 'X' | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | const styles = StyleSheet.create({ | ||||||
|  |   editButtonContainer: { | ||||||
|  |     position: 'absolute', | ||||||
|  |     width: 24, | ||||||
|  |     height: 24, | ||||||
|  |     bottom: 0, | ||||||
|  |     right: 0, | ||||||
|  |     borderRadius: 12, | ||||||
|  |     alignItems: 'center', | ||||||
|  |     justifyContent: 'center', | ||||||
|  |     backgroundColor: colors.gray5, | ||||||
|  |   }, | ||||||
|  |   avatarImage: { | ||||||
|  |     width: 80, | ||||||
|  |     height: 80, | ||||||
|  |     borderRadius: 40, | ||||||
|  |   }, | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | @ -1,10 +1,67 @@ | ||||||
| import React from 'react' | import React, {useCallback} from 'react' | ||||||
|  | import {StyleSheet, View, TouchableOpacity, Alert, Image} from 'react-native' | ||||||
| import Svg, {Rect, Defs, LinearGradient, Stop} from 'react-native-svg' | import Svg, {Rect, Defs, LinearGradient, Stop} from 'react-native-svg' | ||||||
|  | import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' | ||||||
| import {getGradient} from '../../lib/asset-gen' | import {getGradient} from '../../lib/asset-gen' | ||||||
|  | import {colors} from '../../lib/styles' | ||||||
|  | import { | ||||||
|  |   openCamera, | ||||||
|  |   openCropper, | ||||||
|  |   openPicker, | ||||||
|  | } from 'react-native-image-crop-picker' | ||||||
|  | import {IMAGES_ENABLED} from '../../../build-flags' | ||||||
| 
 | 
 | ||||||
| export function UserBanner({handle}: {handle: string}) { | export function UserBanner({ | ||||||
|  |   handle, | ||||||
|  |   userBanner, | ||||||
|  |   setUserBanner, | ||||||
|  | }: { | ||||||
|  |   handle: string | ||||||
|  |   userBanner: string | null | ||||||
|  |   setUserBanner?: React.Dispatch<React.SetStateAction<string | null>> | ||||||
|  | }) { | ||||||
|   const gradient = getGradient(handle) |   const gradient = getGradient(handle) | ||||||
|   return ( | 
 | ||||||
|  |   const handleEditBanner = useCallback(() => { | ||||||
|  |     Alert.alert('Select upload method', '', [ | ||||||
|  |       { | ||||||
|  |         text: 'Take a new photo', | ||||||
|  |         onPress: () => { | ||||||
|  |           openCamera({ | ||||||
|  |             mediaType: 'photo', | ||||||
|  |             cropping: true, | ||||||
|  |             width: 1500, | ||||||
|  |             height: 500, | ||||||
|  |           }).then(item => { | ||||||
|  |             if (setUserBanner != null) { | ||||||
|  |               setUserBanner(item.path) | ||||||
|  |             } | ||||||
|  |           }) | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         text: 'Select from gallery', | ||||||
|  |         onPress: () => { | ||||||
|  |           openPicker({ | ||||||
|  |             mediaType: 'photo', | ||||||
|  |           }).then(async item => { | ||||||
|  |             await openCropper({ | ||||||
|  |               mediaType: 'photo', | ||||||
|  |               path: item.path, | ||||||
|  |               width: 1500, | ||||||
|  |               height: 500, | ||||||
|  |             }).then(croppedItem => { | ||||||
|  |               if (setUserBanner != null) { | ||||||
|  |                 setUserBanner(croppedItem.path) | ||||||
|  |               } | ||||||
|  |             }) | ||||||
|  |           }) | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|  |     ]) | ||||||
|  |   }, [setUserBanner]) | ||||||
|  | 
 | ||||||
|  |   const renderSvg = () => ( | ||||||
|     <Svg width="100%" height="120" viewBox="50 0 200 100"> |     <Svg width="100%" height="120" viewBox="50 0 200 100"> | ||||||
|       <Defs> |       <Defs> | ||||||
|         <LinearGradient id="grad" x1="0" y1="0" x2="1" y2="1"> |         <LinearGradient id="grad" x1="0" y1="0" x2="1" y2="1"> | ||||||
|  | @ -20,4 +77,48 @@ export function UserBanner({handle}: {handle: string}) { | ||||||
|       <Rect x="0" y="0" width="400" height="100" fill="url(#grad2)" /> |       <Rect x="0" y="0" width="400" height="100" fill="url(#grad2)" /> | ||||||
|     </Svg> |     </Svg> | ||||||
|   ) |   ) | ||||||
|  | 
 | ||||||
|  |   // setUserBanner is only passed as prop on the EditProfile component
 | ||||||
|  |   return setUserBanner != null && IMAGES_ENABLED ? ( | ||||||
|  |     <TouchableOpacity onPress={handleEditBanner}> | ||||||
|  |       {userBanner != null ? ( | ||||||
|  |         <Image style={styles.bannerImage} source={{uri: userBanner}} /> | ||||||
|  |       ) : ( | ||||||
|  |         renderSvg() | ||||||
|  |       )} | ||||||
|  |       <View style={styles.editButtonContainer}> | ||||||
|  |         <FontAwesomeIcon | ||||||
|  |           icon="camera" | ||||||
|  |           size={12} | ||||||
|  |           style={{color: colors.white}} | ||||||
|  |         /> | ||||||
|  |       </View> | ||||||
|  |     </TouchableOpacity> | ||||||
|  |   ) : userBanner != null ? ( | ||||||
|  |     <Image | ||||||
|  |       style={styles.bannerImage} | ||||||
|  |       resizeMode="stretch" | ||||||
|  |       source={{uri: userBanner}} | ||||||
|  |     /> | ||||||
|  |   ) : ( | ||||||
|  |     renderSvg() | ||||||
|  |   ) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | const styles = StyleSheet.create({ | ||||||
|  |   editButtonContainer: { | ||||||
|  |     position: 'absolute', | ||||||
|  |     width: 24, | ||||||
|  |     height: 24, | ||||||
|  |     bottom: 8, | ||||||
|  |     right: 8, | ||||||
|  |     borderRadius: 12, | ||||||
|  |     alignItems: 'center', | ||||||
|  |     justifyContent: 'center', | ||||||
|  |     backgroundColor: colors.gray5, | ||||||
|  |   }, | ||||||
|  |   bannerImage: { | ||||||
|  |     width: '100%', | ||||||
|  |     height: 120, | ||||||
|  |   }, | ||||||
|  | }) | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue