import { useRouter } from 'next/router' import { useToast } from './toast' import { usePaidMutation, paidActionCacheMods } from './use-paid-mutation' import useCrossposter from './use-crossposter' import { useCallback } from 'react' import { normalizeForwards, toastUpsertSuccessMessages } from '@/lib/form' import { RETRY_PAID_ACTION } from '@/fragments/paidAction' import gql from 'graphql-tag' import { USER_ID } from '@/lib/constants' import { useMe } from './me' // this is intented to be compatible with upsert item mutations // so that it can be reused for all post types and comments and we don't have // to maintain several copies of the same code // it's a bit much for an abstraction ... but it makes it easy to modify item-payment UX // and other side effects like crossposting and redirection // ... or I just spent too much time in this code and this is overcooked export default function useItemSubmit (mutation, { item, sub, onSuccessfulSubmit, navigateOnSubmit = true, extraValues = {}, paidMutationOptions = { } } = {}) { const router = useRouter() const toaster = useToast() const crossposter = useCrossposter() const [upsertItem] = usePaidMutation(mutation) const { me } = useMe() return useCallback( async ({ boost, crosspost, title, options, bounty, status, ...values }, { resetForm }) => { if (options) { // remove existing poll options since else they will be appended as duplicates options = options.slice(item?.poll?.options?.length || 0).filter(o => o.trim().length > 0) } const hmacEdit = item?.id && Number(item.user.id) === USER_ID.anon && !me if (hmacEdit) { const invParams = window.localStorage.getItem(`item:${item.id}:hash:hmac`) if (invParams) { const [hash, hmac] = invParams.split(':') values.hash = hash values.hmac = hmac } } const { data, error, payError } = await upsertItem({ variables: { id: item?.id, sub: item?.subName || sub?.name, boost: boost ? Number(boost) : item?.boost ? Number(item.boost) : undefined, bounty: bounty ? Number(bounty) : undefined, status: status === 'STOPPED' ? 'STOPPED' : 'ACTIVE', title: title?.trim(), options, ...values, forward: normalizeForwards(values.forward), ...extraValues }, // if not a comment, we want the qr to persist on navigation persistOnNavigate: navigateOnSubmit, ...paidMutationOptions, onPayError: (e, cache, { data }) => { paidActionCacheMods.onPayError(e, cache, { data }) paidMutationOptions?.onPayError?.(e, cache, { data }) }, onPaid: (cache, { data }) => { paidActionCacheMods.onPaid(cache, { data }) paidMutationOptions?.onPaid?.(cache, { data }) }, onCompleted: (data) => { onSuccessfulSubmit?.(data, { resetForm }) paidMutationOptions?.onCompleted?.(data) saveItemInvoiceHmac(data) } }) if (error) throw error if (payError) return // we don't know the mutation name, so we have to extract the result const response = Object.values(data)[0] const postId = response?.result?.id if (crosspost && postId) { await crossposter(postId) } toastUpsertSuccessMessages(toaster, data, Object.keys(data)[0], values.text) // if we're not a comment, we want to redirect after the mutation if (navigateOnSubmit) { if (item) { await router.push(`/items/${item.id}`) } else { await router.push(sub ? `/~${sub.name}/recent` : '/recent') } } }, [me, upsertItem, router, crossposter, item, sub, onSuccessfulSubmit, navigateOnSubmit, extraValues, paidMutationOptions] ) } export function useRetryCreateItem ({ id }) { const [retryPaidAction] = usePaidMutation( RETRY_PAID_ACTION, { ...paidActionCacheMods, update: (cache, { data }) => { const response = Object.values(data)[0] if (!response?.invoice) return cache.modify({ id: `Item:${id}`, fields: { // this is a bit of a hack just to update the reference to the new invoice invoice: () => cache.writeFragment({ id: `Invoice:${response.invoice.id}`, fragment: gql` fragment _ on Invoice { bolt11 } `, data: { bolt11: response.invoice.bolt11 } }) }, optimistic: true }) paidActionCacheMods?.update?.(cache, { data }) } } ) return retryPaidAction } function saveItemInvoiceHmac (mutationData) { const response = Object.values(mutationData)[0] if (!response?.invoice) return const id = response.result.id const { hash, hmac } = response.invoice if (id && hash && hmac) { window.localStorage.setItem(`item:${id}:hash:hmac`, `${hash}:${hmac}`) } }