Merge branch 'eslint-rules' into 'develop'

Change ESLint rules, lint

See merge request soapbox-pub/soapbox!2298
This commit is contained in:
marcin mikołajczak 2023-02-16 00:06:57 +00:00
commit b3585bb348
337 changed files with 1301 additions and 1284 deletions

View file

@ -260,12 +260,29 @@ module.exports = {
}, },
], ],
'@typescript-eslint/no-duplicate-imports': 'error', '@typescript-eslint/no-duplicate-imports': 'error',
'@typescript-eslint/member-delimiter-style': [
'error',
{
multiline: {
delimiter: 'none',
},
singleline: {
delimiter: 'comma',
},
},
],
'promise/catch-or-return': 'error', 'promise/catch-or-return': 'error',
'react-hooks/rules-of-hooks': 'error', 'react-hooks/rules-of-hooks': 'error',
'tailwindcss/classnames-order': 'error', 'tailwindcss/classnames-order': [
'error',
{
classRegex: '^(base|container|icon|item|list|outer|wrapper)?[c|C]lass(Name)?$',
config: 'tailwind.config.cjs',
},
],
'tailwindcss/migration-from-tailwind-2': 'error', 'tailwindcss/migration-from-tailwind-2': 'error',
}, },
overrides: [ overrides: [

View file

@ -228,7 +228,7 @@ const fetchAccountFail = (id: string | null, error: AxiosError) => ({
}); });
type FollowAccountOpts = { type FollowAccountOpts = {
reblogs?: boolean, reblogs?: boolean
notify?: boolean notify?: boolean
}; };

View file

@ -34,8 +34,8 @@ type ExportDataActions = {
| typeof EXPORT_BLOCKS_FAIL | typeof EXPORT_BLOCKS_FAIL
| typeof EXPORT_MUTES_REQUEST | typeof EXPORT_MUTES_REQUEST
| typeof EXPORT_MUTES_SUCCESS | typeof EXPORT_MUTES_SUCCESS
| typeof EXPORT_MUTES_FAIL, | typeof EXPORT_MUTES_FAIL
error?: any, error?: any
} }
function fileExport(content: string, fileName: string) { function fileExport(content: string, fileName: string) {

View file

@ -11,25 +11,25 @@ export const FAMILIAR_FOLLOWERS_FETCH_SUCCESS = 'FAMILIAR_FOLLOWERS_FETCH_SUCCES
export const FAMILIAR_FOLLOWERS_FETCH_FAIL = 'FAMILIAR_FOLLOWERS_FETCH_FAIL'; export const FAMILIAR_FOLLOWERS_FETCH_FAIL = 'FAMILIAR_FOLLOWERS_FETCH_FAIL';
type FamiliarFollowersFetchRequestAction = { type FamiliarFollowersFetchRequestAction = {
type: typeof FAMILIAR_FOLLOWERS_FETCH_REQUEST, type: typeof FAMILIAR_FOLLOWERS_FETCH_REQUEST
id: string, id: string
} }
type FamiliarFollowersFetchRequestSuccessAction = { type FamiliarFollowersFetchRequestSuccessAction = {
type: typeof FAMILIAR_FOLLOWERS_FETCH_SUCCESS, type: typeof FAMILIAR_FOLLOWERS_FETCH_SUCCESS
id: string, id: string
accounts: Array<APIEntity>, accounts: Array<APIEntity>
} }
type FamiliarFollowersFetchRequestFailAction = { type FamiliarFollowersFetchRequestFailAction = {
type: typeof FAMILIAR_FOLLOWERS_FETCH_FAIL, type: typeof FAMILIAR_FOLLOWERS_FETCH_FAIL
id: string, id: string
error: any, error: any
} }
type AccountsImportAction = { type AccountsImportAction = {
type: typeof ACCOUNTS_IMPORT, type: typeof ACCOUNTS_IMPORT
accounts: Array<APIEntity>, accounts: Array<APIEntity>
} }
export type FamiliarFollowersActions = FamiliarFollowersFetchRequestAction | FamiliarFollowersFetchRequestSuccessAction | FamiliarFollowersFetchRequestFailAction | AccountsImportAction export type FamiliarFollowersActions = FamiliarFollowersFetchRequestAction | FamiliarFollowersFetchRequestSuccessAction | FamiliarFollowersFetchRequestFailAction | AccountsImportAction

View file

@ -27,8 +27,8 @@ type ImportDataActions = {
| typeof IMPORT_BLOCKS_FAIL | typeof IMPORT_BLOCKS_FAIL
| typeof IMPORT_MUTES_REQUEST | typeof IMPORT_MUTES_REQUEST
| typeof IMPORT_MUTES_SUCCESS | typeof IMPORT_MUTES_SUCCESS
| typeof IMPORT_MUTES_FAIL, | typeof IMPORT_MUTES_FAIL
error?: any, error?: any
config?: string config?: string
} }

View file

@ -37,8 +37,8 @@ const subscribe = (registration: ServiceWorkerRegistration, getState: () => Root
}); });
const unsubscribe = ({ registration, subscription }: { const unsubscribe = ({ registration, subscription }: {
registration: ServiceWorkerRegistration, registration: ServiceWorkerRegistration
subscription: PushSubscription | null, subscription: PushSubscription | null
}) => }) =>
subscription ? subscription.unsubscribe().then(() => registration) : new Promise<ServiceWorkerRegistration>(r => r(registration)); subscription ? subscription.unsubscribe().then(() => registration) : new Promise<ServiceWorkerRegistration>(r => r(registration));
@ -82,8 +82,8 @@ const register = () =>
.then(getPushSubscription) .then(getPushSubscription)
// @ts-ignore // @ts-ignore
.then(({ registration, subscription }: { .then(({ registration, subscription }: {
registration: ServiceWorkerRegistration, registration: ServiceWorkerRegistration
subscription: PushSubscription | null, subscription: PushSubscription | null
}) => { }) => {
if (subscription !== null) { if (subscription !== null) {
// We have a subscription, check if it is still valid // We have a subscription, check if it is still valid

View file

@ -21,7 +21,7 @@ const REPORT_BLOCK_CHANGE = 'REPORT_BLOCK_CHANGE';
const REPORT_RULE_CHANGE = 'REPORT_RULE_CHANGE'; const REPORT_RULE_CHANGE = 'REPORT_RULE_CHANGE';
type ReportedEntity = { type ReportedEntity = {
status?: Status, status?: Status
chatMessage?: ChatMessage chatMessage?: ChatMessage
} }

View file

@ -18,7 +18,7 @@ const FE_NAME = 'soapbox_fe';
/** Options when changing/saving settings. */ /** Options when changing/saving settings. */
type SettingOpts = { type SettingOpts = {
/** Whether to display an alert when settings are saved. */ /** Whether to display an alert when settings are saved. */
showAlert?: boolean, showAlert?: boolean
} }
const messages = defineMessages({ const messages = defineMessages({

View file

@ -81,7 +81,7 @@ const updateChatQuery = (chat: IChat) => {
}; };
interface StreamOpts { interface StreamOpts {
statContext?: IStatContext, statContext?: IStatContext
} }
const connectTimelineStream = ( const connectTimelineStream = (

View file

@ -31,14 +31,14 @@ const AGE: Challenge = 'age';
export type Challenge = 'age' | 'sms' | 'email' export type Challenge = 'age' | 'sms' | 'email'
type Challenges = { type Challenges = {
email?: 0 | 1, email?: 0 | 1
sms?: 0 | 1, sms?: 0 | 1
age?: 0 | 1, age?: 0 | 1
} }
type Verification = { type Verification = {
token?: string, token?: string
challenges?: Challenges, challenges?: Challenges
challengeTypes?: Array<'age' | 'sms' | 'email'> challengeTypes?: Array<'age' | 'sms' | 'email'>
}; };

View file

@ -1,7 +1,7 @@
import React from 'react'; import React from 'react';
interface IInlineSVG { interface IInlineSVG {
loader?: JSX.Element, loader?: JSX.Element
} }
const InlineSVG: React.FC<IInlineSVG> = ({ loader }): JSX.Element => { const InlineSVG: React.FC<IInlineSVG> = ({ loader }): JSX.Element => {

View file

@ -12,9 +12,9 @@ const messages = defineMessages({
interface IAccountSearch { interface IAccountSearch {
/** Callback when a searched account is chosen. */ /** Callback when a searched account is chosen. */
onSelected: (accountId: string) => void, onSelected: (accountId: string) => void
/** Override the default placeholder of the input. */ /** Override the default placeholder of the input. */
placeholder?: string, placeholder?: string
} }
/** Input to search for accounts. */ /** Input to search for accounts. */

View file

@ -17,8 +17,8 @@ import type { StatusApprovalStatus } from 'soapbox/normalizers/status';
import type { Account as AccountEntity } from 'soapbox/types/entities'; import type { Account as AccountEntity } from 'soapbox/types/entities';
interface IInstanceFavicon { interface IInstanceFavicon {
account: AccountEntity, account: AccountEntity
disabled?: boolean, disabled?: boolean
} }
const messages = defineMessages({ const messages = defineMessages({
@ -53,7 +53,7 @@ const InstanceFavicon: React.FC<IInstanceFavicon> = ({ account, disabled }) => {
}; };
interface IProfilePopper { interface IProfilePopper {
condition: boolean, condition: boolean
wrapper: (children: React.ReactNode) => React.ReactNode wrapper: (children: React.ReactNode) => React.ReactNode
children: React.ReactNode children: React.ReactNode
} }
@ -67,30 +67,30 @@ const ProfilePopper: React.FC<IProfilePopper> = ({ condition, wrapper, children
}; };
export interface IAccount { export interface IAccount {
account: AccountEntity, account: AccountEntity
action?: React.ReactElement, action?: React.ReactElement
actionAlignment?: 'center' | 'top', actionAlignment?: 'center' | 'top'
actionIcon?: string, actionIcon?: string
actionTitle?: string, actionTitle?: string
/** Override other actions for specificity like mute/unmute. */ /** Override other actions for specificity like mute/unmute. */
actionType?: 'muting' | 'blocking' | 'follow_request', actionType?: 'muting' | 'blocking' | 'follow_request'
avatarSize?: number, avatarSize?: number
hidden?: boolean, hidden?: boolean
hideActions?: boolean, hideActions?: boolean
id?: string, id?: string
onActionClick?: (account: any) => void, onActionClick?: (account: any) => void
showProfileHoverCard?: boolean, showProfileHoverCard?: boolean
timestamp?: string, timestamp?: string
timestampUrl?: string, timestampUrl?: string
futureTimestamp?: boolean, futureTimestamp?: boolean
withAccountNote?: boolean, withAccountNote?: boolean
withDate?: boolean, withDate?: boolean
withLinkToProfile?: boolean, withLinkToProfile?: boolean
withRelationship?: boolean, withRelationship?: boolean
showEdit?: boolean, showEdit?: boolean
approvalStatus?: StatusApprovalStatus, approvalStatus?: StatusApprovalStatus
emoji?: string, emoji?: string
note?: string, note?: string
} }
const Account = ({ const Account = ({
@ -143,7 +143,7 @@ const Account = ({
title={actionTitle} title={actionTitle}
onClick={handleAction} onClick={handleAction}
className='bg-transparent text-gray-600 hover:text-gray-700 dark:text-gray-600 dark:hover:text-gray-500' className='bg-transparent text-gray-600 hover:text-gray-700 dark:text-gray-600 dark:hover:text-gray-500'
iconClassName='w-4 h-4' iconClassName='h-4 w-4'
/> />
); );
} }

View file

@ -15,8 +15,8 @@ const obfuscatedCount = (count: number) => {
}; };
interface IAnimatedNumber { interface IAnimatedNumber {
value: number; value: number
obfuscate?: boolean; obfuscate?: boolean
} }
const AnimatedNumber: React.FC<IAnimatedNumber> = ({ value, obfuscate }) => { const AnimatedNumber: React.FC<IAnimatedNumber> = ({ value, obfuscate }) => {

View file

@ -4,7 +4,7 @@ import { useHistory } from 'react-router-dom';
import type { Announcement as AnnouncementEntity, Mention as MentionEntity } from 'soapbox/types/entities'; import type { Announcement as AnnouncementEntity, Mention as MentionEntity } from 'soapbox/types/entities';
interface IAnnouncementContent { interface IAnnouncementContent {
announcement: AnnouncementEntity; announcement: AnnouncementEntity
} }
const AnnouncementContent: React.FC<IAnnouncementContent> = ({ announcement }) => { const AnnouncementContent: React.FC<IAnnouncementContent> = ({ announcement }) => {

View file

@ -11,10 +11,10 @@ import type { Map as ImmutableMap } from 'immutable';
import type { Announcement as AnnouncementEntity } from 'soapbox/types/entities'; import type { Announcement as AnnouncementEntity } from 'soapbox/types/entities';
interface IAnnouncement { interface IAnnouncement {
announcement: AnnouncementEntity; announcement: AnnouncementEntity
addReaction: (id: string, name: string) => void; addReaction: (id: string, name: string) => void
removeReaction: (id: string, name: string) => void; removeReaction: (id: string, name: string) => void
emojiMap: ImmutableMap<string, ImmutableMap<string, string>>; emojiMap: ImmutableMap<string, ImmutableMap<string, string>>
} }
const Announcement: React.FC<IAnnouncement> = ({ announcement, addReaction, removeReaction, emojiMap }) => { const Announcement: React.FC<IAnnouncement> = ({ announcement, addReaction, removeReaction, emojiMap }) => {

View file

@ -7,9 +7,9 @@ import { joinPublicPath } from 'soapbox/utils/static';
import type { Map as ImmutableMap } from 'immutable'; import type { Map as ImmutableMap } from 'immutable';
interface IEmoji { interface IEmoji {
emoji: string; emoji: string
emojiMap: ImmutableMap<string, ImmutableMap<string, string>>; emojiMap: ImmutableMap<string, ImmutableMap<string, string>>
hovered: boolean; hovered: boolean
} }
const Emoji: React.FC<IEmoji> = ({ emoji, emojiMap, hovered }) => { const Emoji: React.FC<IEmoji> = ({ emoji, emojiMap, hovered }) => {

View file

@ -10,12 +10,12 @@ import type { Map as ImmutableMap } from 'immutable';
import type { AnnouncementReaction } from 'soapbox/types/entities'; import type { AnnouncementReaction } from 'soapbox/types/entities';
interface IReaction { interface IReaction {
announcementId: string; announcementId: string
reaction: AnnouncementReaction; reaction: AnnouncementReaction
emojiMap: ImmutableMap<string, ImmutableMap<string, string>>; emojiMap: ImmutableMap<string, ImmutableMap<string, string>>
addReaction: (id: string, name: string) => void; addReaction: (id: string, name: string) => void
removeReaction: (id: string, name: string) => void; removeReaction: (id: string, name: string) => void
style: React.CSSProperties; style: React.CSSProperties
} }
const Reaction: React.FC<IReaction> = ({ announcementId, reaction, addReaction, removeReaction, emojiMap, style }) => { const Reaction: React.FC<IReaction> = ({ announcementId, reaction, addReaction, removeReaction, emojiMap, style }) => {

View file

@ -13,11 +13,11 @@ import type { Emoji } from 'soapbox/components/autosuggest-emoji';
import type { AnnouncementReaction } from 'soapbox/types/entities'; import type { AnnouncementReaction } from 'soapbox/types/entities';
interface IReactionsBar { interface IReactionsBar {
announcementId: string; announcementId: string
reactions: ImmutableList<AnnouncementReaction>; reactions: ImmutableList<AnnouncementReaction>
emojiMap: ImmutableMap<string, ImmutableMap<string, string>>; emojiMap: ImmutableMap<string, ImmutableMap<string, string>>
addReaction: (id: string, name: string) => void; addReaction: (id: string, name: string) => void
removeReaction: (id: string, name: string) => void; removeReaction: (id: string, name: string) => void
} }
const ReactionsBar: React.FC<IReactionsBar> = ({ announcementId, reactions, addReaction, removeReaction, emojiMap }) => { const ReactionsBar: React.FC<IReactionsBar> = ({ announcementId, reactions, addReaction, removeReaction, emojiMap }) => {

View file

@ -12,16 +12,16 @@ import type { InputThemes } from 'soapbox/components/ui/input/input';
const noOp = () => { }; const noOp = () => { };
interface IAutosuggestAccountInput { interface IAutosuggestAccountInput {
onChange: React.ChangeEventHandler<HTMLInputElement>, onChange: React.ChangeEventHandler<HTMLInputElement>
onSelected: (accountId: string) => void, onSelected: (accountId: string) => void
autoFocus?: boolean, autoFocus?: boolean
value: string, value: string
limit?: number, limit?: number
className?: string, className?: string
autoSelect?: boolean, autoSelect?: boolean
menu?: Menu, menu?: Menu
onKeyDown?: React.KeyboardEventHandler, onKeyDown?: React.KeyboardEventHandler
theme?: InputThemes, theme?: InputThemes
} }
const AutosuggestAccountInput: React.FC<IAutosuggestAccountInput> = ({ const AutosuggestAccountInput: React.FC<IAutosuggestAccountInput> = ({

View file

@ -4,19 +4,19 @@ import unicodeMapping from 'soapbox/features/emoji/emoji-unicode-mapping-light';
import { joinPublicPath } from 'soapbox/utils/static'; import { joinPublicPath } from 'soapbox/utils/static';
export type Emoji = { export type Emoji = {
id: string, id: string
custom: boolean, custom: boolean
imageUrl: string, imageUrl: string
native: string, native: string
colons: string, colons: string
} }
type UnicodeMapping = { type UnicodeMapping = {
filename: string, filename: string
} }
interface IAutosuggestEmoji { interface IAutosuggestEmoji {
emoji: Emoji, emoji: Emoji
} }
const AutosuggestEmoji: React.FC<IAutosuggestEmoji> = ({ emoji }) => { const AutosuggestEmoji: React.FC<IAutosuggestEmoji> = ({ emoji }) => {

View file

@ -16,23 +16,23 @@ import type { InputThemes } from 'soapbox/components/ui/input/input';
export type AutoSuggestion = string | Emoji; export type AutoSuggestion = string | Emoji;
export interface IAutosuggestInput extends Pick<React.HTMLAttributes<HTMLInputElement>, 'onChange' | 'onKeyUp' | 'onKeyDown'> { export interface IAutosuggestInput extends Pick<React.HTMLAttributes<HTMLInputElement>, 'onChange' | 'onKeyUp' | 'onKeyDown'> {
value: string, value: string
suggestions: ImmutableList<any>, suggestions: ImmutableList<any>
disabled?: boolean, disabled?: boolean
placeholder?: string, placeholder?: string
onSuggestionSelected: (tokenStart: number, lastToken: string | null, suggestion: AutoSuggestion) => void, onSuggestionSelected: (tokenStart: number, lastToken: string | null, suggestion: AutoSuggestion) => void
onSuggestionsClearRequested: () => void, onSuggestionsClearRequested: () => void
onSuggestionsFetchRequested: (token: string) => void, onSuggestionsFetchRequested: (token: string) => void
autoFocus: boolean, autoFocus: boolean
autoSelect: boolean, autoSelect: boolean
className?: string, className?: string
id?: string, id?: string
searchTokens: string[], searchTokens: string[]
maxLength?: number, maxLength?: number
menu?: Menu, menu?: Menu
renderSuggestion?: React.FC<{ id: string }>, renderSuggestion?: React.FC<{ id: string }>
hidePortal?: boolean, hidePortal?: boolean
theme?: InputThemes, theme?: InputThemes
} }
export default class AutosuggestInput extends ImmutablePureComponent<IAutosuggestInput> { export default class AutosuggestInput extends ImmutablePureComponent<IAutosuggestInput> {

View file

@ -19,7 +19,7 @@ export const ADDRESS_ICONS: Record<string, string> = {
}; };
interface IAutosuggestLocation { interface IAutosuggestLocation {
id: string, id: string
} }
const AutosuggestLocation: React.FC<IAutosuggestLocation> = ({ id }) => { const AutosuggestLocation: React.FC<IAutosuggestLocation> = ({ id }) => {

View file

@ -14,23 +14,23 @@ import AutosuggestEmoji, { Emoji } from './autosuggest-emoji';
import type { List as ImmutableList } from 'immutable'; import type { List as ImmutableList } from 'immutable';
interface IAutosuggesteTextarea { interface IAutosuggesteTextarea {
id?: string, id?: string
value: string, value: string
suggestions: ImmutableList<string>, suggestions: ImmutableList<string>
disabled: boolean, disabled: boolean
placeholder: string, placeholder: string
onSuggestionSelected: (tokenStart: number, token: string | null, value: string | undefined) => void, onSuggestionSelected: (tokenStart: number, token: string | null, value: string | undefined) => void
onSuggestionsClearRequested: () => void, onSuggestionsClearRequested: () => void
onSuggestionsFetchRequested: (token: string | number) => void, onSuggestionsFetchRequested: (token: string | number) => void
onChange: React.ChangeEventHandler<HTMLTextAreaElement>, onChange: React.ChangeEventHandler<HTMLTextAreaElement>
onKeyUp?: React.KeyboardEventHandler<HTMLTextAreaElement>, onKeyUp?: React.KeyboardEventHandler<HTMLTextAreaElement>
onKeyDown?: React.KeyboardEventHandler<HTMLTextAreaElement>, onKeyDown?: React.KeyboardEventHandler<HTMLTextAreaElement>
onPaste: (files: FileList) => void, onPaste: (files: FileList) => void
autoFocus: boolean, autoFocus: boolean
onFocus: () => void, onFocus: () => void
onBlur?: () => void, onBlur?: () => void
condensed?: boolean, condensed?: boolean
children: React.ReactNode, children: React.ReactNode
} }
class AutosuggestTextarea extends ImmutablePureComponent<IAutosuggesteTextarea> { class AutosuggestTextarea extends ImmutablePureComponent<IAutosuggesteTextarea> {

View file

@ -2,8 +2,8 @@ import clsx from 'clsx';
import React from 'react'; import React from 'react';
interface IBadge { interface IBadge {
title: React.ReactNode, title: React.ReactNode
slug: string, slug: string
} }
/** Badge to display on a user's profile. */ /** Badge to display on a user's profile. */
const Badge: React.FC<IBadge> = ({ title, slug }) => { const Badge: React.FC<IBadge> = ({ title, slug }) => {

View file

@ -15,9 +15,9 @@ const messages = defineMessages({
}); });
interface IBirthdayInput { interface IBirthdayInput {
value?: string, value?: string
onChange: (value: string) => void, onChange: (value: string) => void
required?: boolean, required?: boolean
} }
const BirthdayInput: React.FC<IBirthdayInput> = ({ value, onChange, required }) => { const BirthdayInput: React.FC<IBirthdayInput> = ({ value, onChange, required }) => {
@ -56,15 +56,15 @@ const BirthdayInput: React.FC<IBirthdayInput> = ({ value, onChange, required })
nextYearButtonDisabled, nextYearButtonDisabled,
date, date,
}: { }: {
decreaseMonth(): void, decreaseMonth(): void
increaseMonth(): void, increaseMonth(): void
prevMonthButtonDisabled: boolean, prevMonthButtonDisabled: boolean
nextMonthButtonDisabled: boolean, nextMonthButtonDisabled: boolean
decreaseYear(): void, decreaseYear(): void
increaseYear(): void, increaseYear(): void
prevYearButtonDisabled: boolean, prevYearButtonDisabled: boolean
nextYearButtonDisabled: boolean, nextYearButtonDisabled: boolean
date: Date, date: Date
}) => { }) => {
return ( return (
<div className='flex flex-col gap-2'> <div className='flex flex-col gap-2'>

View file

@ -3,18 +3,18 @@ import React, { useRef, useEffect } from 'react';
interface IBlurhash { interface IBlurhash {
/** Hash to render */ /** Hash to render */
hash: string | null | undefined, hash: string | null | undefined
/** Width of the blurred region in pixels. Defaults to 32. */ /** Width of the blurred region in pixels. Defaults to 32. */
width?: number, width?: number
/** Height of the blurred region in pixels. Defaults to width. */ /** Height of the blurred region in pixels. Defaults to width. */
height?: number, height?: number
/** /**
* Whether dummy mode is enabled. If enabled, nothing is rendered * Whether dummy mode is enabled. If enabled, nothing is rendered
* and canvas left untouched. * and canvas left untouched.
*/ */
dummy?: boolean, dummy?: boolean
/** className of the canvas element. */ /** className of the canvas element. */
className?: string, className?: string
} }
/** /**

View file

@ -5,7 +5,7 @@ import { Button, HStack, Input } from './ui';
interface ICopyableInput { interface ICopyableInput {
/** Text to be copied. */ /** Text to be copied. */
value: string, value: string
} }
/** An input with copy abilities. */ /** An input with copy abilities. */

View file

@ -12,7 +12,7 @@ const messages = defineMessages({
}); });
interface IDomain { interface IDomain {
domain: string, domain: string
} }
const Domain: React.FC<IDomain> = ({ domain }) => { const Domain: React.FC<IDomain> = ({ domain }) => {

View file

@ -31,10 +31,10 @@ interface Props extends ReturnType<typeof mapStateToProps> {
} }
type State = { type State = {
hasError: boolean, hasError: boolean
error: any, error: any
componentStack: any, componentStack: any
browser?: Bowser.Parser.Parser, browser?: Bowser.Parser.Parser
} }
class ErrorBoundary extends React.PureComponent<Props, State> { class ErrorBoundary extends React.PureComponent<Props, State> {

View file

@ -3,14 +3,14 @@ import React, { useEffect, useRef } from 'react';
import { isIOS } from 'soapbox/is-mobile'; import { isIOS } from 'soapbox/is-mobile';
interface IExtendedVideoPlayer { interface IExtendedVideoPlayer {
src: string, src: string
alt?: string, alt?: string
width?: number, width?: number
height?: number, height?: number
time?: number, time?: number
controls?: boolean, controls?: boolean
muted?: boolean, muted?: boolean
onClick?: () => void, onClick?: () => void
} }
const ExtendedVideoPlayer: React.FC<IExtendedVideoPlayer> = ({ src, alt, time, controls, muted, onClick }) => { const ExtendedVideoPlayer: React.FC<IExtendedVideoPlayer> = ({ src, alt, time, controls, muted, onClick }) => {

View file

@ -9,9 +9,9 @@ import clsx from 'clsx';
import React from 'react'; import React from 'react';
export interface IForkAwesomeIcon extends React.HTMLAttributes<HTMLLIElement> { export interface IForkAwesomeIcon extends React.HTMLAttributes<HTMLLIElement> {
id: string, id: string
className?: string, className?: string
fixedWidth?: boolean, fixedWidth?: boolean
} }
const ForkAwesomeIcon: React.FC<IForkAwesomeIcon> = ({ id, className, fixedWidth, ...rest }) => { const ForkAwesomeIcon: React.FC<IForkAwesomeIcon> = ({ id, className, fixedWidth, ...rest }) => {

View file

@ -10,7 +10,7 @@ import { HStack, Stack, Text } from './ui';
import type { Tag } from 'soapbox/types/entities'; import type { Tag } from 'soapbox/types/entities';
interface IHashtag { interface IHashtag {
hashtag: Tag, hashtag: Tag
} }
const Hashtag: React.FC<IHashtag> = ({ hashtag }) => { const Hashtag: React.FC<IHashtag> = ({ hashtag }) => {

View file

@ -15,10 +15,10 @@ const showProfileHoverCard = debounce((dispatch, ref, accountId) => {
}, 600); }, 600);
interface IHoverRefWrapper { interface IHoverRefWrapper {
accountId: string, accountId: string
inline?: boolean, inline?: boolean
className?: string, className?: string
children: React.ReactNode, children: React.ReactNode
} }
/** Makes a profile hover card appear when the wrapped element is hovered. */ /** Makes a profile hover card appear when the wrapped element is hovered. */

View file

@ -14,10 +14,10 @@ const showStatusHoverCard = debounce((dispatch, ref, statusId) => {
}, 300); }, 300);
interface IHoverStatusWrapper { interface IHoverStatusWrapper {
statusId: any, statusId: any
inline: boolean, inline: boolean
className?: string, className?: string
children: React.ReactNode, children: React.ReactNode
} }
/** Makes a status hover card appear when the wrapped element is hovered. */ /** Makes a status hover card appear when the wrapped element is hovered. */

View file

@ -4,10 +4,10 @@ import Icon, { IIcon } from 'soapbox/components/icon';
import { Counter } from 'soapbox/components/ui'; import { Counter } from 'soapbox/components/ui';
interface IIconWithCounter extends React.HTMLAttributes<HTMLDivElement> { interface IIconWithCounter extends React.HTMLAttributes<HTMLDivElement> {
count: number, count: number
countMax?: number countMax?: number
icon?: string; icon?: string
src?: string; src?: string
} }
const IconWithCounter: React.FC<IIconWithCounter> = ({ icon, count, countMax, ...rest }) => { const IconWithCounter: React.FC<IIconWithCounter> = ({ icon, count, countMax, ...rest }) => {

View file

@ -8,10 +8,10 @@ import React from 'react';
import InlineSVG from 'react-inlinesvg'; // eslint-disable-line no-restricted-imports import InlineSVG from 'react-inlinesvg'; // eslint-disable-line no-restricted-imports
export interface IIcon extends React.HTMLAttributes<HTMLDivElement> { export interface IIcon extends React.HTMLAttributes<HTMLDivElement> {
src: string, src: string
id?: string, id?: string
alt?: string, alt?: string
className?: string, className?: string
} }
const Icon: React.FC<IIcon> = ({ src, alt, className, ...rest }) => { const Icon: React.FC<IIcon> = ({ src, alt, className, ...rest }) => {

View file

@ -16,9 +16,9 @@ const List: React.FC<IList> = ({ children }) => (
); );
interface IListItem { interface IListItem {
label: React.ReactNode, label: React.ReactNode
hint?: React.ReactNode, hint?: React.ReactNode
onClick?(): void, onClick?(): void
onSelect?(): void onSelect?(): void
isSelected?: boolean isSelected?: boolean
children?: React.ReactNode children?: React.ReactNode

View file

@ -8,9 +8,9 @@ const messages = defineMessages({
}); });
interface ILoadGap { interface ILoadGap {
disabled?: boolean, disabled?: boolean
maxId: string, maxId: string
onClick: (id: string) => void, onClick: (id: string) => void
} }
const LoadGap: React.FC<ILoadGap> = ({ disabled, maxId, onClick }) => { const LoadGap: React.FC<ILoadGap> = ({ disabled, maxId, onClick }) => {

View file

@ -4,9 +4,9 @@ import { FormattedMessage } from 'react-intl';
import { Button } from 'soapbox/components/ui'; import { Button } from 'soapbox/components/ui';
interface ILoadMore { interface ILoadMore {
onClick: React.MouseEventHandler, onClick: React.MouseEventHandler
disabled?: boolean, disabled?: boolean
visible?: Boolean, visible?: Boolean
} }
const LoadMore: React.FC<ILoadMore> = ({ onClick, disabled, visible = true }) => { const LoadMore: React.FC<ILoadMore> = ({ onClick, disabled, visible = true }) => {

View file

@ -18,7 +18,7 @@ const messages = defineMessages({
}); });
interface ILocationSearch { interface ILocationSearch {
onSelected: (locationId: string) => void, onSelected: (locationId: string) => void
} }
const LocationSearch: React.FC<ILocationSearch> = ({ onSelected }) => { const LocationSearch: React.FC<ILocationSearch> = ({ onSelected }) => {

View file

@ -19,21 +19,21 @@ const ATTACHMENT_LIMIT = 4;
const MAX_FILENAME_LENGTH = 45; const MAX_FILENAME_LENGTH = 45;
interface Dimensions { interface Dimensions {
w: Property.Width | number, w: Property.Width | number
h: Property.Height | number, h: Property.Height | number
t?: Property.Top, t?: Property.Top
r?: Property.Right, r?: Property.Right
b?: Property.Bottom, b?: Property.Bottom
l?: Property.Left, l?: Property.Left
float?: Property.Float, float?: Property.Float
pos?: Property.Position, pos?: Property.Position
} }
interface SizeData { interface SizeData {
style: React.CSSProperties, style: React.CSSProperties
itemsDimensions: Dimensions[], itemsDimensions: Dimensions[]
size: number, size: number
width: number, width: number
} }
const withinLimits = (aspectRatio: number) => { const withinLimits = (aspectRatio: number) => {
@ -48,16 +48,16 @@ const shouldLetterbox = (attachment: Attachment): boolean => {
}; };
interface IItem { interface IItem {
attachment: Attachment, attachment: Attachment
standalone?: boolean, standalone?: boolean
index: number, index: number
size: number, size: number
onClick: (index: number) => void, onClick: (index: number) => void
displayWidth?: number, displayWidth?: number
visible: boolean, visible: boolean
dimensions: Dimensions, dimensions: Dimensions
last?: boolean, last?: boolean
total: number, total: number
} }
const Item: React.FC<IItem> = ({ const Item: React.FC<IItem> = ({
@ -261,16 +261,16 @@ const Item: React.FC<IItem> = ({
}; };
interface IMediaGallery { interface IMediaGallery {
sensitive?: boolean, sensitive?: boolean
media: ImmutableList<Attachment>, media: ImmutableList<Attachment>
height?: number, height?: number
onOpenMedia: (media: ImmutableList<Attachment>, index: number) => void, onOpenMedia: (media: ImmutableList<Attachment>, index: number) => void
defaultWidth?: number, defaultWidth?: number
cacheWidth?: (width: number) => void, cacheWidth?: (width: number) => void
visible?: boolean, visible?: boolean
onToggleVisibility?: () => void, onToggleVisibility?: () => void
displayMedia?: string, displayMedia?: string
compact: boolean, compact: boolean
} }
const MediaGallery: React.FC<IMediaGallery> = (props) => { const MediaGallery: React.FC<IMediaGallery> = (props) => {

View file

@ -39,10 +39,10 @@ export const checkEventComposeContent = (compose?: ReturnType<typeof ReducerComp
}; };
interface IModalRoot { interface IModalRoot {
onCancel?: () => void, onCancel?: () => void
onClose: (type?: ModalType) => void, onClose: (type?: ModalType) => void
type: ModalType, type: ModalType
children: React.ReactNode, children: React.ReactNode
} }
const ModalRoot: React.FC<IModalRoot> = ({ children, onCancel, onClose, type }) => { const ModalRoot: React.FC<IModalRoot> = ({ children, onCancel, onClose, type }) => {

View file

@ -2,8 +2,8 @@ import clsx from 'clsx';
import React from 'react'; import React from 'react';
interface IOutlineBox extends React.HTMLAttributes<HTMLDivElement> { interface IOutlineBox extends React.HTMLAttributes<HTMLDivElement> {
children: React.ReactNode, children: React.ReactNode
className?: string, className?: string
} }
/** Wraps children in a container with an outline. */ /** Wraps children in a container with an outline. */

View file

@ -16,9 +16,9 @@ const messages = defineMessages({
}); });
interface IPollFooter { interface IPollFooter {
poll: PollEntity, poll: PollEntity
showResults: boolean, showResults: boolean
selected: Selected, selected: Selected
} }
const PollFooter: React.FC<IPollFooter> = ({ poll, showResults, selected }): JSX.Element => { const PollFooter: React.FC<IPollFooter> = ({ poll, showResults, selected }): JSX.Element => {

View file

@ -29,7 +29,7 @@ const PollPercentageBar: React.FC<{ percent: number, leading: boolean }> = ({ pe
}; };
interface IPollOptionText extends IPollOption { interface IPollOptionText extends IPollOption {
percent: number, percent: number
} }
const PollOptionText: React.FC<IPollOptionText> = ({ poll, option, index, active, onToggle }) => { const PollOptionText: React.FC<IPollOptionText> = ({ poll, option, index, active, onToggle }) => {
@ -95,12 +95,12 @@ const PollOptionText: React.FC<IPollOptionText> = ({ poll, option, index, active
}; };
interface IPollOption { interface IPollOption {
poll: PollEntity, poll: PollEntity
option: PollOptionEntity, option: PollOptionEntity
index: number, index: number
showResults?: boolean, showResults?: boolean
active: boolean, active: boolean
onToggle: (value: number) => void, onToggle: (value: number) => void
} }
const PollOption: React.FC<IPollOption> = (props): JSX.Element | null => { const PollOption: React.FC<IPollOption> = (props): JSX.Element | null => {

View file

@ -13,8 +13,8 @@ import PollOption from './poll-option';
export type Selected = Record<number, boolean>; export type Selected = Record<number, boolean>;
interface IPoll { interface IPoll {
id: string, id: string
status?: string, status?: string
} }
const messages = defineMessages({ const messages = defineMessages({

View file

@ -54,7 +54,7 @@ const handleMouseLeave = (dispatch: AppDispatch): React.MouseEventHandler => {
}; };
interface IProfileHoverCard { interface IProfileHoverCard {
visible: boolean, visible: boolean
} }
/** Popup profile preview that appears when hovering avatars and display names. */ /** Popup profile preview that appears when hovering avatars and display names. */

View file

@ -2,10 +2,10 @@ import clsx from 'clsx';
import React from 'react'; import React from 'react';
interface IProgressCircle { interface IProgressCircle {
progress: number, progress: number
radius?: number, radius?: number
stroke?: number, stroke?: number
title?: string, title?: string
} }
const ProgressCircle: React.FC<IProgressCircle> = ({ progress, radius = 12, stroke = 4, title }) => { const ProgressCircle: React.FC<IProgressCircle> = ({ progress, radius = 12, stroke = 4, title }) => {

View file

@ -4,10 +4,10 @@ import PTRComponent from 'react-simple-pull-to-refresh';
import { Spinner } from 'soapbox/components/ui'; import { Spinner } from 'soapbox/components/ui';
interface IPullToRefresh { interface IPullToRefresh {
onRefresh?: () => Promise<any>; onRefresh?: () => Promise<any>
refreshingContent?: JSX.Element | string; refreshingContent?: JSX.Element | string
pullingContent?: JSX.Element | string; pullingContent?: JSX.Element | string
children: React.ReactNode; children: React.ReactNode
} }
/** /**

View file

@ -3,7 +3,7 @@ import React from 'react';
import PullToRefresh from './pull-to-refresh'; import PullToRefresh from './pull-to-refresh';
interface IPullable { interface IPullable {
children: React.ReactNode, children: React.ReactNode
} }
/** /**

View file

@ -23,11 +23,11 @@ const messages = defineMessages({
interface IQuotedStatus { interface IQuotedStatus {
/** The quoted status entity. */ /** The quoted status entity. */
status?: StatusEntity, status?: StatusEntity
/** Callback when cancelled (during compose). */ /** Callback when cancelled (during compose). */
onCancel?: Function, onCancel?: Function
/** Whether the status is shown in the post composer. */ /** Whether the status is shown in the post composer. */
compose?: boolean, compose?: boolean
} }
/** Status embedded in a quote post. */ /** Status embedded in a quote post. */

View file

@ -16,11 +16,11 @@ const RadioGroup = ({ onChange, children }: IRadioGroup) => {
}; };
interface IRadioItem { interface IRadioItem {
label: React.ReactNode, label: React.ReactNode
hint?: React.ReactNode, hint?: React.ReactNode
value: string, value: string
checked: boolean, checked: boolean
onChange?: React.ChangeEventHandler, onChange?: React.ChangeEventHandler
} }
const RadioItem: React.FC<IRadioItem> = ({ label, hint, checked = false, onChange, value }) => { const RadioItem: React.FC<IRadioItem> = ({ label, hint, checked = false, onChange, value }) => {

View file

@ -113,14 +113,14 @@ const timeRemainingString = (intl: IntlShape, date: Date, now: number) => {
}; };
interface RelativeTimestampProps extends IText { interface RelativeTimestampProps extends IText {
intl: IntlShape, intl: IntlShape
timestamp: string, timestamp: string
year?: number, year?: number
futureDate?: boolean, futureDate?: boolean
} }
interface RelativeTimestampState { interface RelativeTimestampState {
now: number, now: number
} }
/** Displays a timestamp compared to the current time, eg "1m" for one minute ago. */ /** Displays a timestamp compared to the current time, eg "1m" for one minute ago. */

View file

@ -2,13 +2,13 @@ import React, { useCallback, useEffect, useRef, useState } from 'react';
interface ISafeEmbed { interface ISafeEmbed {
/** Styles for the outer frame element. */ /** Styles for the outer frame element. */
className?: string, className?: string
/** Space-separate list of restrictions to ALLOW for the iframe. */ /** Space-separate list of restrictions to ALLOW for the iframe. */
sandbox?: string, sandbox?: string
/** Unique title for the iframe. */ /** Unique title for the iframe. */
title: string, title: string
/** HTML body to embed. */ /** HTML body to embed. */
html?: string, html?: string
} }
/** Safely embeds arbitrary HTML content on the page (by putting it in an iframe). */ /** Safely embeds arbitrary HTML content on the page (by putting it in an iframe). */

View file

@ -9,15 +9,15 @@ import { useSettings } from 'soapbox/hooks';
interface IScrollTopButton { interface IScrollTopButton {
/** Callback when clicked, and also when scrolled to the top. */ /** Callback when clicked, and also when scrolled to the top. */
onClick: () => void, onClick: () => void
/** Number of unread items. */ /** Number of unread items. */
count: number, count: number
/** Message to display in the button (should contain a `{count}` value). */ /** Message to display in the button (should contain a `{count}` value). */
message: MessageDescriptor, message: MessageDescriptor
/** Distance from the top of the screen (scrolling down) before the button appears. */ /** Distance from the top of the screen (scrolling down) before the button appears. */
threshold?: number, threshold?: number
/** Distance from the top of the screen (scrolling up) before the action is triggered. */ /** Distance from the top of the screen (scrolling up) before the action is triggered. */
autoloadThreshold?: number, autoloadThreshold?: number
} }
/** Floating new post counter above timelines, clicked to scroll to top. */ /** Floating new post counter above timelines, clicked to scroll to top. */

View file

@ -10,14 +10,14 @@ import { Card, Spinner } from './ui';
/** Custom Viruoso component context. */ /** Custom Viruoso component context. */
type Context = { type Context = {
itemClassName?: string, itemClassName?: string
listClassName?: string, listClassName?: string
} }
/** Scroll position saved in sessionStorage. */ /** Scroll position saved in sessionStorage. */
type SavedScrollPosition = { type SavedScrollPosition = {
index: number, index: number
offset: number, offset: number
} }
/** Custom Virtuoso Item component representing a single scrollable item. */ /** Custom Virtuoso Item component representing a single scrollable item. */
@ -37,44 +37,44 @@ const List: Components<JSX.Element, Context>['List'] = React.forwardRef((props,
interface IScrollableList extends VirtuosoProps<any, any> { interface IScrollableList extends VirtuosoProps<any, any> {
/** Unique key to preserve the scroll position when navigating back. */ /** Unique key to preserve the scroll position when navigating back. */
scrollKey?: string, scrollKey?: string
/** Pagination callback when the end of the list is reached. */ /** Pagination callback when the end of the list is reached. */
onLoadMore?: () => void, onLoadMore?: () => void
/** Whether the data is currently being fetched. */ /** Whether the data is currently being fetched. */
isLoading?: boolean, isLoading?: boolean
/** Whether to actually display the loading state. */ /** Whether to actually display the loading state. */
showLoading?: boolean, showLoading?: boolean
/** Whether we expect an additional page of data. */ /** Whether we expect an additional page of data. */
hasMore?: boolean, hasMore?: boolean
/** Additional element to display at the top of the list. */ /** Additional element to display at the top of the list. */
prepend?: React.ReactNode, prepend?: React.ReactNode
/** Whether to display the prepended element. */ /** Whether to display the prepended element. */
alwaysPrepend?: boolean, alwaysPrepend?: boolean
/** Message to display when the list is loaded but empty. */ /** Message to display when the list is loaded but empty. */
emptyMessage?: React.ReactNode, emptyMessage?: React.ReactNode
/** Scrollable content. */ /** Scrollable content. */
children: Iterable<React.ReactNode>, children: Iterable<React.ReactNode>
/** Callback when the list is scrolled to the top. */ /** Callback when the list is scrolled to the top. */
onScrollToTop?: () => void, onScrollToTop?: () => void
/** Callback when the list is scrolled. */ /** Callback when the list is scrolled. */
onScroll?: () => void, onScroll?: () => void
/** Placeholder component to render while loading. */ /** Placeholder component to render while loading. */
placeholderComponent?: React.ComponentType | React.NamedExoticComponent, placeholderComponent?: React.ComponentType | React.NamedExoticComponent
/** Number of placeholders to render while loading. */ /** Number of placeholders to render while loading. */
placeholderCount?: number, placeholderCount?: number
/** /**
* Pull to refresh callback. * Pull to refresh callback.
* @deprecated Put a PTR around the component instead. * @deprecated Put a PTR around the component instead.
*/ */
onRefresh?: () => Promise<any>, onRefresh?: () => Promise<any>
/** Extra class names on the Virtuoso element. */ /** Extra class names on the Virtuoso element. */
className?: string, className?: string
/** Class names on each item container. */ /** Class names on each item container. */
itemClassName?: string, itemClassName?: string
/** `id` attribute on the Virtuoso element. */ /** `id` attribute on the Virtuoso element. */
id?: string, id?: string
/** CSS styles on the Virtuoso element. */ /** CSS styles on the Virtuoso element. */
style?: React.CSSProperties, style?: React.CSSProperties
/** Whether to use the window to scroll the content instead of Virtuoso's container. */ /** Whether to use the window to scroll the content instead of Virtuoso's container. */
useWindowScroll?: boolean useWindowScroll?: boolean
} }

View file

@ -43,11 +43,11 @@ const messages = defineMessages({
}); });
interface ISidebarLink { interface ISidebarLink {
href?: string, href?: string
to?: string, to?: string
icon: string, icon: string
text: string | JSX.Element, text: string | JSX.Element
onClick: React.EventHandler<React.MouseEvent>, onClick: React.EventHandler<React.MouseEvent>
} }
const SidebarLink: React.FC<ISidebarLink> = ({ href, to, icon, text, onClick }) => { const SidebarLink: React.FC<ISidebarLink> = ({ href, to, icon, text, onClick }) => {

View file

@ -6,17 +6,17 @@ import { Icon, Text } from './ui';
interface ISidebarNavigationLink { interface ISidebarNavigationLink {
/** Notification count, if any. */ /** Notification count, if any. */
count?: number, count?: number
/** Optional max to cap count (ie: N+) */ /** Optional max to cap count (ie: N+) */
countMax?: number countMax?: number
/** URL to an SVG icon. */ /** URL to an SVG icon. */
icon: string, icon: string
/** Link label. */ /** Link label. */
text: React.ReactNode, text: React.ReactNode
/** Route to an internal page. */ /** Route to an internal page. */
to?: string, to?: string
/** Callback when the link is clicked. */ /** Callback when the link is clicked. */
onClick?: React.EventHandler<React.MouseEvent>, onClick?: React.EventHandler<React.MouseEvent>
} }
/** Desktop sidebar navigation link. */ /** Desktop sidebar navigation link. */

View file

@ -5,9 +5,9 @@ import { useSoapboxConfig, useSettings, useTheme } from 'soapbox/hooks';
interface ISiteLogo extends React.ComponentProps<'img'> { interface ISiteLogo extends React.ComponentProps<'img'> {
/** Extra class names for the <img> element. */ /** Extra class names for the <img> element. */
className?: string, className?: string
/** Override theme setting for <SitePreview /> */ /** Override theme setting for <SitePreview /> */
theme?: 'dark' | 'light', theme?: 'dark' | 'light'
} }
/** Display the most appropriate site logo based on the theme and configuration. */ /** Display the most appropriate site logo based on the theme and configuration. */

View file

@ -97,10 +97,10 @@ const messages = defineMessages({
}); });
interface IStatusActionBar { interface IStatusActionBar {
status: Status, status: Status
withLabels?: boolean, withLabels?: boolean
expandable?: boolean, expandable?: boolean
space?: 'expand' | 'compact', space?: 'expand' | 'compact'
} }
const StatusActionBar: React.FC<IStatusActionBar> = ({ const StatusActionBar: React.FC<IStatusActionBar> = ({

View file

@ -12,7 +12,7 @@ const COLORS = {
type Color = keyof typeof COLORS; type Color = keyof typeof COLORS;
interface IStatusActionCounter { interface IStatusActionCounter {
count: number, count: number
} }
/** Action button numerical counter, eg "5" likes. */ /** Action button numerical counter, eg "5" likes. */
@ -25,14 +25,14 @@ const StatusActionCounter: React.FC<IStatusActionCounter> = ({ count = 0 }): JSX
}; };
interface IStatusActionButton extends React.ButtonHTMLAttributes<HTMLButtonElement> { interface IStatusActionButton extends React.ButtonHTMLAttributes<HTMLButtonElement> {
iconClassName?: string, iconClassName?: string
icon: string, icon: string
count?: number, count?: number
active?: boolean, active?: boolean
color?: Color, color?: Color
filled?: boolean, filled?: boolean
emoji?: string, emoji?: string
text?: React.ReactNode, text?: React.ReactNode
} }
const StatusActionButton = React.forwardRef<HTMLButtonElement, IStatusActionButton>((props, ref): JSX.Element => { const StatusActionButton = React.forwardRef<HTMLButtonElement, IStatusActionButton>((props, ref): JSX.Element => {

View file

@ -20,7 +20,7 @@ const MAX_HEIGHT = 642; // 20px * 32 (+ 2px padding at the top)
const BIG_EMOJI_LIMIT = 10; const BIG_EMOJI_LIMIT = 10;
interface IReadMoreButton { interface IReadMoreButton {
onClick: React.MouseEventHandler, onClick: React.MouseEventHandler
} }
/** Button to expand a truncated status (due to too much content) */ /** Button to expand a truncated status (due to too much content) */
@ -32,11 +32,11 @@ const ReadMoreButton: React.FC<IReadMoreButton> = ({ onClick }) => (
); );
interface IStatusContent { interface IStatusContent {
status: Status, status: Status
onClick?: () => void, onClick?: () => void
collapsable?: boolean, collapsable?: boolean
translatable?: boolean, translatable?: boolean
textSize?: Sizes, textSize?: Sizes
} }
/** Renders the text content of a status */ /** Renders the text content of a status */

View file

@ -15,7 +15,7 @@ import { showStatusHoverCard } from './hover-status-wrapper';
import { Card, CardBody } from './ui'; import { Card, CardBody } from './ui';
interface IStatusHoverCard { interface IStatusHoverCard {
visible: boolean, visible: boolean
} }
/** Popup status preview that appears when hovering reply to */ /** Popup status preview that appears when hovering reply to */

View file

@ -23,31 +23,31 @@ import type { Ad as AdEntity } from 'soapbox/types/soapbox';
interface IStatusList extends Omit<IScrollableList, 'onLoadMore' | 'children'> { interface IStatusList extends Omit<IScrollableList, 'onLoadMore' | 'children'> {
/** Unique key to preserve the scroll position when navigating back. */ /** Unique key to preserve the scroll position when navigating back. */
scrollKey: string, scrollKey: string
/** List of status IDs to display. */ /** List of status IDs to display. */
statusIds: ImmutableOrderedSet<string>, statusIds: ImmutableOrderedSet<string>
/** Last _unfiltered_ status ID (maxId) for pagination. */ /** Last _unfiltered_ status ID (maxId) for pagination. */
lastStatusId?: string, lastStatusId?: string
/** Pinned statuses to show at the top of the feed. */ /** Pinned statuses to show at the top of the feed. */
featuredStatusIds?: ImmutableOrderedSet<string>, featuredStatusIds?: ImmutableOrderedSet<string>
/** Pagination callback when the end of the list is reached. */ /** Pagination callback when the end of the list is reached. */
onLoadMore?: (lastStatusId: string) => void, onLoadMore?: (lastStatusId: string) => void
/** Whether the data is currently being fetched. */ /** Whether the data is currently being fetched. */
isLoading: boolean, isLoading: boolean
/** Whether the server did not return a complete page. */ /** Whether the server did not return a complete page. */
isPartial?: boolean, isPartial?: boolean
/** Whether we expect an additional page of data. */ /** Whether we expect an additional page of data. */
hasMore: boolean, hasMore: boolean
/** Message to display when the list is loaded but empty. */ /** Message to display when the list is loaded but empty. */
emptyMessage: React.ReactNode, emptyMessage: React.ReactNode
/** ID of the timeline in Redux. */ /** ID of the timeline in Redux. */
timelineId?: string, timelineId?: string
/** Whether to display a gap or border between statuses in the list. */ /** Whether to display a gap or border between statuses in the list. */
divideType?: 'space' | 'border', divideType?: 'space' | 'border'
/** Whether to display ads. */ /** Whether to display ads. */
showAds?: boolean, showAds?: boolean
/** Whether to show group information. */ /** Whether to show group information. */
showGroup?: boolean, showGroup?: boolean
} }
/** Feed of statuses, built atop ScrollableList. */ /** Feed of statuses, built atop ScrollableList. */

View file

@ -15,15 +15,15 @@ import type { Status, Attachment } from 'soapbox/types/entities';
interface IStatusMedia { interface IStatusMedia {
/** Status entity to render media for. */ /** Status entity to render media for. */
status: Status, status: Status
/** Whether to display compact media. */ /** Whether to display compact media. */
muted?: boolean, muted?: boolean
/** Callback when compact media is clicked. */ /** Callback when compact media is clicked. */
onClick?: () => void, onClick?: () => void
/** Whether or not the media is concealed behind a NSFW banner. */ /** Whether or not the media is concealed behind a NSFW banner. */
showMedia?: boolean, showMedia?: boolean
/** Callback when visibility is toggled (eg clicked through NSFW). */ /** Callback when visibility is toggled (eg clicked through NSFW). */
onToggleVisibility?: () => void, onToggleVisibility?: () => void
} }
/** Render media attachments for a status. */ /** Render media attachments for a status. */

View file

@ -8,8 +8,8 @@ import { isUserTouching } from 'soapbox/is-mobile';
import { getReactForStatus } from 'soapbox/utils/emoji-reacts'; import { getReactForStatus } from 'soapbox/utils/emoji-reacts';
interface IStatusReactionWrapper { interface IStatusReactionWrapper {
statusId: string, statusId: string
children: JSX.Element, children: JSX.Element
} }
/** Provides emoji reaction functionality to the underlying button component */ /** Provides emoji reaction functionality to the underlying button component */

View file

@ -10,8 +10,8 @@ import { useAppDispatch } from 'soapbox/hooks';
import type { Account, Status } from 'soapbox/types/entities'; import type { Account, Status } from 'soapbox/types/entities';
interface IStatusReplyMentions { interface IStatusReplyMentions {
status: Status, status: Status
hoverable?: boolean, hoverable?: boolean
} }
const StatusReplyMentions: React.FC<IStatusReplyMentions> = ({ status, hoverable = true }) => { const StatusReplyMentions: React.FC<IStatusReplyMentions> = ({ status, hoverable = true }) => {

View file

@ -38,22 +38,22 @@ const messages = defineMessages({
}); });
export interface IStatus { export interface IStatus {
id?: string, id?: string
avatarSize?: number, avatarSize?: number
status: StatusEntity, status: StatusEntity
onClick?: () => void, onClick?: () => void
muted?: boolean, muted?: boolean
hidden?: boolean, hidden?: boolean
unread?: boolean, unread?: boolean
onMoveUp?: (statusId: string, featured?: boolean) => void, onMoveUp?: (statusId: string, featured?: boolean) => void
onMoveDown?: (statusId: string, featured?: boolean) => void, onMoveDown?: (statusId: string, featured?: boolean) => void
focusable?: boolean, focusable?: boolean
featured?: boolean, featured?: boolean
hideActionBar?: boolean, hideActionBar?: boolean
hoverable?: boolean, hoverable?: boolean
variant?: 'default' | 'rounded', variant?: 'default' | 'rounded'
showGroup?: boolean, showGroup?: boolean
accountAction?: React.ReactElement, accountAction?: React.ReactElement
} }
const Status: React.FC<IStatus> = (props) => { const Status: React.FC<IStatus> = (props) => {

View file

@ -5,17 +5,17 @@ import { useSettings } from 'soapbox/hooks';
interface IStillImage { interface IStillImage {
/** Image alt text. */ /** Image alt text. */
alt?: string, alt?: string
/** Extra class names for the outer <div> container. */ /** Extra class names for the outer <div> container. */
className?: string, className?: string
/** URL to the image */ /** URL to the image */
src: string, src: string
/** Extra CSS styles on the outer <div> element. */ /** Extra CSS styles on the outer <div> element. */
style?: React.CSSProperties, style?: React.CSSProperties
/** Whether to display the image contained vs filled in its container. */ /** Whether to display the image contained vs filled in its container. */
letterboxed?: boolean, letterboxed?: boolean
/** Whether to show the file extension in the corner. */ /** Whether to show the file extension in the corner. */
showExt?: boolean, showExt?: boolean
} }
/** Renders images on a canvas, only playing GIFs if autoPlayGif is enabled. */ /** Renders images on a canvas, only playing GIFs if autoPlayGif is enabled. */
@ -80,7 +80,7 @@ const StillImage: React.FC<IStillImage> = ({ alt, className, src, style, letterb
interface IExtensionBadge { interface IExtensionBadge {
/** File extension. */ /** File extension. */
ext: string, ext: string
} }
/** Badge displaying a file extension. */ /** Badge displaying a file extension. */

View file

@ -6,13 +6,13 @@ import IconWithCounter from 'soapbox/components/icon-with-counter';
import { Icon, Text } from 'soapbox/components/ui'; import { Icon, Text } from 'soapbox/components/ui';
interface IThumbNavigationLink { interface IThumbNavigationLink {
count?: number, count?: number
countMax?: number, countMax?: number
src: string, src: string
text: string | React.ReactElement, text: string | React.ReactElement
to: string, to: string
exact?: boolean, exact?: boolean
paths?: Array<string>, paths?: Array<string>
} }
const ThumbNavigationLink: React.FC<IThumbNavigationLink> = ({ count, countMax, src, text, to, exact, paths }): JSX.Element => { const ThumbNavigationLink: React.FC<IThumbNavigationLink> = ({ count, countMax, src, text, to, exact, paths }): JSX.Element => {

View file

@ -5,9 +5,9 @@ import { FormattedMessage } from 'react-intl';
import { Text } from 'soapbox/components/ui'; import { Text } from 'soapbox/components/ui';
interface ITombstone { interface ITombstone {
id: string, id: string
onMoveUp: (statusId: string) => void, onMoveUp: (statusId: string) => void
onMoveDown: (statusId: string) => void, onMoveDown: (statusId: string) => void
} }
/** Represents a deleted item. */ /** Represents a deleted item. */

View file

@ -11,7 +11,7 @@ import { Stack, Button, Text } from './ui';
import type { Account, Status } from 'soapbox/types/entities'; import type { Account, Status } from 'soapbox/types/entities';
interface ITranslateButton { interface ITranslateButton {
status: Status, status: Status
} }
const TranslateButton: React.FC<ITranslateButton> = ({ status }) => { const TranslateButton: React.FC<ITranslateButton> = ({ status }) => {

View file

@ -16,11 +16,11 @@ const messages = defineMessages({
}); });
interface IAccordion { interface IAccordion {
headline: React.ReactNode, headline: React.ReactNode
children?: React.ReactNode, children?: React.ReactNode
menu?: Menu, menu?: Menu
expanded?: boolean, expanded?: boolean
onToggle?: (value: boolean) => void, onToggle?: (value: boolean) => void
} }
/** /**

View file

@ -7,11 +7,11 @@ const AVATAR_SIZE = 42;
interface IAvatar { interface IAvatar {
/** URL to the avatar image. */ /** URL to the avatar image. */
src: string, src: string
/** Width and height of the avatar in pixels. */ /** Width and height of the avatar in pixels. */
size?: number, size?: number
/** Extra class names for the div surrounding the avatar image. */ /** Extra class names for the div surrounding the avatar image. */
className?: string, className?: string
} }
/** Round profile avatar for accounts. */ /** Round profile avatar for accounts. */

View file

@ -2,9 +2,9 @@ import clsx from 'clsx';
import React from 'react'; import React from 'react';
interface IBanner { interface IBanner {
theme: 'frosted' | 'opaque', theme: 'frosted' | 'opaque'
children: React.ReactNode, children: React.ReactNode
className?: string, className?: string
} }
/** Displays a sticky full-width banner at the bottom of the screen. */ /** Displays a sticky full-width banner at the bottom of the screen. */

View file

@ -10,27 +10,27 @@ import type { ButtonSizes, ButtonThemes } from './useButtonStyles';
interface IButton { interface IButton {
/** Whether this button expands the width of its container. */ /** Whether this button expands the width of its container. */
block?: boolean, block?: boolean
/** Elements inside the <button> */ /** Elements inside the <button> */
children?: React.ReactNode, children?: React.ReactNode
/** Extra class names for the button. */ /** Extra class names for the button. */
className?: string, className?: string
/** Prevent the button from being clicked. */ /** Prevent the button from being clicked. */
disabled?: boolean, disabled?: boolean
/** URL to an SVG icon to render inside the button. */ /** URL to an SVG icon to render inside the button. */
icon?: string, icon?: string
/** Action when the button is clicked. */ /** Action when the button is clicked. */
onClick?: (event: React.MouseEvent<HTMLButtonElement>) => void, onClick?: (event: React.MouseEvent<HTMLButtonElement>) => void
/** A predefined button size. */ /** A predefined button size. */
size?: ButtonSizes, size?: ButtonSizes
/** Text inside the button. Takes precedence over `children`. */ /** Text inside the button. Takes precedence over `children`. */
text?: React.ReactNode, text?: React.ReactNode
/** Makes the button into a navlink, if provided. */ /** Makes the button into a navlink, if provided. */
to?: string, to?: string
/** Styles the button visually with a predefined theme. */ /** Styles the button visually with a predefined theme. */
theme?: ButtonThemes, theme?: ButtonThemes
/** Whether this button should submit a form by default. */ /** Whether this button should submit a form by default. */
type?: 'button' | 'submit', type?: 'button' | 'submit'
} }
/** Customizable button element with various themes. */ /** Customizable button element with various themes. */

View file

@ -42,7 +42,7 @@ const Card = React.forwardRef<HTMLDivElement, ICard>(({ children, variant = 'def
)); ));
interface ICardHeader { interface ICardHeader {
backHref?: string, backHref?: string
onBackClick?: (event: React.MouseEvent) => void onBackClick?: (event: React.MouseEvent) => void
className?: string className?: string
children?: React.ReactNode children?: React.ReactNode

View file

@ -35,15 +35,15 @@ const ColumnHeader: React.FC<IColumnHeader> = ({ label, backHref, className }) =
export interface IColumn { export interface IColumn {
/** Route the back button goes to. */ /** Route the back button goes to. */
backHref?: string, backHref?: string
/** Column title text. */ /** Column title text. */
label?: string, label?: string
/** Whether this column should have a transparent background. */ /** Whether this column should have a transparent background. */
transparent?: boolean, transparent?: boolean
/** Whether this column should have a title and back button. */ /** Whether this column should have a title and back button. */
withHeader?: boolean, withHeader?: boolean
/** Extra class name for top <div> element. */ /** Extra class name for top <div> element. */
className?: string, className?: string
/** Ref forwarded to column. */ /** Ref forwarded to column. */
ref?: React.Ref<HTMLDivElement> ref?: React.Ref<HTMLDivElement>
/** Children to display in the column. */ /** Children to display in the column. */

View file

@ -4,7 +4,7 @@ import { shortNumberFormat } from 'soapbox/utils/numbers';
interface ICounter { interface ICounter {
/** Number this counter should display. */ /** Number this counter should display. */
count: number, count: number
/** Optional max number (ie: N+) */ /** Optional max number (ie: N+) */
countMax?: number countMax?: number
} }

View file

@ -9,13 +9,13 @@ import { useSoapboxConfig } from 'soapbox/hooks';
interface IEmojiButton { interface IEmojiButton {
/** Unicode emoji character. */ /** Unicode emoji character. */
emoji: string, emoji: string
/** Event handler when the emoji is clicked. */ /** Event handler when the emoji is clicked. */
onClick(emoji: string): void onClick(emoji: string): void
/** Extra class name on the <button> element. */ /** Extra class name on the <button> element. */
className?: string, className?: string
/** Tab order of the button. */ /** Tab order of the button. */
tabIndex?: number, tabIndex?: number
} }
/** Clickable emoji button that scales when hovered. */ /** Clickable emoji button that scales when hovered. */

View file

@ -5,7 +5,7 @@ import { joinPublicPath } from 'soapbox/utils/static';
interface IEmoji extends React.ImgHTMLAttributes<HTMLImageElement> { interface IEmoji extends React.ImgHTMLAttributes<HTMLImageElement> {
/** Unicode emoji character. */ /** Unicode emoji character. */
emoji: string, emoji: string
} }
/** A single emoji image. */ /** A single emoji image. */

View file

@ -7,11 +7,11 @@ import Stack from '../stack/stack';
interface IFormGroup { interface IFormGroup {
/** Input label message. */ /** Input label message. */
labelText?: React.ReactNode, labelText?: React.ReactNode
/** Input label tooltip message. */ /** Input label tooltip message. */
labelTitle?: string, labelTitle?: string
/** Input hint message. */ /** Input hint message. */
hintText?: React.ReactNode, hintText?: React.ReactNode
/** Input errors. */ /** Input errors. */
errors?: string[] errors?: string[]
/** Elements to display within the FormGroup. */ /** Elements to display within the FormGroup. */

View file

@ -2,11 +2,11 @@ import React from 'react';
interface IForm { interface IForm {
/** Form submission event handler. */ /** Form submission event handler. */
onSubmit?: (event: React.FormEvent) => void, onSubmit?: (event: React.FormEvent) => void
/** Class name override for the <form> element. */ /** Class name override for the <form> element. */
className?: string, className?: string
/** Elements to display within the Form. */ /** Elements to display within the Form. */
children: React.ReactNode, children: React.ReactNode
} }
/** Form element with custom styles. */ /** Form element with custom styles. */

View file

@ -44,7 +44,7 @@ interface IHStack extends Pick<React.HTMLAttributes<HTMLDivElement>, 'onClick'>
/** Whether to let the flexbox grow. */ /** Whether to let the flexbox grow. */
grow?: boolean grow?: boolean
/** HTML element to use for container. */ /** HTML element to use for container. */
element?: keyof JSX.IntrinsicElements, element?: keyof JSX.IntrinsicElements
/** Extra CSS styles for the <div> */ /** Extra CSS styles for the <div> */
style?: React.CSSProperties style?: React.CSSProperties
/** Whether to let the flexbox wrap onto multiple lines. */ /** Whether to let the flexbox wrap onto multiple lines. */

View file

@ -6,15 +6,15 @@ import Text from '../text/text';
interface IIconButton extends React.ButtonHTMLAttributes<HTMLButtonElement> { interface IIconButton extends React.ButtonHTMLAttributes<HTMLButtonElement> {
/** Class name for the <svg> icon. */ /** Class name for the <svg> icon. */
iconClassName?: string, iconClassName?: string
/** URL to the svg icon. */ /** URL to the svg icon. */
src: string, src: string
/** Text to display next ot the button. */ /** Text to display next ot the button. */
text?: string, text?: string
/** Don't render a background behind the icon. */ /** Don't render a background behind the icon. */
transparent?: boolean, transparent?: boolean
/** Predefined styles to display for the button. */ /** Predefined styles to display for the button. */
theme?: 'seamless' | 'outlined', theme?: 'seamless' | 'outlined'
/** Override the data-testid */ /** Override the data-testid */
'data-testid'?: string 'data-testid'?: string
} }

View file

@ -6,17 +6,17 @@ import SvgIcon from './svg-icon';
interface IIcon extends Pick<React.SVGAttributes<SVGAElement>, 'strokeWidth'> { interface IIcon extends Pick<React.SVGAttributes<SVGAElement>, 'strokeWidth'> {
/** Class name for the <svg> element. */ /** Class name for the <svg> element. */
className?: string, className?: string
/** Number to display a counter over the icon. */ /** Number to display a counter over the icon. */
count?: number, count?: number
/** Optional max to cap count (ie: N+) */ /** Optional max to cap count (ie: N+) */
countMax?: number, countMax?: number
/** Tooltip text for the icon. */ /** Tooltip text for the icon. */
alt?: string, alt?: string
/** URL to the svg icon. */ /** URL to the svg icon. */
src: string, src: string
/** Width and height of the icon in pixels. */ /** Width and height of the icon in pixels. */
size?: number, size?: number
} }
/** Renders and SVG icon with optional counter. */ /** Renders and SVG icon with optional counter. */

View file

@ -3,13 +3,13 @@ import InlineSVG from 'react-inlinesvg'; // eslint-disable-line no-restricted-im
interface ISvgIcon { interface ISvgIcon {
/** Class name for the <svg> */ /** Class name for the <svg> */
className?: string, className?: string
/** Tooltip text for the icon. */ /** Tooltip text for the icon. */
alt?: string, alt?: string
/** URL to the svg file. */ /** URL to the svg file. */
src: string, src: string
/** Width and height of the icon in pixels. */ /** Width and height of the icon in pixels. */
size?: number, size?: number
} }
/** Renders an inline SVG with an empty frame loading state */ /** Renders an inline SVG with an empty frame loading state */

View file

@ -16,29 +16,29 @@ type InputThemes = 'normal' | 'search'
interface IInput extends Pick<React.InputHTMLAttributes<HTMLInputElement>, 'maxLength' | 'onChange' | 'onBlur' | 'type' | 'autoComplete' | 'autoCorrect' | 'autoCapitalize' | 'required' | 'disabled' | 'onClick' | 'readOnly' | 'min' | 'pattern' | 'onKeyDown' | 'onKeyUp' | 'onFocus' | 'style' | 'id'> { interface IInput extends Pick<React.InputHTMLAttributes<HTMLInputElement>, 'maxLength' | 'onChange' | 'onBlur' | 'type' | 'autoComplete' | 'autoCorrect' | 'autoCapitalize' | 'required' | 'disabled' | 'onClick' | 'readOnly' | 'min' | 'pattern' | 'onKeyDown' | 'onKeyUp' | 'onFocus' | 'style' | 'id'> {
/** Put the cursor into the input on mount. */ /** Put the cursor into the input on mount. */
autoFocus?: boolean, autoFocus?: boolean
/** The initial text in the input. */ /** The initial text in the input. */
defaultValue?: string, defaultValue?: string
/** Extra class names for the <input> element. */ /** Extra class names for the <input> element. */
className?: string, className?: string
/** Extra class names for the outer <div> element. */ /** Extra class names for the outer <div> element. */
outerClassName?: string, outerClassName?: string
/** URL to the svg icon. Cannot be used with prepend. */ /** URL to the svg icon. Cannot be used with prepend. */
icon?: string, icon?: string
/** Internal input name. */ /** Internal input name. */
name?: string, name?: string
/** Text to display before a value is entered. */ /** Text to display before a value is entered. */
placeholder?: string, placeholder?: string
/** Text in the input. */ /** Text in the input. */
value?: string | number, value?: string | number
/** Change event handler for the input. */ /** Change event handler for the input. */
onChange?: (event: React.ChangeEvent<HTMLInputElement>) => void, onChange?: (event: React.ChangeEvent<HTMLInputElement>) => void
/** An element to display as prefix to input. Cannot be used with icon. */ /** An element to display as prefix to input. Cannot be used with icon. */
prepend?: React.ReactElement, prepend?: React.ReactElement
/** An element to display as suffix to input. Cannot be used with password type. */ /** An element to display as suffix to input. Cannot be used with password type. */
append?: React.ReactElement, append?: React.ReactElement
/** Theme to style the input with. */ /** Theme to style the input with. */
theme?: InputThemes, theme?: InputThemes
} }
/** Form input element. */ /** Form input element. */

View file

@ -14,9 +14,9 @@ interface ILayout {
} }
interface LayoutComponent extends React.FC<ILayout> { interface LayoutComponent extends React.FC<ILayout> {
Sidebar: React.FC<ISidebar>, Sidebar: React.FC<ISidebar>
Main: React.FC<React.HTMLAttributes<HTMLDivElement>>, Main: React.FC<React.HTMLAttributes<HTMLDivElement>>
Aside: React.FC<IAside>, Aside: React.FC<IAside>
} }
/** Layout container, to hold Sidebar, Main, and Aside. */ /** Layout container, to hold Sidebar, Main, and Aside. */

View file

@ -25,36 +25,36 @@ const widths = {
interface IModal { interface IModal {
/** Callback when the modal is cancelled. */ /** Callback when the modal is cancelled. */
cancelAction?: () => void, cancelAction?: () => void
/** Cancel button text. */ /** Cancel button text. */
cancelText?: React.ReactNode, cancelText?: React.ReactNode
/** URL to an SVG icon for the close button. */ /** URL to an SVG icon for the close button. */
closeIcon?: string, closeIcon?: string
/** Position of the close button. */ /** Position of the close button. */
closePosition?: 'left' | 'right', closePosition?: 'left' | 'right'
/** Callback when the modal is confirmed. */ /** Callback when the modal is confirmed. */
confirmationAction?: (event?: React.MouseEvent<HTMLButtonElement>) => void, confirmationAction?: (event?: React.MouseEvent<HTMLButtonElement>) => void
/** Whether the confirmation button is disabled. */ /** Whether the confirmation button is disabled. */
confirmationDisabled?: boolean, confirmationDisabled?: boolean
/** Confirmation button text. */ /** Confirmation button text. */
confirmationText?: React.ReactNode, confirmationText?: React.ReactNode
/** Confirmation button theme. */ /** Confirmation button theme. */
confirmationTheme?: ButtonThemes, confirmationTheme?: ButtonThemes
/** Whether to use full width style for confirmation button. */ /** Whether to use full width style for confirmation button. */
confirmationFullWidth?: boolean, confirmationFullWidth?: boolean
/** Callback when the modal is closed. */ /** Callback when the modal is closed. */
onClose?: () => void, onClose?: () => void
/** Callback when the secondary action is chosen. */ /** Callback when the secondary action is chosen. */
secondaryAction?: (event?: React.MouseEvent<HTMLButtonElement>) => void, secondaryAction?: (event?: React.MouseEvent<HTMLButtonElement>) => void
/** Secondary button text. */ /** Secondary button text. */
secondaryText?: React.ReactNode, secondaryText?: React.ReactNode
secondaryDisabled?: boolean, secondaryDisabled?: boolean
/** Don't focus the "confirm" button on mount. */ /** Don't focus the "confirm" button on mount. */
skipFocus?: boolean, skipFocus?: boolean
/** Title text for the modal. */ /** Title text for the modal. */
title?: React.ReactNode, title?: React.ReactNode
width?: keyof typeof widths, width?: keyof typeof widths
children?: React.ReactNode, children?: React.ReactNode
} }
/** Displays a modal dialog box. */ /** Displays a modal dialog box. */

View file

@ -3,8 +3,8 @@ import React from 'react';
import { COUNTRY_CODES, CountryCode } from 'soapbox/utils/phone'; import { COUNTRY_CODES, CountryCode } from 'soapbox/utils/phone';
interface ICountryCodeDropdown { interface ICountryCodeDropdown {
countryCode: CountryCode, countryCode: CountryCode
onChange(countryCode: CountryCode): void, onChange(countryCode: CountryCode): void
} }
/** Dropdown menu to select a country code. */ /** Dropdown menu to select a country code. */

View file

@ -9,11 +9,11 @@ import CountryCodeDropdown from './country-code-dropdown';
interface IPhoneInput extends Pick<React.InputHTMLAttributes<HTMLInputElement>, 'required' | 'autoFocus'> { interface IPhoneInput extends Pick<React.InputHTMLAttributes<HTMLInputElement>, 'required' | 'autoFocus'> {
/** E164 phone number. */ /** E164 phone number. */
value?: string, value?: string
/** Change handler which receives the E164 phone string. */ /** Change handler which receives the E164 phone string. */
onChange?: (phone: string | undefined) => void, onChange?: (phone: string | undefined) => void
/** Country code that's selected on mount. */ /** Country code that's selected on mount. */
defaultCountryCode?: CountryCode, defaultCountryCode?: CountryCode
} }
/** Internationalized phone input with country code picker. */ /** Internationalized phone input with country code picker. */

View file

@ -2,7 +2,7 @@ import clsx from 'clsx';
import React from 'react'; import React from 'react';
interface ISelect extends React.SelectHTMLAttributes<HTMLSelectElement> { interface ISelect extends React.SelectHTMLAttributes<HTMLSelectElement> {
children: Iterable<React.ReactNode>, children: Iterable<React.ReactNode>
} }
/** Multiple-select dropdown. */ /** Multiple-select dropdown. */

View file

@ -8,7 +8,7 @@ import './spinner.css';
interface ISpinner { interface ISpinner {
/** Width and height of the spinner in pixels. */ /** Width and height of the spinner in pixels. */
size?: number, size?: number
/** Whether to display "Loading..." beneath the spinner. */ /** Whether to display "Loading..." beneath the spinner. */
withText?: boolean withText?: boolean
} }

View file

@ -39,7 +39,7 @@ interface IStack extends React.HTMLAttributes<HTMLDivElement> {
/** Whether to let the flexbox grow. */ /** Whether to let the flexbox grow. */
grow?: boolean grow?: boolean
/** HTML element to use for container. */ /** HTML element to use for container. */
element?: keyof JSX.IntrinsicElements, element?: keyof JSX.IntrinsicElements
} }
/** Vertical stack of child elements. */ /** Vertical stack of child elements. */

View file

@ -14,27 +14,27 @@ const messages = defineMessages({
/** Type of the inner Streamfield input component. */ /** Type of the inner Streamfield input component. */
export type StreamfieldComponent<T> = React.ComponentType<{ export type StreamfieldComponent<T> = React.ComponentType<{
value: T, value: T
onChange: (value: T) => void, onChange: (value: T) => void
}>; }>;
interface IStreamfield { interface IStreamfield {
/** Array of values for the streamfield. */ /** Array of values for the streamfield. */
values: any[], values: any[]
/** Input label message. */ /** Input label message. */
label?: React.ReactNode, label?: React.ReactNode
/** Input hint message. */ /** Input hint message. */
hint?: React.ReactNode, hint?: React.ReactNode
/** Callback to add an item. */ /** Callback to add an item. */
onAddItem?: () => void, onAddItem?: () => void
/** Callback to remove an item by index. */ /** Callback to remove an item by index. */
onRemoveItem?: (i: number) => void, onRemoveItem?: (i: number) => void
/** Callback when values are changed. */ /** Callback when values are changed. */
onChange: (values: any[]) => void, onChange: (values: any[]) => void
/** Input to render for each value. */ /** Input to render for each value. */
component: StreamfieldComponent<any>, component: StreamfieldComponent<any>
/** Maximum number of allowed inputs. */ /** Maximum number of allowed inputs. */
maxItems?: number, maxItems?: number
} }
/** List of inputs that can be added or removed. */ /** List of inputs that can be added or removed. */
@ -72,7 +72,7 @@ const Streamfield: React.FC<IStreamfield> = ({
<Component key={i} onChange={handleChange(i)} value={value} /> <Component key={i} onChange={handleChange(i)} value={value} />
{onRemoveItem && ( {onRemoveItem && (
<IconButton <IconButton
iconClassName='w-4 h-4' iconClassName='h-4 w-4'
className='bg-transparent text-gray-400 hover:text-gray-600' className='bg-transparent text-gray-400 hover:text-gray-600'
src={require('@tabler/icons/x.svg')} src={require('@tabler/icons/x.svg')}
onClick={() => onRemoveItem(i)} onClick={() => onRemoveItem(i)}

View file

@ -18,7 +18,7 @@ const AnimatedContext = React.createContext(null);
interface IAnimatedInterface { interface IAnimatedInterface {
/** Callback when a tab is chosen. */ /** Callback when a tab is chosen. */
onChange(index: number): void, onChange(index: number): void
/** Default tab index. */ /** Default tab index. */
defaultIndex: number defaultIndex: number
children: React.ReactNode children: React.ReactNode
@ -63,13 +63,13 @@ const AnimatedTabs: React.FC<IAnimatedInterface> = ({ children, ...rest }) => {
interface IAnimatedTab { interface IAnimatedTab {
/** ARIA role. */ /** ARIA role. */
role: 'button', role: 'button'
/** Element to represent the tab. */ /** Element to represent the tab. */
as: 'a' | 'button', as: 'a' | 'button'
/** Route to visit when the tab is chosen. */ /** Route to visit when the tab is chosen. */
href?: string, href?: string
/** Tab title text. */ /** Tab title text. */
title: string, title: string
/** Index value of the tab. */ /** Index value of the tab. */
index: number index: number
} }
@ -104,26 +104,26 @@ const AnimatedTab: React.FC<IAnimatedTab> = ({ index, ...props }) => {
/** Structure to represent a tab. */ /** Structure to represent a tab. */
export type Item = { export type Item = {
/** Tab text. */ /** Tab text. */
text: React.ReactNode, text: React.ReactNode
/** Tab tooltip text. */ /** Tab tooltip text. */
title?: string, title?: string
/** URL to visit when the tab is selected. */ /** URL to visit when the tab is selected. */
href?: string, href?: string
/** Route to visit when the tab is selected. */ /** Route to visit when the tab is selected. */
to?: string, to?: string
/** Callback when the tab is selected. */ /** Callback when the tab is selected. */
action?: () => void, action?: () => void
/** Display a counter over the tab. */ /** Display a counter over the tab. */
count?: number, count?: number
/** Unique name for this tab. */ /** Unique name for this tab. */
name: string name: string
} }
interface ITabs { interface ITabs {
/** Array of structured tab items. */ /** Array of structured tab items. */
items: Item[], items: Item[]
/** Name of the active tab item. */ /** Name of the active tab item. */
activeItem: string, activeItem: string
} }
/** Animated tabs component. */ /** Animated tabs component. */

View file

@ -5,9 +5,9 @@ import HStack from '../hstack/hstack';
import Tag from './tag'; import Tag from './tag';
interface ITagInput { interface ITagInput {
tags: string[], tags: string[]
onChange: (tags: string[]) => void, onChange: (tags: string[]) => void
placeholder?: string, placeholder?: string
} }
/** Manage a list of tags. */ /** Manage a list of tags. */

View file

@ -5,9 +5,9 @@ import Text from '../text/text';
interface ITag { interface ITag {
/** Name of the tag. */ /** Name of the tag. */
tag: string, tag: string
/** Callback when the X icon is pressed. */ /** Callback when the X icon is pressed. */
onDelete: (tag: string) => void, onDelete: (tag: string) => void
} }
/** A single editable Tag (used by TagInput). */ /** A single editable Tag (used by TagInput). */
@ -17,7 +17,7 @@ const Tag: React.FC<ITag> = ({ tag, onDelete }) => {
<Text theme='white'>{tag}</Text> <Text theme='white'>{tag}</Text>
<IconButton <IconButton
iconClassName='w-4 h-4' iconClassName='h-4 w-4'
src={require('@tabler/icons/x.svg')} src={require('@tabler/icons/x.svg')}
onClick={() => onDelete(tag)} onClick={() => onDelete(tag)}
transparent transparent

Some files were not shown because too many files have changed in this diff Show more