* Add modservice screen and profile-header-card * Drop the guidelines for now * Remove ununsed constants * Add label & label group descriptions * Not found state * Reorg, add icon * Subheader * Header * Complete header * Clean up * Add all groups * Fix scroll view * Dialogs side quest * Remove log * Add (WIP) debug mod page * Dialog solution * Add note * Clean up and reorganize localized moderation strings * Memoize * Add example * Add first ReportDialog screen * Report dialog step 2 * Submit * Integrate updates * Move moderation screen * Migrate buttons * Migrate everything * Rough sketch * Fix types * Update atoms values * Abstract ModerationServiceCard * Hook up data to settings page * Handle subscription * Rough enablement * Rough enablement * Some validation, fixes * More work on the mod debug screen * Hook up data * Update invalidation * Hook up data to ReportDialog * Fix native error * Refactor/rewrite the entire moderation-application system * Fix toggles * Add copyright and other option to report * Handle reports on profile vs content * Little cleanup * Get post hiding back in gear * Better loading flow on Mod screen * Clean up Mod screen * Clean up ProfileMod screen * Handle muting correctly * Update enablement on ProfileMod screen * Improve Moderation screen and dialog * Styling, handle disabled labelers * Rework list of labels on own content * Use moderateNotification() * ReportDialog updates * Fix button overflow * Simplify the ProfileModerationService ui * Mod screen design * Move moderation card from the profile header to a tab * Small tweaks to the moderation screen * Enable toggle on mod page * Add notifs to debugmod and dont filter notifs from followed users * Add moderator-service profile view * Wire up more of the modservice data to profiles * A bunch of speculative non-working UI * Cleanup: delete old code * Update ModerationDetailsDialog * Update ReportDialog * Update LabelsOnMe dialog * Handle ReportDialog load better * Rename LabelsOnMeDialog, fix close * Experiment to put labeling under a tab of a normal profile * Moderator variation of profile * Remove dead code and start moving toward latest modsdk * Remove a bunch of now-dead label strings * Update ModDebug to be a bit more intuitive and support custom labels * Minor ui tweaks * Improve consistency of display name blurring * Fix profile-card warning rendering * More debugmod UI tuning * Update to use new labeler semantics * Delete some dead code and do some refactoring * Update profile to pull from labeler definition * Implement new label config controls (wip) * Tweak ui * Implement preference controls on labelers * Rework label pref ui * Get moderation screen working * Add asyncstorage query persistence * Implement label handling * Small cleanup * Implement Likes dialog * Fix: remove text outside of text element * Cleanup * Fix likes dialog on mobile * Implement the label appeal flow * Get report flow working again with temporarily fixed report options * Update onboarding * Enforce limit of ten labeler subscriptions * Fix type errors * Fix lint errors * Improve types of RQ * Some work on Likes dialog, needs discussion * Bit of ReportDialog cleanup * Replace non-single-path SVG * Update nudity descriptions * Update to use new sdk updates * Add adult-content-enabled behavior to label config * Use the default setting of custom labels * Handle global moderation label prefs with the global settings * Fix missing postAuthor * Fix empty moderation page * Add mutewords control back to Mod screen * Tweak adult setting styles * Remove deprecated global labels * Handle underage users on mod screen * Adjust font sizes * Swap in RichText * Like button improvements * Tweaks to Labeler profile * Design tweaks for mod pref dialog * Add tertiary button color * Switch moderation UIs to tertiary color * Update mutewords and hiddenposts to use the new sdk * Add test-environment mod authority * Switch 'gore' to 'graphic-media' * Move nudity out of the adult content control * Remove focus styles from buttons - let the browser behavior handle it * Fixes to the adult content age-gating in moderaiton * Ditch tertiary button color, lighten secondary button * Fix some colors * Remove focused overrides from toggles * Liked by screen * Rework the moderationlabelpref * Fix optimistic like * Cleanup * Change how onboarding handles adult content enabled/disabled * Add special handling of the mod authorities * Tweaks * Update the default labeler avatar to a shield * Add route to go server * Avoid dups due to bad config * Fix attrs * Fix: dont try to detect link/label mismatches on post meta * Correctly show the label behavior when adult content is disabled * Readd the local hiddenPosts handling * WIP * Fix bad merge * Conten hider design tweaks * Fix text string breakage * Adjust source text in ContentHider * Fix link bug * Design tweaks to ContentHider and ModDetailsDialog * Adjust spacing of inform badges * Adjust spacing of embeds in posts * Style tweaks to post/profile alerts * Labels on me and dialog * Remove bad focus styles from post dropdown * Better spacing solution * Tune moderation UIs * Moderation UI tweaks for mobile * Move labelers query on Mod screen * Update to use new SDK appLabelers semantics * Implement report submission * Replace the report modal entirely with the report dialog * Add @ to mod details dialog handle * Bump SDK package * Remove silly type * Add to AWS build CI * Fix ToggleButton overflow * Clean up ModServiceCard, rename to LabelingServiceCard * Hackfix to translate gore labels to graphic-media * Tune content hider sizing on web desktop * Handle self labels * Fix spacing below text-only posts * Fix: send appeals to the right labeler * Give mod page links interactive states * Fix references * Remove focus handling * Remove remnant * Remove the like count from the subscribed labeler listing * Bump @atproto/api@0.11.1 * Remove extra @ * Fix: persist labels to local storage to reduce coverage gaps * update dipendencies * revert dipendencies * Add some explainers on how blocking affects labelers * Tweak copy * Fix underline color in header * Fix profile menu * Handle card overflow * Remove metrics from header * Mute 'account' not 'user' * Show metrics if self * Show the labels tab on logged out view * Fix bad merge * Use purple theming on labelers * Tighten space on LabelerCard * Set staleTime to 6hrs for labeler details * Memoize the memoizers * Drop staleTime to 60s * Move label defs into a context to reduce recomputes * Submit view tweaks * Move labeler fetch below auth * Mitigation: hardcode the bluesky moderation labeler name * Bump sdk * Add missing translated string Co-authored-by: Takayuki KUSANO <65759+tkusano@users.noreply.github.com> * Add missing translated string Co-authored-by: Takayuki KUSANO <65759+tkusano@users.noreply.github.com> * Hailey's fix for incorrect profile tabs Co-authored-by: Hailey <me@haileyok.com> * Feedback * Fix borders, add bottom space * Hailey's fix pt 2 Co-authored-by: Hailey <me@haileyok.com> * Fix post tabs * Integrate feedback pt 1 Co-authored-by: Takayuki KUSANO <65759+tkusano@users.noreply.github.com> * Integrate feedback pt 2 Co-authored-by: Takayuki KUSANO <65759+tkusano@users.noreply.github.com> * Integrate feedback pt 3 Co-authored-by: Takayuki KUSANO <65759+tkusano@users.noreply.github.com> * Integrate feedback pt 4 Co-authored-by: Takayuki KUSANO <65759+tkusano@users.noreply.github.com> * Integrate feedback pt 5 Co-authored-by: Takayuki KUSANO <65759+tkusano@users.noreply.github.com> * Integrate feedback pt 6 Co-authored-by: Takayuki KUSANO <65759+tkusano@users.noreply.github.com> * Integrate feedback pt 7 Co-authored-by: Takayuki KUSANO <65759+tkusano@users.noreply.github.com> * Integrate feedback pt 8 Co-authored-by: Takayuki KUSANO <65759+tkusano@users.noreply.github.com> * Format * Integrate new bday modal * Use public agent for getServices * Update casing --------- Co-authored-by: Eric Bailey <git@esb.lol> Co-authored-by: Takayuki KUSANO <65759+tkusano@users.noreply.github.com> Co-authored-by: Hailey <me@haileyok.com>
293 lines
6.9 KiB
TypeScript
293 lines
6.9 KiB
TypeScript
/* eslint-disable react/prop-types */
|
|
|
|
import React from 'react'
|
|
import {View, Pressable, ViewStyle, StyleProp} from 'react-native'
|
|
import * as DropdownMenu from '@radix-ui/react-dropdown-menu'
|
|
import {msg} from '@lingui/macro'
|
|
import {useLingui} from '@lingui/react'
|
|
|
|
import * as Dialog from '#/components/Dialog'
|
|
import {useInteractionState} from '#/components/hooks/useInteractionState'
|
|
import {atoms as a, useTheme, flatten, web} from '#/alf'
|
|
import {Text} from '#/components/Typography'
|
|
|
|
import {
|
|
ContextType,
|
|
TriggerProps,
|
|
ItemProps,
|
|
GroupProps,
|
|
ItemTextProps,
|
|
ItemIconProps,
|
|
RadixPassThroughTriggerProps,
|
|
} from '#/components/Menu/types'
|
|
import {Context} from '#/components/Menu/context'
|
|
import {Portal} from '#/components/Portal'
|
|
|
|
export function useMenuControl(): Dialog.DialogControlProps {
|
|
const id = React.useId()
|
|
const [isOpen, setIsOpen] = React.useState(false)
|
|
|
|
return React.useMemo(
|
|
() => ({
|
|
id,
|
|
ref: {current: null},
|
|
isOpen,
|
|
open() {
|
|
setIsOpen(true)
|
|
},
|
|
close() {
|
|
setIsOpen(false)
|
|
},
|
|
}),
|
|
[id, isOpen, setIsOpen],
|
|
)
|
|
}
|
|
|
|
export function useMemoControlContext() {
|
|
return React.useContext(Context)
|
|
}
|
|
|
|
export function Root({
|
|
children,
|
|
control,
|
|
}: React.PropsWithChildren<{
|
|
control?: Dialog.DialogOuterProps['control']
|
|
}>) {
|
|
const {_} = useLingui()
|
|
const defaultControl = useMenuControl()
|
|
const context = React.useMemo<ContextType>(
|
|
() => ({
|
|
control: control || defaultControl,
|
|
}),
|
|
[control, defaultControl],
|
|
)
|
|
const onOpenChange = React.useCallback(
|
|
(open: boolean) => {
|
|
if (context.control.isOpen && !open) {
|
|
context.control.close()
|
|
} else if (!context.control.isOpen && open) {
|
|
context.control.open()
|
|
}
|
|
},
|
|
[context.control],
|
|
)
|
|
|
|
return (
|
|
<Context.Provider value={context}>
|
|
{context.control.isOpen && (
|
|
<Portal>
|
|
<Pressable
|
|
style={[a.fixed, a.inset_0, a.z_50]}
|
|
onPress={() => context.control.close()}
|
|
accessibilityHint=""
|
|
accessibilityLabel={_(
|
|
msg`Context menu backdrop, click to close the menu.`,
|
|
)}
|
|
/>
|
|
</Portal>
|
|
)}
|
|
<DropdownMenu.Root
|
|
open={context.control.isOpen}
|
|
onOpenChange={onOpenChange}>
|
|
{children}
|
|
</DropdownMenu.Root>
|
|
</Context.Provider>
|
|
)
|
|
}
|
|
|
|
const RadixTriggerPassThrough = React.forwardRef(
|
|
(
|
|
props: {
|
|
children: (
|
|
props: RadixPassThroughTriggerProps & {
|
|
ref: React.Ref<any>
|
|
},
|
|
) => React.ReactNode
|
|
},
|
|
ref,
|
|
) => {
|
|
// @ts-expect-error Radix provides no types of this stuff
|
|
return props.children({...props, ref})
|
|
},
|
|
)
|
|
RadixTriggerPassThrough.displayName = 'RadixTriggerPassThrough'
|
|
|
|
export function Trigger({children, label}: TriggerProps) {
|
|
const {control} = React.useContext(Context)
|
|
const {
|
|
state: hovered,
|
|
onIn: onMouseEnter,
|
|
onOut: onMouseLeave,
|
|
} = useInteractionState()
|
|
const {state: focused, onIn: onFocus, onOut: onBlur} = useInteractionState()
|
|
|
|
return (
|
|
<DropdownMenu.Trigger asChild>
|
|
<RadixTriggerPassThrough>
|
|
{props =>
|
|
children({
|
|
isNative: false,
|
|
control,
|
|
state: {
|
|
hovered,
|
|
focused,
|
|
pressed: false,
|
|
},
|
|
props: {
|
|
...props,
|
|
// disable on web, use `onPress`
|
|
onPointerDown: () => false,
|
|
onPress: () =>
|
|
control.isOpen ? control.close() : control.open(),
|
|
onFocus: onFocus,
|
|
onBlur: onBlur,
|
|
onMouseEnter,
|
|
onMouseLeave,
|
|
accessibilityLabel: label,
|
|
},
|
|
})
|
|
}
|
|
</RadixTriggerPassThrough>
|
|
</DropdownMenu.Trigger>
|
|
)
|
|
}
|
|
|
|
export function Outer({
|
|
children,
|
|
style,
|
|
}: React.PropsWithChildren<{
|
|
showCancel?: boolean
|
|
style?: StyleProp<ViewStyle>
|
|
}>) {
|
|
const t = useTheme()
|
|
|
|
return (
|
|
<DropdownMenu.Portal>
|
|
<DropdownMenu.Content sideOffset={5} loop aria-label="Test">
|
|
<View
|
|
style={[
|
|
a.rounded_sm,
|
|
a.p_xs,
|
|
t.name === 'light' ? t.atoms.bg : t.atoms.bg_contrast_25,
|
|
t.atoms.shadow_md,
|
|
style,
|
|
]}>
|
|
{children}
|
|
</View>
|
|
|
|
{/* Disabled until we can fix positioning
|
|
<DropdownMenu.Arrow
|
|
className="DropdownMenuArrow"
|
|
fill={
|
|
(t.name === 'light' ? t.atoms.bg : t.atoms.bg_contrast_25)
|
|
.backgroundColor
|
|
}
|
|
/>
|
|
*/}
|
|
</DropdownMenu.Content>
|
|
</DropdownMenu.Portal>
|
|
)
|
|
}
|
|
|
|
export function Item({children, label, onPress, ...rest}: ItemProps) {
|
|
const t = useTheme()
|
|
const {control} = React.useContext(Context)
|
|
const {
|
|
state: hovered,
|
|
onIn: onMouseEnter,
|
|
onOut: onMouseLeave,
|
|
} = useInteractionState()
|
|
const {state: focused, onIn: onFocus, onOut: onBlur} = useInteractionState()
|
|
|
|
return (
|
|
<DropdownMenu.Item asChild>
|
|
<Pressable
|
|
{...rest}
|
|
className="radix-dropdown-item"
|
|
accessibilityHint=""
|
|
accessibilityLabel={label}
|
|
onPress={e => {
|
|
onPress(e)
|
|
|
|
/**
|
|
* Ported forward from Radix
|
|
* @see https://www.radix-ui.com/primitives/docs/components/dropdown-menu#item
|
|
*/
|
|
if (!e.defaultPrevented) {
|
|
control.close()
|
|
}
|
|
}}
|
|
onFocus={onFocus}
|
|
onBlur={onBlur}
|
|
// need `flatten` here for Radix compat
|
|
style={flatten([
|
|
a.flex_row,
|
|
a.align_center,
|
|
a.gap_lg,
|
|
a.py_sm,
|
|
a.rounded_xs,
|
|
{minHeight: 32, paddingHorizontal: 10},
|
|
web({outline: 0}),
|
|
(hovered || focused) && [
|
|
web({outline: '0 !important'}),
|
|
t.name === 'light'
|
|
? t.atoms.bg_contrast_25
|
|
: t.atoms.bg_contrast_50,
|
|
],
|
|
])}
|
|
{...web({
|
|
onMouseEnter,
|
|
onMouseLeave,
|
|
})}>
|
|
{children}
|
|
</Pressable>
|
|
</DropdownMenu.Item>
|
|
)
|
|
}
|
|
|
|
export function ItemText({children, style}: ItemTextProps) {
|
|
const t = useTheme()
|
|
return (
|
|
<Text style={[a.flex_1, a.font_bold, t.atoms.text_contrast_high, style]}>
|
|
{children}
|
|
</Text>
|
|
)
|
|
}
|
|
|
|
export function ItemIcon({icon: Comp, position = 'left'}: ItemIconProps) {
|
|
const t = useTheme()
|
|
return (
|
|
<Comp
|
|
size="md"
|
|
fill={t.atoms.text_contrast_medium.color}
|
|
style={[
|
|
position === 'left' && {
|
|
marginLeft: -2,
|
|
},
|
|
position === 'right' && {
|
|
marginRight: -2,
|
|
marginLeft: 12,
|
|
},
|
|
]}
|
|
/>
|
|
)
|
|
}
|
|
|
|
export function Group({children}: GroupProps) {
|
|
return children
|
|
}
|
|
|
|
export function Divider() {
|
|
const t = useTheme()
|
|
return (
|
|
<DropdownMenu.Separator
|
|
style={flatten([
|
|
a.my_xs,
|
|
t.atoms.bg_contrast_100,
|
|
{
|
|
height: 1,
|
|
},
|
|
])}
|
|
/>
|
|
)
|
|
}
|