-
Notifications
You must be signed in to change notification settings - Fork 2.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[Form Provider Refactor] AddDebitCardPage #29983
Changes from 22 commits
b4d942f
a348d1b
38d9184
00e12fc
b05c8ba
c290dbb
64dbcd4
196689d
b418a4c
280e8f6
677c2a5
377cdb9
c206be1
c2f54d3
06a6f27
d93b649
1142573
de75ffa
8dfc203
211934d
674a0dc
89b00f6
58f09ab
6a4f127
0f6688b
b35ebf5
890dd54
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,7 +1,7 @@ | ||
import _ from 'underscore'; | ||
import React, {useEffect, useMemo, useRef, useState} from 'react'; | ||
import React, {useCallback, useEffect, useMemo, useRef, useState} from 'react'; | ||
import PropTypes from 'prop-types'; | ||
import {Keyboard, LogBox, ScrollView, View, Text, ActivityIndicator} from 'react-native'; | ||
import {ActivityIndicator, Keyboard, LogBox, ScrollView, Text, View} from 'react-native'; | ||
import {GooglePlacesAutocomplete} from 'react-native-google-places-autocomplete'; | ||
import lodashGet from 'lodash/get'; | ||
import compose from '../../libs/compose'; | ||
|
@@ -140,27 +140,46 @@ const defaultProps = { | |
resultTypes: 'address', | ||
}; | ||
|
||
// Do not convert to class component! It's been tried before and presents more challenges than it's worth. | ||
// Relevant thread: https://expensify.slack.com/archives/C03TQ48KC/p1634088400387400 | ||
// Reference: https://github.com/FaridSafi/react-native-google-places-autocomplete/issues/609#issuecomment-886133839 | ||
function AddressSearch(props) { | ||
function AddressSearch({ | ||
canUseCurrentLocation, | ||
containerStyles, | ||
defaultValue, | ||
errorText, | ||
hint, | ||
innerRef, | ||
inputID, | ||
isLimitedToUSA, | ||
label, | ||
maxInputLength, | ||
network, | ||
onBlur, | ||
onInputChange, | ||
onPress, | ||
predefinedPlaces, | ||
preferredLocale, | ||
renamedInputKeys, | ||
resultTypes, | ||
shouldSaveDraft, | ||
translate, | ||
value, | ||
}) { | ||
const [displayListViewBorder, setDisplayListViewBorder] = useState(false); | ||
const [isTyping, setIsTyping] = useState(false); | ||
const [isFocused, setIsFocused] = useState(false); | ||
const [searchValue, setSearchValue] = useState(props.value || props.defaultValue || ''); | ||
const [searchValue, setSearchValue] = useState(value || defaultValue || ''); | ||
const [locationErrorCode, setLocationErrorCode] = useState(null); | ||
const [isFetchingCurrentLocation, setIsFetchingCurrentLocation] = useState(false); | ||
const shouldTriggerGeolocationCallbacks = useRef(true); | ||
const containerRef = useRef(); | ||
const query = useMemo( | ||
() => ({ | ||
language: props.preferredLocale, | ||
types: props.resultTypes, | ||
components: props.isLimitedToUSA ? 'country:us' : undefined, | ||
language: preferredLocale, | ||
types: resultTypes, | ||
components: isLimitedToUSA ? 'country:us' : undefined, | ||
}), | ||
[props.preferredLocale, props.resultTypes, props.isLimitedToUSA], | ||
[preferredLocale, resultTypes, isLimitedToUSA], | ||
); | ||
const shouldShowCurrentLocationButton = props.canUseCurrentLocation && searchValue.trim().length === 0 && isFocused; | ||
const shouldShowCurrentLocationButton = canUseCurrentLocation && searchValue.trim().length === 0 && isFocused; | ||
|
||
const saveLocationDetails = (autocompleteData, details) => { | ||
const addressComponents = details.address_components; | ||
|
@@ -169,7 +188,7 @@ function AddressSearch(props) { | |
// to this component which don't match the usual properties coming from auto-complete. In that case, only a limited | ||
// amount of data massaging needs to happen for what the parent expects to get from this function. | ||
if (_.size(details)) { | ||
props.onPress({ | ||
onPress({ | ||
address: lodashGet(details, 'description'), | ||
lat: lodashGet(details, 'geometry.location.lat', 0), | ||
lng: lodashGet(details, 'geometry.location.lng', 0), | ||
|
@@ -256,7 +275,7 @@ function AddressSearch(props) { | |
|
||
// Not all pages define the Address Line 2 field, so in that case we append any additional address details | ||
// (e.g. Apt #) to Address Line 1 | ||
if (subpremise && typeof props.renamedInputKeys.street2 === 'undefined') { | ||
if (subpremise && typeof renamedInputKeys.street2 === 'undefined') { | ||
values.street += `, ${subpremise}`; | ||
} | ||
|
||
|
@@ -265,19 +284,19 @@ function AddressSearch(props) { | |
values.country = country; | ||
} | ||
|
||
if (props.inputID) { | ||
_.each(values, (value, key) => { | ||
const inputKey = lodashGet(props.renamedInputKeys, key, key); | ||
if (inputID) { | ||
_.each(values, (inputValue, key) => { | ||
const inputKey = lodashGet(renamedInputKeys, key, key); | ||
if (!inputKey) { | ||
return; | ||
} | ||
props.onInputChange(value, inputKey); | ||
onInputChange(inputValue, inputKey); | ||
}); | ||
} else { | ||
props.onInputChange(values); | ||
onInputChange(values); | ||
} | ||
|
||
props.onPress(values); | ||
onPress(values); | ||
}; | ||
|
||
/** Gets the user's current location and registers success/error callbacks */ | ||
|
@@ -307,7 +326,7 @@ function AddressSearch(props) { | |
lng: successData.coords.longitude, | ||
address: CONST.YOUR_LOCATION_TEXT, | ||
}; | ||
props.onPress(location); | ||
onPress(location); | ||
}, | ||
(errorData) => { | ||
if (!shouldTriggerGeolocationCallbacks.current) { | ||
|
@@ -325,16 +344,16 @@ function AddressSearch(props) { | |
}; | ||
|
||
const renderHeaderComponent = () => | ||
props.predefinedPlaces.length > 0 && ( | ||
predefinedPlaces.length > 0 && ( | ||
<> | ||
{/* This will show current location button in list if there are some recent destinations */} | ||
{shouldShowCurrentLocationButton && ( | ||
<CurrentLocationButton | ||
onPress={getCurrentLocation} | ||
isDisabled={props.network.isOffline} | ||
isDisabled={network.isOffline} | ||
/> | ||
)} | ||
{!props.value && <Text style={[styles.textLabel, styles.colorMuted, styles.pv2, styles.ph3, styles.overflowAuto]}>{props.translate('common.recentDestinations')}</Text>} | ||
{!value && <Text style={[styles.textLabel, styles.colorMuted, styles.pv2, styles.ph3, styles.overflowAuto]}>{translate('common.recentDestinations')}</Text>} | ||
</> | ||
); | ||
|
||
|
@@ -346,6 +365,26 @@ function AddressSearch(props) { | |
}; | ||
}, []); | ||
|
||
const listEmptyComponent = useCallback( | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think we can use There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think the function is best suited for a child to be used as a functional component. Otherwise, the react lifecycle will not work properly and the prop used inside There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @luacmartins I think I think it's safer to use callback and the value of the prop needs to be a function. |
||
() => | ||
network.isOffline || !isTyping ? null : ( | ||
<Text style={[styles.textLabel, styles.colorMuted, styles.pv4, styles.ph3, styles.overflowAuto]}>{translate('common.noResultsFound')}</Text> | ||
), | ||
[isTyping, translate, network.isOffline], | ||
); | ||
|
||
const listLoader = useCallback( | ||
kowczarz marked this conversation as resolved.
Show resolved
Hide resolved
|
||
() => ( | ||
<View style={[styles.pv4]}> | ||
<ActivityIndicator | ||
color={themeColors.spinner} | ||
size="small" | ||
/> | ||
</View> | ||
), | ||
[], | ||
); | ||
|
||
return ( | ||
/* | ||
* The GooglePlacesAutocomplete component uses a VirtualizedList internally, | ||
|
@@ -372,20 +411,10 @@ function AddressSearch(props) { | |
fetchDetails | ||
suppressDefaultStyles | ||
enablePoweredByContainer={false} | ||
predefinedPlaces={props.predefinedPlaces} | ||
listEmptyComponent={ | ||
props.network.isOffline || !isTyping ? null : ( | ||
<Text style={[styles.textLabel, styles.colorMuted, styles.pv4, styles.ph3, styles.overflowAuto]}>{props.translate('common.noResultsFound')}</Text> | ||
) | ||
} | ||
listLoaderComponent={ | ||
<View style={[styles.pv4]}> | ||
<ActivityIndicator | ||
color={themeColors.spinner} | ||
size="small" | ||
/> | ||
</View> | ||
} | ||
predefinedPlaces={predefinedPlaces} | ||
listEmptyComponent={listEmptyComponent} | ||
listLoaderComponent={listLoader} | ||
renderHeaderComponent={renderHeaderComponent} | ||
renderRow={(data) => { | ||
const title = data.isPredefinedPlace ? data.name : data.structured_formatting.main_text; | ||
const subtitle = data.isPredefinedPlace ? data.description : data.structured_formatting.secondary_text; | ||
|
@@ -396,7 +425,6 @@ function AddressSearch(props) { | |
</View> | ||
); | ||
}} | ||
renderHeaderComponent={renderHeaderComponent} | ||
onPress={(data, details) => { | ||
saveLocationDetails(data, details); | ||
setIsTyping(false); | ||
|
@@ -411,34 +439,31 @@ function AddressSearch(props) { | |
query={query} | ||
requestUrl={{ | ||
useOnPlatform: 'all', | ||
url: props.network.isOffline ? null : ApiUtils.getCommandURL({command: 'Proxy_GooglePlaces&proxyUrl='}), | ||
url: network.isOffline ? null : ApiUtils.getCommandURL({command: 'Proxy_GooglePlaces&proxyUrl='}), | ||
}} | ||
textInputProps={{ | ||
InputComp: TextInput, | ||
ref: (node) => { | ||
if (!props.innerRef) { | ||
if (!innerRef) { | ||
return; | ||
} | ||
|
||
if (_.isFunction(props.innerRef)) { | ||
props.innerRef(node); | ||
if (_.isFunction(innerRef)) { | ||
innerRef(node); | ||
return; | ||
} | ||
|
||
// eslint-disable-next-line no-param-reassign | ||
props.innerRef.current = node; | ||
innerRef.current = node; | ||
}, | ||
label: props.label, | ||
containerStyles: props.containerStyles, | ||
errorText: props.errorText, | ||
hint: | ||
displayListViewBorder || (props.predefinedPlaces.length === 0 && shouldShowCurrentLocationButton) || (props.canUseCurrentLocation && isTyping) | ||
? undefined | ||
: props.hint, | ||
value: props.value, | ||
defaultValue: props.defaultValue, | ||
inputID: props.inputID, | ||
shouldSaveDraft: props.shouldSaveDraft, | ||
label, | ||
containerStyles, | ||
errorText, | ||
hint: displayListViewBorder || (predefinedPlaces.length === 0 && shouldShowCurrentLocationButton) || (canUseCurrentLocation && isTyping) ? undefined : hint, | ||
value, | ||
defaultValue, | ||
inputID, | ||
shouldSaveDraft, | ||
onFocus: () => { | ||
setIsFocused(true); | ||
}, | ||
|
@@ -448,24 +473,24 @@ function AddressSearch(props) { | |
setIsFocused(false); | ||
setIsTyping(false); | ||
} | ||
props.onBlur(); | ||
onBlur(); | ||
}, | ||
autoComplete: 'off', | ||
onInputChange: (text) => { | ||
setSearchValue(text); | ||
setIsTyping(true); | ||
if (props.inputID) { | ||
props.onInputChange(text); | ||
if (inputID) { | ||
onInputChange(text); | ||
} else { | ||
props.onInputChange({street: text}); | ||
onInputChange({street: text}); | ||
} | ||
|
||
// If the text is empty and we have no predefined places, we set displayListViewBorder to false to prevent UI flickering | ||
if (_.isEmpty(text) && _.isEmpty(props.predefinedPlaces)) { | ||
if (_.isEmpty(text) && _.isEmpty(predefinedPlaces)) { | ||
setDisplayListViewBorder(false); | ||
} | ||
}, | ||
maxLength: props.maxInputLength, | ||
maxLength: maxInputLength, | ||
spellCheck: false, | ||
}} | ||
styles={{ | ||
|
@@ -486,17 +511,18 @@ function AddressSearch(props) { | |
}} | ||
inbetweenCompo={ | ||
// We want to show the current location button even if there are no recent destinations | ||
props.predefinedPlaces.length === 0 && shouldShowCurrentLocationButton ? ( | ||
predefinedPlaces.length === 0 && shouldShowCurrentLocationButton ? ( | ||
<View style={[StyleUtils.getGoogleListViewStyle(true), styles.overflowAuto, styles.borderLeft, styles.borderRight]}> | ||
<CurrentLocationButton | ||
onPress={getCurrentLocation} | ||
isDisabled={props.network.isOffline} | ||
isDisabled={network.isOffline} | ||
/> | ||
</View> | ||
) : ( | ||
<></> | ||
) | ||
} | ||
placeholder="" | ||
kowczarz marked this conversation as resolved.
Show resolved
Hide resolved
|
||
/> | ||
<LocationErrorMessage | ||
onClose={() => setLocationErrorCode(null)} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I had to spread props to avoid disabling eslint exhaustive-deps rule in line 372.