Merge branch 'ten-milly' into main

zio/dev^2
Paul Frazee 2024-09-13 08:57:41 -07:00
commit c7231537f1
25 changed files with 1183 additions and 125 deletions

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"><path fill="#000" fill-rule="evenodd" d="M12 3a1 1 0 0 1 1 1v8.086l1.793-1.793a1 1 0 1 1 1.414 1.414l-3.5 3.5a1 1 0 0 1-1.414 0l-3.5-3.5a1 1 0 1 1 1.414-1.414L11 12.086V4a1 1 0 0 1 1-1ZM4 14a1 1 0 0 1 1 1v4h14v-4a1 1 0 1 1 2 0v5a1 1 0 0 1-1 1H4a1 1 0 0 1-1-1v-5a1 1 0 0 1 1-1Z" clip-rule="evenodd"/></svg>

After

Width:  |  Height:  |  Size: 378 B

View File

@ -57,6 +57,7 @@ import * as Toast from '#/view/com/util/Toast'
import {Shell} from '#/view/shell'
import {ThemeProvider as Alf} from '#/alf'
import {useColorModeTheme} from '#/alf/util/useColorModeTheme'
import {NuxDialogs} from '#/components/dialogs/nuxs'
import {useStarterPackEntry} from '#/components/hooks/useStarterPackEntry'
import {Provider as IntentDialogProvider} from '#/components/intents/IntentDialogs'
import {Provider as PortalProvider} from '#/components/Portal'
@ -131,6 +132,7 @@ function InnerApp() {
style={s.h100pct}>
<TestCtrls />
<Shell />
<NuxDialogs />
</GestureHandlerRootView>
</ProgressGuideProvider>
</MutedThreadsProvider>

View File

@ -46,6 +46,7 @@ import {ToastContainer} from '#/view/com/util/Toast.web'
import {Shell} from '#/view/shell/index'
import {ThemeProvider as Alf} from '#/alf'
import {useColorModeTheme} from '#/alf/util/useColorModeTheme'
import {NuxDialogs} from '#/components/dialogs/nuxs'
import {useStarterPackEntry} from '#/components/hooks/useStarterPackEntry'
import {Provider as IntentDialogProvider} from '#/components/intents/IntentDialogs'
import {Provider as PortalProvider} from '#/components/Portal'
@ -113,6 +114,7 @@ function InnerApp() {
<SafeAreaProvider>
<ProgressGuideProvider>
<Shell />
<NuxDialogs />
</ProgressGuideProvider>
</SafeAreaProvider>
</MutedThreadsProvider>

View File

@ -18,9 +18,17 @@ export * from '#/alf/util/themeSelector'
export const Context = React.createContext<{
themeName: ThemeName
theme: Theme
themes: ReturnType<typeof createThemes>
}>({
themeName: 'light',
theme: defaultTheme,
themes: createThemes({
hues: {
primary: BLUE_HUE,
negative: RED_HUE,
positive: GREEN_HUE,
},
}),
})
export function ThemeProvider({
@ -42,18 +50,22 @@ export function ThemeProvider({
<Context.Provider
value={React.useMemo(
() => ({
themes,
themeName: themeName,
theme: theme,
}),
[theme, themeName],
[theme, themeName, themes],
)}>
{children}
</Context.Provider>
)
}
export function useTheme() {
return React.useContext(Context).theme
export function useTheme(theme?: ThemeName) {
const ctx = React.useContext(Context)
return React.useMemo(() => {
return theme ? ctx.themes[theme] : ctx.theme
}, [theme, ctx])
}
export function useBreakpoints() {

View File

@ -256,7 +256,7 @@ export const ScrollableInner = React.forwardRef<
borderTopLeftRadius: 40,
borderTopRightRadius: 40,
},
flatten(style),
style,
]}
contentContainerStyle={a.pb_4xl}
ref={ref}>

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,650 @@
import React from 'react'
import {View} from 'react-native'
import Animated, {FadeIn} from 'react-native-reanimated'
import ViewShot from 'react-native-view-shot'
import {Image} from 'expo-image'
import {requestMediaLibraryPermissionsAsync} from 'expo-image-picker'
import * as MediaLibrary from 'expo-media-library'
import {moderateProfile} from '@atproto/api'
import {msg, Trans} from '@lingui/macro'
import {useLingui} from '@lingui/react'
import {networkRetry} from '#/lib/async/retry'
import {getCanvas} from '#/lib/canvas'
import {shareUrl} from '#/lib/sharing'
import {sanitizeDisplayName} from '#/lib/strings/display-names'
import {sanitizeHandle} from '#/lib/strings/handles'
import {isIOS, isNative} from '#/platform/detection'
import {useModerationOpts} from '#/state/preferences/moderation-opts'
import {useProfileQuery} from '#/state/queries/profile'
import {useAgent, useSession} from '#/state/session'
import {useComposerControls} from 'state/shell'
import {formatCount} from '#/view/com/util/numeric/format'
import {Logomark} from '#/view/icons/Logomark'
import * as Toast from 'view/com/util/Toast'
import {
atoms as a,
ThemeProvider,
tokens,
useBreakpoints,
useTheme,
} from '#/alf'
import {Button, ButtonIcon, ButtonText} from '#/components/Button'
import * as Dialog from '#/components/Dialog'
import {useNuxDialogContext} from '#/components/dialogs/nuxs'
import {OnePercent} from '#/components/dialogs/nuxs/TenMillion/icons/OnePercent'
import {PointOnePercent} from '#/components/dialogs/nuxs/TenMillion/icons/PointOnePercent'
import {TenPercent} from '#/components/dialogs/nuxs/TenMillion/icons/TenPercent'
import {Divider} from '#/components/Divider'
import {GradientFill} from '#/components/GradientFill'
import {ArrowOutOfBox_Stroke2_Corner0_Rounded as Share} from '#/components/icons/ArrowOutOfBox'
import {Download_Stroke2_Corner0_Rounded as Download} from '#/components/icons/Download'
import {Image_Stroke2_Corner0_Rounded as ImageIcon} from '#/components/icons/Image'
import {Loader} from '#/components/Loader'
import {Text} from '#/components/Typography'
const DEBUG = false
const RATIO = 8 / 10
const WIDTH = 2000
const HEIGHT = WIDTH * RATIO
function getFontSize(count: number) {
const length = count.toString().length
if (length < 7) {
return 80
} else if (length < 5) {
return 100
} else {
return 70
}
}
function getPercentBadge(percent: number) {
if (percent <= 0.001) {
return PointOnePercent
} else if (percent <= 0.01) {
return OnePercent
} else if (percent <= 0.1) {
return TenPercent
}
return null
}
function Frame({children}: {children: React.ReactNode}) {
return (
<View
style={[
a.relative,
a.w_full,
a.overflow_hidden,
{
paddingTop: '80%',
},
]}>
{children}
</View>
)
}
export function TenMillion() {
const agent = useAgent()
const nuxDialogs = useNuxDialogContext()
const [userNumber, setUserNumber] = React.useState<number>(0)
const fetching = React.useRef(false)
React.useEffect(() => {
async function fetchUserNumber() {
const isBlueskyHosted = agent.sessionManager.pdsUrl
?.toString()
.includes('bsky.network')
if (isBlueskyHosted && agent.session?.accessJwt) {
const res = await fetch(
`https://bsky.social/xrpc/com.atproto.temp.getSignupNumber`,
{
headers: {
Authorization: `Bearer ${agent.session.accessJwt}`,
},
},
)
if (!res.ok) {
throw new Error('Network request failed')
}
const data = await res.json()
if (data.number) {
setUserNumber(data.number)
} else {
// should be rare
nuxDialogs.dismissActiveNux()
}
}
}
if (!fetching.current) {
fetching.current = true
networkRetry(3, fetchUserNumber).catch(() => {
nuxDialogs.dismissActiveNux()
})
}
}, [
agent.sessionManager.pdsUrl,
agent.session?.accessJwt,
setUserNumber,
nuxDialogs.dismissActiveNux,
nuxDialogs,
])
return userNumber ? <TenMillionInner userNumber={userNumber} /> : null
}
export function TenMillionInner({userNumber}: {userNumber: number}) {
const t = useTheme()
const lightTheme = useTheme('light')
const {_, i18n} = useLingui()
const control = Dialog.useDialogControl()
const {gtMobile} = useBreakpoints()
const {openComposer} = useComposerControls()
const {currentAccount} = useSession()
const {
isLoading: isProfileLoading,
data: profile,
error: profileError,
} = useProfileQuery({
did: currentAccount!.did,
})
const moderationOpts = useModerationOpts()
const nuxDialogs = useNuxDialogContext()
const moderation = React.useMemo(() => {
return profile && moderationOpts
? moderateProfile(profile, moderationOpts)
: undefined
}, [profile, moderationOpts])
const [uri, setUri] = React.useState<string | null>(null)
const percent = userNumber / 10_000_000
const Badge = getPercentBadge(percent)
const isLoadingData = isProfileLoading || !moderation || !profile
const isLoadingImage = !uri
const error: string = React.useMemo(() => {
if (profileError) {
return _(
msg`Oh no! We weren't able to generate an image for you to share. Rest assured, we're glad you're here 🦋`,
)
}
return ''
}, [_, profileError])
/*
* Opening and closing
*/
React.useEffect(() => {
const timeout = setTimeout(() => {
control.open()
}, 3e3)
return () => {
clearTimeout(timeout)
}
}, [control])
const onClose = React.useCallback(() => {
nuxDialogs.dismissActiveNux()
}, [nuxDialogs])
/*
* Actions
*/
const sharePost = React.useCallback(() => {
if (uri) {
control.close(() => {
setTimeout(() => {
openComposer({
text: _(
msg`Bluesky now has over 10 million users, and I was #${i18n.number(
userNumber,
)}!`,
), // TODO
imageUris: [
{
uri,
width: WIDTH,
height: HEIGHT,
},
],
})
}, 1e3)
})
}
}, [_, i18n, control, openComposer, uri, userNumber])
const onNativeShare = React.useCallback(() => {
if (uri) {
control.close(() => {
shareUrl(uri)
})
}
}, [uri, control])
const onNativeDownload = React.useCallback(async () => {
if (uri) {
const res = await requestMediaLibraryPermissionsAsync()
if (!res) {
Toast.show(
_(
msg`You must grant access to your photo library to save the image.`,
),
'xmark',
)
return
}
try {
await MediaLibrary.createAssetAsync(uri)
Toast.show(_(msg`Image saved to your camera roll!`))
} catch (e: unknown) {
console.log(e)
Toast.show(_(msg`An error occurred while saving the image!`), 'xmark')
return
}
}
}, [_, uri])
const onWebDownload = React.useCallback(async () => {
if (uri) {
const canvas = await getCanvas(uri)
const imgHref = canvas
.toDataURL('image/png')
.replace('image/png', 'image/octet-stream')
const link = document.createElement('a')
link.setAttribute('download', `Bluesky 10M Users.png`)
link.setAttribute('href', imgHref)
link.click()
}
}, [uri])
/*
* Canvas stuff
*/
const imageRef = React.useRef<ViewShot>(null)
const captureInProgress = React.useRef(false)
const onCanvasReady = React.useCallback(async () => {
if (
imageRef.current &&
imageRef.current.capture &&
!captureInProgress.current
) {
captureInProgress.current = true
const uri = await imageRef.current.capture()
setUri(uri)
}
}, [setUri])
const canvas = isLoadingData ? null : (
<View
style={[
a.absolute,
a.overflow_hidden,
DEBUG
? {
width: 600,
height: 600 * RATIO,
}
: {
width: 1,
height: 1,
},
]}>
<View style={{width: 600}}>
<ThemeProvider theme="light">
<Frame>
<ViewShot
ref={imageRef}
options={{width: WIDTH, height: HEIGHT}}
style={[a.absolute, a.inset_0]}>
<View
onLayout={onCanvasReady}
style={[
a.absolute,
a.inset_0,
a.align_center,
a.justify_center,
{
top: -1,
bottom: -1,
left: -1,
right: -1,
paddingVertical: 48,
paddingHorizontal: 48,
},
]}>
<GradientFill gradient={tokens.gradients.bonfire} />
<View
style={[
a.flex_1,
a.w_full,
a.align_center,
a.justify_center,
a.rounded_md,
{
backgroundColor: 'white',
shadowRadius: 32,
shadowOpacity: 0.1,
elevation: 24,
shadowColor: tokens.gradients.bonfire.values[0][1],
},
]}>
<View
style={[
a.absolute,
a.px_xl,
a.py_xl,
{
top: 0,
left: 0,
},
]}>
<Logomark fill={t.palette.primary_500} width={36} />
</View>
{/* Centered content */}
<View
style={[
{
paddingBottom: isNative ? 0 : 24,
},
]}>
<Text
style={[
a.text_md,
a.font_bold,
a.text_center,
a.pb_sm,
lightTheme.atoms.text_contrast_medium,
]}>
<Trans>
Celebrating {formatCount(i18n, 10000000)} users
</Trans>{' '}
🎉
</Text>
<View style={[a.flex_row, a.align_start]}>
<Text
style={[
a.absolute,
{
color: t.palette.primary_500,
fontSize: 32,
fontWeight: '900',
width: 32,
top: isNative ? -10 : 0,
left: 0,
transform: [
{
translateX: -16,
},
],
},
]}>
#
</Text>
<Text
style={[
a.relative,
a.text_center,
{
fontStyle: 'italic',
fontSize: getFontSize(userNumber),
lineHeight: getFontSize(userNumber),
fontWeight: '900',
letterSpacing: -2,
},
]}>
{i18n.number(userNumber)}
</Text>
</View>
{Badge && (
<View
style={[
a.absolute,
{
width: 64,
height: 64,
top: isNative ? 75 : 85,
right: '5%',
transform: [
{
rotate: '8deg',
},
],
},
]}>
<Badge fill={t.palette.primary_500} />
</View>
)}
</View>
{/* End centered content */}
<View
style={[
a.absolute,
a.px_xl,
a.py_xl,
{
bottom: 0,
left: 0,
right: 0,
},
]}>
<View style={[a.flex_row, a.align_center, a.gap_sm]}>
{/*
<UserAvatar
size={36}
avatar={profile.avatar}
moderation={moderation.ui('avatar')}
onLoad={onCanvasReady}
/>
*/}
<View style={[a.gap_2xs, a.flex_1]}>
<Text
style={[
a.flex_1,
a.text_sm,
a.font_bold,
a.leading_tight,
{maxWidth: '60%'},
]}>
{sanitizeDisplayName(
profile.displayName ||
sanitizeHandle(profile.handle),
moderation.ui('displayName'),
)}
</Text>
<View
style={[a.flex_row, a.justify_between, a.gap_4xl]}>
<Text
numberOfLines={1}
style={[
a.flex_1,
a.text_sm,
a.font_semibold,
a.leading_snug,
lightTheme.atoms.text_contrast_medium,
]}>
{sanitizeHandle(profile.handle, '@')}
</Text>
{profile.createdAt && (
<Text
numberOfLines={1}
ellipsizeMode="head"
style={[
a.flex_1,
a.text_sm,
a.font_semibold,
a.leading_snug,
a.text_right,
lightTheme.atoms.text_contrast_low,
]}>
<Trans>
Joined{' '}
{i18n.date(profile.createdAt, {
month: 'short',
day: 'numeric',
year: 'numeric',
})}
</Trans>
</Text>
)}
</View>
</View>
</View>
</View>
</View>
</View>
</ViewShot>
</Frame>
</ThemeProvider>
</View>
</View>
)
return (
<Dialog.Outer control={control} onClose={onClose}>
<Dialog.ScrollableInner
label={_(msg`Ten Million`)}
style={[
{
padding: 0,
paddingTop: 0,
},
]}>
<View
style={[
a.rounded_md,
a.overflow_hidden,
isNative && {
borderTopLeftRadius: 40,
borderTopRightRadius: 40,
},
]}>
<Frame>
<View
style={[a.absolute, a.inset_0, a.align_center, a.justify_center]}>
<GradientFill gradient={tokens.gradients.bonfire} />
{error ? (
<View>
<Text
style={[
a.text_md,
a.leading_snug,
a.text_center,
a.pb_md,
{
maxWidth: 300,
},
]}>
(°°)
</Text>
<Text
style={[
a.text_xl,
a.font_bold,
a.leading_snug,
a.text_center,
{
maxWidth: 300,
},
]}>
{error}
</Text>
</View>
) : isLoadingData || isLoadingImage ? (
<Loader size="xl" fill="white" />
) : (
<Animated.View
entering={FadeIn.duration(150)}
style={[a.w_full, a.h_full]}>
<Image
accessibilityIgnoresInvertColors
source={{uri}}
style={[a.w_full, a.h_full]}
/>
</Animated.View>
)}
</View>
</Frame>
{canvas}
<View style={[gtMobile ? a.p_2xl : a.p_xl]}>
<Text
style={[
a.text_5xl,
a.leading_tight,
a.pb_lg,
{
fontWeight: '900',
},
]}>
<Trans>Thanks for being one of our first 10 million users.</Trans>
</Text>
<Text style={[a.leading_snug, a.text_lg, a.pb_xl]}>
<Trans>
Together, we're rebuilding the social internet. We're glad
you're here.
</Trans>
</Text>
<Divider />
<View
style={[
a.flex_row,
a.align_center,
a.justify_end,
a.gap_md,
a.pt_xl,
]}>
<Text style={[a.text_md, a.italic, t.atoms.text_contrast_medium]}>
<Trans>Brag a little!</Trans>
</Text>
<Button
disabled={isLoadingImage}
label={
isNative && isIOS
? _(msg`Share image externally`)
: _(msg`Download image`)
}
size="large"
variant="solid"
color="secondary"
shape="square"
onPress={
isNative
? isIOS
? onNativeShare
: onNativeDownload
: onWebDownload
}>
<ButtonIcon icon={isNative && isIOS ? Share : Download} />
</Button>
<Button
disabled={isLoadingImage}
label={_(msg`Share image in post`)}
size="large"
variant="solid"
color="primary"
onPress={sharePost}>
<ButtonText>{_(msg`Share`)}</ButtonText>
<ButtonIcon position="right" icon={ImageIcon} />
</Button>
</View>
</View>
</View>
<Dialog.Close />
</Dialog.ScrollableInner>
</Dialog.Outer>
)
}

View File

@ -0,0 +1,151 @@
import React from 'react'
import {useGate} from '#/lib/statsig/statsig'
import {logger} from '#/logger'
import {
Nux,
useNuxs,
useRemoveNuxsMutation,
useUpsertNuxMutation,
} from '#/state/queries/nuxs'
import {useSession} from '#/state/session'
import {isSnoozed, snooze, unsnooze} from '#/components/dialogs/nuxs/snoozing'
import {TenMillion} from '#/components/dialogs/nuxs/TenMillion'
import {IS_DEV} from '#/env'
type Context = {
activeNux: Nux | undefined
dismissActiveNux: () => void
}
/**
* If we fail to complete a NUX here, it may show again on next reload,
* or if prefs state updates. If `true`, this fallback ensures that the last
* shown NUX won't show again, at least for this session.
*
* This is temporary, and only needed for the 10Milly dialog rn, since we
* aren't snoozing that one in device storage.
*/
let __isSnoozedFallback = false
const queuedNuxs: {
id: Nux
enabled(props: {gate: ReturnType<typeof useGate>}): boolean
/**
* TEMP only intended for use with the 10Milly dialog rn, since there are no
* other NUX dialogs configured
*/
unsafe_disableSnooze: boolean
}[] = [
{
id: Nux.TenMillionDialog,
enabled({gate}) {
return gate('ten_million_dialog')
},
unsafe_disableSnooze: true,
},
]
const Context = React.createContext<Context>({
activeNux: undefined,
dismissActiveNux: () => {},
})
export function useNuxDialogContext() {
return React.useContext(Context)
}
export function NuxDialogs() {
const {hasSession} = useSession()
return hasSession ? <Inner /> : null
}
function Inner() {
const gate = useGate()
const {nuxs} = useNuxs()
const [snoozed, setSnoozed] = React.useState(() => {
return isSnoozed()
})
const [activeNux, setActiveNux] = React.useState<Nux | undefined>()
const {mutateAsync: upsertNux} = useUpsertNuxMutation()
const {mutate: removeNuxs} = useRemoveNuxsMutation()
const snoozeNuxDialog = React.useCallback(() => {
snooze()
setSnoozed(true)
}, [setSnoozed])
const dismissActiveNux = React.useCallback(() => {
if (!activeNux) return
setActiveNux(undefined)
}, [activeNux, setActiveNux])
if (IS_DEV && typeof window !== 'undefined') {
// @ts-ignore
window.clearNuxDialog = (id: Nux) => {
if (!IS_DEV || !id) return
removeNuxs([id])
unsnooze()
}
}
React.useEffect(() => {
if (__isSnoozedFallback) return
if (snoozed) return
if (!nuxs) return
for (const {id, enabled, unsafe_disableSnooze} of queuedNuxs) {
const nux = nuxs.find(nux => nux.id === id)
// check if completed first
if (nux && nux.completed) continue
// then check gate (track exposure)
if (!enabled({gate})) continue
// we have a winner
setActiveNux(id)
/**
* TEMP only intended for use with the 10Milly dialog rn, since there are no
* other NUX dialogs configured
*/
if (!unsafe_disableSnooze) {
// immediately snooze for a day
snoozeNuxDialog()
}
// immediately update remote data (affects next reload)
upsertNux({
id,
completed: true,
data: undefined,
}).catch(e => {
logger.error(`NUX dialogs: failed to upsert '${id}' NUX`, {
safeMessage: e.message,
})
/*
* TEMP only intended for use with the 10Milly dialog rn
*/
if (unsafe_disableSnooze) {
__isSnoozedFallback = true
}
})
break
}
}, [nuxs, snoozed, snoozeNuxDialog, upsertNux, gate])
const ctx = React.useMemo(() => {
return {
activeNux,
dismissActiveNux,
}
}, [activeNux, dismissActiveNux])
return (
<Context.Provider value={ctx}>
{activeNux === Nux.TenMillionDialog && <TenMillion />}
</Context.Provider>
)
}

View File

@ -0,0 +1,22 @@
import {simpleAreDatesEqual} from '#/lib/strings/time'
import {device} from '#/storage'
export function snooze() {
device.set(['lastNuxDialog'], new Date().toISOString())
}
export function unsnooze() {
device.set(['lastNuxDialog'], undefined)
}
export function isSnoozed() {
const lastNuxDialog = device.get(['lastNuxDialog'])
if (!lastNuxDialog) return false
const last = new Date(lastNuxDialog)
const now = new Date()
// already snoozed today
if (simpleAreDatesEqual(last, now)) {
return true
}
return false
}

View File

@ -0,0 +1,5 @@
import {createSinglePathSVG} from './TEMPLATE'
export const Download_Stroke2_Corner0_Rounded = createSinglePathSVG({
path: 'M12 3a1 1 0 0 1 1 1v8.086l1.793-1.793a1 1 0 1 1 1.414 1.414l-3.5 3.5a1 1 0 0 1-1.414 0l-3.5-3.5a1 1 0 1 1 1.414-1.414L11 12.086V4a1 1 0 0 1 1-1ZM4 14a1 1 0 0 1 1 1v4h14v-4a1 1 0 1 1 2 0v5a1 1 0 0 1-1 1H4a1 1 0 0 1-1-1v-5a1 1 0 0 1 1-1Z',
})

15
src/lib/canvas.ts 100644
View File

@ -0,0 +1,15 @@
export const getCanvas = (base64: string): Promise<HTMLCanvasElement> => {
return new Promise(resolve => {
const image = new Image()
image.onload = () => {
const canvas = document.createElement('canvas')
canvas.width = image.width
canvas.height = image.height
const ctx = canvas.getContext('2d')
ctx?.drawImage(image, 0, 0)
resolve(canvas)
}
image.src = base64
})
}

View File

@ -71,7 +71,7 @@ export function useIntentHandler() {
}, [incomingUrl, composeIntent, verifyEmailIntent])
}
function useComposeIntent() {
export function useComposeIntent() {
const closeAllActiveElements = useCloseAllActiveElements()
const {openComposer} = useComposerControls()
const {hasSession} = useSession()
@ -97,6 +97,10 @@ function useComposeIntent() {
if (part.includes('https://') || part.includes('http://')) {
return false
}
console.log({
part,
text: VALID_IMAGE_REGEX.test(part),
})
// We also should just filter out cases that don't have all the info we need
return VALID_IMAGE_REGEX.test(part)
})

View File

@ -1,3 +1,5 @@
export type Gate =
// Keep this alphabetic please.
'debug_show_feedcontext' | 'suggested_feeds_interstitial'
| 'debug_show_feedcontext'
| 'suggested_feeds_interstitial'
| 'ten_million_dialog'

View File

@ -115,7 +115,7 @@ msgstr "<0><1>テキストとタグ</1>中の</0>{0}"
msgid "{0} joined this week"
msgstr "今週、{0}人が参加しました"
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:593
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:637
msgid "{0} of {1}"
msgstr "{0} / {1}"
@ -123,7 +123,7 @@ msgstr "{0} / {1}"
msgid "{0} people have used this starter pack!"
msgstr "{0}人がこのスターターパックを使用しました!"
#: src/view/com/util/UserAvatar.tsx:419
#: src/view/com/util/UserAvatar.tsx:423
msgid "{0}'s avatar"
msgstr "{0}のアバター"
@ -372,7 +372,7 @@ msgstr "アカウントを追加"
msgid "Add alt text"
msgstr "ALTテキストを追加"
#: src/view/com/composer/videos/SubtitleDialog.tsx:107
#: src/view/com/composer/videos/SubtitleDialog.tsx:109
msgid "Add alt text (optional)"
msgstr "ALTテキストを追加オプション"
@ -493,9 +493,9 @@ msgid "ALT"
msgstr "ALT"
#: src/view/com/composer/GifAltText.tsx:144
#: src/view/com/composer/videos/SubtitleDialog.tsx:54
#: src/view/com/composer/videos/SubtitleDialog.tsx:102
#: src/view/com/composer/videos/SubtitleDialog.tsx:106
#: src/view/com/composer/videos/SubtitleDialog.tsx:56
#: src/view/com/composer/videos/SubtitleDialog.tsx:104
#: src/view/com/composer/videos/SubtitleDialog.tsx:108
#: src/view/com/modals/EditImage.tsx:316
#: src/view/screens/AccessibilitySettings.tsx:87
msgid "Alt text"
@ -522,11 +522,11 @@ msgstr "以前のメールアドレス{0}にメールが送信されました。
msgid "An error has occurred"
msgstr "エラーが発生しました"
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:369
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:413
msgid "An error occurred"
msgstr "エラーが発生しました"
#: src/state/queries/video/video.ts:227
#: src/state/queries/video/video.ts:232
msgid "An error occurred while compressing the video."
msgstr "ビデオの圧縮中にエラーが発生しました。"
@ -534,11 +534,11 @@ msgstr "ビデオの圧縮中にエラーが発生しました。"
msgid "An error occurred while generating your starter pack. Want to try again?"
msgstr "スターターパックの生成中にエラーが発生しました。再度試しますか?"
#: src/view/com/util/post-embeds/VideoEmbed.tsx:213
#: src/view/com/util/post-embeds/VideoEmbed.tsx:215
msgid "An error occurred while loading the video. Please try again later."
msgstr "ビデオの読み込み時にエラーが発生しました。時間をおいてもう一度お試しください。"
#: src/view/com/util/post-embeds/VideoEmbed.web.tsx:170
#: src/view/com/util/post-embeds/VideoEmbed.web.tsx:174
msgid "An error occurred while loading the video. Please try again."
msgstr "ビデオの読み込み時にエラーが発生しました。もう一度お試しください。"
@ -547,7 +547,7 @@ msgstr "ビデオの読み込み時にエラーが発生しました。もう一
msgid "An error occurred while saving the QR code!"
msgstr "QRコードの保存中にエラーが発生しました"
#: src/view/com/composer/videos/SelectVideoBtn.tsx:61
#: src/view/com/composer/videos/SelectVideoBtn.tsx:68
msgid "An error occurred while selecting the video"
msgstr "ビデオの選択中にエラーが発生しました"
@ -556,7 +556,7 @@ msgstr "ビデオの選択中にエラーが発生しました"
msgid "An error occurred while trying to follow all"
msgstr "すべてフォローしようとしたらエラーが発生しました"
#: src/state/queries/video/video.ts:194
#: src/state/queries/video/video.ts:199
msgid "An error occurred while uploading the video."
msgstr "ビデオのアップロード中にエラーが発生しました。"
@ -702,7 +702,7 @@ msgstr "あなたのフィードから{0}を削除してもよろしいですか
msgid "Are you sure you want to remove this from your feeds?"
msgstr "本当にこのフィードをあなたのフィードから削除したいですか?"
#: src/view/com/composer/Composer.tsx:837
#: src/view/com/composer/Composer.tsx:841
msgid "Are you sure you'd like to discard this draft?"
msgstr "本当にこの下書きを破棄しますか?"
@ -844,6 +844,10 @@ msgstr "Bluesky は、ホスティング プロバイダーを選択できるオ
msgid "Bluesky is better with friends!"
msgstr "Blueskyは友達と一緒のほうが楽しい"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:201
msgid "Bluesky now has over 10 million users, and I was #{0}!"
msgstr "Bluesky のユーザー数は現在 1,000 万人を超えており、私は #{0} でした。"
#: src/components/StarterPack/ProfileStarterPacks.tsx:282
msgid "Bluesky will choose a set of recommended accounts from people in your network."
msgstr "Blueskyはあなたのつながっているユーザーからおすすめのアカウントを選びます。"
@ -865,6 +869,10 @@ msgstr "画像のぼかしとフィードからのフィルタリング"
msgid "Books"
msgstr "書籍"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:583
msgid "Brag a little!"
msgstr "ちょっと自慢してみよう!"
#: src/components/FeedInterstitials.tsx:346
msgid "Browse more accounts on the Explore page"
msgstr "検索ページでさらにアカウントを見る"
@ -927,8 +935,8 @@ msgstr "英数字、スペース、ハイフン、アンダースコアのみが
#: src/components/Prompt.tsx:124
#: src/components/TagMenu/index.tsx:282
#: src/screens/Deactivated.tsx:161
#: src/view/com/composer/Composer.tsx:590
#: src/view/com/composer/Composer.tsx:605
#: src/view/com/composer/Composer.tsx:595
#: src/view/com/composer/Composer.tsx:610
#: src/view/com/modals/ChangeEmail.tsx:213
#: src/view/com/modals/ChangeEmail.tsx:215
#: src/view/com/modals/ChangeHandle.tsx:148
@ -996,14 +1004,18 @@ msgstr "リンク先のウェブサイトを開くことをキャンセル"
msgid "Cannot interact with a blocked user"
msgstr "ブロックしたユーザーとはやりとりできません"
#: src/view/com/composer/videos/SubtitleDialog.tsx:133
#: src/view/com/composer/videos/SubtitleDialog.tsx:135
msgid "Captions (.vtt)"
msgstr "キャプション(.vtt"
#: src/view/com/composer/videos/SubtitleDialog.tsx:54
#: src/view/com/composer/videos/SubtitleDialog.tsx:56
msgid "Captions & alt text"
msgstr "キャプションALTテキスト"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:336
msgid "Celebrating {0} users"
msgstr "{0} 人のユーザーを祝福"
#: src/view/com/modals/VerifyEmail.tsx:160
msgid "Change"
msgstr "変更"
@ -1238,7 +1250,7 @@ msgstr "下部のナビゲーションバーを閉じる"
msgid "Closes password update alert"
msgstr "パスワード更新アラートを閉じる"
#: src/view/com/composer/Composer.tsx:602
#: src/view/com/composer/Composer.tsx:607
msgid "Closes post composer and discards post draft"
msgstr "投稿の編集画面を閉じて下書きを削除する"
@ -1277,7 +1289,7 @@ msgstr "初期設定を完了してアカウントを使い始める"
msgid "Complete the challenge"
msgstr "テストをクリアしてください"
#: src/view/com/composer/Composer.tsx:710
#: src/view/com/composer/Composer.tsx:715
msgid "Compose posts up to {MAX_GRAPHEME_LENGTH} characters in length"
msgstr "{MAX_GRAPHEME_LENGTH}文字までの投稿を作成"
@ -1495,7 +1507,7 @@ msgstr "リストの読み込みに失敗しました"
msgid "Could not mute chat"
msgstr "チャットのミュートに失敗しました"
#: src/view/com/composer/videos/VideoPreview.web.tsx:45
#: src/view/com/composer/videos/VideoPreview.web.tsx:56
msgid "Could not process your video"
msgstr "ビデオを処理できませんでした"
@ -1604,7 +1616,7 @@ msgstr "ダークモード"
msgid "Dark theme"
msgstr "ダークテーマ"
#: src/screens/Signup/StepInfo/index.tsx:191
#: src/screens/Signup/StepInfo/index.tsx:192
msgid "Date of birth"
msgstr "生年月日"
@ -1738,7 +1750,7 @@ msgstr "引用投稿を切り離しますか?"
msgid "Dialog: adjust who can interact with this post"
msgstr "ダイアログ:この投稿に誰が反応できるか調整"
#: src/view/com/composer/Composer.tsx:351
#: src/view/com/composer/Composer.tsx:356
msgid "Did you want to say anything?"
msgstr "なにか言いたいことはあった?"
@ -1763,7 +1775,7 @@ msgstr "メールでの2要素認証を無効化"
msgid "Disable haptic feedback"
msgstr "触覚フィードバックを無効化"
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:335
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:379
msgid "Disable subtitles"
msgstr "サブタイトル(字幕)を無効にする"
@ -1776,11 +1788,11 @@ msgstr "サブタイトル(字幕)を無効にする"
msgid "Disabled"
msgstr "無効"
#: src/view/com/composer/Composer.tsx:839
#: src/view/com/composer/Composer.tsx:843
msgid "Discard"
msgstr "破棄"
#: src/view/com/composer/Composer.tsx:836
#: src/view/com/composer/Composer.tsx:840
msgid "Discard draft?"
msgstr "下書きを削除しますか?"
@ -1806,7 +1818,7 @@ msgstr "新しいフィードを探す"
#~ msgid "Dismiss"
#~ msgstr "消す"
#: src/view/com/composer/Composer.tsx:1106
#: src/view/com/composer/Composer.tsx:1110
msgid "Dismiss error"
msgstr "エラーを消す"
@ -1858,8 +1870,8 @@ msgstr "ドメインを確認しました!"
#: src/screens/Onboarding/StepProfile/index.tsx:325
#: src/view/com/auth/server-input/index.tsx:169
#: src/view/com/auth/server-input/index.tsx:170
#: src/view/com/composer/videos/SubtitleDialog.tsx:167
#: src/view/com/composer/videos/SubtitleDialog.tsx:177
#: src/view/com/composer/videos/SubtitleDialog.tsx:171
#: src/view/com/composer/videos/SubtitleDialog.tsx:181
#: src/view/com/modals/AddAppPasswords.tsx:243
#: src/view/com/modals/AltImage.tsx:141
#: src/view/com/modals/crop-image/CropImage.web.tsx:177
@ -1891,6 +1903,10 @@ msgstr "Blueskyをダウンロード"
msgid "Download CAR file"
msgstr "CARファイルをダウンロード"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:591
msgid "Download image"
msgstr "画像をダウンロード"
#: src/view/com/composer/text-input/TextInput.web.tsx:269
msgid "Drop to add images"
msgstr "ドロップして画像を追加する"
@ -1952,7 +1968,7 @@ msgctxt "action"
msgid "Edit"
msgstr "編集"
#: src/view/com/util/UserAvatar.tsx:328
#: src/view/com/util/UserAvatar.tsx:332
#: src/view/com/util/UserBanner.tsx:92
msgid "Edit avatar"
msgstr "アバターを編集"
@ -2112,7 +2128,7 @@ msgstr "有効にするメディアプレイヤー"
msgid "Enable priority notifications"
msgstr "優先通知を有効にする"
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:336
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:380
msgid "Enable subtitles"
msgstr "サブタイトル(字幕)を有効にする"
@ -2130,7 +2146,7 @@ msgstr "有効"
msgid "End of feed"
msgstr "フィードの終わり"
#: src/view/com/composer/videos/SubtitleDialog.tsx:157
#: src/view/com/composer/videos/SubtitleDialog.tsx:161
msgid "Ensure you have selected a language for each subtitle file."
msgstr "各字幕ファイルに言語が選択されてることを確認してください。"
@ -2232,7 +2248,7 @@ msgstr "フォローしているユーザーは除外"
msgid "Excludes users you follow"
msgstr "フォローしているユーザーは除外"
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:353
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:397
msgid "Exit fullscreen"
msgstr "全画面表示を終了"
@ -2667,7 +2683,7 @@ msgctxt "from-feed"
msgid "From <0/>"
msgstr "<0/>から"
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:354
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:398
msgid "Fullscreen"
msgstr "全画面表示"
@ -2696,7 +2712,7 @@ msgstr "開始"
msgid "Getting started"
msgstr "入門"
#: src/components/MediaPreview.tsx:119
#: src/components/MediaPreview.tsx:120
msgid "GIF"
msgstr "GIF"
@ -3098,7 +3114,7 @@ msgstr "招待、ただし個人的なもの"
msgid "It's just you right now! Add more people to your starter pack by searching above."
msgstr "今はあなただけ!上で検索してスターターパックにより多くのユーザーを追加してください。"
#: src/view/com/composer/Composer.tsx:1125
#: src/view/com/composer/Composer.tsx:1129
msgid "Job ID: {0}"
msgstr "ジョブID{0}"
@ -3117,6 +3133,10 @@ msgstr "Blueskyに参加"
msgid "Join the conversation"
msgstr "会話に参加"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:461
msgid "Joined {0}"
msgstr "{0} 参加"
#: src/screens/Onboarding/index.tsx:21
#: src/screens/Onboarding/state.ts:89
msgid "Journalism"
@ -3580,8 +3600,12 @@ msgstr "映画"
msgid "Music"
msgstr "音楽"
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:389
msgctxt "video"
msgid "Mute"
msgstr ""
#: src/components/TagMenu/index.tsx:263
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:345
msgid "Mute"
msgstr "ミュート"
@ -4042,6 +4066,10 @@ msgstr "ちょっと!"
msgid "Oh no! Something went wrong."
msgstr "ちょっと!何らかの問題が発生したようです。"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:171
msgid "Oh no! We weren't able to generate an image for you to share. Rest assured, we're glad you're here 🦋"
msgstr "ああ、残念!共有するための画像を生成できませんでした。ご安心ください、ここに来てくれて嬉しいです🦋"
#: src/screens/Profile/Header/ProfileHeaderLabeler.tsx:339
msgid "OK"
msgstr "OK"
@ -4062,7 +4090,7 @@ msgstr "<0><1/><2><3/></2></0>"
msgid "Onboarding reset"
msgstr "オンボーディングのリセット"
#: src/view/com/composer/Composer.tsx:667
#: src/view/com/composer/Composer.tsx:672
msgid "One or more images is missing alt text."
msgstr "つもしくは複数の画像にALTテキストがありません。"
@ -4113,8 +4141,8 @@ msgid "Open conversation options"
msgstr "会話のオプションを開く"
#: src/screens/Messages/Conversation/MessageInput.web.tsx:165
#: src/view/com/composer/Composer.tsx:819
#: src/view/com/composer/Composer.tsx:820
#: src/view/com/composer/Composer.tsx:823
#: src/view/com/composer/Composer.tsx:824
msgid "Open emoji picker"
msgstr "絵文字を入力"
@ -4283,11 +4311,11 @@ msgid "Opens the threads preferences"
msgstr "スレッドの設定を開く"
#: src/view/com/notifications/FeedItem.tsx:551
#: src/view/com/util/UserAvatar.tsx:420
#: src/view/com/util/UserAvatar.tsx:424
msgid "Opens this profile"
msgstr "プロフィールを開く"
#: src/view/com/composer/videos/SelectVideoBtn.tsx:81
#: src/view/com/composer/videos/SelectVideoBtn.tsx:88
msgid "Opens video picker"
msgstr "ビデオの選択画面を開く"
@ -4365,11 +4393,11 @@ msgid "Password updated!"
msgstr "パスワードが更新されました!"
#: src/view/com/util/post-embeds/GifEmbed.tsx:44
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:322
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:366
msgid "Pause"
msgstr "一時停止"
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:275
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:319
msgid "Pause video"
msgstr "ビデオを一時停止"
@ -4429,7 +4457,7 @@ msgid "Pinned to your feeds"
msgstr "フィードにピン留めしました"
#: src/view/com/util/post-embeds/GifEmbed.tsx:44
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:323
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:367
msgid "Play"
msgstr "再生"
@ -4441,8 +4469,8 @@ msgstr "{0}を再生"
msgid "Play or pause the GIF"
msgstr "GIFの再生や一時停止"
#: src/view/com/util/post-embeds/VideoEmbed.tsx:187
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:276
#: src/view/com/util/post-embeds/VideoEmbed.tsx:189
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:320
msgid "Play video"
msgstr "ビデオを再生"
@ -4514,7 +4542,7 @@ msgstr "@{0}としてサインインしてください"
msgid "Please Verify Your Email"
msgstr "メールアドレスを確認してください"
#: src/view/com/composer/Composer.tsx:355
#: src/view/com/composer/Composer.tsx:360
msgid "Please wait for your link card to finish loading"
msgstr "リンクカードが読み込まれるまでお待ちください"
@ -4527,8 +4555,8 @@ msgstr "政治"
msgid "Porn"
msgstr "ポルノ"
#: src/view/com/composer/Composer.tsx:642
#: src/view/com/composer/Composer.tsx:649
#: src/view/com/composer/Composer.tsx:647
#: src/view/com/composer/Composer.tsx:654
msgctxt "action"
msgid "Post"
msgstr "投稿"
@ -4698,11 +4726,11 @@ msgstr "ユーザーを一括でミュートまたはブロックする、公開
msgid "Public, shareable lists which can drive feeds."
msgstr "フィードとして利用できる、公開された共有可能なリスト。"
#: src/view/com/composer/Composer.tsx:627
#: src/view/com/composer/Composer.tsx:632
msgid "Publish post"
msgstr "投稿を公開"
#: src/view/com/composer/Composer.tsx:627
#: src/view/com/composer/Composer.tsx:632
msgid "Publish reply"
msgstr "返信を公開"
@ -4832,7 +4860,7 @@ msgstr "アカウントを削除"
msgid "Remove attachment"
msgstr "添付を削除"
#: src/view/com/util/UserAvatar.tsx:387
#: src/view/com/util/UserAvatar.tsx:391
msgid "Remove Avatar"
msgstr "アバターを削除"
@ -4900,7 +4928,7 @@ msgstr "引用を削除"
msgid "Remove repost"
msgstr "リポストを削除"
#: src/view/com/composer/videos/SubtitleDialog.tsx:260
#: src/view/com/composer/videos/SubtitleDialog.tsx:264
msgid "Remove subtitle file"
msgstr "字幕ファイルを削除"
@ -4961,7 +4989,7 @@ msgstr "返信できません"
msgid "Replies to this post are disabled."
msgstr "この投稿への返信は無効化されています。"
#: src/view/com/composer/Composer.tsx:640
#: src/view/com/composer/Composer.tsx:645
msgctxt "action"
msgid "Reply"
msgstr "返信"
@ -5415,7 +5443,7 @@ msgstr "Blueskyの求人を見る"
msgid "See this guide"
msgstr "ガイドを見る"
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:587
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:631
msgid "Seek slider"
msgstr "シークバー"
@ -5455,7 +5483,7 @@ msgstr "GIF「{0}」を選ぶ"
msgid "Select how long to mute this word for."
msgstr "このワードをどのくらいの間ミュートするのかを選択。"
#: src/view/com/composer/videos/SubtitleDialog.tsx:245
#: src/view/com/composer/videos/SubtitleDialog.tsx:249
msgid "Select language..."
msgstr "言語を選択…"
@ -5487,7 +5515,7 @@ msgstr "報告先のモデレーションサービスを選んでください"
msgid "Select the service that hosts your data."
msgstr "データをホストするサービスを選択します。"
#: src/view/com/composer/videos/SelectVideoBtn.tsx:80
#: src/view/com/composer/videos/SelectVideoBtn.tsx:87
msgid "Select video"
msgstr "ビデオを選択"
@ -5503,7 +5531,7 @@ msgstr "登録されたフィードに含める言語を選択します。選択
msgid "Select your app language for the default text to display in the app."
msgstr "アプリに表示されるデフォルトのテキストの言語を選択"
#: src/screens/Signup/StepInfo/index.tsx:192
#: src/screens/Signup/StepInfo/index.tsx:193
msgid "Select your date of birth"
msgstr "生年月日を選択"
@ -5643,6 +5671,7 @@ msgstr "性的行為または性的なヌード。"
msgid "Sexually Suggestive"
msgstr "性的にきわどい"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:607
#: src/components/StarterPack/QrCodeDialog.tsx:177
#: src/screens/StarterPack/StarterPackScreen.tsx:411
#: src/screens/StarterPack/StarterPackScreen.tsx:582
@ -5679,6 +5708,14 @@ msgstr "とにかく共有"
msgid "Share feed"
msgstr "フィードを共有"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:590
msgid "Share image externally"
msgstr "画像を外部に共有する"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:602
msgid "Share image in post"
msgstr "投稿で画像を共有"
#: src/components/StarterPack/ShareDialog.tsx:124
#: src/components/StarterPack/ShareDialog.tsx:131
#: src/screens/StarterPack/StarterPackScreen.tsx:586
@ -5950,8 +5987,8 @@ msgstr "何らかの問題が発生したようなので、もう一度お試し
msgid "Something went wrong!"
msgstr "何らかの問題が発生したようです!"
#: src/App.native.tsx:102
#: src/App.web.tsx:83
#: src/App.native.tsx:103
#: src/App.web.tsx:84
msgid "Sorry! Your session expired. Please log in again."
msgstr "大変申し訳ありません!セッションの有効期限が切れました。もう一度ログインしてください。"
@ -6153,6 +6190,10 @@ msgstr "ジョークを言って!"
msgid "Tell us a little more"
msgstr "もう少し教えて"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:487
msgid "Ten Million"
msgstr "1000万"
#: src/view/shell/desktop/RightNav.tsx:90
msgid "Terms"
msgstr "条件"
@ -6186,6 +6227,10 @@ msgstr "テキストの入力フィールド"
msgid "Thank you. Your report has been sent."
msgstr "ありがとうございます。あなたの報告は送信されました。"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:562
msgid "Thanks for being one of our first 10 million users."
msgstr "最初の 1,000 万人のユーザーの 1 人になっていただきありがとうございます。"
#: src/components/intents/VerifyEmailIntentDialog.tsx:74
msgid "Thanks, you have successfully verified your email address."
msgstr ""
@ -6271,7 +6316,7 @@ msgstr "投稿が削除された可能性があります。"
msgid "The Privacy Policy has been moved to <0/>"
msgstr "プライバシーポリシーは<0/>に移動しました"
#: src/state/queries/video/video.ts:222
#: src/state/queries/video/video.ts:227
msgid "The selected video is larger than 50MB."
msgstr "選択したビデオのサイズが50MBを超えています。"
@ -6608,7 +6653,7 @@ msgstr "メールでの2要素認証を無効にするには、メールアド
msgid "To report a conversation, please report one of its messages via the conversation screen. This lets our moderators understand the context of your issue."
msgstr "会話を報告するには、会話の画面からメッセージのうちの一つを報告してください。それによって問題の文脈をモデレーターが理解できるようになります。"
#: src/view/com/composer/videos/SelectVideoBtn.tsx:106
#: src/view/com/composer/videos/SelectVideoBtn.tsx:113
msgid "To upload videos to Bluesky, you must first verify your email."
msgstr "Blueskyにビデオをアップロードするには、まずメールアドレスを確認しなくてはなりません。"
@ -6616,6 +6661,10 @@ msgstr "Blueskyにビデオをアップロードするには、まずメール
msgid "To whom would you like to send this report?"
msgstr "この報告を誰に送りたいですか?"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:566
msgid "Together, we're rebuilding the social internet. We're glad you're here."
msgstr "私たちは一緒にソーシャル インターネットを再構築しています。ご参加いただきありがとうございます。"
#: src/view/com/util/forms/DropdownButton.tsx:255
msgid "Toggle dropdown"
msgstr "ドロップダウンを切り替え"
@ -6739,8 +6788,12 @@ msgstr "アカウントのフォローを解除"
msgid "Unlike this feed"
msgstr "このフィードからいいねを外す"
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:388
msgctxt "video"
msgid "Unmute"
msgstr ""
#: src/components/TagMenu/index.tsx:263
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:344
#: src/view/screens/ProfileList.tsx:689
msgid "Unmute"
msgstr "ミュートを解除"
@ -6767,7 +6820,7 @@ msgstr "会話のミュートを解除"
msgid "Unmute thread"
msgstr "スレッドのミュートを解除"
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:273
#: src/view/com/util/post-embeds/VideoEmbedInner/VideoWebControls.tsx:317
msgid "Unmute video"
msgstr "ビデオのミュートを解除"
@ -6809,7 +6862,7 @@ msgstr "このラベラーの登録を解除"
msgid "Unsubscribed from list"
msgstr "リストの登録を解除しました"
#: src/state/queries/video/video.ts:240
#: src/state/queries/video/video.ts:245
msgid "Unsupported video type: {mimeType}"
msgstr "サポートしていないビデオ形式:{mimeType}"
@ -6846,20 +6899,20 @@ msgstr "代わりに写真をアップロード"
msgid "Upload a text file to:"
msgstr "テキストファイルのアップロード先:"
#: src/view/com/util/UserAvatar.tsx:355
#: src/view/com/util/UserAvatar.tsx:358
#: src/view/com/util/UserAvatar.tsx:359
#: src/view/com/util/UserAvatar.tsx:362
#: src/view/com/util/UserBanner.tsx:123
#: src/view/com/util/UserBanner.tsx:126
msgid "Upload from Camera"
msgstr "カメラからアップロード"
#: src/view/com/util/UserAvatar.tsx:372
#: src/view/com/util/UserAvatar.tsx:376
#: src/view/com/util/UserBanner.tsx:140
msgid "Upload from Files"
msgstr "ファイルからアップロード"
#: src/view/com/util/UserAvatar.tsx:366
#: src/view/com/util/UserAvatar.tsx:370
#: src/view/com/util/UserAvatar.tsx:374
#: src/view/com/util/UserBanner.tsx:134
#: src/view/com/util/UserBanner.tsx:138
msgid "Upload from Library"
@ -6988,7 +7041,7 @@ msgstr "このコンテンツやプロフィールにいいねをしているユ
msgid "Value:"
msgstr "値:"
#: src/view/com/composer/videos/SelectVideoBtn.tsx:104
#: src/view/com/composer/videos/SelectVideoBtn.tsx:111
msgid "Verified email required"
msgstr "メールアドレスの確認が必要"
@ -7017,7 +7070,7 @@ msgstr "メールアドレスを確認"
msgid "Verify New Email"
msgstr "新しいメールアドレスを確認"
#: src/view/com/composer/videos/SelectVideoBtn.tsx:108
#: src/view/com/composer/videos/SelectVideoBtn.tsx:115
msgid "Verify now"
msgstr "確認する"
@ -7047,11 +7100,11 @@ msgstr "ビデオの処理に失敗"
msgid "Video Games"
msgstr "ビデオゲーム"
#: src/view/com/util/post-embeds/VideoEmbed.web.tsx:163
#: src/view/com/util/post-embeds/VideoEmbed.web.tsx:167
msgid "Video not found."
msgstr "ビデオが見つかりません。"
#: src/view/com/composer/videos/SubtitleDialog.tsx:99
#: src/view/com/composer/videos/SubtitleDialog.tsx:101
msgid "Video settings"
msgstr "ビデオの設定"
@ -7059,6 +7112,11 @@ msgstr "ビデオの設定"
msgid "Video: {0}"
msgstr "ビデオ:{0}"
#: src/view/com/composer/videos/SelectVideoBtn.tsx:58
#: src/view/com/composer/videos/VideoPreview.web.tsx:44
msgid "Videos must be less than 60 seconds long"
msgstr ""
#: src/screens/Profile/Header/Shell.tsx:113
msgid "View {0}'s avatar"
msgstr "{0}のアバターを表示"
@ -7224,7 +7282,7 @@ msgstr "大変申し訳ありませんが、現在ミュートされたワード
msgid "We're sorry, but your search could not be completed. Please try again in a few minutes."
msgstr "大変申し訳ありませんが、検索を完了できませんでした。数分後に再試行してください。"
#: src/view/com/composer/Composer.tsx:417
#: src/view/com/composer/Composer.tsx:422
msgid "We're sorry! The post you are replying to has been deleted."
msgstr "大変申し訳ありません!返信しようとしている投稿は削除されました。"
@ -7255,7 +7313,7 @@ msgstr "あなたのスターターパックを何と呼びたいですか?"
#: src/view/com/auth/SplashScreen.tsx:40
#: src/view/com/auth/SplashScreen.web.tsx:86
#: src/view/com/composer/Composer.tsx:512
#: src/view/com/composer/Composer.tsx:517
msgid "What's up?"
msgstr "最近どう?"
@ -7322,11 +7380,11 @@ msgstr "ワイド"
msgid "Write a message"
msgstr "メッセージを書く"
#: src/view/com/composer/Composer.tsx:708
#: src/view/com/composer/Composer.tsx:713
msgid "Write post"
msgstr "投稿を書く"
#: src/view/com/composer/Composer.tsx:511
#: src/view/com/composer/Composer.tsx:516
#: src/view/com/post-thread/PostThreadComposePrompt.tsx:42
msgid "Write your reply"
msgstr "返信を書く"
@ -7646,15 +7704,19 @@ msgstr "あなたのアカウント"
msgid "Your account has been deleted"
msgstr "あなたのアカウントは削除されました"
#: src/state/queries/video/video.ts:185
msgid "Your account is not yet old enough to upload videos. Please try again later."
msgstr ""
#: src/view/screens/Settings/ExportCarDialog.tsx:65
msgid "Your account repository, containing all public data records, can be downloaded as a \"CAR\" file. This file does not include media embeds, such as images, or your private data, which must be fetched separately."
msgstr "あなたのアカウントの公開データの全記録を含むリポジトリは、「CAR」ファイルとしてダウンロードできます。このファイルには、画像などのメディア埋め込み、また非公開のデータは含まれていないため、それらは個別に取得する必要があります。"
#: src/screens/Signup/StepInfo/index.tsx:180
#: src/screens/Signup/StepInfo/index.tsx:181
msgid "Your birth date"
msgstr "生年月日"
#: src/view/com/util/post-embeds/VideoEmbed.web.tsx:167
#: src/view/com/util/post-embeds/VideoEmbed.web.tsx:171
msgid "Your browser does not support the video format. Please try a different browser."
msgstr "利用中のブラウザがこのビデオ形式をサポートしていません。他のブラウザをお試しください。"
@ -7705,7 +7767,7 @@ msgstr "ミュートしたワード"
msgid "Your password has been changed successfully!"
msgstr "パスワードの変更が完了しました!"
#: src/view/com/composer/Composer.tsx:463
#: src/view/com/composer/Composer.tsx:468
msgid "Your post has been published"
msgstr "投稿を公開しました"
@ -7721,7 +7783,7 @@ msgstr "あなたのプロフィール"
msgid "Your profile, posts, feeds, and lists will no longer be visible to other Bluesky users. You can reactivate your account at any time by logging in."
msgstr "あなたのプロフィール、投稿、フィード、そしてリストは他のBlueskyユーザーに見えなくなります。ログインすることでいつでもアカウントを再有効化できます。"
#: src/view/com/composer/Composer.tsx:462
#: src/view/com/composer/Composer.tsx:467
msgid "Your reply has been published"
msgstr "返信を公開しました"

View File

@ -139,7 +139,7 @@ msgstr "{0} pessoas já usaram este pacote inicial!"
#~ msgid "{0} your feeds"
#~ msgstr "{0} seus feeds"
#: src/view/com/util/UserAvatar.tsx:419
#: src/view/com/util/UserAvatar.tsx:423
msgid "{0}'s avatar"
msgstr "Avatar de {0}"
@ -656,7 +656,7 @@ msgstr "Ocorreu um erro ao gerar seu pacote inicial. Quer tentar novamente?"
msgid "An error occurred while loading the video. Please try again later."
msgstr "Ocorreu um erro ao carregar o vídeo. Tente novamente mais tarde."
#: src/view/com/util/post-embeds/VideoEmbed.web.tsx:171
#: src/view/com/util/post-embeds/VideoEmbed.web.tsx:174
msgid "An error occurred while loading the video. Please try again."
msgstr "Ocorreu um erro ao carregar o vídeo. Tente novamente."
@ -1005,6 +1005,10 @@ msgstr "Bluesky é melhor com amigos!"
#~ msgid "Bluesky is public."
#~ msgstr "Bluesky é público."
#: src/components/dialogs/nuxs/TenMillion/index.tsx:201
msgid "Bluesky now has over 10 million users, and I was #{0}!"
msgstr "O Bluesky agora tem mais de 10 milhões de usuários, e eu tinha #{0}!"
#: src/components/StarterPack/ProfileStarterPacks.tsx:282
msgid "Bluesky will choose a set of recommended accounts from people in your network."
msgstr "O Bluesky escolherá um conjunto de contas recomendadas dentre as pessoas em sua rede."
@ -1026,6 +1030,10 @@ msgstr "Desfocar imagens e filtrar dos feeds"
msgid "Books"
msgstr "Livros"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:583
msgid "Brag a little!"
msgstr "Gabe-se um pouco!"
#: src/components/FeedInterstitials.tsx:346
msgid "Browse more accounts on the Explore page"
msgstr "Navegue por mais contas na página Explorar"
@ -1173,6 +1181,10 @@ msgstr "Legendas (.vtt)"
msgid "Captions & alt text"
msgstr "Legendas e texto alt"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:336
msgid "Celebrating {0} users"
msgstr "Comemorando {0} usuários"
#: src/view/com/modals/VerifyEmail.tsx:160
msgid "Change"
msgstr "Alterar"
@ -1867,7 +1879,7 @@ msgstr "Tema escuro"
#~ msgid "Dark Theme"
#~ msgstr "Modo Escuro"
#: src/screens/Signup/StepInfo/index.tsx:191
#: src/screens/Signup/StepInfo/index.tsx:192
msgid "Date of birth"
msgstr "Data de nascimento"
@ -2174,6 +2186,10 @@ msgstr "Baixe o Bluesky"
msgid "Download CAR file"
msgstr "Baixar arquivo CAR"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:591
msgid "Download image"
msgstr "Baixar imagem"
#: src/view/com/composer/text-input/TextInput.web.tsx:269
msgid "Drop to add images"
msgstr "Solte para adicionar imagens"
@ -2239,7 +2255,7 @@ msgctxt "action"
msgid "Edit"
msgstr "Editar"
#: src/view/com/util/UserAvatar.tsx:328
#: src/view/com/util/UserAvatar.tsx:332
#: src/view/com/util/UserBanner.tsx:92
msgid "Edit avatar"
msgstr "Editar avatar"
@ -3545,6 +3561,10 @@ msgstr "Crie uma conta no Bluesky"
msgid "Join the conversation"
msgstr "Participe da conversa"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:461
msgid "Joined {0}"
msgstr "Juntou-se {0}"
#: src/screens/Onboarding/index.tsx:21
#: src/screens/Onboarding/state.ts:89
msgid "Journalism"
@ -4551,6 +4571,10 @@ msgstr "Opa!"
msgid "Oh no! Something went wrong."
msgstr "Opa! Algo deu errado."
#: src/components/dialogs/nuxs/TenMillion/index.tsx:171
msgid "Oh no! We weren't able to generate an image for you to share. Rest assured, we're glad you're here 🦋"
msgstr "Ah, não! Não conseguimos gerar uma imagem para você compartilhar. Fique tranquilo, estamos felizes que você esteja aqui 🦋"
#: src/screens/Profile/Header/ProfileHeaderLabeler.tsx:339
msgid "OK"
msgstr "OK"
@ -4821,7 +4845,7 @@ msgid "Opens the threads preferences"
msgstr "Abre as preferências de threads"
#: src/view/com/notifications/FeedItem.tsx:551
#: src/view/com/util/UserAvatar.tsx:420
#: src/view/com/util/UserAvatar.tsx:424
msgid "Opens this profile"
msgstr "Abre este perfil"
@ -5410,7 +5434,7 @@ msgstr "Remover conta"
msgid "Remove attachment"
msgstr ""
#: src/view/com/util/UserAvatar.tsx:387
#: src/view/com/util/UserAvatar.tsx:391
msgid "Remove Avatar"
msgstr "Remover avatar"
@ -6153,7 +6177,7 @@ msgstr "Selecione quais idiomas você deseja ver nos seus feeds. Se nenhum for s
msgid "Select your app language for the default text to display in the app."
msgstr "Selecione o idioma do seu aplicativo"
#: src/screens/Signup/StepInfo/index.tsx:192
#: src/screens/Signup/StepInfo/index.tsx:193
msgid "Select your date of birth"
msgstr "Selecione sua data de nascimento"
@ -6321,6 +6345,7 @@ msgstr "Atividade sexual ou nudez erótica."
msgid "Sexually Suggestive"
msgstr "Sexualmente Sugestivo"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:607
#: src/components/StarterPack/QrCodeDialog.tsx:177
#: src/screens/StarterPack/StarterPackScreen.tsx:411
#: src/screens/StarterPack/StarterPackScreen.tsx:582
@ -6357,6 +6382,14 @@ msgstr "Compartilhar assim"
msgid "Share feed"
msgstr "Compartilhar feed"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:590
msgid "Share image externally"
msgstr "Compartilhar imagem externamente"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:602
msgid "Share image in post"
msgstr "Compartilhe a imagem na postagem"
#: src/components/StarterPack/ShareDialog.tsx:124
#: src/components/StarterPack/ShareDialog.tsx:131
#: src/screens/StarterPack/StarterPackScreen.tsx:586
@ -6668,8 +6701,8 @@ msgstr "Algo deu errado. Por favor, tente novamente."
msgid "Something went wrong!"
msgstr "Algo deu errado!"
#: src/App.native.tsx:102
#: src/App.web.tsx:83
#: src/App.native.tsx:103
#: src/App.web.tsx:84
msgid "Sorry! Your session expired. Please log in again."
msgstr "Opa! Sua sessão expirou. Por favor, entre novamente."
@ -6912,6 +6945,10 @@ msgstr "Conte uma piada!"
msgid "Tell us a little more"
msgstr "Conte-nos um pouco mais"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:487
msgid "Ten Million"
msgstr "Dez milhões"
#: src/view/shell/desktop/RightNav.tsx:90
msgid "Terms"
msgstr "Termos"
@ -6949,6 +6986,10 @@ msgstr "Campo de entrada de texto"
msgid "Thank you. Your report has been sent."
msgstr "Obrigado. Sua denúncia foi enviada."
#: src/components/dialogs/nuxs/TenMillion/index.tsx:562
msgid "Thanks for being one of our first 10 million users."
msgstr "Obrigado por ser um dos nossos primeiros 10 milhões de usuários."
#: src/components/intents/VerifyEmailIntentDialog.tsx:74
msgid "Thanks, you have successfully verified your email address."
msgstr ""
@ -7437,6 +7478,10 @@ msgstr "Para enviar vídeos para o Bluesky, você deve primeiro verificar seu e-
msgid "To whom would you like to send this report?"
msgstr "Para quem você gostaria de enviar esta denúncia?"
#: src/components/dialogs/nuxs/TenMillion/index.tsx:566
msgid "Together, we're rebuilding the social internet. We're glad you're here."
msgstr "Juntos, estamos reconstruindo a internet social. Estamos felizes que você esteja aqui."
#: src/components/dialogs/MutedWords.tsx:112
#~ msgid "Toggle between muted word options."
#~ msgstr "Alternar entre opções de uma palavra silenciada"
@ -7691,20 +7736,20 @@ msgstr "Enviar uma foto"
msgid "Upload a text file to:"
msgstr "Carregar um arquivo de texto para:"
#: src/view/com/util/UserAvatar.tsx:355
#: src/view/com/util/UserAvatar.tsx:358
#: src/view/com/util/UserAvatar.tsx:359
#: src/view/com/util/UserAvatar.tsx:362
#: src/view/com/util/UserBanner.tsx:123
#: src/view/com/util/UserBanner.tsx:126
msgid "Upload from Camera"
msgstr "Tirar uma foto"
#: src/view/com/util/UserAvatar.tsx:372
#: src/view/com/util/UserAvatar.tsx:376
#: src/view/com/util/UserBanner.tsx:140
msgid "Upload from Files"
msgstr "Carregar um arquivo"
#: src/view/com/util/UserAvatar.tsx:366
#: src/view/com/util/UserAvatar.tsx:370
#: src/view/com/util/UserAvatar.tsx:374
#: src/view/com/util/UserBanner.tsx:134
#: src/view/com/util/UserBanner.tsx:138
msgid "Upload from Library"
@ -7904,7 +7949,7 @@ msgstr "Falha no processamento do vídeo"
msgid "Video Games"
msgstr "Games"
#: src/view/com/util/post-embeds/VideoEmbed.web.tsx:164
#: src/view/com/util/post-embeds/VideoEmbed.web.tsx:167
msgid "Video not found."
msgstr "Vídeo não encontrado."
@ -8572,11 +8617,11 @@ msgstr "Sua conta ainda não tem idade suficiente para enviar vídeos. Por favor
msgid "Your account repository, containing all public data records, can be downloaded as a \"CAR\" file. This file does not include media embeds, such as images, or your private data, which must be fetched separately."
msgstr "O repositório da sua conta, contendo todos os seus dados públicos, pode ser baixado como um arquivo \"CAR\". Este arquivo não inclui imagens ou dados privados, estes devem ser exportados separadamente."
#: src/screens/Signup/StepInfo/index.tsx:180
#: src/screens/Signup/StepInfo/index.tsx:181
msgid "Your birth date"
msgstr "Sua data de nascimento"
#: src/view/com/util/post-embeds/VideoEmbed.web.tsx:168
#: src/view/com/util/post-embeds/VideoEmbed.web.tsx:171
msgid "Your browser does not support the video format. Please try a different browser."
msgstr "Seu navegador não suporta o formato de vídeo. Por favor, tente um navegador diferente."

View File

@ -3,27 +3,16 @@ import zod from 'zod'
import {BaseNux} from '#/state/queries/nuxs/types'
export enum Nux {
One = 'one',
Two = 'two',
TenMillionDialog = 'TenMillionDialog',
}
export const nuxNames = new Set(Object.values(Nux))
export type AppNux =
| BaseNux<{
id: Nux.One
data: {
likes: number
}
}>
| BaseNux<{
id: Nux.Two
export type AppNux = BaseNux<{
id: Nux.TenMillionDialog
data: undefined
}>
export const NuxSchemas = {
[Nux.One]: zod.object({
likes: zod.number(),
}),
[Nux.Two]: undefined,
export const NuxSchemas: Record<Nux, zod.ZodObject<any> | undefined> = {
[Nux.TenMillionDialog]: undefined,
}

View File

@ -57,6 +57,7 @@ export function useUpsertNuxMutation() {
const agent = useAgent()
return useMutation({
retry: 3,
mutationFn: async (nux: AppNux) => {
await agent.bskyAppUpsertNux(serializeAppNux(nux))
// triggers a refetch
@ -72,6 +73,7 @@ export function useRemoveNuxsMutation() {
const agent = useAgent()
return useMutation({
retry: 3,
mutationFn: async (ids: string[]) => {
await agent.bskyAppRemoveNuxs(ids)
// triggers a refetch

View File

@ -4,6 +4,4 @@ export type Data = Record<string, unknown> | undefined
export type BaseNux<
T extends Pick<AppBskyActorDefs.Nux, 'id' | 'expiresAt'> & {data: Data},
> = T & {
completed: boolean
}
> = Pick<AppBskyActorDefs.Nux, 'id' | 'completed' | 'expiresAt'> & T

View File

@ -1,4 +1,6 @@
/**
* Device data that's specific to the device and does not vary based account
*/
export type Device = {}
export type Device = {
lastNuxDialog: string | undefined
}

View File

@ -43,6 +43,7 @@ interface BaseUserAvatarProps {
interface UserAvatarProps extends BaseUserAvatarProps {
moderation?: ModerationUI
usePlainRNImage?: boolean
onLoad?: () => void
}
interface EditableUserAvatarProps extends BaseUserAvatarProps {
@ -174,6 +175,7 @@ let UserAvatar = ({
avatar,
moderation,
usePlainRNImage = false,
onLoad,
}: UserAvatarProps): React.ReactNode => {
const pal = usePalette('default')
const backgroundColor = pal.colors.backgroundLight
@ -224,6 +226,7 @@ let UserAvatar = ({
uri: hackModifyThumbnailPath(avatar, size < 90),
}}
blurRadius={moderation?.blur ? BLUR_AMOUNT : 0}
onLoad={onLoad}
/>
) : (
<HighPriorityImage
@ -234,6 +237,7 @@ let UserAvatar = ({
uri: hackModifyThumbnailPath(avatar, size < 90),
}}
blurRadius={moderation?.blur ? BLUR_AMOUNT : 0}
onLoad={onLoad}
/>
)}
{alert}

View File

@ -0,0 +1,29 @@
import React from 'react'
import Svg, {Path, PathProps, SvgProps} from 'react-native-svg'
import {usePalette} from '#/lib/hooks/usePalette'
const ratio = 54 / 61
export function Logomark({
fill,
...rest
}: {fill?: PathProps['fill']} & SvgProps) {
const pal = usePalette('default')
// @ts-ignore it's fiiiiine
const size = parseInt(rest.width || 32)
return (
<Svg
fill="none"
viewBox="0 0 61 54"
{...rest}
width={size}
height={Number(size) * ratio}>
<Path
fill={fill || pal.text.color}
d="M13.223 3.602C20.215 8.832 27.738 19.439 30.5 25.13c2.762-5.691 10.284-16.297 17.278-21.528C52.824-.172 61-3.093 61 6.2c0 1.856-1.068 15.59-1.694 17.82-2.178 7.752-10.112 9.73-17.17 8.532 12.337 2.092 15.475 9.021 8.697 15.95-12.872 13.159-18.5-3.302-19.943-7.52-.264-.773-.388-1.135-.39-.827-.002-.308-.126.054-.39.827-1.442 4.218-7.071 20.679-19.943 7.52-6.778-6.929-3.64-13.858 8.697-15.95-7.058 1.197-14.992-.78-17.17-8.532C1.068 21.79 0 8.056 0 6.2 0-3.093 8.176-.172 13.223 3.602Z"
/>
</Svg>
)
}

View File

@ -63,6 +63,7 @@ export function Composer({}: {winHeight: number}) {
mention={state.mention}
openEmojiPicker={onOpenPicker}
text={state.text}
imageUris={state.imageUris}
/>
</View>
<EmojiPicker state={pickerState} close={onClosePicker} />