import {useMemo, useState} from 'react'
import {View} from 'react-native'
import {
type AppBskyActorDefs,
AppBskyActorStatus,
type AppBskyEmbedExternal,
} from '@atproto/api'
import {msg, Trans} from '@lingui/macro'
import {useLingui} from '@lingui/react'
import {differenceInMinutes} from 'date-fns'
import {cleanError} from '#/lib/strings/errors'
import {definitelyUrl} from '#/lib/strings/url-helpers'
import {useTickEveryMinute} from '#/state/shell'
import {atoms as a, platform, useTheme, web} from '#/alf'
import {Admonition} from '#/components/Admonition'
import {Button, ButtonIcon, ButtonText} from '#/components/Button'
import * as Dialog from '#/components/Dialog'
import * as TextField from '#/components/forms/TextField'
import {Clock_Stroke2_Corner0_Rounded as ClockIcon} from '#/components/icons/Clock'
import {Warning_Stroke2_Corner0_Rounded as WarningIcon} from '#/components/icons/Warning'
import {Loader} from '#/components/Loader'
import {Text} from '#/components/Typography'
import {LinkPreview} from './LinkPreview'
import {
useLiveLinkMetaQuery,
useRemoveLiveStatusMutation,
useUpsertLiveStatusMutation,
} from './queries'
import {displayDuration, useDebouncedValue} from './utils'
export function EditLiveDialog({
control,
status,
embed,
}: {
control: Dialog.DialogControlProps
status: AppBskyActorDefs.StatusView
embed: AppBskyEmbedExternal.View
}) {
return (
)
}
function DialogInner({
status,
embed,
}: {
status: AppBskyActorDefs.StatusView
embed: AppBskyEmbedExternal.View
}) {
const control = Dialog.useDialogContext()
const {_, i18n} = useLingui()
const t = useTheme()
const [liveLink, setLiveLink] = useState(embed.external.uri)
const [liveLinkError, setLiveLinkError] = useState('')
const tick = useTickEveryMinute()
const liveLinkUrl = definitelyUrl(liveLink)
const debouncedUrl = useDebouncedValue(liveLinkUrl, 500)
const isDirty = liveLinkUrl !== embed.external.uri
const {
data: linkMeta,
isSuccess: hasValidLinkMeta,
isLoading: linkMetaLoading,
error: linkMetaError,
} = useLiveLinkMetaQuery(debouncedUrl)
const record = useMemo(() => {
if (!AppBskyActorStatus.isRecord(status.record)) return null
const validation = AppBskyActorStatus.validateRecord(status.record)
if (validation.success) {
return validation.value
}
return null
}, [status])
const {
mutate: goLive,
isPending: isGoingLive,
error: goLiveError,
} = useUpsertLiveStatusMutation(
record?.durationMinutes ?? 0,
linkMeta,
record?.createdAt,
)
const {
mutate: removeLiveStatus,
isPending: isRemovingLiveStatus,
error: removeLiveStatusError,
} = useRemoveLiveStatusMutation()
const {minutesUntilExpiry, expiryDateTime} = useMemo(() => {
tick!
const expiry = new Date(status.expiresAt ?? new Date())
return {
expiryDateTime: expiry,
minutesUntilExpiry: differenceInMinutes(expiry, new Date()),
}
}, [tick, status.expiresAt])
const submitDisabled =
isGoingLive ||
!hasValidLinkMeta ||
debouncedUrl !== liveLinkUrl ||
isRemovingLiveStatus
return (
You are Live
{typeof record?.durationMinutes === 'number' ? (
Expires in {displayDuration(i18n, minutesUntilExpiry)} at{' '}
{i18n.date(expiryDateTime, {
hour: 'numeric',
minute: '2-digit',
hour12: true,
})}
) : (
No expiry set
)}
Live link
setLiveLinkError('')}
onBlur={() => {
if (!definitelyUrl(liveLink)) {
setLiveLinkError('Invalid URL')
}
}}
returnKeyType="done"
autoCapitalize="none"
autoComplete="url"
autoCorrect={false}
onSubmitEditing={() => {
if (isDirty && !submitDisabled) {
goLive()
}
}}
/>
{(liveLinkError || linkMetaError) && (
{liveLinkError ? (
This is not a valid link
) : (
cleanError(linkMetaError)
)}
)}
{goLiveError && (
{cleanError(goLiveError)}
)}
{removeLiveStatusError && (
{cleanError(removeLiveStatusError)}
)}
{isDirty ? (
) : (
)}
)
}