import { Checkbox, Form, Input, MarkdownInput } from './form' import Row from 'react-bootstrap/Row' import Col from 'react-bootstrap/Col' import InputGroup from 'react-bootstrap/InputGroup' import Image from 'react-bootstrap/Image' import BootstrapForm from 'react-bootstrap/Form' import Alert from 'react-bootstrap/Alert' import { useCallback, useEffect, useState } from 'react' import Info from './info' import AccordianItem from './accordian-item' import styles from '@/styles/post.module.css' import { useLazyQuery, gql, useMutation } from '@apollo/client' import { useRouter } from 'next/router' import Link from 'next/link' import { usePrice } from './price' import Avatar from './avatar' import { jobSchema } from '@/lib/validate' import { MAX_TITLE_LENGTH, MEDIA_URL } from '@/lib/constants' import { useToast } from './toast' import { toastUpsertSuccessMessages } from '@/lib/form' import { ItemButtonBar } from './post' import { useFormikContext } from 'formik' function satsMin2Mo (minute) { return minute * 30 * 24 * 60 } function PriceHint ({ monthly }) { const { price, fiatSymbol } = usePrice() if (!price || !monthly) { return null } const fixed = (n, f) => Number.parseFloat(n).toFixed(f) const fiat = fixed((price / 100000000) * monthly, 0) return {monthly} sats/mo which is {fiatSymbol}{fiat}/mo } // need to recent list items export default function JobForm ({ item, sub }) { const storageKeyPrefix = item ? undefined : `${sub.name}-job` const router = useRouter() const toaster = useToast() const [logoId, setLogoId] = useState(item?.uploadId) const [upsertJob] = useMutation(gql` mutation upsertJob($sub: String!, $id: ID, $title: String!, $company: String!, $location: String, $remote: Boolean, $text: String!, $url: String!, $maxBid: Int!, $status: String, $logo: Int, $hash: String, $hmac: String) { upsertJob(sub: $sub, id: $id, title: $title, company: $company, location: $location, remote: $remote, text: $text, url: $url, maxBid: $maxBid, status: $status, logo: $logo, hash: $hash, hmac: $hmac) { id deleteScheduledAt reminderScheduledAt } }` ) const onSubmit = useCallback( async ({ maxBid, start, stop, ...values }) => { let status if (start) { status = 'ACTIVE' } else if (stop) { status = 'STOPPED' } const { data, error } = await upsertJob({ variables: { id: item?.id, sub: item?.subName || sub?.name, maxBid: Number(maxBid), status, logo: Number(logoId), ...values } }) if (error) { throw new Error({ message: error.toString() }) } if (item) { await router.push(`/items/${item.id}`) } else { await router.push(`/~${sub.name}/recent`) } toastUpsertSuccessMessages(toaster, data, 'upsertJob', !!item, values.text) }, [upsertJob, router, logoId] ) return ( <>
> ) } const FormStatus = { DIRTY: 'dirty', ERROR: 'error' } function PromoteJob ({ item, sub }) { const formik = useFormikContext() const [show, setShow] = useState(false) const [monthly, setMonthly] = useState(satsMin2Mo(item?.maxBid || 0)) const [getAuctionPosition, { data }] = useLazyQuery(gql` query AuctionPosition($id: ID, $bid: Int!) { auctionPosition(sub: "${item?.subName || sub?.name}", id: $id, bid: $bid) }`, { fetchPolicy: 'cache-and-network' }) const position = data?.auctionPosition useEffect(() => { const initialMaxBid = Number(item?.maxBid) || 0 getAuctionPosition({ variables: { id: item?.id, bid: initialMaxBid } }) setMonthly(satsMin2Mo(initialMaxBid)) }, []) useEffect(() => { if (formik?.values?.maxBid !== 0) { setShow(FormStatus.DIRTY) } }, [formik?.values]) useEffect(() => { const hasMaxBidError = !!formik?.errors?.maxBid // if it's open we don't want to collapse on submit setShow(show => show || (hasMaxBidError && formik?.isSubmitting && FormStatus.ERROR)) }, [formik?.isSubmitting]) return (