* 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>
566 lines
15 KiB
TypeScript
566 lines
15 KiB
TypeScript
import React from 'react'
|
|
import {
|
|
Pressable,
|
|
Text,
|
|
PressableProps,
|
|
TextProps,
|
|
ViewStyle,
|
|
AccessibilityProps,
|
|
View,
|
|
TextStyle,
|
|
StyleSheet,
|
|
StyleProp,
|
|
} from 'react-native'
|
|
import LinearGradient from 'react-native-linear-gradient'
|
|
|
|
import {useTheme, atoms as a, tokens, android, flatten} from '#/alf'
|
|
import {Props as SVGIconProps} from '#/components/icons/common'
|
|
import {normalizeTextStyles} from '#/components/Typography'
|
|
|
|
export type ButtonVariant = 'solid' | 'outline' | 'ghost' | 'gradient'
|
|
export type ButtonColor =
|
|
| 'primary'
|
|
| 'secondary'
|
|
| 'negative'
|
|
| 'gradient_sky'
|
|
| 'gradient_midnight'
|
|
| 'gradient_sunrise'
|
|
| 'gradient_sunset'
|
|
| 'gradient_nordic'
|
|
| 'gradient_bonfire'
|
|
export type ButtonSize = 'tiny' | 'small' | 'medium' | 'large'
|
|
export type ButtonShape = 'round' | 'square' | 'default'
|
|
export type VariantProps = {
|
|
/**
|
|
* The style variation of the button
|
|
*/
|
|
variant?: ButtonVariant
|
|
/**
|
|
* The color of the button
|
|
*/
|
|
color?: ButtonColor
|
|
/**
|
|
* The size of the button
|
|
*/
|
|
size?: ButtonSize
|
|
/**
|
|
* The shape of the button
|
|
*/
|
|
shape?: ButtonShape
|
|
}
|
|
|
|
export type ButtonState = {
|
|
hovered: boolean
|
|
focused: boolean
|
|
pressed: boolean
|
|
disabled: boolean
|
|
}
|
|
|
|
export type ButtonContext = VariantProps & ButtonState
|
|
|
|
export type ButtonProps = Pick<
|
|
PressableProps,
|
|
'disabled' | 'onPress' | 'testID'
|
|
> &
|
|
AccessibilityProps &
|
|
VariantProps & {
|
|
testID?: string
|
|
label: string
|
|
style?: StyleProp<ViewStyle>
|
|
children:
|
|
| React.ReactNode
|
|
| string
|
|
| ((context: ButtonContext) => React.ReactNode | string)
|
|
}
|
|
export type ButtonTextProps = TextProps & VariantProps & {disabled?: boolean}
|
|
|
|
const Context = React.createContext<VariantProps & ButtonState>({
|
|
hovered: false,
|
|
focused: false,
|
|
pressed: false,
|
|
disabled: false,
|
|
})
|
|
|
|
export function useButtonContext() {
|
|
return React.useContext(Context)
|
|
}
|
|
|
|
export function Button({
|
|
children,
|
|
variant,
|
|
color,
|
|
size,
|
|
shape = 'default',
|
|
label,
|
|
disabled = false,
|
|
style,
|
|
...rest
|
|
}: ButtonProps) {
|
|
const t = useTheme()
|
|
const [state, setState] = React.useState({
|
|
pressed: false,
|
|
hovered: false,
|
|
focused: false,
|
|
})
|
|
|
|
const onPressIn = React.useCallback(() => {
|
|
setState(s => ({
|
|
...s,
|
|
pressed: true,
|
|
}))
|
|
}, [setState])
|
|
const onPressOut = React.useCallback(() => {
|
|
setState(s => ({
|
|
...s,
|
|
pressed: false,
|
|
}))
|
|
}, [setState])
|
|
const onHoverIn = React.useCallback(() => {
|
|
setState(s => ({
|
|
...s,
|
|
hovered: true,
|
|
}))
|
|
}, [setState])
|
|
const onHoverOut = React.useCallback(() => {
|
|
setState(s => ({
|
|
...s,
|
|
hovered: false,
|
|
}))
|
|
}, [setState])
|
|
const onFocus = React.useCallback(() => {
|
|
setState(s => ({
|
|
...s,
|
|
focused: true,
|
|
}))
|
|
}, [setState])
|
|
const onBlur = React.useCallback(() => {
|
|
setState(s => ({
|
|
...s,
|
|
focused: false,
|
|
}))
|
|
}, [setState])
|
|
|
|
const {baseStyles, hoverStyles} = React.useMemo(() => {
|
|
const baseStyles: ViewStyle[] = []
|
|
const hoverStyles: ViewStyle[] = []
|
|
const light = t.name === 'light'
|
|
|
|
if (color === 'primary') {
|
|
if (variant === 'solid') {
|
|
if (!disabled) {
|
|
baseStyles.push({
|
|
backgroundColor: t.palette.primary_500,
|
|
})
|
|
hoverStyles.push({
|
|
backgroundColor: t.palette.primary_600,
|
|
})
|
|
} else {
|
|
baseStyles.push({
|
|
backgroundColor: t.palette.primary_700,
|
|
})
|
|
}
|
|
} else if (variant === 'outline') {
|
|
baseStyles.push(a.border, t.atoms.bg, {
|
|
borderWidth: 1,
|
|
})
|
|
|
|
if (!disabled) {
|
|
baseStyles.push(a.border, {
|
|
borderColor: t.palette.primary_500,
|
|
})
|
|
hoverStyles.push(a.border, {
|
|
backgroundColor: light
|
|
? t.palette.primary_50
|
|
: t.palette.primary_950,
|
|
})
|
|
} else {
|
|
baseStyles.push(a.border, {
|
|
borderColor: light ? t.palette.primary_200 : t.palette.primary_900,
|
|
})
|
|
}
|
|
} else if (variant === 'ghost') {
|
|
if (!disabled) {
|
|
baseStyles.push(t.atoms.bg)
|
|
hoverStyles.push({
|
|
backgroundColor: light
|
|
? t.palette.primary_100
|
|
: t.palette.primary_900,
|
|
})
|
|
}
|
|
}
|
|
} else if (color === 'secondary') {
|
|
if (variant === 'solid') {
|
|
if (!disabled) {
|
|
baseStyles.push({
|
|
backgroundColor: t.palette.contrast_25,
|
|
})
|
|
hoverStyles.push({
|
|
backgroundColor: t.palette.contrast_50,
|
|
})
|
|
} else {
|
|
baseStyles.push({
|
|
backgroundColor: t.palette.contrast_100,
|
|
})
|
|
}
|
|
} else if (variant === 'outline') {
|
|
baseStyles.push(a.border, t.atoms.bg, {
|
|
borderWidth: 1,
|
|
})
|
|
|
|
if (!disabled) {
|
|
baseStyles.push(a.border, {
|
|
borderColor: t.palette.contrast_300,
|
|
})
|
|
hoverStyles.push(t.atoms.bg_contrast_50)
|
|
} else {
|
|
baseStyles.push(a.border, {
|
|
borderColor: t.palette.contrast_200,
|
|
})
|
|
}
|
|
} else if (variant === 'ghost') {
|
|
if (!disabled) {
|
|
baseStyles.push(t.atoms.bg)
|
|
hoverStyles.push({
|
|
backgroundColor: t.palette.contrast_100,
|
|
})
|
|
}
|
|
}
|
|
} else if (color === 'negative') {
|
|
if (variant === 'solid') {
|
|
if (!disabled) {
|
|
baseStyles.push({
|
|
backgroundColor: t.palette.negative_500,
|
|
})
|
|
hoverStyles.push({
|
|
backgroundColor: t.palette.negative_600,
|
|
})
|
|
} else {
|
|
baseStyles.push({
|
|
backgroundColor: t.palette.negative_700,
|
|
})
|
|
}
|
|
} else if (variant === 'outline') {
|
|
baseStyles.push(a.border, t.atoms.bg, {
|
|
borderWidth: 1,
|
|
})
|
|
|
|
if (!disabled) {
|
|
baseStyles.push(a.border, {
|
|
borderColor: t.palette.negative_500,
|
|
})
|
|
hoverStyles.push(a.border, {
|
|
backgroundColor: light
|
|
? t.palette.negative_50
|
|
: t.palette.negative_975,
|
|
})
|
|
} else {
|
|
baseStyles.push(a.border, {
|
|
borderColor: light
|
|
? t.palette.negative_200
|
|
: t.palette.negative_900,
|
|
})
|
|
}
|
|
} else if (variant === 'ghost') {
|
|
if (!disabled) {
|
|
baseStyles.push(t.atoms.bg)
|
|
hoverStyles.push({
|
|
backgroundColor: light
|
|
? t.palette.negative_100
|
|
: t.palette.negative_975,
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
if (shape === 'default') {
|
|
if (size === 'large') {
|
|
baseStyles.push({paddingVertical: 15}, a.px_2xl, a.rounded_sm, a.gap_md)
|
|
} else if (size === 'medium') {
|
|
baseStyles.push({paddingVertical: 12}, a.px_2xl, a.rounded_sm, a.gap_md)
|
|
} else if (size === 'small') {
|
|
baseStyles.push({paddingVertical: 9}, a.px_lg, a.rounded_sm, a.gap_sm)
|
|
} else if (size === 'tiny') {
|
|
baseStyles.push({paddingVertical: 4}, a.px_sm, a.rounded_xs, a.gap_xs)
|
|
}
|
|
} else if (shape === 'round' || shape === 'square') {
|
|
if (size === 'large') {
|
|
if (shape === 'round') {
|
|
baseStyles.push({height: 54, width: 54})
|
|
} else {
|
|
baseStyles.push({height: 50, width: 50})
|
|
}
|
|
} else if (size === 'small') {
|
|
baseStyles.push({height: 40, width: 40})
|
|
} else if (size === 'tiny') {
|
|
baseStyles.push({height: 20, width: 20})
|
|
}
|
|
|
|
if (shape === 'round') {
|
|
baseStyles.push(a.rounded_full)
|
|
} else if (shape === 'square') {
|
|
if (size === 'tiny') {
|
|
baseStyles.push(a.rounded_xs)
|
|
} else {
|
|
baseStyles.push(a.rounded_sm)
|
|
}
|
|
}
|
|
}
|
|
|
|
return {
|
|
baseStyles,
|
|
hoverStyles,
|
|
}
|
|
}, [t, variant, color, size, shape, disabled])
|
|
|
|
const {gradientColors, gradientHoverColors, gradientLocations} =
|
|
React.useMemo(() => {
|
|
const colors: string[] = []
|
|
const hoverColors: string[] = []
|
|
const locations: number[] = []
|
|
const gradient = {
|
|
primary: tokens.gradients.sky,
|
|
secondary: tokens.gradients.sky,
|
|
negative: tokens.gradients.sky,
|
|
gradient_sky: tokens.gradients.sky,
|
|
gradient_midnight: tokens.gradients.midnight,
|
|
gradient_sunrise: tokens.gradients.sunrise,
|
|
gradient_sunset: tokens.gradients.sunset,
|
|
gradient_nordic: tokens.gradients.nordic,
|
|
gradient_bonfire: tokens.gradients.bonfire,
|
|
}[color || 'primary']
|
|
|
|
if (variant === 'gradient') {
|
|
colors.push(...gradient.values.map(([_, color]) => color))
|
|
hoverColors.push(...gradient.values.map(_ => gradient.hover_value))
|
|
locations.push(...gradient.values.map(([location, _]) => location))
|
|
}
|
|
|
|
return {
|
|
gradientColors: colors,
|
|
gradientHoverColors: hoverColors,
|
|
gradientLocations: locations,
|
|
}
|
|
}, [variant, color])
|
|
|
|
const context = React.useMemo<ButtonContext>(
|
|
() => ({
|
|
...state,
|
|
variant,
|
|
color,
|
|
size,
|
|
disabled: disabled || false,
|
|
}),
|
|
[state, variant, color, size, disabled],
|
|
)
|
|
|
|
const flattenedBaseStyles = flatten(baseStyles)
|
|
|
|
return (
|
|
<Pressable
|
|
role="button"
|
|
accessibilityHint={undefined} // optional
|
|
{...rest}
|
|
aria-label={label}
|
|
aria-pressed={state.pressed}
|
|
accessibilityLabel={label}
|
|
disabled={disabled || false}
|
|
accessibilityState={{
|
|
disabled: disabled || false,
|
|
}}
|
|
style={[
|
|
a.flex_row,
|
|
a.align_center,
|
|
a.justify_center,
|
|
flattenedBaseStyles,
|
|
...(state.hovered || state.pressed ? hoverStyles : []),
|
|
flatten(style),
|
|
]}
|
|
onPressIn={onPressIn}
|
|
onPressOut={onPressOut}
|
|
onHoverIn={onHoverIn}
|
|
onHoverOut={onHoverOut}
|
|
onFocus={onFocus}
|
|
onBlur={onBlur}>
|
|
{variant === 'gradient' && (
|
|
<View
|
|
style={[
|
|
a.absolute,
|
|
a.inset_0,
|
|
a.overflow_hidden,
|
|
{borderRadius: flattenedBaseStyles.borderRadius},
|
|
]}>
|
|
<LinearGradient
|
|
colors={
|
|
state.hovered || state.pressed
|
|
? gradientHoverColors
|
|
: gradientColors
|
|
}
|
|
locations={gradientLocations}
|
|
start={{x: 0, y: 0}}
|
|
end={{x: 1, y: 1}}
|
|
style={[a.absolute, a.inset_0]}
|
|
/>
|
|
</View>
|
|
)}
|
|
<Context.Provider value={context}>
|
|
{typeof children === 'string' ? (
|
|
<ButtonText>{children}</ButtonText>
|
|
) : typeof children === 'function' ? (
|
|
children(context)
|
|
) : (
|
|
children
|
|
)}
|
|
</Context.Provider>
|
|
</Pressable>
|
|
)
|
|
}
|
|
|
|
export function useSharedButtonTextStyles() {
|
|
const t = useTheme()
|
|
const {color, variant, disabled, size} = useButtonContext()
|
|
return React.useMemo(() => {
|
|
const baseStyles: TextStyle[] = []
|
|
const light = t.name === 'light'
|
|
|
|
if (color === 'primary') {
|
|
if (variant === 'solid') {
|
|
if (!disabled) {
|
|
baseStyles.push({color: t.palette.white})
|
|
} else {
|
|
baseStyles.push({color: t.palette.white, opacity: 0.5})
|
|
}
|
|
} else if (variant === 'outline') {
|
|
if (!disabled) {
|
|
baseStyles.push({
|
|
color: light ? t.palette.primary_600 : t.palette.primary_500,
|
|
})
|
|
} else {
|
|
baseStyles.push({color: t.palette.primary_600, opacity: 0.5})
|
|
}
|
|
} else if (variant === 'ghost') {
|
|
if (!disabled) {
|
|
baseStyles.push({color: t.palette.primary_600})
|
|
} else {
|
|
baseStyles.push({color: t.palette.primary_600, opacity: 0.5})
|
|
}
|
|
}
|
|
} else if (color === 'secondary') {
|
|
if (variant === 'solid' || variant === 'gradient') {
|
|
if (!disabled) {
|
|
baseStyles.push({
|
|
color: t.palette.contrast_700,
|
|
})
|
|
} else {
|
|
baseStyles.push({
|
|
color: t.palette.contrast_400,
|
|
})
|
|
}
|
|
} else if (variant === 'outline') {
|
|
if (!disabled) {
|
|
baseStyles.push({
|
|
color: t.palette.contrast_600,
|
|
})
|
|
} else {
|
|
baseStyles.push({
|
|
color: t.palette.contrast_300,
|
|
})
|
|
}
|
|
} else if (variant === 'ghost') {
|
|
if (!disabled) {
|
|
baseStyles.push({
|
|
color: t.palette.contrast_600,
|
|
})
|
|
} else {
|
|
baseStyles.push({
|
|
color: t.palette.contrast_300,
|
|
})
|
|
}
|
|
}
|
|
} else if (color === 'negative') {
|
|
if (variant === 'solid' || variant === 'gradient') {
|
|
if (!disabled) {
|
|
baseStyles.push({color: t.palette.white})
|
|
} else {
|
|
baseStyles.push({color: t.palette.white, opacity: 0.5})
|
|
}
|
|
} else if (variant === 'outline') {
|
|
if (!disabled) {
|
|
baseStyles.push({color: t.palette.negative_400})
|
|
} else {
|
|
baseStyles.push({color: t.palette.negative_400, opacity: 0.5})
|
|
}
|
|
} else if (variant === 'ghost') {
|
|
if (!disabled) {
|
|
baseStyles.push({color: t.palette.negative_400})
|
|
} else {
|
|
baseStyles.push({color: t.palette.negative_400, opacity: 0.5})
|
|
}
|
|
}
|
|
} else {
|
|
if (!disabled) {
|
|
baseStyles.push({color: t.palette.white})
|
|
} else {
|
|
baseStyles.push({color: t.palette.white, opacity: 0.5})
|
|
}
|
|
}
|
|
|
|
if (size === 'large') {
|
|
baseStyles.push(a.text_md, android({paddingBottom: 1}))
|
|
} else if (size === 'tiny') {
|
|
baseStyles.push(a.text_xs, android({paddingBottom: 1}))
|
|
} else {
|
|
baseStyles.push(a.text_sm, android({paddingBottom: 1}))
|
|
}
|
|
|
|
return StyleSheet.flatten(baseStyles)
|
|
}, [t, variant, color, size, disabled])
|
|
}
|
|
|
|
export function ButtonText({children, style, ...rest}: ButtonTextProps) {
|
|
const textStyles = useSharedButtonTextStyles()
|
|
|
|
return (
|
|
<Text
|
|
{...rest}
|
|
style={normalizeTextStyles([
|
|
a.font_bold,
|
|
a.text_center,
|
|
textStyles,
|
|
style,
|
|
])}>
|
|
{children}
|
|
</Text>
|
|
)
|
|
}
|
|
|
|
export function ButtonIcon({
|
|
icon: Comp,
|
|
position,
|
|
size: iconSize,
|
|
}: {
|
|
icon: React.ComponentType<SVGIconProps>
|
|
position?: 'left' | 'right'
|
|
size?: SVGIconProps['size']
|
|
}) {
|
|
const {size, disabled} = useButtonContext()
|
|
const textStyles = useSharedButtonTextStyles()
|
|
|
|
return (
|
|
<View
|
|
style={[
|
|
a.z_20,
|
|
{
|
|
opacity: disabled ? 0.7 : 1,
|
|
marginLeft: position === 'left' ? -2 : 0,
|
|
marginRight: position === 'right' ? -2 : 0,
|
|
},
|
|
]}>
|
|
<Comp
|
|
size={
|
|
iconSize ?? (size === 'large' ? 'md' : size === 'tiny' ? 'xs' : 'sm')
|
|
}
|
|
style={[{color: textStyles.color, pointerEvents: 'none'}]}
|
|
/>
|
|
</View>
|
|
)
|
|
}
|