Co-authored-by: surfdude29 <149612116+surfdude29@users.noreply.github.com> Co-authored-by: Adrov Igor <nucleartux@gmail.com>
107 lines
2.7 KiB
TypeScript
107 lines
2.7 KiB
TypeScript
import React from 'react'
|
|
import {
|
|
AppBskyActorDefs,
|
|
AppBskyEmbedRecord,
|
|
AppBskyRichtextFacet,
|
|
ModerationDecision,
|
|
} from '@atproto/api'
|
|
import {msg} from '@lingui/macro'
|
|
import {useLingui} from '@lingui/react'
|
|
|
|
import {useNonReactiveCallback} from '#/lib/hooks/useNonReactiveCallback'
|
|
import * as Toast from '#/view/com/util/Toast'
|
|
|
|
export interface ComposerOptsPostRef {
|
|
uri: string
|
|
cid: string
|
|
text: string
|
|
author: AppBskyActorDefs.ProfileViewBasic
|
|
embed?: AppBskyEmbedRecord.ViewRecord['embed']
|
|
moderation?: ModerationDecision
|
|
}
|
|
export interface ComposerOptsQuote {
|
|
uri: string
|
|
cid: string
|
|
text: string
|
|
facets?: AppBskyRichtextFacet.Main[]
|
|
indexedAt: string
|
|
author: AppBskyActorDefs.ProfileViewBasic
|
|
embeds?: AppBskyEmbedRecord.ViewRecord['embeds']
|
|
}
|
|
export interface ComposerOpts {
|
|
replyTo?: ComposerOptsPostRef
|
|
onPost?: (postUri: string | undefined) => void
|
|
quote?: ComposerOptsQuote
|
|
quoteCount?: number
|
|
mention?: string // handle of user to mention
|
|
openEmojiPicker?: (pos: DOMRect | undefined) => void
|
|
text?: string
|
|
imageUris?: {uri: string; width: number; height: number}[]
|
|
}
|
|
|
|
type StateContext = ComposerOpts | undefined
|
|
type ControlsContext = {
|
|
openComposer: (opts: ComposerOpts) => void
|
|
closeComposer: () => boolean
|
|
}
|
|
|
|
const stateContext = React.createContext<StateContext>(undefined)
|
|
const controlsContext = React.createContext<ControlsContext>({
|
|
openComposer(_opts: ComposerOpts) {},
|
|
closeComposer() {
|
|
return false
|
|
},
|
|
})
|
|
|
|
export function Provider({children}: React.PropsWithChildren<{}>) {
|
|
const {_} = useLingui()
|
|
const [state, setState] = React.useState<StateContext>()
|
|
|
|
const openComposer = useNonReactiveCallback((opts: ComposerOpts) => {
|
|
const author = opts.replyTo?.author || opts.quote?.author
|
|
const isBlocked = Boolean(
|
|
author &&
|
|
(author.viewer?.blocking ||
|
|
author.viewer?.blockedBy ||
|
|
author.viewer?.blockingByList),
|
|
)
|
|
if (isBlocked) {
|
|
Toast.show(
|
|
_(msg`Cannot interact with a blocked user`),
|
|
'exclamation-circle',
|
|
)
|
|
} else {
|
|
setState(opts)
|
|
}
|
|
})
|
|
|
|
const closeComposer = useNonReactiveCallback(() => {
|
|
let wasOpen = !!state
|
|
setState(undefined)
|
|
return wasOpen
|
|
})
|
|
|
|
const api = React.useMemo(
|
|
() => ({
|
|
openComposer,
|
|
closeComposer,
|
|
}),
|
|
[openComposer, closeComposer],
|
|
)
|
|
|
|
return (
|
|
<stateContext.Provider value={state}>
|
|
<controlsContext.Provider value={api}>
|
|
{children}
|
|
</controlsContext.Provider>
|
|
</stateContext.Provider>
|
|
)
|
|
}
|
|
|
|
export function useComposerState() {
|
|
return React.useContext(stateContext)
|
|
}
|
|
|
|
export function useComposerControls() {
|
|
return React.useContext(controlsContext)
|
|
}
|