diff --git a/src/app/Components/Artist/ArtistAbout/ArtistAbout.tsx b/src/app/Components/Artist/ArtistAbout/ArtistAbout.tsx index ec864daa38d..2439b6a16ae 100644 --- a/src/app/Components/Artist/ArtistAbout/ArtistAbout.tsx +++ b/src/app/Components/Artist/ArtistAbout/ArtistAbout.tsx @@ -1,11 +1,12 @@ import { ContextModule, OwnerType } from "@artsy/cohesion" -import { Join, Spacer, Tabs } from "@artsy/palette-mobile" +import { Flex, Join, Spacer, Tabs } from "@artsy/palette-mobile" import { ArtistAbout_artist$data } from "__generated__/ArtistAbout_artist.graphql" import { Articles } from "app/Components/Artist/Articles/Articles" import { ArtistAboutEmpty } from "app/Components/Artist/ArtistAbout/ArtistAboutEmpty" import { ArtistAboutRelatedGenes } from "app/Components/Artist/ArtistAbout/ArtistAboutRelatedGenes" -import { Biography } from "app/Components/Artist/Biography" +import { Biography, MAX_WIDTH_BIO } from "app/Components/Artist/Biography" import { RelatedArtistsRail } from "app/Components/Artist/RelatedArtistsRail" +import { SectionTitle } from "app/Components/SectionTitle" import { ArtistSeriesMoreSeriesFragmentContainer } from "app/Scenes/ArtistSeries/ArtistSeriesMoreSeries" import { extractNodes } from "app/utils/extractNodes" import { createFragmentContainer, graphql } from "react-relay" @@ -46,7 +47,10 @@ export const ArtistAbout: React.FC = ({ artist }) => { {!!hasBiography && ( <> - + + + + )} {!!hasInsights && } diff --git a/src/app/Components/Artist/Biography.tsx b/src/app/Components/Artist/Biography.tsx index 530dd28c1bb..be6f3088f37 100644 --- a/src/app/Components/Artist/Biography.tsx +++ b/src/app/Components/Artist/Biography.tsx @@ -1,19 +1,18 @@ -import { Flex, Text } from "@artsy/palette-mobile" +import { Text } from "@artsy/palette-mobile" import { Biography_artist$key } from "__generated__/Biography_artist.graphql" import { HTML } from "app/Components/HTML" -import { SectionTitle } from "app/Components/SectionTitle" import { useState } from "react" import { graphql, useFragment } from "react-relay" const MAX_CHARS = 250 - -export const MAX_WIDTH = 650 +export const MAX_WIDTH_BIO = 650 interface BiographyProps { artist: Biography_artist$key + variant?: "sm" | "xs" } -export const Biography: React.FC = ({ artist }) => { +export const Biography: React.FC = ({ artist, variant = "sm" }) => { const [expanded, setExpanded] = useState(false) const data = useFragment(query, artist) @@ -26,21 +25,20 @@ export const Biography: React.FC = ({ artist }) => { const canExpand = text.length > MAX_CHARS return ( - - + <> MAX_CHARS && !expanded ? "... " : " " }`} - variant="sm" + variant={variant} /> {!!canExpand && ( - setExpanded((e) => !e)} mt={-1}> + setExpanded((e) => !e)} mt={-1} variant={variant}> {expanded ? "Read Less" : "Read More"} )} - + ) } diff --git a/src/app/Components/ArtistFollowButton.tsx b/src/app/Components/ArtistFollowButton.tsx new file mode 100644 index 00000000000..987ba87e42e --- /dev/null +++ b/src/app/Components/ArtistFollowButton.tsx @@ -0,0 +1,61 @@ +import { FollowButton } from "@artsy/palette-mobile" +import { ArtistFollowButtonQuery } from "__generated__/ArtistFollowButtonQuery.graphql" +import { ArtistFollowButton_artist$key } from "__generated__/ArtistFollowButton_artist.graphql" +import { useFollowArtist } from "app/utils/mutations/useFollowArtist" +import { FC } from "react" +import { useLazyLoadQuery, graphql, useFragment } from "react-relay" + +interface ArtistFollowButtonProps { + artist: ArtistFollowButton_artist$key +} + +export const ArtistFollowButton: FC = ({ artist }) => { + const data = useFragment(fragment, artist) + const [commitMutation, isInFlight] = useFollowArtist() + + if (!data) { + return null + } + + const handleOnPress = () => { + commitMutation({ + variables: { input: { artistID: data.internalID, unfollow: data.isFollowed } }, + updater: (store) => { + store.get(data.internalID)?.setValue(!data.isFollowed, "isFollowed") + }, + }) + } + + return +} + +const fragment = graphql` + fragment ArtistFollowButton_artist on Artist { + internalID @required(action: NONE) + isFollowed @required(action: NONE) + } +` + +interface ArtistFollowButtonQueryRendererProps { + artistID: string +} + +export const ArtistFollowButtonQueryRenderer: FC = ({ + artistID, +}) => { + const data = useLazyLoadQuery(query, { id: artistID }) + + if (!data.artist) { + return + } + + return +} + +const query = graphql` + query ArtistFollowButtonQuery($id: String!) { + artist(id: $id) { + ...ArtistFollowButton_artist + } + } +` diff --git a/src/app/Components/ArtistListItemShort.tsx b/src/app/Components/ArtistListItemShort.tsx new file mode 100644 index 00000000000..4e397b9c37b --- /dev/null +++ b/src/app/Components/ArtistListItemShort.tsx @@ -0,0 +1,100 @@ +import { EntityHeader, Flex, Touchable } from "@artsy/palette-mobile" +import { ArtistListItemShortQuery } from "__generated__/ArtistListItemShortQuery.graphql" +import { ArtistListItemShort_artist$key } from "__generated__/ArtistListItemShort_artist.graphql" +import { ArtistFollowButtonQueryRenderer } from "app/Components/ArtistFollowButton" +import { ReadMore } from "app/Components/ReadMore" +import { navigate } from "app/system/navigation/navigate" +import { truncatedTextLimit } from "app/utils/hardware" +import { withSuspense } from "app/utils/hooks/withSuspense" +import { FC } from "react" +import { graphql, useFragment, useLazyLoadQuery } from "react-relay" + +interface ArtistListItemShortProps { + artist: ArtistListItemShort_artist$key +} + +export const ArtistListItemShort: FC = ({ artist }) => { + const data = useFragment(fragment, artist) + + if (!data) { + return null + } + + const image = data.coverArtwork?.image?.cropped?.url ?? undefined + const bio = data?.biographyBlurb?.text + const bioTextLimit = truncatedTextLimit() + + const handleOnPress = () => { + navigate(data.href) + } + + return ( + <> + + + } + /> + + + + {!!bio && ( + + )} + + ) +} + +const fragment = graphql` + fragment ArtistListItemShort_artist on Artist { + internalID @required(action: NONE) + name @required(action: NONE) + initials @required(action: NONE) + href @required(action: NONE) + formattedNationalityAndBirthday + + coverArtwork { + image { + cropped(height: 45, width: 45) { + url + } + } + } + biographyBlurb { + text + } + } +` + +interface ArtworkListItemShortWithSuspenseProps { + artistSlug: string +} + +export const ArtworkListItemShortWithSuspense = withSuspense( + { + Component: ({ artistSlug }) => { + const data = useLazyLoadQuery(query, { slug: artistSlug }) + + if (!data?.artist) { + return null + } + + return + }, + LoadingFallback: () => null, + ErrorFallback: () => null, + } +) + +const query = graphql` + query ArtistListItemShortQuery($slug: String!) { + artist(id: $slug) { + ...ArtistListItemShort_artist + name + } + } +` diff --git a/src/app/Components/Expandable.tsx b/src/app/Components/Expandable.tsx index 8ee8add4000..fadc414ee49 100644 --- a/src/app/Components/Expandable.tsx +++ b/src/app/Components/Expandable.tsx @@ -1,5 +1,5 @@ import { ChevronIcon, Collapse, Flex, Text, Touchable } from "@artsy/palette-mobile" -import { MAX_WIDTH } from "app/Components/Artist/Biography" +import { MAX_WIDTH_BIO } from "app/Components/Artist/Biography" import { MotiView } from "moti" import { useState } from "react" @@ -30,7 +30,12 @@ export const Expandable: React.FC = ({ } return ( - + handleToggle()} accessibilityRole="togglebutton" diff --git a/src/app/Components/PartnerEntityHeader.tsx b/src/app/Components/PartnerEntityHeader.tsx index ff31bd5f7bd..73946f50e53 100644 --- a/src/app/Components/PartnerEntityHeader.tsx +++ b/src/app/Components/PartnerEntityHeader.tsx @@ -43,7 +43,7 @@ export const PartnerEntityHeader: React.FC = ({ partne export const PartnerEntityHeaderFragmentContainer = createFragmentContainer(PartnerEntityHeader, { partner: graphql` fragment PartnerEntityHeader_partner on Partner { - ...PartnerFollowButton_partner + ...PartnerFollowButton_deprecated_partner href name cities diff --git a/src/app/Components/PartnerFollowButton.tsx b/src/app/Components/PartnerFollowButton.tsx new file mode 100644 index 00000000000..efd1d49ab45 --- /dev/null +++ b/src/app/Components/PartnerFollowButton.tsx @@ -0,0 +1,70 @@ +import { FollowButton } from "@artsy/palette-mobile" +import { PartnerFollowButtonQuery } from "__generated__/PartnerFollowButtonQuery.graphql" +import { PartnerFollowButton_partner$key } from "__generated__/PartnerFollowButton_partner.graphql" +import { useFollowProfile } from "app/utils/mutations/useFollowProfile" +import { FC } from "react" +import { graphql, useFragment, useLazyLoadQuery } from "react-relay" + +interface PartnerFollowButtonProps { + partner: PartnerFollowButton_partner$key +} + +export const PartnerFollowButton: FC = ({ partner }) => { + const data = useFragment(fragment, partner) + const { followProfile, isInFlight } = useFollowProfile({ + id: data?.profile.id ?? "", + internalID: data?.profile.internalID ?? "", + isFollowed: !!data?.profile.isFollowed, + }) + + if (!data) { + return null + } + + const handleOnPress = () => { + followProfile() + } + + return ( + + ) +} + +const fragment = graphql` + fragment PartnerFollowButton_partner on Partner { + internalID @required(action: NONE) + profile @required(action: NONE) { + id @required(action: NONE) + internalID @required(action: NONE) + isFollowed + } + } +` + +interface PartnerFollowButtonQueryRendererProps { + partnerID: string +} + +export const PartnerFollowButtonQueryRenderer: FC = ({ + partnerID, +}) => { + const data = useLazyLoadQuery(query, { id: partnerID }) + + if (!data.partner) { + return + } + + return +} + +const query = graphql` + query PartnerFollowButtonQuery($id: String!) { + partner(id: $id) { + ...PartnerFollowButton_partner + } + } +` diff --git a/src/app/Components/PartnerListItemShort.tsx b/src/app/Components/PartnerListItemShort.tsx new file mode 100644 index 00000000000..651c1e9fc9f --- /dev/null +++ b/src/app/Components/PartnerListItemShort.tsx @@ -0,0 +1,120 @@ +import { EntityHeader, Flex, Text, Touchable } from "@artsy/palette-mobile" +import { PartnerListItemShortQuery } from "__generated__/PartnerListItemShortQuery.graphql" +import { PartnerListItemShort_partner$key } from "__generated__/PartnerListItemShort_partner.graphql" +import { PartnerFollowButtonQueryRenderer } from "app/Components/PartnerFollowButton" +import { sortByDistance } from "app/Scenes/GalleriesForYou/helpers" +import { navigate } from "app/system/navigation/navigate" +import { extractNodes } from "app/utils/extractNodes" +import { Location, useLocation } from "app/utils/hooks/useLocation" +import { withSuspense } from "app/utils/hooks/withSuspense" +import { pluralize } from "app/utils/pluralize" +import { FC } from "react" +import { graphql, useFragment, useLazyLoadQuery } from "react-relay" + +interface PartnerListItemShortProps { + partner: PartnerListItemShort_partner$key +} + +export const PartnerListItemShort: FC = ({ partner }) => { + const data = useFragment(fragment, partner) + const { location } = useLocation() + + if (!data) { + return null + } + + const image = data.profile?.image?.cropped?.url ?? undefined + const locations = extractNodes(data.locationsConnection) + const sortedLocations = location + ? sortByDistance(locations as { coordinates?: Location }[], location) + : locations + + const handleOnPress = () => { + navigate(data.href) + } + + return ( + <> + + + + {!!sortedLocations[0] && ( + + {sortedLocations[0].city} + {!!(sortedLocations.length > 1) && + ` and ${sortedLocations.length - 1} more ${pluralize( + "location", + sortedLocations.length - 1 + )}`} + + )} + + } + RightButton={} + /> + + + + ) +} + +const fragment = graphql` + fragment PartnerListItemShort_partner on Partner { + internalID @required(action: NONE) + name @required(action: NONE) + initials @required(action: NONE) + href @required(action: NONE) + + profile { + image { + cropped(height: 45, width: 45) { + url + } + } + } + locationsConnection(first: 20) { + edges { + node { + city + coordinates { + lat + lng + } + } + } + } + } +` + +interface PartnerListItemShortWithSuspenseProps { + partnerID: string +} + +export const PartnerListItemShortWithSuspense = withSuspense( + { + Component: ({ partnerID }) => { + const data = useLazyLoadQuery(query, { id: partnerID }) + + if (!data?.partner) { + return null + } + + return + }, + LoadingFallback: () => null, + ErrorFallback: () => null, + } +) + +const query = graphql` + query PartnerListItemShortQuery($id: String!) { + partner(id: $id) { + ...PartnerListItemShort_partner + } + } +` diff --git a/src/app/Scenes/Activity/components/PartnerShowOpenedNotification.tsx b/src/app/Scenes/Activity/components/PartnerShowOpenedNotification.tsx index c8f6248a95d..9b65f81a924 100644 --- a/src/app/Scenes/Activity/components/PartnerShowOpenedNotification.tsx +++ b/src/app/Scenes/Activity/components/PartnerShowOpenedNotification.tsx @@ -27,7 +27,7 @@ export const PartnerShowOpenedNotification: FC = ({ artwork }) => { const { medium, dimensions, framed, editionOf, editionSets, isUnlisted } = artwork - const dimensionsPresent = (dimensions: any) => - /\d/.test(dimensions?.in) || /\d/.test(dimensions?.cm) - - const getFrameString = (frameDetails?: string | null, isUnlisted?: boolean) => { - if (frameDetails !== "Included") { - if (isUnlisted) { - return "Frame not included" - } else { - return - } - } - - return `Frame ${frameDetails.toLowerCase()}` - } - return ( @@ -81,3 +66,18 @@ export const ArtworkDimensionsClassificationAndAuthenticityFragmentContainer = } `, }) + +export const getFrameString = (frameDetails?: string | null, isUnlisted?: boolean) => { + if (frameDetails !== "Included") { + if (isUnlisted) { + return "Frame not included" + } else { + return + } + } + + return `Frame ${frameDetails.toLowerCase()}` +} + +export const dimensionsPresent = (dimensions: any) => + /\d/.test(dimensions?.in) || /\d/.test(dimensions?.cm) diff --git a/src/app/Scenes/Artwork/Components/ArtworkHeader.tsx b/src/app/Scenes/Artwork/Components/ArtworkHeader.tsx index 4b4ad3a48f1..965a103f0c4 100644 --- a/src/app/Scenes/Artwork/Components/ArtworkHeader.tsx +++ b/src/app/Scenes/Artwork/Components/ArtworkHeader.tsx @@ -144,7 +144,7 @@ export const ArtworkHeaderFragmentContainer = createFragmentContainer(ArtworkHea artists(shallow: true) { name } - partner { + partner(shallow: true) { name } } diff --git a/src/app/Scenes/Artwork/Components/ArtworkPartnerOfferNote.tsx b/src/app/Scenes/Artwork/Components/ArtworkPartnerOfferNote.tsx index 17e6397bfed..97db28a315e 100644 --- a/src/app/Scenes/Artwork/Components/ArtworkPartnerOfferNote.tsx +++ b/src/app/Scenes/Artwork/Components/ArtworkPartnerOfferNote.tsx @@ -58,7 +58,7 @@ export const ArtworkPartnerOfferNote: React.FC = ( const artworkFragment = graphql` fragment ArtworkPartnerOfferNote_artwork on Artwork { - partner { + partner(shallow: true) { profile { icon { url(version: "square140") diff --git a/src/app/Scenes/Artwork/Components/CommercialButtons/CompleteProfilePrompt.tsx b/src/app/Scenes/Artwork/Components/CommercialButtons/CompleteProfilePrompt.tsx index ff4f9bb6c75..fd708e25547 100644 --- a/src/app/Scenes/Artwork/Components/CommercialButtons/CompleteProfilePrompt.tsx +++ b/src/app/Scenes/Artwork/Components/CommercialButtons/CompleteProfilePrompt.tsx @@ -28,7 +28,7 @@ export const CompleteProfilePrompt: React.FC = ({ const artworkFragment = graphql` fragment CompleteProfilePrompt_artwork on Artwork { - partner { + partner(shallow: true) { name } } diff --git a/src/app/Scenes/Artwork/Components/CommercialButtons/InquiryModal.tsx b/src/app/Scenes/Artwork/Components/CommercialButtons/InquiryModal.tsx index 24ff808a1c4..412f6678f98 100644 --- a/src/app/Scenes/Artwork/Components/CommercialButtons/InquiryModal.tsx +++ b/src/app/Scenes/Artwork/Components/CommercialButtons/InquiryModal.tsx @@ -218,7 +218,7 @@ const artworkFragment = graphql` internalID question } - partner { + partner(shallow: true) { name } } diff --git a/src/app/Scenes/Artwork/Components/PartnerCard.tsx b/src/app/Scenes/Artwork/Components/PartnerCard.tsx index e2d23ef6e0d..cfc224af40c 100644 --- a/src/app/Scenes/Artwork/Components/PartnerCard.tsx +++ b/src/app/Scenes/Artwork/Components/PartnerCard.tsx @@ -103,7 +103,7 @@ export const PartnerCardFragmentContainer = createFragmentContainer(PartnerCard, isBenefit isGalleryAuction } - partner { + partner(shallow: true) { cities isDefaultProfilePublic type diff --git a/src/app/Scenes/Artwork/Components/PrivateArtwork/PrivateArtworkExclusiveAccess.tsx b/src/app/Scenes/Artwork/Components/PrivateArtwork/PrivateArtworkExclusiveAccess.tsx index fb8c2020f7d..31ab897ae67 100644 --- a/src/app/Scenes/Artwork/Components/PrivateArtwork/PrivateArtworkExclusiveAccess.tsx +++ b/src/app/Scenes/Artwork/Components/PrivateArtwork/PrivateArtworkExclusiveAccess.tsx @@ -15,7 +15,7 @@ export const PrivateArtworkExclusiveAccess: React.FC = ({ : locations const { followProfile, isInFlight } = useFollowProfile({ - id: profile?.id!, - internalID: profile?.internalID!, - isFollowd: profile?.isFollowed!, + id: profile?.id ?? "", + internalID: profile?.internalID ?? "", + isFollowed: !!profile?.isFollowed, onCompleted: onFollow, }) diff --git a/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryAboutTheWorkTab.tsx b/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryAboutTheWorkTab.tsx new file mode 100644 index 00000000000..9b749fb4aae --- /dev/null +++ b/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryAboutTheWorkTab.tsx @@ -0,0 +1,326 @@ +import { + ArtworkIcon, + CertificateIcon, + EnvelopeIcon, + Flex, + FlexProps, + LinkText, + Skeleton, + SkeletonBox, + SkeletonText, + Spacer, + Text, + TextProps, + useSpace, +} from "@artsy/palette-mobile" +import { BottomSheetScrollView } from "@gorhom/bottom-sheet" +import { InfiniteDiscoveryAboutTheWorkTab_artwork$key } from "__generated__/InfiniteDiscoveryAboutTheWorkTab_artwork.graphql" +import { InfiniteDiscoveryBottomSheetTabsQuery } from "__generated__/InfiniteDiscoveryBottomSheetTabsQuery.graphql" +import { MyProfileEditModal_me$key } from "__generated__/MyProfileEditModal_me.graphql" +import { useSendInquiry_me$key } from "__generated__/useSendInquiry_me.graphql" +import { ArtistListItemShort } from "app/Components/ArtistListItemShort" +import { Divider } from "app/Components/Bidding/Components/Divider" +import { PartnerListItemShort } from "app/Components/PartnerListItemShort" +import { dimensionsPresent } from "app/Scenes/Artwork/Components/ArtworkDimensionsClassificationAndAuthenticity/ArtworkDimensionsClassificationAndAuthenticity" +import { ContactGalleryButton } from "app/Scenes/Artwork/Components/CommercialButtons/ContactGalleryButton" +import { aboutTheWorkQuery } from "app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheet" +import { navigate } from "app/system/navigation/navigate" +import { FC } from "react" +import { graphql, PreloadedQuery, useFragment, usePreloadedQuery } from "react-relay" + +interface AboutTheWorkTabProps { + artwork: InfiniteDiscoveryAboutTheWorkTab_artwork$key + me: MyProfileEditModal_me$key & useSendInquiry_me$key +} + +// TODO: export TAB_BAR_HEIGHT from palette-mobile +export const AboutTheWorkTab: FC = ({ artwork, me }) => { + const data = useFragment(fragment, artwork) + const space = useSpace() + + if (!data) { + return null + } + + const attributionClass = data.attributionClass?.shortArrayDescription + const hasCertificateOfAuthenticity = data.hasCertificateOfAuthenticity && !data.isBiddable + + return ( + + + + {!!attributionClass?.length && ( + + + + {attributionClass[0]}{" "} + navigate(`/artwork-classifications`)}> + {attributionClass[1]} + + + + )} + + {!!hasCertificateOfAuthenticity && ( + + + + Includes a{" "} + navigate(`/artwork-certificate-of-authenticity`)} + > + Certificate of Authenticity + + + + )} + + + {!!attributionClass?.length && !!hasCertificateOfAuthenticity && } + + + + Materials + {data.medium} + + + {dimensionsPresent(data.dimensions) && ( + + Dimensions + {`${data.dimensions?.in} | ${data.dimensions?.cm}`} + + )} + + + Rarity + {data.attributionClass?.name} + + + {!!data.mediumType?.name && ( + + Medium + {data.mediumType?.name} + + )} + + {!!data.condition?.displayText && ( + + Condition + {data.condition.displayText} + + )} + + {!!data.signatureInfo?.details && ( + + Signature + {data.signatureInfo.details} + + )} + + {!!data.certificateOfAuthenticity?.details && ( + + Certificate of Authenticity + {data.certificateOfAuthenticity.details} + + )} + + {!!data.publisher && ( + + Publisher + {data.publisher} + + )} + + + Frame + {data.isFramed ? "Frame included" : "Frame not included"} + + + + + + + {!!data.artists?.length && ( + Artist{data.artists.length > 1 ? `s` : ``} + )} + + {data.artists?.map((artist, index) => { + if (!artist) { + return null + } + + return + })} + + + + + + + Gallery + + + + + Questions about this piece? + + } + /> + + + + + + + + ) +} + +const fragment = graphql` + fragment InfiniteDiscoveryAboutTheWorkTab_artwork on Artwork { + ...ContactGalleryButton_artwork + + attributionClass { + shortArrayDescription + } + hasCertificateOfAuthenticity + isBiddable + + medium + dimensions { + in + cm + } + attributionClass { + name + } + mediumType { + name + } + condition { + description + displayText + value + } + signatureInfo { + details + } + certificateOfAuthenticity { + details + } + publisher + isFramed + + artists(shallow: true) @required(action: NONE) { + ...ArtistListItemShort_artist + } + + partner(shallow: true) @required(action: NONE) { + ...PartnerListItemShort_partner + } + } +` + +interface InfiniteDiscoveryAboutTheWorkTabProps { + queryRef: PreloadedQuery +} + +export const InfiniteDiscoveryAboutTheWorkTab: FC = ({ + queryRef, +}) => { + const data = usePreloadedQuery(aboutTheWorkQuery, queryRef) + + if (!data?.artwork || !data?.me) { + return + } + + return +} + +export const InfiniteDiscoveryAboutTheWorkTabSkeleton: FC = () => { + const space = useSpace() + + return ( + + + + + + + Classification + + + + + Authenticity + + + + + + + {[ + "Materials", + "Dimensions", + "Rarity", + "Medium", + "Condition", + "Signature", + "Certificate", + "Publisher", + ].map((label) => ( + + + {label} + + {label + "a".repeat(Math.random() * 10)} + + ))} + + + + + + Artist + + + + + + Artist Name + birthdate + + + + + + + {"Biogr ".repeat(20)} + + + + + ) +} + +const labelStyle = { + width: "35%", + variant: "xs", + color: "black60", +} satisfies TextProps | FlexProps + +const valueStyle = { + width: "65%", + variant: "xs", +} satisfies TextProps | FlexProps diff --git a/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheet.tsx b/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheet.tsx new file mode 100644 index 00000000000..53bc887fb83 --- /dev/null +++ b/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheet.tsx @@ -0,0 +1,96 @@ +import { ArrowUpIcon, Flex, Text } from "@artsy/palette-mobile" +import { InfiniteDiscoveryBottomSheetTabsQuery } from "__generated__/InfiniteDiscoveryBottomSheetTabsQuery.graphql" +import { AutomountedBottomSheetModal } from "app/Components/BottomSheet/AutomountedBottomSheetModal" +import { InfiniteDiscoveryBottomSheetFooterQueryRenderer } from "app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheetFooter" +import { + InfiniteDiscoveryTabs, + InfiniteDiscoveryTabsSkeleton, +} from "app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheetTabs" +import { FC, useEffect, useState } from "react" +import { Dimensions } from "react-native" +import { Gesture, GestureDetector } from "react-native-gesture-handler" +import { runOnJS } from "react-native-reanimated" +import { useSafeAreaInsets } from "react-native-safe-area-context" +import { graphql, useQueryLoader } from "react-relay" + +interface InfiniteDiscoveryBottomSheetProps { + // TODO: should come from the context + artworkID: string +} + +export const InfiniteDiscoveryBottomSheet: FC = ({ + artworkID, +}) => { + const [visible, setVisible] = useState(false) + const { bottom } = useSafeAreaInsets() + const [queryRef, loadQuery] = + useQueryLoader(aboutTheWorkQuery) + + useEffect(() => { + if (!queryRef) { + loadQuery({ id: artworkID }) + } + }, []) + + const pan = Gesture.Pan().onUpdate((event) => { + if (event.translationY < TRANSLATE_Y_THRESHOLD) { + runOnJS(setVisible)(true) + } + }) + + const handleOnDismiss = () => { + setVisible(false) + } + + return ( + <> + { + if (!queryRef) { + return null + } + return + }} + > + {/* This if is to make TS happy, usePreloadedQuery will always require a queryRef */} + {!queryRef ? ( + + ) : ( + + )} + + + + + + + Swipe up for more details + + + + ) +} + +const { height } = Dimensions.get("screen") + +const SNAP_POINTS = [height * 0.88] +const TRANSLATE_Y_THRESHOLD = -50 + +export const aboutTheWorkQuery = graphql` + query InfiniteDiscoveryBottomSheetTabsQuery($id: String!) { + me { + ...useSendInquiry_me + ...MyProfileEditModal_me + ...BidButton_me + ...InfiniteDiscoveryBottomSheetFooter_me + } + artwork(id: $id) { + ...InfiniteDiscoveryAboutTheWorkTab_artwork + ...InfiniteDiscoveryBottomSheetFooter_artwork + } + } +` diff --git a/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheetFooter.tsx b/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheetFooter.tsx new file mode 100644 index 00000000000..ecd509cb8b6 --- /dev/null +++ b/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheetFooter.tsx @@ -0,0 +1,166 @@ +import { Flex, Skeleton, SkeletonBox, SkeletonText, useColor } from "@artsy/palette-mobile" +import { BottomSheetFooter, BottomSheetFooterProps } from "@gorhom/bottom-sheet" +import { + InfiniteDiscoveryBottomSheetFooter_artwork$data, + InfiniteDiscoveryBottomSheetFooter_artwork$key, +} from "__generated__/InfiniteDiscoveryBottomSheetFooter_artwork.graphql" +import { InfiniteDiscoveryBottomSheetFooter_me$key } from "__generated__/InfiniteDiscoveryBottomSheetFooter_me.graphql" +import { InfiniteDiscoveryBottomSheetTabsQuery } from "__generated__/InfiniteDiscoveryBottomSheetTabsQuery.graphql" +import { Divider } from "app/Components/Bidding/Components/Divider" +import { currentTimerState } from "app/Components/Bidding/Components/Timer" +import { artworkModel, ArtworkStoreProvider } from "app/Scenes/Artwork/ArtworkStore" +import { ArtworkCommercialButtons } from "app/Scenes/Artwork/Components/ArtworkCommercialButtons" +import { ArtworkPrice } from "app/Scenes/Artwork/Components/ArtworkPrice" +import { aboutTheWorkQuery } from "app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheet" +import { + AuctionWebsocketChannelInfo, + AuctionWebsocketContextProvider, +} from "app/utils/Websockets/auctions/AuctionSocketContext" +import { FC } from "react" +import { useSafeAreaInsets } from "react-native-safe-area-context" +import { graphql, PreloadedQuery, useFragment, usePreloadedQuery } from "react-relay" + +interface InfiniteDiscoveryBottomSheetFooterProps extends BottomSheetFooterProps { + artwork: InfiniteDiscoveryBottomSheetFooter_artwork$key + me: InfiniteDiscoveryBottomSheetFooter_me$key +} + +export const InfiniteDiscoveryBottomSheetFooter: FC = ({ + artwork: _artwork, + me: _me, + ...bottomSheetFooterProps +}) => { + const { bottom } = useSafeAreaInsets() + const color = useColor() + + const artwork = useFragment(artworkFragment, _artwork) + const me = useFragment(meFragment, _me) + + if (!artwork || !me) { + return null + } + + const partnerOffer = me.partnerOffersConnection?.edges?.[0]?.node + + const initialAuctionTimer = getInitialAuctionTimerState(artwork) + const socketChannelInfo: AuctionWebsocketChannelInfo = { + channel: "SalesChannel", + sale_id: artwork.sale?.internalID, + } + const websocketEnabled = !!artwork.sale?.extendedBiddingIntervalMinutes + + return ( + + + + + + + + + + + + + ) +} + +const artworkFragment = graphql` + fragment InfiniteDiscoveryBottomSheetFooter_artwork on Artwork { + ...ArtworkPrice_artwork + ...ArtworkCommercialButtons_artwork + + isInAuction + sale { + internalID + isPreview + isClosed + liveStartAt + extendedBiddingIntervalMinutes + } + } +` + +const meFragment = graphql` + fragment InfiniteDiscoveryBottomSheetFooter_me on Me { + ...ArtworkCommercialButtons_me + ...MyProfileEditModal_me + ...useSendInquiry_me + ...BidButton_me + + partnerOffersConnection { + edges { + node { + ...ArtworkPrice_partnerOffer + ...ArtworkCommercialButtons_partnerOffer + } + } + } + } +` + +interface InfiniteDiscoveryBottomSheetFooterQueryRendererProps extends BottomSheetFooterProps { + queryRef: PreloadedQuery +} + +export const InfiniteDiscoveryBottomSheetFooterQueryRenderer: FC< + InfiniteDiscoveryBottomSheetFooterQueryRendererProps +> = ({ queryRef, ...rest }) => { + const data = usePreloadedQuery(aboutTheWorkQuery, queryRef) + + if (!data.artwork || !data.me) { + return + } + + return +} + +const getInitialAuctionTimerState = ( + artwork: NonNullable +) => { + if (!artwork.isInAuction) { + return null + } + + const sale = artwork.sale + + return currentTimerState({ + isPreview: sale?.isPreview || undefined, + isClosed: sale?.isClosed || undefined, + liveStartsAt: sale?.liveStartAt || undefined, + }) +} + +export const InfiniteDiscoveryBottomSheetFooterSkeleton: FC = (props) => { + const { bottom } = useSafeAreaInsets() + const color = useColor() + return ( + + + + + $1000.00 + + + + + + + + ) +} diff --git a/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheetTabs.tsx b/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheetTabs.tsx new file mode 100644 index 00000000000..cbca804ce33 --- /dev/null +++ b/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheetTabs.tsx @@ -0,0 +1,41 @@ +import { Skeleton, SkeletonText, Tabs } from "@artsy/palette-mobile" +import { InfiniteDiscoveryBottomSheetTabsQuery } from "__generated__/InfiniteDiscoveryBottomSheetTabsQuery.graphql" +import { + InfiniteDiscoveryAboutTheWorkTab, + InfiniteDiscoveryAboutTheWorkTabSkeleton, +} from "app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryAboutTheWorkTab" +import { InfiniteDiscoveryOtherWorksTab } from "app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryOtherWorksTab" +import { FC } from "react" +import { PreloadedQuery } from "react-relay" + +interface InfiniteDiscoveryOtherWorksTabProps { + queryRef: PreloadedQuery +} + +export const InfiniteDiscoveryTabs: FC = ({ queryRef }) => { + return ( + + + + + + + + + ) +} + +export const InfiniteDiscoveryTabsSkeleton: FC = () => { + return ( + + + + + + + Other works + + + + ) +} diff --git a/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryOtherWorksTab.tsx b/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryOtherWorksTab.tsx new file mode 100644 index 00000000000..03680f29910 --- /dev/null +++ b/src/app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryOtherWorksTab.tsx @@ -0,0 +1,6 @@ +import { Text } from "@artsy/palette-mobile" +import { FC } from "react" + +export const InfiniteDiscoveryOtherWorksTab: FC = () => { + return Other works +} diff --git a/src/app/Scenes/InfiniteDiscovery/Components/__tests__/InfiniteDiscoveryAboutTheWorkTab.tests.tsx b/src/app/Scenes/InfiniteDiscovery/Components/__tests__/InfiniteDiscoveryAboutTheWorkTab.tests.tsx new file mode 100644 index 00000000000..5c7f3b21c81 --- /dev/null +++ b/src/app/Scenes/InfiniteDiscovery/Components/__tests__/InfiniteDiscoveryAboutTheWorkTab.tests.tsx @@ -0,0 +1,154 @@ +import { screen, waitForElementToBeRemoved } from "@testing-library/react-native" +import { InfiniteDiscoveryAboutTheWorkTabTestQuery } from "__generated__/InfiniteDiscoveryAboutTheWorkTabTestQuery.graphql" +import { AboutTheWorkTab } from "app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryAboutTheWorkTab" +import { setupTestWrapper } from "app/utils/tests/setupTestWrapper" +import { Suspense } from "react" +import { Text } from "react-native" +import { graphql } from "react-relay" + +describe("AboutTheWorkTab", () => { + const { renderWithRelay } = setupTestWrapper({ + Component: ({ artwork, me }: any) => { + return ( + Loading...}> + + + ) + }, + query: graphql` + query InfiniteDiscoveryAboutTheWorkTabTestQuery @relay_test_operation { + artwork(id: "artwork-id") { + ...InfiniteDiscoveryAboutTheWorkTab_artwork + } + me { + ...MyProfileEditModal_me + ...useSendInquiry_me + } + } + `, + preloaded: true, + }) + + it("renders all artwork details when available", () => { + const { mockResolveLastOperation } = renderWithRelay() + + mockResolveLastOperation({ Artwork: () => artwork }) + + expect(screen.getByText("Oil on canvas")).toBeOnTheScreen() + expect(screen.getByText("20 × 24 in | 50.8 × 61 cm")).toBeOnTheScreen() + expect(screen.getByText("Unique")).toBeOnTheScreen() + expect(screen.getByText("Painting")).toBeOnTheScreen() + expect(screen.getByText("Excellent condition")).toBeOnTheScreen() + expect(screen.getByText("Signed and dated lower right")).toBeOnTheScreen() + expect(screen.getByText("Includes gallery certificate")).toBeOnTheScreen() + expect(screen.getByText("Test Publisher")).toBeOnTheScreen() + expect(screen.getByText("Frame included")).toBeOnTheScreen() + }) + + it("renders certificate of authenticity section when available", () => { + const { mockResolveLastOperation } = renderWithRelay() + + mockResolveLastOperation({ Artwork: () => artwork }) + + expect(screen.getByText("Includes a Certificate of Authenticity")).toBeOnTheScreen() + }) + + it("does not render optional fields when data is missing", async () => { + const { mockResolveLastOperation } = renderWithRelay() + + mockResolveLastOperation({ + Artwork: () => ({ + ...artwork, + hasCertificateOfAuthenticity: false, + condition: null, + signatureInfo: null, + publisher: null, + }), + }) + mockResolveLastOperation({}) + + await waitForElementToBeRemoved(() => screen.queryByText("Loading...")) + + expect(screen.queryByText("Includes a Certificate of Authenticity")).not.toBeOnTheScreen() + expect(screen.queryByText("Condition")).not.toBeOnTheScreen() + expect(screen.queryByText("Signature")).not.toBeOnTheScreen() + expect(screen.queryByText("Publisher")).not.toBeOnTheScreen() + }) + + it("shows frame not included when isFramed is false", () => { + const { mockResolveLastOperation } = renderWithRelay() + + mockResolveLastOperation({ Artwork: () => ({ ...artwork, isFramed: false }) }) + + expect(screen.getByText("Frame not included")).toBeOnTheScreen() + }) + + describe("classification and authenticity section", () => { + it("shows attribution class when available", async () => { + const { mockResolveLastOperation } = renderWithRelay() + mockResolveLastOperation({ Artwork: () => artwork }) + mockResolveLastOperation({}) + + await waitForElementToBeRemoved(() => screen.queryByText("Loading...")) + + expect(screen.getByText("This is a unique work")).toBeOnTheScreen() + }) + + it("shows certificate icon when work has certificate and is not biddable", async () => { + const { mockResolveLastOperation } = renderWithRelay() + mockResolveLastOperation({ Artwork: () => artwork }) + mockResolveLastOperation({}) + + await waitForElementToBeRemoved(() => screen.queryByText("Loading...")) + + expect(screen.getByTestId("certificate-icon")).toBeOnTheScreen() + }) + + it("does not show certificate icon for biddable works", () => { + renderWithRelay({ Artwork: () => ({ ...artwork, isBiddable: true }) }) + + expect(screen.queryByTestId("certificate-icon")).not.toBeOnTheScreen() + }) + }) +}) + +const artwork = { + medium: "Oil on canvas", + dimensions: { + in: "20 × 24 in", + cm: "50.8 × 61 cm", + }, + attributionClass: { + name: "Unique", + shortArrayDescription: ["This is a", "unique work"], + }, + mediumType: { + name: "Painting", + }, + condition: { + displayText: "Excellent condition", + }, + signatureInfo: { + details: "Signed and dated lower right", + }, + certificateOfAuthenticity: { + details: "Includes gallery certificate", + }, + publisher: "Test Publisher", + isFramed: true, + hasCertificateOfAuthenticity: true, + isBiddable: false, + artists: [ + { + slug: "test-artist", + }, + ], + partner: { + name: "Test Gallery", + profile: { + icon: { + url: "https://example.com/image.jpg", + }, + }, + }, +} diff --git a/src/app/Scenes/InfiniteDiscovery/Components/__tests__/InfiniteDiscoveryBottomSheetFooter.tests.tsx b/src/app/Scenes/InfiniteDiscovery/Components/__tests__/InfiniteDiscoveryBottomSheetFooter.tests.tsx new file mode 100644 index 00000000000..473100a7486 --- /dev/null +++ b/src/app/Scenes/InfiniteDiscovery/Components/__tests__/InfiniteDiscoveryBottomSheetFooter.tests.tsx @@ -0,0 +1,26 @@ +import { InfiniteDiscoveryBottomSheetFooter } from "app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheetFooter" +import { setupTestWrapper } from "app/utils/tests/setupTestWrapper" +import { graphql } from "react-relay" + +describe("InfiniteDiscoveryBottomSheetFooter", () => { + const { renderWithRelay } = setupTestWrapper({ + Component: ({ artwork, me }: any) => { + // @ts-expect-error TODO: fix shared value prop + return + }, + query: graphql` + query InfiniteDiscoveryBottomSheetFooterTestQuery @relay_test_operation { + artwork(id: "artwork-id") { + ...InfiniteDiscoveryBottomSheetFooter_artwork + } + me { + ...InfiniteDiscoveryBottomSheetFooter_me + } + } + `, + }) + + it.skip("renders", async () => { + renderWithRelay() + }) +}) diff --git a/src/app/Scenes/InfiniteDiscovery/InfiniteDiscovery.tsx b/src/app/Scenes/InfiniteDiscovery/InfiniteDiscovery.tsx index 1fd31e82a4c..d5042076e3e 100644 --- a/src/app/Scenes/InfiniteDiscovery/InfiniteDiscovery.tsx +++ b/src/app/Scenes/InfiniteDiscovery/InfiniteDiscovery.tsx @@ -11,7 +11,8 @@ import { useTheme, } from "@artsy/palette-mobile" import { FancySwiper } from "app/Components/FancySwiper/FancySwiper" -import { navigate } from "app/system/navigation/navigate" +import { InfiniteDiscoveryBottomSheet } from "app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheet" +import { goBack } from "app/system/navigation/navigate" import { extractNodes } from "app/utils/extractNodes" import { NoFallback, withSuspense } from "app/utils/hooks/withSuspense" import { useMemo, useState } from "react" @@ -45,7 +46,7 @@ export const InfiniteDiscovery: React.FC = () => { } const handleExitPressed = () => { - navigate("/home-view") + goBack() } const handleSwipedRight = () => { @@ -122,6 +123,8 @@ export const InfiniteDiscovery: React.FC = () => { onSwipeRight={handleSwipedRight} onSwipeLeft={handleSwipedLeft} /> + + ) diff --git a/src/app/Scenes/InfiniteDiscovery/__tests__/InfiniteDiscovery.tests.tsx b/src/app/Scenes/InfiniteDiscovery/__tests__/InfiniteDiscovery.tests.tsx index 919c127ca4b..cf30989cea9 100644 --- a/src/app/Scenes/InfiniteDiscovery/__tests__/InfiniteDiscovery.tests.tsx +++ b/src/app/Scenes/InfiniteDiscovery/__tests__/InfiniteDiscovery.tests.tsx @@ -3,14 +3,15 @@ import { infiniteDiscoveryQuery, InfiniteDiscoveryQueryRenderer, } from "app/Scenes/InfiniteDiscovery/InfiniteDiscovery" -import { navigate } from "app/system/navigation/navigate" +import { goBack } from "app/system/navigation/navigate" import { setupTestWrapper } from "app/utils/tests/setupTestWrapper" jest.mock("app/system/navigation/navigate") +jest.mock("app/Scenes/InfiniteDiscovery/Components/InfiniteDiscoveryBottomSheet", () => ({ + InfiniteDiscoveryBottomSheet: () => null, +})) describe("InfiniteDiscovery", () => { - const mockNavigate = navigate as jest.Mock - beforeEach(() => { jest.clearAllMocks() }) @@ -54,7 +55,7 @@ describe("InfiniteDiscovery", () => { }) renderWithRelay(marketingCollection) fireEvent.press(screen.getByText("Exit")) - expect(mockNavigate).toHaveBeenCalledWith("/home-view") + expect(goBack).toHaveBeenCalledTimes(1) }) }) diff --git a/src/app/Scenes/Partner/Components/PartnerFollowButton.tests.tsx b/src/app/Scenes/Partner/Components/PartnerFollowButton.tests.tsx index fda50f630d5..26d9efe6613 100644 --- a/src/app/Scenes/Partner/Components/PartnerFollowButton.tests.tsx +++ b/src/app/Scenes/Partner/Components/PartnerFollowButton.tests.tsx @@ -9,7 +9,7 @@ describe("PartnerFollowButton", () => { query: graphql` query PartnerFollowButtonTestQuery @relay_test_operation { partner(id: "white-cube") { - ...PartnerFollowButton_partner + ...PartnerFollowButton_deprecated_partner } } `, diff --git a/src/app/Scenes/Partner/Components/PartnerFollowButton.tsx b/src/app/Scenes/Partner/Components/PartnerFollowButton.tsx index 67b02c3f84a..a096c5ab292 100644 --- a/src/app/Scenes/Partner/Components/PartnerFollowButton.tsx +++ b/src/app/Scenes/Partner/Components/PartnerFollowButton.tsx @@ -1,12 +1,12 @@ import { ButtonProps, FollowButton } from "@artsy/palette-mobile" import { PartnerFollowButtonFollowMutation } from "__generated__/PartnerFollowButtonFollowMutation.graphql" -import { PartnerFollowButton_partner$data } from "__generated__/PartnerFollowButton_partner.graphql" +import { PartnerFollowButton_deprecated_partner$data } from "__generated__/PartnerFollowButton_deprecated_partner.graphql" import { Schema, Track, track as _track } from "app/utils/track" import React from "react" import { commitMutation, createFragmentContainer, graphql, RelayProp } from "react-relay" interface Props { - partner: PartnerFollowButton_partner$data + partner: PartnerFollowButton_deprecated_partner$data relay: RelayProp size?: ButtonProps["size"] } @@ -17,6 +17,10 @@ interface State { const track: Track = _track +/** + * @deprecated in favor of PartnerFollowButtonWithSuspense + * @reason moving away personalized data into different queries + */ @track() export class PartnerFollowButton extends React.Component { state = { isFollowedChanging: false } @@ -107,7 +111,7 @@ export class PartnerFollowButton extends React.Component { export const PartnerFollowButtonFragmentContainer = createFragmentContainer(PartnerFollowButton, { partner: graphql` - fragment PartnerFollowButton_partner on Partner { + fragment PartnerFollowButton_deprecated_partner on Partner { internalID slug profile { diff --git a/src/app/Scenes/Partner/Components/PartnerHeader.tsx b/src/app/Scenes/Partner/Components/PartnerHeader.tsx index befb136e515..2e41ccaf4a4 100644 --- a/src/app/Scenes/Partner/Components/PartnerHeader.tsx +++ b/src/app/Scenes/Partner/Components/PartnerHeader.tsx @@ -67,7 +67,7 @@ export const PartnerHeaderContainer = createFragmentContainer(PartnerHeader, { counts { eligibleArtworks } - ...PartnerFollowButton_partner + ...PartnerFollowButton_deprecated_partner } `, }) diff --git a/src/app/utils/mutations/useFollowArtist.ts b/src/app/utils/mutations/useFollowArtist.ts index 5ffae9f0444..39b18d229bb 100644 --- a/src/app/utils/mutations/useFollowArtist.ts +++ b/src/app/utils/mutations/useFollowArtist.ts @@ -1,10 +1,12 @@ +import { useFollowArtistMutation } from "__generated__/useFollowArtistMutation.graphql" import { graphql, useMutation } from "react-relay" export const useFollowArtist = () => { - return useMutation(graphql` + return useMutation(graphql` mutation useFollowArtistMutation($input: FollowArtistInput!) { followArtist(input: $input) { artist { + ...ArtistFollowButton_artist ...RelatedArtistsRailCell_relatedArtist } } diff --git a/src/app/utils/mutations/useFollowProfile.ts b/src/app/utils/mutations/useFollowProfile.ts index af304b6e9b1..d235f9cd811 100644 --- a/src/app/utils/mutations/useFollowProfile.ts +++ b/src/app/utils/mutations/useFollowProfile.ts @@ -3,32 +3,32 @@ import { useMutation, graphql } from "react-relay" export interface FollowProfileOptions { id: string internalID: string - isFollowd: boolean | null - onCompleted?: (isFollowd: boolean) => void + isFollowed: boolean | null + onCompleted?: (isFollowed: boolean) => void onError?: () => void } export const useFollowProfile = ({ id, internalID, - isFollowd, + isFollowed, onCompleted, onError, }: FollowProfileOptions) => { const [commit, isInFlight] = useMutation(Mutation) - const nextFollowdState = !isFollowd + const nextFollowedState = !isFollowed const followProfile = () => { commit({ variables: { input: { profileID: internalID, - unfollow: !!isFollowd, + unfollow: !!isFollowed, }, }, onCompleted: () => { - onCompleted?.(nextFollowdState) + onCompleted?.(nextFollowedState) }, onError, optimisticResponse: { @@ -36,13 +36,13 @@ export const useFollowProfile = ({ profile: { id, internalID, - isFollowed: nextFollowdState, + isFollowed: nextFollowedState, }, }, }, optimisticUpdater: (store) => { const profile = store.get(id) - profile?.setValue(nextFollowdState, "isFollowd") + profile?.setValue(nextFollowedState, "isFollowed") }, }) }