From 9d96a8b4a384e29b5a2132dcd207e0bb6a7dc6e1 Mon Sep 17 00:00:00 2001 From: Yauheni Pasiukevich Date: Fri, 19 Jan 2024 16:45:38 +0100 Subject: [PATCH] [TS migration] Migrate 'Details' page to TypeScript --- src/components/OfflineWithFeedback.tsx | 2 +- src/libs/UserUtils.ts | 2 +- src/pages/{DetailsPage.js => DetailsPage.tsx} | 126 +++++++----------- src/styles/utils/addOutlineWidth/types.ts | 4 +- 4 files changed, 51 insertions(+), 83 deletions(-) rename src/pages/{DetailsPage.js => DetailsPage.tsx} (72%) diff --git a/src/components/OfflineWithFeedback.tsx b/src/components/OfflineWithFeedback.tsx index 975d154b885b..cd1df11be471 100644 --- a/src/components/OfflineWithFeedback.tsx +++ b/src/components/OfflineWithFeedback.tsx @@ -21,7 +21,7 @@ import MessagesRow from './MessagesRow'; type OfflineWithFeedbackProps = ChildrenProps & { /** The type of action that's pending */ - pendingAction?: OnyxCommon.PendingAction; + pendingAction?: OnyxCommon.PendingAction | null; /** Determine whether to hide the component's children if deletion is pending */ shouldHideOnDelete?: boolean; diff --git a/src/libs/UserUtils.ts b/src/libs/UserUtils.ts index a8c918bc5def..9228090f17a0 100644 --- a/src/libs/UserUtils.ts +++ b/src/libs/UserUtils.ts @@ -184,7 +184,7 @@ function getAvatarUrl(avatarSource: AvatarSource | undefined, accountID: number) * Avatars uploaded by users will have a _128 appended so that the asset server returns a small version. * This removes that part of the URL so the full version of the image can load. */ -function getFullSizeAvatar(avatarSource: AvatarSource, accountID: number): AvatarSource { +function getFullSizeAvatar(avatarSource: AvatarSource | undefined, accountID: number): AvatarSource { const source = getAvatar(avatarSource, accountID); if (typeof source !== 'string') { return source; diff --git a/src/pages/DetailsPage.js b/src/pages/DetailsPage.tsx similarity index 72% rename from src/pages/DetailsPage.js rename to src/pages/DetailsPage.tsx index a4cafd59cb73..973fd5d4aba6 100755 --- a/src/pages/DetailsPage.js +++ b/src/pages/DetailsPage.tsx @@ -1,10 +1,9 @@ +import type {StackScreenProps} from '@react-navigation/stack'; import Str from 'expensify-common/lib/str'; -import lodashGet from 'lodash/get'; -import PropTypes from 'prop-types'; import React from 'react'; import {ScrollView, View} from 'react-native'; import {withOnyx} from 'react-native-onyx'; -import _ from 'underscore'; +import type {OnyxEntry} from 'react-native-onyx'; import AttachmentModal from '@components/AttachmentModal'; import AutoUpdateTime from '@components/AutoUpdateTime'; import Avatar from '@components/Avatar'; @@ -18,77 +17,51 @@ import PressableWithoutFocus from '@components/Pressable/PressableWithoutFocus'; import ScreenWrapper from '@components/ScreenWrapper'; import Text from '@components/Text'; import UserDetailsTooltip from '@components/UserDetailsTooltip'; -import withLocalize, {withLocalizePropTypes} from '@components/withLocalize'; +import useLocalize from '@hooks/useLocalize'; import useThemeStyles from '@hooks/useThemeStyles'; -import compose from '@libs/compose'; import * as PersonalDetailsUtils from '@libs/PersonalDetailsUtils'; import {parsePhoneNumber} from '@libs/PhoneNumber'; import * as ReportUtils from '@libs/ReportUtils'; import * as UserUtils from '@libs/UserUtils'; +import type {DetailsNavigatorParamList} from '@navigation/types'; import * as Report from '@userActions/Report'; import CONST from '@src/CONST'; +import type {TranslationPaths} from '@src/languages/types'; import ONYXKEYS from '@src/ONYXKEYS'; -import personalDetailsPropType from './personalDetailsPropType'; - -const matchType = PropTypes.shape({ - params: PropTypes.shape({ - /** login passed via route /details/:login */ - login: PropTypes.string, - - /** report ID passed */ - reportID: PropTypes.string, - }), -}); - -const propTypes = { - /* Onyx Props */ +import type SCREENS from '@src/SCREENS'; +import type {PersonalDetails, PersonalDetailsList, Session} from '@src/types/onyx'; +type DetailsPageOnyxProps = { /** The personal details of the person who is logged in */ - personalDetails: personalDetailsPropType, - - /** Route params */ - route: matchType.isRequired, + personalDetails: OnyxEntry; /** Session info for the currently logged in user. */ - session: PropTypes.shape({ - /** Currently logged in user accountID */ - accountID: PropTypes.number, - }), - - ...withLocalizePropTypes, + session: OnyxEntry; }; -const defaultProps = { - // When opening someone else's profile (via deep link) before login, this is empty - personalDetails: {}, - session: { - accountID: 0, - }, -}; +type DetailsPageProps = DetailsPageOnyxProps & StackScreenProps; /** * Gets the phone number to display for SMS logins - * - * @param {Object} details - * @param {String} details.login - * @param {String} details.displayName - * @returns {String} */ -const getPhoneNumber = (details) => { +const getPhoneNumber = ({login = '', displayName = ''}: PersonalDetails): string | undefined => { // If the user hasn't set a displayName, it is set to their phone number, so use that - const parsedPhoneNumber = parsePhoneNumber(details.displayName); + const parsedPhoneNumber = parsePhoneNumber(displayName); if (parsedPhoneNumber.possible) { - return parsedPhoneNumber.number.e164; + return parsedPhoneNumber?.number?.e164; } // If the user has set a displayName, get the phone number from the SMS login - return details.login ? Str.removeSMSDomain(details.login) : ''; + return login ? Str.removeSMSDomain(login) : ''; }; -function DetailsPage(props) { +function DetailsPage({personalDetails, route, session}: DetailsPageProps) { const styles = useThemeStyles(); - const login = lodashGet(props.route.params, 'login', ''); - let details = _.find(props.personalDetails, (detail) => detail.login === login.toLowerCase()); + const {translate, formatPhoneNumber} = useLocalize(); + const login = route.params?.login ?? ''; + const sessionAccountID = session?.accountID ?? 0; + + let details = Object.values(personalDetails ?? {}).find((personalDetail) => personalDetail?.login === login.toLowerCase()); if (!details) { if (login === CONST.EMAIL.CONCIERGE) { @@ -116,44 +89,44 @@ function DetailsPage(props) { if (pronouns && pronouns.startsWith(CONST.PRONOUNS.PREFIX)) { const localeKey = pronouns.replace(CONST.PRONOUNS.PREFIX, ''); - pronouns = props.translate(`pronouns.${localeKey}`); + pronouns = translate(`pronouns.${localeKey}` as TranslationPaths); } const phoneNumber = getPhoneNumber(details); const phoneOrEmail = isSMSLogin ? getPhoneNumber(details) : details.login; const displayName = PersonalDetailsUtils.getDisplayNameOrDefault(details, '', false); - const isCurrentUser = props.session.accountID === details.accountID; + const isCurrentUser = sessionAccountID === details.accountID; return ( - - + + {details ? ( {({show}) => ( - + @@ -173,11 +146,11 @@ function DetailsPage(props) { style={[styles.textLabelSupporting, styles.mb1]} numberOfLines={1} > - {props.translate(isSMSLogin ? 'common.phoneNumber' : 'common.email')} + {translate(isSMSLogin ? 'common.phoneNumber' : 'common.email')} - {isSMSLogin ? props.formatPhoneNumber(phoneNumber) : details.login} + {isSMSLogin ? formatPhoneNumber(phoneNumber ?? '') : details.login} @@ -188,16 +161,16 @@ function DetailsPage(props) { style={[styles.textLabelSupporting, styles.mb1]} numberOfLines={1} > - {props.translate('profilePage.preferredPronouns')} + {translate('profilePage.preferredPronouns')} {pronouns} ) : null} - {shouldShowLocalTime && } + {shouldShowLocalTime && } {!isCurrentUser && ( Report.navigateToAndOpenReport([login])} @@ -213,18 +186,13 @@ function DetailsPage(props) { ); } -DetailsPage.propTypes = propTypes; -DetailsPage.defaultProps = defaultProps; DetailsPage.displayName = 'DetailsPage'; -export default compose( - withLocalize, - withOnyx({ - personalDetails: { - key: ONYXKEYS.PERSONAL_DETAILS_LIST, - }, - session: { - key: ONYXKEYS.SESSION, - }, - }), -)(DetailsPage); +export default withOnyx({ + personalDetails: { + key: ONYXKEYS.PERSONAL_DETAILS_LIST, + }, + session: { + key: ONYXKEYS.SESSION, + }, +})(DetailsPage); diff --git a/src/styles/utils/addOutlineWidth/types.ts b/src/styles/utils/addOutlineWidth/types.ts index 45975b72dc8a..d3a2538cd1b2 100644 --- a/src/styles/utils/addOutlineWidth/types.ts +++ b/src/styles/utils/addOutlineWidth/types.ts @@ -1,6 +1,6 @@ -import type {TextStyle} from 'react-native'; +import type {TextStyle, ViewStyle} from 'react-native'; import type {ThemeColors} from '@styles/theme/types'; -type AddOutlineWidth = (theme: ThemeColors, obj: TextStyle, val?: number, hasError?: boolean) => TextStyle; +type AddOutlineWidth = (theme: ThemeColors, obj: TStyle, val?: number, hasError?: boolean) => TStyle; export default AddOutlineWidth;