Merge branch 'tailwind-order' into 'develop'
Tailwind order Closes #1349 See merge request soapbox-pub/soapbox!2244
This commit is contained in:
commit
0ee8c8eba1
197 changed files with 536 additions and 522 deletions
|
@ -5,6 +5,7 @@ module.exports = {
|
||||||
'eslint:recommended',
|
'eslint:recommended',
|
||||||
'plugin:import/typescript',
|
'plugin:import/typescript',
|
||||||
'plugin:compat/recommended',
|
'plugin:compat/recommended',
|
||||||
|
'plugin:tailwindcss/recommended',
|
||||||
],
|
],
|
||||||
|
|
||||||
env: {
|
env: {
|
||||||
|
@ -61,6 +62,9 @@ module.exports = {
|
||||||
'URL', // core-js
|
'URL', // core-js
|
||||||
'URLSearchParams', // core-js
|
'URLSearchParams', // core-js
|
||||||
],
|
],
|
||||||
|
tailwindcss: {
|
||||||
|
config: 'tailwind.config.cjs',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
rules: {
|
rules: {
|
||||||
|
@ -235,18 +239,7 @@ module.exports = {
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
'import/newline-after-import': 'error',
|
'import/newline-after-import': 'error',
|
||||||
'import/no-extraneous-dependencies': [
|
'import/no-extraneous-dependencies': 'error',
|
||||||
'error',
|
|
||||||
// {
|
|
||||||
// devDependencies: [
|
|
||||||
// 'webpack/**',
|
|
||||||
// 'app/soapbox/test_setup.js',
|
|
||||||
// 'app/soapbox/test_helpers.js',
|
|
||||||
// 'app/**/__tests__/**',
|
|
||||||
// 'app/**/__mocks__/**',
|
|
||||||
// ],
|
|
||||||
// },
|
|
||||||
],
|
|
||||||
'import/no-unresolved': 'error',
|
'import/no-unresolved': 'error',
|
||||||
'import/no-webpack-loader-syntax': 'error',
|
'import/no-webpack-loader-syntax': 'error',
|
||||||
'import/order': [
|
'import/order': [
|
||||||
|
@ -271,6 +264,9 @@ module.exports = {
|
||||||
'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/migration-from-tailwind-2': 'error',
|
||||||
},
|
},
|
||||||
overrides: [
|
overrides: [
|
||||||
{
|
{
|
||||||
|
|
|
@ -72,7 +72,7 @@ const AccountSearch: React.FC<IAccountSearch> = ({ onSelected, ...rest }) => {
|
||||||
<div
|
<div
|
||||||
role='button'
|
role='button'
|
||||||
tabIndex={0}
|
tabIndex={0}
|
||||||
className='absolute inset-y-0 right-0 px-3 flex items-center cursor-pointer'
|
className='absolute inset-y-0 right-0 flex cursor-pointer items-center px-3'
|
||||||
onClick={handleClear}
|
onClick={handleClear}
|
||||||
>
|
>
|
||||||
<SvgIcon
|
<SvgIcon
|
||||||
|
|
|
@ -43,11 +43,11 @@ const InstanceFavicon: React.FC<IInstanceFavicon> = ({ account, disabled }) => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<button
|
<button
|
||||||
className='w-4 h-4 flex-none focus:ring-primary-500 focus:ring-2 focus:ring-offset-2'
|
className='h-4 w-4 flex-none focus:ring-2 focus:ring-primary-500 focus:ring-offset-2'
|
||||||
onClick={handleClick}
|
onClick={handleClick}
|
||||||
disabled={disabled}
|
disabled={disabled}
|
||||||
>
|
>
|
||||||
<img src={account.favicon} alt='' title={account.domain} className='w-full max-h-full' />
|
<img src={account.favicon} alt='' title={account.domain} className='max-h-full w-full' />
|
||||||
</button>
|
</button>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -147,7 +147,7 @@ const Account = ({
|
||||||
src={actionIcon}
|
src={actionIcon}
|
||||||
title={actionTitle}
|
title={actionTitle}
|
||||||
onClick={handleAction}
|
onClick={handleAction}
|
||||||
className='bg-transparent text-gray-600 dark:text-gray-600 hover:text-gray-700 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='w-4 h-4'
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
@ -193,7 +193,7 @@ const Account = ({
|
||||||
const LinkEl: any = withLinkToProfile ? Link : 'div';
|
const LinkEl: any = withLinkToProfile ? Link : 'div';
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div data-testid='account' className='flex-shrink-0 group block w-full' ref={overflowRef}>
|
<div data-testid='account' className='group block w-full shrink-0' ref={overflowRef}>
|
||||||
<HStack alignItems={actionAlignment} justifyContent='between'>
|
<HStack alignItems={actionAlignment} justifyContent='between'>
|
||||||
<HStack alignItems={withAccountNote || note ? 'top' : 'center'} space={3}>
|
<HStack alignItems={withAccountNote || note ? 'top' : 'center'} space={3}>
|
||||||
<ProfilePopper
|
<ProfilePopper
|
||||||
|
@ -208,14 +208,14 @@ const Account = ({
|
||||||
<Avatar src={account.avatar} size={avatarSize} />
|
<Avatar src={account.avatar} size={avatarSize} />
|
||||||
{emoji && (
|
{emoji && (
|
||||||
<Emoji
|
<Emoji
|
||||||
className='w-5 h-5 absolute -bottom-1.5 -right-1.5'
|
className='absolute -bottom-1.5 -right-1.5 h-5 w-5'
|
||||||
emoji={emoji}
|
emoji={emoji}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</LinkEl>
|
</LinkEl>
|
||||||
</ProfilePopper>
|
</ProfilePopper>
|
||||||
|
|
||||||
<div className='flex-grow'>
|
<div className='grow'>
|
||||||
<ProfilePopper
|
<ProfilePopper
|
||||||
condition={showProfileHoverCard}
|
condition={showProfileHoverCard}
|
||||||
wrapper={(children) => <HoverRefWrapper accountId={account.id} inline>{children}</HoverRefWrapper>}
|
wrapper={(children) => <HoverRefWrapper accountId={account.id} inline>{children}</HoverRefWrapper>}
|
||||||
|
|
|
@ -50,7 +50,7 @@ const AnimatedNumber: React.FC<IAnimatedNumber> = ({ value, obfuscate }) => {
|
||||||
return (
|
return (
|
||||||
<TransitionMotion styles={styles} willEnter={willEnter} willLeave={willLeave}>
|
<TransitionMotion styles={styles} willEnter={willEnter} willLeave={willLeave}>
|
||||||
{items => (
|
{items => (
|
||||||
<span className='inline-flex flex-col items-stretch relative overflow-hidden'>
|
<span className='relative inline-flex flex-col items-stretch overflow-hidden'>
|
||||||
{items.map(({ key, data, style }) => (
|
{items.map(({ key, data, style }) => (
|
||||||
<span key={key} style={{ position: (direction * style.y) > 0 ? 'absolute' : 'static', transform: `translateY(${style.y * 100}%)` }}>{obfuscate ? obfuscatedCount(data) : <FormattedNumber value={data} />}</span>
|
<span key={key} style={{ position: (direction * style.y) > 0 ? 'absolute' : 'static', transform: `translateY(${style.y * 100}%)` }}>{obfuscate ? obfuscatedCount(data) : <FormattedNumber value={data} />}</span>
|
||||||
))}
|
))}
|
||||||
|
|
|
@ -24,7 +24,7 @@ const Emoji: React.FC<IEmoji> = ({ emoji, emojiMap, hovered }) => {
|
||||||
return (
|
return (
|
||||||
<img
|
<img
|
||||||
draggable='false'
|
draggable='false'
|
||||||
className='emojione block m-0'
|
className='emojione m-0 block'
|
||||||
alt={emoji}
|
alt={emoji}
|
||||||
title={title}
|
title={title}
|
||||||
src={joinPublicPath(`packs/emoji/${filename}.svg`)}
|
src={joinPublicPath(`packs/emoji/${filename}.svg`)}
|
||||||
|
@ -37,7 +37,7 @@ const Emoji: React.FC<IEmoji> = ({ emoji, emojiMap, hovered }) => {
|
||||||
return (
|
return (
|
||||||
<img
|
<img
|
||||||
draggable='false'
|
draggable='false'
|
||||||
className='emojione block m-0'
|
className='emojione m-0 block'
|
||||||
alt={shortCode}
|
alt={shortCode}
|
||||||
title={shortCode}
|
title={shortCode}
|
||||||
src={filename as string}
|
src={filename as string}
|
||||||
|
|
|
@ -113,7 +113,7 @@ const BirthdayInput: React.FC<IBirthdayInput> = ({ value, onChange, required })
|
||||||
const handleChange = (date: Date) => onChange(date ? new Date(date.getTime() - (date.getTimezoneOffset() * 60000)).toISOString().slice(0, 10) : '');
|
const handleChange = (date: Date) => onChange(date ? new Date(date.getTime() - (date.getTimezoneOffset() * 60000)).toISOString().slice(0, 10) : '');
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='mt-1 relative rounded-md shadow-sm'>
|
<div className='relative mt-1 rounded-md shadow-sm'>
|
||||||
<BundleContainer fetchComponent={DatePicker}>
|
<BundleContainer fetchComponent={DatePicker}>
|
||||||
{Component => (<Component
|
{Component => (<Component
|
||||||
selected={selected}
|
selected={selected}
|
||||||
|
|
|
@ -196,7 +196,7 @@ class DropdownMenu extends React.PureComponent<IDropdownMenu, IDropdownMenuState
|
||||||
data-method={isLogout ? 'delete' : undefined}
|
data-method={isLogout ? 'delete' : undefined}
|
||||||
title={text}
|
title={text}
|
||||||
>
|
>
|
||||||
{icon && <SvgIcon src={icon} className='mr-3 rtl:ml-3 rtl:mr-0 h-5 w-5 flex-none' />}
|
{icon && <SvgIcon src={icon} className='mr-3 h-5 w-5 flex-none rtl:ml-3 rtl:mr-0' />}
|
||||||
|
|
||||||
<span className='truncate'>{text}</span>
|
<span className='truncate'>{text}</span>
|
||||||
|
|
||||||
|
|
|
@ -113,17 +113,17 @@ class ErrorBoundary extends React.PureComponent<Props, State> {
|
||||||
const errorText = this.getErrorText();
|
const errorText = this.getErrorText();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='h-screen pt-16 pb-12 flex flex-col bg-white dark:bg-primary-900'>
|
<div className='flex h-screen flex-col bg-white pt-16 pb-12 dark:bg-primary-900'>
|
||||||
<main className='flex-grow flex flex-col justify-center max-w-7xl w-full mx-auto px-4 sm:px-6 lg:px-8'>
|
<main className='mx-auto flex w-full max-w-7xl grow flex-col justify-center px-4 sm:px-6 lg:px-8'>
|
||||||
<div className='flex-shrink-0 flex justify-center'>
|
<div className='flex shrink-0 justify-center'>
|
||||||
<a href='/' className='inline-flex'>
|
<a href='/' className='inline-flex'>
|
||||||
<SiteLogo alt='Logo' className='h-12 w-auto cursor-pointer' />
|
<SiteLogo alt='Logo' className='h-12 w-auto cursor-pointer' />
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className='py-8'>
|
<div className='py-8'>
|
||||||
<div className='text-center max-w-xl mx-auto space-y-2'>
|
<div className='mx-auto max-w-xl space-y-2 text-center'>
|
||||||
<h1 className='text-3xl font-extrabold text-gray-900 dark:text-gray-500 tracking-tight sm:text-4xl'>
|
<h1 className='text-3xl font-extrabold tracking-tight text-gray-900 dark:text-gray-500 sm:text-4xl'>
|
||||||
<FormattedMessage id='alert.unexpected.message' defaultMessage='Something went wrong.' />
|
<FormattedMessage id='alert.unexpected.message' defaultMessage='Something went wrong.' />
|
||||||
</h1>
|
</h1>
|
||||||
<p className='text-lg text-gray-700 dark:text-gray-600'>
|
<p className='text-lg text-gray-700 dark:text-gray-600'>
|
||||||
|
@ -132,7 +132,7 @@ class ErrorBoundary extends React.PureComponent<Props, State> {
|
||||||
defaultMessage="We're sorry for the interruption. If the problem persists, please reach out to our support team. You may also try to {clearCookies} (this will log you out)."
|
defaultMessage="We're sorry for the interruption. If the problem persists, please reach out to our support team. You may also try to {clearCookies} (this will log you out)."
|
||||||
values={{
|
values={{
|
||||||
clearCookies: (
|
clearCookies: (
|
||||||
<a href='/' onClick={this.clearCookies} className='text-primary-600 dark:text-accent-blue hover:underline'>
|
<a href='/' onClick={this.clearCookies} className='text-primary-600 hover:underline dark:text-accent-blue'>
|
||||||
<FormattedMessage
|
<FormattedMessage
|
||||||
id='alert.unexpected.clear_cookies'
|
id='alert.unexpected.clear_cookies'
|
||||||
defaultMessage='clear cookies and browser data'
|
defaultMessage='clear cookies and browser data'
|
||||||
|
@ -150,7 +150,7 @@ class ErrorBoundary extends React.PureComponent<Props, State> {
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
<div className='mt-10'>
|
<div className='mt-10'>
|
||||||
<a href='/' className='text-base font-medium text-primary-600 dark:text-accent-blue hover:underline'>
|
<a href='/' className='text-base font-medium text-primary-600 hover:underline dark:text-accent-blue'>
|
||||||
<FormattedMessage id='alert.unexpected.return_home' defaultMessage='Return Home' />
|
<FormattedMessage id='alert.unexpected.return_home' defaultMessage='Return Home' />
|
||||||
<span aria-hidden='true'> →</span>
|
<span aria-hidden='true'> →</span>
|
||||||
</a>
|
</a>
|
||||||
|
@ -158,11 +158,11 @@ class ErrorBoundary extends React.PureComponent<Props, State> {
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{!isProduction && (
|
{!isProduction && (
|
||||||
<div className='py-16 max-w-lg mx-auto space-y-4'>
|
<div className='mx-auto max-w-lg space-y-4 py-16'>
|
||||||
{errorText && (
|
{errorText && (
|
||||||
<textarea
|
<textarea
|
||||||
ref={this.setTextareaRef}
|
ref={this.setTextareaRef}
|
||||||
className='h-48 p-4 shadow-sm bg-gray-100 text-gray-900 dark:text-gray-100 dark:bg-gray-800 focus:ring-2 focus:ring-primary-500 focus:border-primary-500 block w-full sm:text-sm border-gray-300 dark:border-gray-700 rounded-md font-mono'
|
className='block h-48 w-full rounded-md border-gray-300 bg-gray-100 p-4 font-mono text-gray-900 shadow-sm focus:border-primary-500 focus:ring-2 focus:ring-primary-500 dark:border-gray-700 dark:bg-gray-800 dark:text-gray-100 sm:text-sm'
|
||||||
value={errorText}
|
value={errorText}
|
||||||
onClick={this.handleCopy}
|
onClick={this.handleCopy}
|
||||||
readOnly
|
readOnly
|
||||||
|
@ -180,11 +180,11 @@ class ErrorBoundary extends React.PureComponent<Props, State> {
|
||||||
</div>
|
</div>
|
||||||
</main>
|
</main>
|
||||||
|
|
||||||
<footer className='flex-shrink-0 max-w-7xl w-full mx-auto px-4 sm:px-6 lg:px-8'>
|
<footer className='mx-auto w-full max-w-7xl shrink-0 px-4 sm:px-6 lg:px-8'>
|
||||||
<HStack justifyContent='center' space={4} element='nav'>
|
<HStack justifyContent='center' space={4} element='nav'>
|
||||||
{links.get('status') && (
|
{links.get('status') && (
|
||||||
<>
|
<>
|
||||||
<a href={links.get('status')} className='text-sm font-medium text-gray-700 dark:text-gray-600 hover:underline'>
|
<a href={links.get('status')} className='text-sm font-medium text-gray-700 hover:underline dark:text-gray-600'>
|
||||||
<FormattedMessage id='alert.unexpected.links.status' defaultMessage='Status' />
|
<FormattedMessage id='alert.unexpected.links.status' defaultMessage='Status' />
|
||||||
</a>
|
</a>
|
||||||
</>
|
</>
|
||||||
|
@ -193,7 +193,7 @@ class ErrorBoundary extends React.PureComponent<Props, State> {
|
||||||
{links.get('help') && (
|
{links.get('help') && (
|
||||||
<>
|
<>
|
||||||
<span className='inline-block border-l border-gray-300' aria-hidden='true' />
|
<span className='inline-block border-l border-gray-300' aria-hidden='true' />
|
||||||
<a href={links.get('help')} className='text-sm font-medium text-gray-700 dark:text-gray-600 hover:underline'>
|
<a href={links.get('help')} className='text-sm font-medium text-gray-700 hover:underline dark:text-gray-600'>
|
||||||
<FormattedMessage id='alert.unexpected.links.help' defaultMessage='Help Center' />
|
<FormattedMessage id='alert.unexpected.links.help' defaultMessage='Help Center' />
|
||||||
</a>
|
</a>
|
||||||
</>
|
</>
|
||||||
|
@ -202,7 +202,7 @@ class ErrorBoundary extends React.PureComponent<Props, State> {
|
||||||
{links.get('support') && (
|
{links.get('support') && (
|
||||||
<>
|
<>
|
||||||
<span className='inline-block border-l border-gray-300' aria-hidden='true' />
|
<span className='inline-block border-l border-gray-300' aria-hidden='true' />
|
||||||
<a href={links.get('support')} className='text-sm font-medium text-gray-700 dark:text-gray-600 hover:underline'>
|
<a href={links.get('support')} className='text-sm font-medium text-gray-700 hover:underline dark:text-gray-600'>
|
||||||
<FormattedMessage id='alert.unexpected.links.support' defaultMessage='Support' />
|
<FormattedMessage id='alert.unexpected.links.support' defaultMessage='Support' />
|
||||||
</a>
|
</a>
|
||||||
</>
|
</>
|
||||||
|
|
|
@ -55,7 +55,7 @@ const EventPreview: React.FC<IEventPreview> = ({ status, className, hideAction,
|
||||||
<div className='absolute top-28 right-3'>
|
<div className='absolute top-28 right-3'>
|
||||||
{floatingAction && action}
|
{floatingAction && action}
|
||||||
</div>
|
</div>
|
||||||
<div className='bg-primary-200 dark:bg-gray-600 h-40'>
|
<div className='h-40 bg-primary-200 dark:bg-gray-600'>
|
||||||
{banner && <img className='h-full w-full object-cover' src={banner.url} alt={intl.formatMessage(messages.eventBanner)} />}
|
{banner && <img className='h-full w-full object-cover' src={banner.url} alt={intl.formatMessage(messages.eventBanner)} />}
|
||||||
</div>
|
</div>
|
||||||
<Stack className='p-2.5' space={2}>
|
<Stack className='p-2.5' space={2}>
|
||||||
|
@ -65,7 +65,7 @@ const EventPreview: React.FC<IEventPreview> = ({ status, className, hideAction,
|
||||||
{!floatingAction && action}
|
{!floatingAction && action}
|
||||||
</HStack>
|
</HStack>
|
||||||
|
|
||||||
<div className='flex gap-y-1 gap-x-2 flex-wrap text-gray-700 dark:text-gray-600'>
|
<div className='flex flex-wrap gap-y-1 gap-x-2 text-gray-700 dark:text-gray-600'>
|
||||||
<HStack alignItems='center' space={2}>
|
<HStack alignItems='center' space={2}>
|
||||||
<Icon src={require('@tabler/icons/user.svg')} />
|
<Icon src={require('@tabler/icons/user.svg')} />
|
||||||
<HStack space={1} alignItems='center' grow>
|
<HStack space={1} alignItems='center' grow>
|
||||||
|
|
|
@ -31,7 +31,7 @@ const GdprBanner: React.FC = () => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Banner theme='opaque' className={classNames('transition-transform', { 'translate-y-full': slideout })}>
|
<Banner theme='opaque' className={classNames('transition-transform', { 'translate-y-full': slideout })}>
|
||||||
<div className='flex flex-col space-y-4 lg:space-y-0 lg:space-x-4 rtl:space-x-reverse lg:flex-row lg:items-center lg:justify-between'>
|
<div className='flex flex-col space-y-4 rtl:space-x-reverse lg:flex-row lg:items-center lg:justify-between lg:space-y-0 lg:space-x-4'>
|
||||||
<Stack space={2}>
|
<Stack space={2}>
|
||||||
<Text size='xl' weight='bold'>
|
<Text size='xl' weight='bold'>
|
||||||
<FormattedMessage id='gdpr.title' defaultMessage='{siteTitle} uses cookies' values={{ siteTitle: instance.title }} />
|
<FormattedMessage id='gdpr.title' defaultMessage='{siteTitle} uses cookies' values={{ siteTitle: instance.title }} />
|
||||||
|
|
|
@ -18,9 +18,9 @@ const GroupCard: React.FC<IGroupCard> = ({ group }) => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='overflow-hidden'>
|
<div className='overflow-hidden'>
|
||||||
<Stack className='bg-white dark:bg-primary-900 border border-solid border-gray-300 dark:border-primary-800 rounded-lg sm:rounded-xl'>
|
<Stack className='rounded-lg border border-solid border-gray-300 bg-white dark:border-primary-800 dark:bg-primary-900 sm:rounded-xl'>
|
||||||
<div className='bg-primary-100 dark:bg-gray-800 h-[120px] relative -m-[1px] mb-0 rounded-t-lg sm:rounded-t-xl'>
|
<div className='relative -m-[1px] mb-0 h-[120px] rounded-t-lg bg-primary-100 dark:bg-gray-800 sm:rounded-t-xl'>
|
||||||
{group.header && <img className='h-full w-full object-cover rounded-t-lg sm:rounded-t-xl' src={group.header} alt={intl.formatMessage(messages.groupHeader)} />}
|
{group.header && <img className='h-full w-full rounded-t-lg object-cover sm:rounded-t-xl' src={group.header} alt={intl.formatMessage(messages.groupHeader)} />}
|
||||||
<div className='absolute left-1/2 bottom-0 -translate-x-1/2 translate-y-1/2'>
|
<div className='absolute left-1/2 bottom-0 -translate-x-1/2 translate-y-1/2'>
|
||||||
<Avatar className='ring-2 ring-white dark:ring-primary-900' src={group.avatar} size={64} />
|
<Avatar className='ring-2 ring-white dark:ring-primary-900' src={group.avatar} size={64} />
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -2,7 +2,7 @@ import React from 'react';
|
||||||
|
|
||||||
/** Fullscreen gradient used as a backdrop to public pages. */
|
/** Fullscreen gradient used as a backdrop to public pages. */
|
||||||
const LandingGradient: React.FC = () => (
|
const LandingGradient: React.FC = () => (
|
||||||
<div className='fixed h-screen w-full bg-gradient-to-tr from-primary-50 dark:from-primary-900/50 via-white dark:via-primary-900 to-gradient-end/10 dark:to-primary-800/50' />
|
<div className='fixed h-screen w-full bg-gradient-to-tr from-primary-50 via-white to-gradient-end/10 dark:from-primary-900/50 dark:via-primary-900 dark:to-primary-800/50' />
|
||||||
);
|
);
|
||||||
|
|
||||||
export default LandingGradient;
|
export default LandingGradient;
|
||||||
|
|
|
@ -4,7 +4,7 @@ import { Link as Comp, LinkProps } from 'react-router-dom';
|
||||||
const Link = (props: LinkProps) => (
|
const Link = (props: LinkProps) => (
|
||||||
<Comp
|
<Comp
|
||||||
{...props}
|
{...props}
|
||||||
className='text-primary-600 dark:text-accent-blue hover:underline'
|
className='text-primary-600 hover:underline dark:text-accent-blue'
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -9,7 +9,7 @@ const LoadingScreen: React.FC = () => {
|
||||||
<div className='fixed h-screen w-screen'>
|
<div className='fixed h-screen w-screen'>
|
||||||
<LandingGradient />
|
<LandingGradient />
|
||||||
|
|
||||||
<div className='fixed d-screen w-screen flex items-center justify-center z-10'>
|
<div className='d-screen fixed z-10 flex w-screen items-center justify-center'>
|
||||||
<div className='p-4'>
|
<div className='p-4'>
|
||||||
<Spinner size={40} withText={false} />
|
<Spinner size={40} withText={false} />
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -171,7 +171,7 @@ const Item: React.FC<IItem> = ({
|
||||||
target='_blank'
|
target='_blank'
|
||||||
>
|
>
|
||||||
<StillImage
|
<StillImage
|
||||||
className='w-full h-full'
|
className='h-full w-full'
|
||||||
src={mediaPreview ? attachment.preview_url : attachment.url}
|
src={mediaPreview ? attachment.preview_url : attachment.url}
|
||||||
alt={attachment.description}
|
alt={attachment.description}
|
||||||
letterboxed={letterboxed}
|
letterboxed={letterboxed}
|
||||||
|
|
|
@ -241,7 +241,7 @@ const ModalRoot: React.FC<IModalRoot> = ({ children, onCancel, onClose, type })
|
||||||
<div
|
<div
|
||||||
role='presentation'
|
role='presentation'
|
||||||
id='modal-overlay'
|
id='modal-overlay'
|
||||||
className='fixed inset-0 bg-gray-500/90 dark:bg-gray-700/90 backdrop-blur'
|
className='fixed inset-0 bg-gray-500/90 backdrop-blur dark:bg-gray-700/90'
|
||||||
onClick={handleOnClose}
|
onClick={handleOnClose}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
|
|
@ -20,7 +20,7 @@ const PollPercentageBar: React.FC<{ percent: number, leading: boolean }> = ({ pe
|
||||||
<Motion defaultStyle={{ width: 0 }} style={{ width: spring(percent, { ...presets.gentle, precision: 0.1 }) }}>
|
<Motion defaultStyle={{ width: 0 }} style={{ width: spring(percent, { ...presets.gentle, precision: 0.1 }) }}>
|
||||||
{({ width }) => (
|
{({ width }) => (
|
||||||
<span
|
<span
|
||||||
className='absolute inset-0 h-full inline-block bg-primary-100 dark:bg-primary-500 rounded-l-md'
|
className='absolute inset-0 inline-block h-full rounded-l-md bg-primary-100 dark:bg-primary-500'
|
||||||
style={{ width: `${width}%` }}
|
style={{ width: `${width}%` }}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
@ -61,8 +61,8 @@ const PollOptionText: React.FC<IPollOptionText> = ({ poll, option, index, active
|
||||||
onChange={handleOptionChange}
|
onChange={handleOptionChange}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div className='grid items-center w-full'>
|
<div className='grid w-full items-center'>
|
||||||
<div className='col-start-1 row-start-1 justify-self-center ml-4 mr-6'>
|
<div className='col-start-1 row-start-1 ml-4 mr-6 justify-self-center'>
|
||||||
<div className='text-primary-600 dark:text-white'>
|
<div className='text-primary-600 dark:text-white'>
|
||||||
<Text
|
<Text
|
||||||
theme='inherit'
|
theme='inherit'
|
||||||
|
@ -72,7 +72,7 @@ const PollOptionText: React.FC<IPollOptionText> = ({ poll, option, index, active
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className='col-start-1 row-start-1 justify-self-end flex items-center'>
|
<div className='col-start-1 row-start-1 flex items-center justify-self-end'>
|
||||||
<span
|
<span
|
||||||
className={classNames('flex items-center justify-center w-6 h-6 flex-none border border-solid rounded-full', {
|
className={classNames('flex items-center justify-center w-6 h-6 flex-none border border-solid rounded-full', {
|
||||||
'bg-primary-600 border-primary-600 dark:bg-primary-300 dark:border-primary-300': active,
|
'bg-primary-600 border-primary-600 dark:bg-primary-300 dark:border-primary-300': active,
|
||||||
|
@ -85,7 +85,7 @@ const PollOptionText: React.FC<IPollOptionText> = ({ poll, option, index, active
|
||||||
aria-label={option.title}
|
aria-label={option.title}
|
||||||
>
|
>
|
||||||
{active && (
|
{active && (
|
||||||
<Icon src={require('@tabler/icons/check.svg')} className='text-white dark:text-primary-900 w-4 h-4' />
|
<Icon src={require('@tabler/icons/check.svg')} className='h-4 w-4 text-white dark:text-primary-900' />
|
||||||
)}
|
)}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
@ -123,7 +123,7 @@ const PollOption: React.FC<IPollOption> = (props): JSX.Element | null => {
|
||||||
<HStack
|
<HStack
|
||||||
justifyContent='between'
|
justifyContent='between'
|
||||||
alignItems='center'
|
alignItems='center'
|
||||||
className='relative p-2 w-full bg-white dark:bg-primary-800 rounded-md overflow-hidden'
|
className='relative w-full overflow-hidden rounded-md bg-white p-2 dark:bg-primary-800'
|
||||||
>
|
>
|
||||||
<PollPercentageBar percent={percent} leading={leading} />
|
<PollPercentageBar percent={percent} leading={leading} />
|
||||||
|
|
||||||
|
@ -141,7 +141,7 @@ const PollOption: React.FC<IPollOption> = (props): JSX.Element | null => {
|
||||||
<Icon
|
<Icon
|
||||||
src={require('@tabler/icons/circle-check.svg')}
|
src={require('@tabler/icons/circle-check.svg')}
|
||||||
alt={intl.formatMessage(messages.voted)}
|
alt={intl.formatMessage(messages.voted)}
|
||||||
className='text-primary-600 dark:text-primary-800 dark:fill-white w-4 h-4'
|
className='h-4 w-4 text-primary-600 dark:fill-white dark:text-primary-800'
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<div className='svg-icon' />
|
<div className='svg-icon' />
|
||||||
|
|
|
@ -123,7 +123,7 @@ export const ProfileHoverCard: React.FC<IProfileHoverCard> = ({ visible = true }
|
||||||
<HStack alignItems='center' space={0.5}>
|
<HStack alignItems='center' space={0.5}>
|
||||||
<Icon
|
<Icon
|
||||||
src={require('@tabler/icons/calendar.svg')}
|
src={require('@tabler/icons/calendar.svg')}
|
||||||
className='w-4 h-4 text-gray-800 dark:text-gray-200'
|
className='h-4 w-4 text-gray-800 dark:text-gray-200'
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Text size='sm'>
|
<Text size='sm'>
|
||||||
|
|
|
@ -83,7 +83,7 @@ const ScrollTopButton: React.FC<IScrollTopButton> = ({
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={classes}>
|
<div className={classes}>
|
||||||
<a className='flex items-center bg-primary-600 hover:bg-primary-700 hover:scale-105 active:scale-100 transition-transform text-white rounded-full px-4 py-2 space-x-1.5 cursor-pointer whitespace-nowrap' onClick={handleClick}>
|
<a className='flex cursor-pointer items-center space-x-1.5 whitespace-nowrap rounded-full bg-primary-600 px-4 py-2 text-white transition-transform hover:scale-105 hover:bg-primary-700 active:scale-100' onClick={handleClick}>
|
||||||
<Icon src={require('@tabler/icons/arrow-bar-to-up.svg')} />
|
<Icon src={require('@tabler/icons/arrow-bar-to-up.svg')} />
|
||||||
|
|
||||||
{(count > 0) && (
|
{(count > 0) && (
|
||||||
|
|
|
@ -53,8 +53,8 @@ interface ISidebarLink {
|
||||||
const SidebarLink: React.FC<ISidebarLink> = ({ href, to, icon, text, onClick }) => {
|
const SidebarLink: React.FC<ISidebarLink> = ({ href, to, icon, text, onClick }) => {
|
||||||
const body = (
|
const body = (
|
||||||
<HStack space={2} alignItems='center'>
|
<HStack space={2} alignItems='center'>
|
||||||
<div className='bg-primary-50 dark:bg-gray-800 relative rounded-full inline-flex p-2'>
|
<div className='relative inline-flex rounded-full bg-primary-50 p-2 dark:bg-gray-800'>
|
||||||
<Icon src={icon} className='text-primary-500 h-5 w-5' />
|
<Icon src={icon} className='h-5 w-5 text-primary-500' />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Text tag='span' weight='medium' theme='inherit'>{text}</Text>
|
<Text tag='span' weight='medium' theme='inherit'>{text}</Text>
|
||||||
|
@ -63,14 +63,14 @@ const SidebarLink: React.FC<ISidebarLink> = ({ href, to, icon, text, onClick })
|
||||||
|
|
||||||
if (to) {
|
if (to) {
|
||||||
return (
|
return (
|
||||||
<NavLink className='group rounded-full text-gray-900 dark:text-gray-100 hover:bg-gray-50 dark:hover:bg-gray-800' to={to} onClick={onClick}>
|
<NavLink className='group rounded-full text-gray-900 hover:bg-gray-50 dark:text-gray-100 dark:hover:bg-gray-800' to={to} onClick={onClick}>
|
||||||
{body}
|
{body}
|
||||||
</NavLink>
|
</NavLink>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<a className='group rounded-full text-gray-900 dark:text-gray-100 hover:bg-gray-50 dark:hover:bg-gray-800' href={href} target='_blank' onClick={onClick}>
|
<a className='group rounded-full text-gray-900 hover:bg-gray-50 dark:text-gray-100 dark:hover:bg-gray-800' href={href} target='_blank' onClick={onClick}>
|
||||||
{body}
|
{body}
|
||||||
</a>
|
</a>
|
||||||
);
|
);
|
||||||
|
@ -165,10 +165,10 @@ const SidebarMenu: React.FC = (): JSX.Element | null => {
|
||||||
src={require('@tabler/icons/x.svg')}
|
src={require('@tabler/icons/x.svg')}
|
||||||
ref={closeButtonRef}
|
ref={closeButtonRef}
|
||||||
iconClassName='h-6 w-6'
|
iconClassName='h-6 w-6'
|
||||||
className='absolute top-0 right-0 -mr-11 mt-2 text-gray-600 dark:text-gray-400 hover:text-gray-600 dark:hover:text-gray-300'
|
className='absolute top-0 right-0 -mr-11 mt-2 text-gray-600 hover:text-gray-600 dark:text-gray-400 dark:hover:text-gray-300'
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div className='relative overflow-y-scroll overflow-auto h-full w-full'>
|
<div className='relative h-full w-full overflow-auto overflow-y-scroll'>
|
||||||
<div className='p-4'>
|
<div className='p-4'>
|
||||||
<Stack space={4}>
|
<Stack space={4}>
|
||||||
<Link to={`/@${account.acct}`} onClick={onClose}>
|
<Link to={`/@${account.acct}`} onClick={onClose}>
|
||||||
|
@ -342,11 +342,11 @@ const SidebarMenu: React.FC = (): JSX.Element | null => {
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
{switcher && (
|
{switcher && (
|
||||||
<div className='border-t-2 border-gray-100 dark:border-gray-800 border-solid'>
|
<div className='border-t-2 border-solid border-gray-100 dark:border-gray-800'>
|
||||||
{otherAccounts.map(account => renderAccount(account))}
|
{otherAccounts.map(account => renderAccount(account))}
|
||||||
|
|
||||||
<NavLink className='flex items-center py-2 space-x-1' to='/login/add' onClick={handleClose}>
|
<NavLink className='flex items-center space-x-1 py-2' to='/login/add' onClick={handleClose}>
|
||||||
<Icon className='text-primary-500 w-4 h-4' src={require('@tabler/icons/plus.svg')} />
|
<Icon className='h-4 w-4 text-primary-500' src={require('@tabler/icons/plus.svg')} />
|
||||||
<Text size='sm' weight='medium'>{intl.formatMessage(messages.addAccount)}</Text>
|
<Text size='sm' weight='medium'>{intl.formatMessage(messages.addAccount)}</Text>
|
||||||
</NavLink>
|
</NavLink>
|
||||||
</div>
|
</div>
|
||||||
|
@ -361,7 +361,7 @@ const SidebarMenu: React.FC = (): JSX.Element | null => {
|
||||||
{/* Dummy element to keep Close Icon visible */}
|
{/* Dummy element to keep Close Icon visible */}
|
||||||
<div
|
<div
|
||||||
aria-hidden
|
aria-hidden
|
||||||
className='w-14 flex-shrink-0'
|
className='w-14 shrink-0'
|
||||||
onClick={handleClose}
|
onClick={handleClose}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -41,8 +41,8 @@ const StatusActionButton = React.forwardRef<HTMLButtonElement, IStatusActionButt
|
||||||
const renderIcon = () => {
|
const renderIcon = () => {
|
||||||
if (emoji) {
|
if (emoji) {
|
||||||
return (
|
return (
|
||||||
<span className='flex w-6 h-6 items-center justify-center'>
|
<span className='flex h-6 w-6 items-center justify-center'>
|
||||||
<Emoji className='w-full h-full p-0.5' emoji={emoji} />
|
<Emoji className='h-full w-full p-0.5' emoji={emoji} />
|
||||||
</span>
|
</span>
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -25,7 +25,7 @@ interface IReadMoreButton {
|
||||||
|
|
||||||
/** Button to expand a truncated status (due to too much content) */
|
/** Button to expand a truncated status (due to too much content) */
|
||||||
const ReadMoreButton: React.FC<IReadMoreButton> = ({ onClick }) => (
|
const ReadMoreButton: React.FC<IReadMoreButton> = ({ onClick }) => (
|
||||||
<button className='flex items-center text-gray-900 dark:text-gray-300 border-0 bg-transparent p-0 pt-2 hover:underline active:underline' onClick={onClick}>
|
<button className='flex items-center border-0 bg-transparent p-0 pt-2 text-gray-900 hover:underline active:underline dark:text-gray-300' onClick={onClick}>
|
||||||
<FormattedMessage id='status.read_more' defaultMessage='Read more' />
|
<FormattedMessage id='status.read_more' defaultMessage='Read more' />
|
||||||
<Icon className='inline-block h-5 w-5' src={require('@tabler/icons/chevron-right.svg')} />
|
<Icon className='inline-block h-5 w-5' src={require('@tabler/icons/chevron-right.svg')} />
|
||||||
</button>
|
</button>
|
||||||
|
|
|
@ -73,7 +73,7 @@ const StatusReplyMentions: React.FC<IStatusReplyMentions> = ({ status, hoverable
|
||||||
|
|
||||||
if (to.size > 2) {
|
if (to.size > 2) {
|
||||||
accounts.push(
|
accounts.push(
|
||||||
<span key='more' className='hover:underline cursor-pointer' role='button' onClick={handleOpenMentionsModal} tabIndex={0}>
|
<span key='more' className='cursor-pointer hover:underline' role='button' onClick={handleOpenMentionsModal} tabIndex={0}>
|
||||||
<FormattedMessage id='reply_mentions.more' defaultMessage='{count} more' values={{ count: to.size - 2 }} />
|
<FormattedMessage id='reply_mentions.more' defaultMessage='{count} more' values={{ count: to.size - 2 }} />
|
||||||
</span>,
|
</span>,
|
||||||
);
|
);
|
||||||
|
@ -93,7 +93,7 @@ const StatusReplyMentions: React.FC<IStatusReplyMentions> = ({ status, hoverable
|
||||||
<HoverStatusWrapper statusId={status.in_reply_to_id} inline>
|
<HoverStatusWrapper statusId={status.in_reply_to_id} inline>
|
||||||
<span
|
<span
|
||||||
key='hoverstatus'
|
key='hoverstatus'
|
||||||
className='hover:underline cursor-pointer'
|
className='cursor-pointer hover:underline'
|
||||||
role='presentation'
|
role='presentation'
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
|
|
|
@ -107,8 +107,8 @@ const SensitiveContentOverlay = React.forwardRef<HTMLDivElement, ISensitiveConte
|
||||||
size='sm'
|
size='sm'
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<div className='flex justify-center items-center max-h-screen'>
|
<div className='flex max-h-screen items-center justify-center'>
|
||||||
<div className='text-center w-3/4 mx-auto space-y-4' ref={ref}>
|
<div className='mx-auto w-3/4 space-y-4 text-center' ref={ref}>
|
||||||
<div className='space-y-1'>
|
<div className='space-y-1'>
|
||||||
<Text theme='white' weight='semibold'>
|
<Text theme='white' weight='semibold'>
|
||||||
{intl.formatMessage(isUnderReview ? messages.underReviewTitle : messages.sensitiveTitle)}
|
{intl.formatMessage(isUnderReview ? messages.underReviewTitle : messages.sensitiveTitle)}
|
||||||
|
|
|
@ -21,7 +21,7 @@ const StatusInfo = (props: IStatusInfo) => {
|
||||||
return (
|
return (
|
||||||
<Container
|
<Container
|
||||||
{...containerProps}
|
{...containerProps}
|
||||||
className='flex items-center text-gray-700 dark:text-gray-600 text-xs font-medium space-x-3 rtl:space-x-reverse hover:underline'
|
className='flex items-center space-x-3 text-xs font-medium text-gray-700 hover:underline rtl:space-x-reverse dark:text-gray-600'
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className='flex justify-end'
|
className='flex justify-end'
|
||||||
|
|
|
@ -70,7 +70,7 @@ const StillImage: React.FC<IStillImage> = ({ alt, className, src, style, letterb
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{(hoverToPlay && showExt) && (
|
{(hoverToPlay && showExt) && (
|
||||||
<div className='group-hover:hidden absolute opacity-90 left-2 bottom-2 pointer-events-none'>
|
<div className='pointer-events-none absolute left-2 bottom-2 opacity-90 group-hover:hidden'>
|
||||||
<ExtensionBadge ext='GIF' />
|
<ExtensionBadge ext='GIF' />
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
@ -86,7 +86,7 @@ interface IExtensionBadge {
|
||||||
/** Badge displaying a file extension. */
|
/** Badge displaying a file extension. */
|
||||||
const ExtensionBadge: React.FC<IExtensionBadge> = ({ ext }) => {
|
const ExtensionBadge: React.FC<IExtensionBadge> = ({ ext }) => {
|
||||||
return (
|
return (
|
||||||
<div className='inline-flex items-center px-2 py-0.5 rounded text-sm font-medium bg-gray-100 dark:bg-gray-800 text-gray-900 dark:text-gray-100'>
|
<div className='inline-flex items-center rounded bg-gray-100 px-2 py-0.5 text-sm font-medium text-gray-900 dark:bg-gray-800 dark:text-gray-100'>
|
||||||
{ext}
|
{ext}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
|
@ -19,7 +19,7 @@ const Tombstone: React.FC<ITombstone> = ({ id, onMoveUp, onMoveDown }) => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<HotKeys handlers={handlers}>
|
<HotKeys handlers={handlers}>
|
||||||
<div className='p-9 flex items-center justify-center sm:rounded-xl bg-gray-100 border border-solid border-gray-200 dark:bg-gray-900 dark:border-gray-800 focusable' tabIndex={0}>
|
<div className='focusable flex items-center justify-center border border-solid border-gray-200 bg-gray-100 p-9 dark:border-gray-800 dark:bg-gray-900 sm:rounded-xl' tabIndex={0}>
|
||||||
<Text>
|
<Text>
|
||||||
<FormattedMessage id='statuses.tombstone' defaultMessage='One or more posts are unavailable.' />
|
<FormattedMessage id='statuses.tombstone' defaultMessage='One or more posts are unavailable.' />
|
||||||
</Text>
|
</Text>
|
||||||
|
|
|
@ -36,13 +36,13 @@ const Accordion: React.FC<IAccordion> = ({ headline, children, menu, expanded =
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='bg-white dark:bg-primary-800 text-gray-900 dark:text-gray-100 rounded-lg shadow dark:shadow-none'>
|
<div className='rounded-lg bg-white text-gray-900 shadow dark:bg-primary-800 dark:text-gray-100 dark:shadow-none'>
|
||||||
<button
|
<button
|
||||||
type='button'
|
type='button'
|
||||||
onClick={handleToggle}
|
onClick={handleToggle}
|
||||||
title={intl.formatMessage(expanded ? messages.collapse : messages.expand)}
|
title={intl.formatMessage(expanded ? messages.collapse : messages.expand)}
|
||||||
aria-expanded={expanded}
|
aria-expanded={expanded}
|
||||||
className='px-4 py-3 font-semibold flex items-center justify-between w-full'
|
className='flex w-full items-center justify-between px-4 py-3 font-semibold'
|
||||||
>
|
>
|
||||||
<span>{headline}</span>
|
<span>{headline}</span>
|
||||||
|
|
||||||
|
@ -55,7 +55,7 @@ const Accordion: React.FC<IAccordion> = ({ headline, children, menu, expanded =
|
||||||
)}
|
)}
|
||||||
<Icon
|
<Icon
|
||||||
src={expanded ? require('@tabler/icons/chevron-up.svg') : require('@tabler/icons/chevron-down.svg')}
|
src={expanded ? require('@tabler/icons/chevron-up.svg') : require('@tabler/icons/chevron-down.svg')}
|
||||||
className='text-gray-700 dark:text-gray-600 h-5 w-5'
|
className='h-5 w-5 text-gray-700 dark:text-gray-600'
|
||||||
/>
|
/>
|
||||||
</HStack>
|
</HStack>
|
||||||
</button>
|
</button>
|
||||||
|
|
|
@ -17,7 +17,7 @@ const Banner: React.FC<IBanner> = ({ theme, children, className }) => {
|
||||||
'bg-white dark:bg-gray-800 text-gray-900 dark:text-gray-100 shadow-3xl dark:shadow-inset': theme === 'opaque',
|
'bg-white dark:bg-gray-800 text-gray-900 dark:text-gray-100 shadow-3xl dark:shadow-inset': theme === 'opaque',
|
||||||
}, className)}
|
}, className)}
|
||||||
>
|
>
|
||||||
<div className='max-w-4xl mx-auto px-4'>
|
<div className='mx-auto max-w-4xl px-4'>
|
||||||
{children}
|
{children}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -63,7 +63,7 @@ const Button = React.forwardRef<HTMLButtonElement, IButton>((props, ref): JSX.El
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return <Icon src={icon} className='w-4 h-4' />;
|
return <Icon src={icon} className='h-4 w-4' />;
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleClick: React.MouseEventHandler<HTMLButtonElement> = React.useCallback((event) => {
|
const handleClick: React.MouseEventHandler<HTMLButtonElement> = React.useCallback((event) => {
|
||||||
|
|
|
@ -64,7 +64,7 @@ const CardHeader: React.FC<ICardHeader> = ({ className, children, backHref, onBa
|
||||||
const backAttributes = backHref ? { to: backHref } : { onClick: onBackClick };
|
const backAttributes = backHref ? { to: backHref } : { onClick: onBackClick };
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Comp {...backAttributes} className='text-gray-900 dark:text-gray-100 focus:ring-primary-500 focus:ring-2' aria-label={intl.formatMessage(messages.back)}>
|
<Comp {...backAttributes} className='text-gray-900 focus:ring-2 focus:ring-primary-500 dark:text-gray-100' aria-label={intl.formatMessage(messages.back)}>
|
||||||
<SvgIcon src={require('@tabler/icons/arrow-left.svg')} className='h-6 w-6 rtl:rotate-180' />
|
<SvgIcon src={require('@tabler/icons/arrow-left.svg')} className='h-6 w-6 rtl:rotate-180' />
|
||||||
<span className='sr-only' data-testid='back-button'>{intl.formatMessage(messages.back)}</span>
|
<span className='sr-only' data-testid='back-button'>{intl.formatMessage(messages.back)}</span>
|
||||||
</Comp>
|
</Comp>
|
||||||
|
|
|
@ -9,7 +9,7 @@ const Checkbox = React.forwardRef<HTMLInputElement, ICheckbox>((props, ref) => {
|
||||||
{...props}
|
{...props}
|
||||||
ref={ref}
|
ref={ref}
|
||||||
type='checkbox'
|
type='checkbox'
|
||||||
className='border-2 dark:bg-gray-900 dark:border-gray-800 focus:ring-primary-500 h-4 w-4 text-primary-600 border-gray-300 rounded'
|
className='h-4 w-4 rounded border-2 border-gray-300 text-primary-600 focus:ring-primary-500 dark:border-gray-800 dark:bg-gray-900'
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
|
@ -12,7 +12,7 @@ interface ICounter {
|
||||||
/** A simple counter for notifications, etc. */
|
/** A simple counter for notifications, etc. */
|
||||||
const Counter: React.FC<ICounter> = ({ count, countMax }) => {
|
const Counter: React.FC<ICounter> = ({ count, countMax }) => {
|
||||||
return (
|
return (
|
||||||
<span className='h-5 min-w-[20px] max-w-[26px] flex items-center justify-center bg-secondary-500 text-xs font-medium text-white rounded-full ring-2 ring-white dark:ring-gray-800'>
|
<span className='flex h-5 min-w-[20px] max-w-[26px] items-center justify-center rounded-full bg-secondary-500 text-xs font-medium text-white ring-2 ring-white dark:ring-gray-800'>
|
||||||
{shortNumberFormat(count, countMax)}
|
{shortNumberFormat(count, countMax)}
|
||||||
</span>
|
</span>
|
||||||
);
|
);
|
||||||
|
|
|
@ -31,7 +31,7 @@ const Datepicker = ({ onChange }: IDatepicker) => {
|
||||||
}, [month, day, year]);
|
}, [month, day, year]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='grid grid-cols-1 gap-y-2 gap-x-2 sm:grid-cols-3'>
|
<div className='grid grid-cols-1 gap-2 sm:grid-cols-3'>
|
||||||
<div className='sm:col-span-1'>
|
<div className='sm:col-span-1'>
|
||||||
<Stack>
|
<Stack>
|
||||||
<Text size='sm' weight='medium' theme='muted'>
|
<Text size='sm' weight='medium' theme='muted'>
|
||||||
|
|
|
@ -13,12 +13,12 @@ interface IDivider {
|
||||||
const Divider = ({ text, textSize = 'md' }: IDivider) => (
|
const Divider = ({ text, textSize = 'md' }: IDivider) => (
|
||||||
<div className='relative' data-testid='divider'>
|
<div className='relative' data-testid='divider'>
|
||||||
<div className='absolute inset-0 flex items-center' aria-hidden='true'>
|
<div className='absolute inset-0 flex items-center' aria-hidden='true'>
|
||||||
<div className='w-full border-t-2 border-gray-100 dark:border-gray-800 border-solid' />
|
<div className='w-full border-t-2 border-solid border-gray-100 dark:border-gray-800' />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{text && (
|
{text && (
|
||||||
<div className='relative flex justify-center'>
|
<div className='relative flex justify-center'>
|
||||||
<span className='px-2 bg-white dark:bg-gray-900 text-gray-700 dark:text-gray-600' data-testid='divider-text'>
|
<span className='bg-white px-2 text-gray-700 dark:bg-gray-900 dark:text-gray-600' data-testid='divider-text'>
|
||||||
<Text size={textSize} tag='span' theme='inherit'>{text}</Text>
|
<Text size={textSize} tag='span' theme='inherit'>{text}</Text>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -18,7 +18,7 @@ interface IEmojiButton {
|
||||||
const EmojiButton: React.FC<IEmojiButton> = ({ emoji, className, onClick, tabIndex }): JSX.Element => {
|
const EmojiButton: React.FC<IEmojiButton> = ({ emoji, className, onClick, tabIndex }): JSX.Element => {
|
||||||
return (
|
return (
|
||||||
<button className={classNames(className)} onClick={onClick} tabIndex={tabIndex}>
|
<button className={classNames(className)} onClick={onClick} tabIndex={tabIndex}>
|
||||||
<Emoji className='w-8 h-8 duration-100 hover:scale-125' emoji={emoji} />
|
<Emoji className='h-8 w-8 duration-100 hover:scale-125' emoji={emoji} />
|
||||||
</button>
|
</button>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -8,7 +8,7 @@ const FileInput = forwardRef<HTMLInputElement, IFileInput>((props, ref) => {
|
||||||
{...props}
|
{...props}
|
||||||
ref={ref}
|
ref={ref}
|
||||||
type='file'
|
type='file'
|
||||||
className='block w-full text-sm text-gray-800 dark:text-gray-200 file:cursor-pointer file:mr-2 file:py-1.5 file:px-3 file:rounded-full file:text-xs file:leading-4 file:font-medium file:border-gray-200 file:border file:border-solid file:bg-white file:text-gray-700 hover:file:bg-gray-100 dark:file:border-gray-800 dark:file:bg-gray-900 dark:file:hover:bg-gray-800 dark:file:text-gray-500'
|
className='block w-full text-sm text-gray-800 file:mr-2 file:cursor-pointer file:rounded-full file:border file:border-solid file:border-gray-200 file:bg-white file:py-1.5 file:px-3 file:text-xs file:font-medium file:leading-4 file:text-gray-700 hover:file:bg-gray-100 dark:text-gray-200 dark:file:border-gray-800 dark:file:bg-gray-900 dark:file:text-gray-500 dark:file:hover:bg-gray-800'
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
|
@ -55,7 +55,7 @@ const FormGroup: React.FC<IFormGroup> = (props) => {
|
||||||
<div>
|
<div>
|
||||||
<p
|
<p
|
||||||
data-testid='form-group-error'
|
data-testid='form-group-error'
|
||||||
className='mt-0.5 text-xs text-danger-900 bg-danger-200 rounded-md inline-block px-2 py-1 relative form-error'
|
className='form-error relative mt-0.5 inline-block rounded-md bg-danger-200 px-2 py-1 text-xs text-danger-900'
|
||||||
>
|
>
|
||||||
{errors.join(', ')}
|
{errors.join(', ')}
|
||||||
</p>
|
</p>
|
||||||
|
@ -92,7 +92,7 @@ const FormGroup: React.FC<IFormGroup> = (props) => {
|
||||||
{hasError && (
|
{hasError && (
|
||||||
<p
|
<p
|
||||||
data-testid='form-group-error'
|
data-testid='form-group-error'
|
||||||
className='mt-0.5 text-xs text-danger-900 bg-danger-200 rounded-md inline-block px-2 py-1 relative form-error'
|
className='form-error relative mt-0.5 inline-block rounded-md bg-danger-200 px-2 py-1 text-xs text-danger-900'
|
||||||
>
|
>
|
||||||
{errors.join(', ')}
|
{errors.join(', ')}
|
||||||
</p>
|
</p>
|
||||||
|
|
|
@ -21,9 +21,9 @@ interface IIcon extends Pick<React.SVGAttributes<SVGAElement>, 'strokeWidth'> {
|
||||||
|
|
||||||
/** Renders and SVG icon with optional counter. */
|
/** Renders and SVG icon with optional counter. */
|
||||||
const Icon: React.FC<IIcon> = ({ src, alt, count, size, countMax, ...filteredProps }): JSX.Element => (
|
const Icon: React.FC<IIcon> = ({ src, alt, count, size, countMax, ...filteredProps }): JSX.Element => (
|
||||||
<div className='flex flex-col flex-shrink-0 relative' data-testid='icon'>
|
<div className='relative flex shrink-0 flex-col' data-testid='icon'>
|
||||||
{count ? (
|
{count ? (
|
||||||
<span className='absolute -top-2 -right-3 min-w-[20px] h-5 flex-shrink-0 whitespace-nowrap flex items-center justify-center break-words'>
|
<span className='absolute -top-2 -right-3 flex h-5 min-w-[20px] shrink-0 items-center justify-center whitespace-nowrap break-words'>
|
||||||
<Counter count={count} countMax={countMax} />
|
<Counter count={count} countMax={countMax} />
|
||||||
</span>
|
</span>
|
||||||
) : null}
|
) : null}
|
||||||
|
|
|
@ -68,7 +68,7 @@ const Input = React.forwardRef<HTMLInputElement, IInput>(
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
{icon ? (
|
{icon ? (
|
||||||
<div className='absolute inset-y-0 left-0 pl-3 flex items-center pointer-events-none'>
|
<div className='pointer-events-none absolute inset-y-0 left-0 flex items-center pl-3'>
|
||||||
<Icon src={icon} className='h-4 w-4 text-gray-700 dark:text-gray-600' aria-hidden='true' />
|
<Icon src={icon} className='h-4 w-4 text-gray-700 dark:text-gray-600' aria-hidden='true' />
|
||||||
</div>
|
</div>
|
||||||
) : null}
|
) : null}
|
||||||
|
@ -95,7 +95,7 @@ const Input = React.forwardRef<HTMLInputElement, IInput>(
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{append ? (
|
{append ? (
|
||||||
<div className='absolute inset-y-0 right-0 rtl:left-0 rtl:right-auto flex items-center pr-3'>
|
<div className='absolute inset-y-0 right-0 flex items-center pr-3 rtl:left-0 rtl:right-auto'>
|
||||||
{append}
|
{append}
|
||||||
</div>
|
</div>
|
||||||
) : null}
|
) : null}
|
||||||
|
@ -108,12 +108,12 @@ const Input = React.forwardRef<HTMLInputElement, IInput>(
|
||||||
intl.formatMessage(messages.showPassword)
|
intl.formatMessage(messages.showPassword)
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<div className='absolute inset-y-0 right-0 rtl:left-0 rtl:right-auto flex items-center'>
|
<div className='absolute inset-y-0 right-0 flex items-center rtl:left-0 rtl:right-auto'>
|
||||||
<button
|
<button
|
||||||
type='button'
|
type='button'
|
||||||
onClick={togglePassword}
|
onClick={togglePassword}
|
||||||
tabIndex={-1}
|
tabIndex={-1}
|
||||||
className='text-gray-700 dark:text-gray-600 hover:text-gray-500 dark:hover:text-gray-400 h-full px-2 focus:ring-primary-500 focus:ring-2'
|
className='h-full px-2 text-gray-700 hover:text-gray-500 focus:ring-2 focus:ring-primary-500 dark:text-gray-600 dark:hover:text-gray-400'
|
||||||
>
|
>
|
||||||
<SvgIcon
|
<SvgIcon
|
||||||
src={revealed ? require('@tabler/icons/eye-off.svg') : require('@tabler/icons/eye.svg')}
|
src={revealed ? require('@tabler/icons/eye-off.svg') : require('@tabler/icons/eye.svg')}
|
||||||
|
|
|
@ -21,8 +21,8 @@ interface LayoutComponent extends React.FC<ILayout> {
|
||||||
|
|
||||||
/** Layout container, to hold Sidebar, Main, and Aside. */
|
/** Layout container, to hold Sidebar, Main, and Aside. */
|
||||||
const Layout: LayoutComponent = ({ children }) => (
|
const Layout: LayoutComponent = ({ children }) => (
|
||||||
<div className='sm:pt-4 relative'>
|
<div className='relative sm:pt-4'>
|
||||||
<div className='max-w-3xl mx-auto sm:px-6 md:max-w-7xl md:px-8 md:grid md:grid-cols-12 md:gap-8'>
|
<div className='mx-auto max-w-3xl sm:px-6 md:grid md:max-w-7xl md:grid-cols-12 md:gap-8 md:px-8'>
|
||||||
{children}
|
{children}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -30,7 +30,7 @@ const Layout: LayoutComponent = ({ children }) => (
|
||||||
|
|
||||||
/** Left sidebar container in the UI. */
|
/** Left sidebar container in the UI. */
|
||||||
const Sidebar: React.FC<ISidebar> = ({ children }) => (
|
const Sidebar: React.FC<ISidebar> = ({ children }) => (
|
||||||
<div className='hidden lg:block lg:col-span-3'>
|
<div className='hidden lg:col-span-3 lg:block'>
|
||||||
<StickyBox offsetTop={80} className='pb-4'>
|
<StickyBox offsetTop={80} className='pb-4'>
|
||||||
{children}
|
{children}
|
||||||
</StickyBox>
|
</StickyBox>
|
||||||
|
@ -50,7 +50,7 @@ const Main: React.FC<React.HTMLAttributes<HTMLDivElement>> = ({ children, classN
|
||||||
|
|
||||||
/** Right sidebar container in the UI. */
|
/** Right sidebar container in the UI. */
|
||||||
const Aside: React.FC<IAside> = ({ children }) => (
|
const Aside: React.FC<IAside> = ({ children }) => (
|
||||||
<aside className='hidden xl:block xl:col-span-3'>
|
<aside className='hidden xl:col-span-3 xl:block'>
|
||||||
<StickyBox offsetTop={80} className='space-y-6 pb-12'>
|
<StickyBox offsetTop={80} className='space-y-6 pb-12'>
|
||||||
{children}
|
{children}
|
||||||
</StickyBox>
|
</StickyBox>
|
||||||
|
|
|
@ -88,7 +88,7 @@ const Modal: React.FC<IModal> = ({
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div data-testid='modal' className={classNames('block w-full p-6 mx-auto text-start align-middle transition-all transform bg-white dark:bg-primary-900 text-gray-900 dark:text-gray-100 shadow-xl rounded-2xl pointer-events-auto', widths[width])}>
|
<div data-testid='modal' className={classNames('block w-full p-6 mx-auto text-start align-middle transition-all transform bg-white dark:bg-primary-900 text-gray-900 dark:text-gray-100 shadow-xl rounded-2xl pointer-events-auto', widths[width])}>
|
||||||
<div className='sm:flex sm:items-start w-full justify-between'>
|
<div className='w-full justify-between sm:flex sm:items-start'>
|
||||||
<div className='w-full'>
|
<div className='w-full'>
|
||||||
{title && (
|
{title && (
|
||||||
<div
|
<div
|
||||||
|
@ -96,7 +96,7 @@ const Modal: React.FC<IModal> = ({
|
||||||
'flex-row-reverse': closePosition === 'left',
|
'flex-row-reverse': closePosition === 'left',
|
||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
<h3 className='flex-grow text-lg leading-6 font-bold text-gray-900 dark:text-white'>
|
<h3 className='grow text-lg font-bold leading-6 text-gray-900 dark:text-white'>
|
||||||
{title}
|
{title}
|
||||||
</h3>
|
</h3>
|
||||||
|
|
||||||
|
@ -105,14 +105,14 @@ const Modal: React.FC<IModal> = ({
|
||||||
src={closeIcon}
|
src={closeIcon}
|
||||||
title={intl.formatMessage(messages.close)}
|
title={intl.formatMessage(messages.close)}
|
||||||
onClick={onClose}
|
onClick={onClose}
|
||||||
className='text-gray-500 hover:text-gray-700 dark:text-gray-300 dark:hover:text-gray-200 rtl:rotate-180'
|
className='text-gray-500 hover:text-gray-700 rtl:rotate-180 dark:text-gray-300 dark:hover:text-gray-200'
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{title ? (
|
{title ? (
|
||||||
<div className='w-full mt-2'>
|
<div className='mt-2 w-full'>
|
||||||
{children}
|
{children}
|
||||||
</div>
|
</div>
|
||||||
) : children}
|
) : children}
|
||||||
|
|
|
@ -12,7 +12,7 @@ const CountryCodeDropdown: React.FC<ICountryCodeDropdown> = ({ countryCode, onCh
|
||||||
return (
|
return (
|
||||||
<select
|
<select
|
||||||
value={countryCode}
|
value={countryCode}
|
||||||
className='h-full py-0 pl-3 pr-7 text-base bg-transparent border-transparent focus:outline-none focus:ring-primary-500 dark:text-white sm:text-sm rounded-md'
|
className='h-full rounded-md border-transparent bg-transparent py-0 pl-3 pr-7 text-base focus:outline-none focus:ring-primary-500 dark:text-white sm:text-sm'
|
||||||
onChange={(event) => onChange(event.target.value as any)}
|
onChange={(event) => onChange(event.target.value as any)}
|
||||||
>
|
>
|
||||||
{COUNTRY_CODES.map((code) => (
|
{COUNTRY_CODES.map((code) => (
|
||||||
|
|
|
@ -6,7 +6,7 @@ interface IProgressBar {
|
||||||
|
|
||||||
/** A horizontal meter filled to the given percentage. */
|
/** A horizontal meter filled to the given percentage. */
|
||||||
const ProgressBar: React.FC<IProgressBar> = ({ progress }) => (
|
const ProgressBar: React.FC<IProgressBar> = ({ progress }) => (
|
||||||
<div className='h-2.5 w-full rounded-full bg-gray-300 dark:bg-primary-800 overflow-hidden'>
|
<div className='h-2.5 w-full overflow-hidden rounded-full bg-gray-300 dark:bg-primary-800'>
|
||||||
<div className='h-full bg-secondary-500' style={{ width: `${Math.floor(progress * 100)}%` }} />
|
<div className='h-full bg-secondary-500' style={{ width: `${Math.floor(progress * 100)}%` }} />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
|
@ -11,7 +11,7 @@ const Select = React.forwardRef<HTMLSelectElement, ISelect>((props, ref) => {
|
||||||
return (
|
return (
|
||||||
<select
|
<select
|
||||||
ref={ref}
|
ref={ref}
|
||||||
className={`w-full pl-3 pr-10 py-2 text-base truncate border-gray-300 dark:border-gray-800 focus:outline-none focus:ring-primary-500 focus:border-primary-500 dark:bg-gray-900 dark:text-gray-100 dark:ring-1 dark:ring-gray-800 dark:focus:ring-primary-500 dark:focus:border-primary-500 sm:text-sm rounded-md disabled:opacity-50 ${className}`}
|
className={`w-full truncate rounded-md border-gray-300 py-2 pl-3 pr-10 text-base focus:border-primary-500 focus:outline-none focus:ring-primary-500 disabled:opacity-50 dark:border-gray-800 dark:bg-gray-900 dark:text-gray-100 dark:ring-1 dark:ring-gray-800 dark:focus:border-primary-500 dark:focus:ring-primary-500 sm:text-sm ${className}`}
|
||||||
{...filteredProps}
|
{...filteredProps}
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
|
|
|
@ -53,14 +53,14 @@ const Slider: React.FC<ISlider> = ({ value, onChange }) => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className='inline-flex cursor-pointer h-6 relative transition'
|
className='relative inline-flex h-6 cursor-pointer transition'
|
||||||
onMouseDown={handleMouseDown}
|
onMouseDown={handleMouseDown}
|
||||||
ref={node}
|
ref={node}
|
||||||
>
|
>
|
||||||
<div className='w-full h-1 bg-primary-200 dark:bg-primary-700 absolute top-1/2 -translate-y-1/2 rounded-full' />
|
<div className='absolute top-1/2 h-1 w-full -translate-y-1/2 rounded-full bg-primary-200 dark:bg-primary-700' />
|
||||||
<div className='h-1 bg-accent-500 absolute top-1/2 -translate-y-1/2 rounded-full' style={{ width: `${value * 100}%` }} />
|
<div className='absolute top-1/2 h-1 -translate-y-1/2 rounded-full bg-accent-500' style={{ width: `${value * 100}%` }} />
|
||||||
<span
|
<span
|
||||||
className='bg-accent-500 absolute rounded-full w-3 h-3 -ml-1.5 top-1/2 -translate-y-1/2 z-10 shadow'
|
className='absolute top-1/2 z-10 -ml-1.5 h-3 w-3 -translate-y-1/2 rounded-full bg-accent-500 shadow'
|
||||||
tabIndex={0}
|
tabIndex={0}
|
||||||
style={{ left: `${value * 100}%` }}
|
style={{ left: `${value * 100}%` }}
|
||||||
/>
|
/>
|
||||||
|
|
|
@ -46,7 +46,7 @@ const AnimatedTabs: React.FC<IAnimatedInterface> = ({ children, ...rest }) => {
|
||||||
ref={ref}
|
ref={ref}
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className='w-full h-[3px] bg-primary-200 dark:bg-primary-700 absolute'
|
className='absolute h-[3px] w-full bg-primary-200 dark:bg-primary-700'
|
||||||
style={{ top }}
|
style={{ top }}
|
||||||
/>
|
/>
|
||||||
<div
|
<div
|
||||||
|
|
|
@ -43,9 +43,9 @@ const TagInput: React.FC<ITagInput> = ({ tags, onChange, placeholder }) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='mt-1 relative shadow-sm flex-grow'>
|
<div className='relative mt-1 grow shadow-sm'>
|
||||||
<HStack
|
<HStack
|
||||||
className='p-2 pb-0 text-gray-900 dark:text-gray-100 placeholder:text-gray-600 dark:placeholder:text-gray-600 block w-full sm:text-sm dark:ring-1 dark:ring-gray-800 focus:ring-primary-500 focus:border-primary-500 dark:focus:ring-primary-500 dark:focus:border-primary-500 rounded-md bg-white dark:bg-gray-900 border-gray-400 dark:border-gray-800'
|
className='block w-full rounded-md border-gray-400 bg-white p-2 pb-0 text-gray-900 placeholder:text-gray-600 focus:border-primary-500 focus:ring-primary-500 dark:border-gray-800 dark:bg-gray-900 dark:text-gray-100 dark:ring-1 dark:ring-gray-800 dark:placeholder:text-gray-600 dark:focus:border-primary-500 dark:focus:ring-primary-500 sm:text-sm'
|
||||||
space={2}
|
space={2}
|
||||||
wrap
|
wrap
|
||||||
>
|
>
|
||||||
|
@ -56,7 +56,7 @@ const TagInput: React.FC<ITagInput> = ({ tags, onChange, placeholder }) => {
|
||||||
))}
|
))}
|
||||||
|
|
||||||
<input
|
<input
|
||||||
className='p-1 mb-2 w-32 h-8 flex-grow bg-transparent outline-none'
|
className='mb-2 h-8 w-32 grow bg-transparent p-1 outline-none'
|
||||||
value={input}
|
value={input}
|
||||||
placeholder={placeholder}
|
placeholder={placeholder}
|
||||||
onChange={e => setInput(e.target.value)}
|
onChange={e => setInput(e.target.value)}
|
||||||
|
|
|
@ -13,7 +13,7 @@ interface ITag {
|
||||||
/** A single editable Tag (used by TagInput). */
|
/** A single editable Tag (used by TagInput). */
|
||||||
const Tag: React.FC<ITag> = ({ tag, onDelete }) => {
|
const Tag: React.FC<ITag> = ({ tag, onDelete }) => {
|
||||||
return (
|
return (
|
||||||
<div className='inline-flex p-1 rounded bg-primary-500 items-center whitespace-nowrap'>
|
<div className='inline-flex items-center whitespace-nowrap rounded bg-primary-500 p-1'>
|
||||||
<Text theme='white'>{tag}</Text>
|
<Text theme='white'>{tag}</Text>
|
||||||
|
|
||||||
<IconButton
|
<IconButton
|
||||||
|
|
|
@ -40,7 +40,7 @@ const Toast = (props: IToast) => {
|
||||||
return (
|
return (
|
||||||
<Icon
|
<Icon
|
||||||
src={require('@tabler/icons/circle-check.svg')}
|
src={require('@tabler/icons/circle-check.svg')}
|
||||||
className='w-6 h-6 text-success-500 dark:text-success-400'
|
className='h-6 w-6 text-success-500 dark:text-success-400'
|
||||||
aria-hidden
|
aria-hidden
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
@ -48,7 +48,7 @@ const Toast = (props: IToast) => {
|
||||||
return (
|
return (
|
||||||
<Icon
|
<Icon
|
||||||
src={require('@tabler/icons/info-circle.svg')}
|
src={require('@tabler/icons/info-circle.svg')}
|
||||||
className='w-6 h-6 text-primary-600 dark:text-accent-blue'
|
className='h-6 w-6 text-primary-600 dark:text-accent-blue'
|
||||||
aria-hidden
|
aria-hidden
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
@ -56,7 +56,7 @@ const Toast = (props: IToast) => {
|
||||||
return (
|
return (
|
||||||
<Icon
|
<Icon
|
||||||
src={require('@tabler/icons/alert-circle.svg')}
|
src={require('@tabler/icons/alert-circle.svg')}
|
||||||
className='w-6 h-6 text-danger-600'
|
className='h-6 w-6 text-danger-600'
|
||||||
aria-hidden
|
aria-hidden
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
@ -112,7 +112,7 @@ const Toast = (props: IToast) => {
|
||||||
<HStack space={4} alignItems='start'>
|
<HStack space={4} alignItems='start'>
|
||||||
<HStack space={3} justifyContent='between' alignItems='start' className='w-0 flex-1'>
|
<HStack space={3} justifyContent='between' alignItems='start' className='w-0 flex-1'>
|
||||||
<HStack space={3} alignItems='start' className='w-0 flex-1'>
|
<HStack space={3} alignItems='start' className='w-0 flex-1'>
|
||||||
<div className='flex-shrink-0'>
|
<div className='shrink-0'>
|
||||||
{renderIcon()}
|
{renderIcon()}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -126,15 +126,15 @@ const Toast = (props: IToast) => {
|
||||||
</HStack>
|
</HStack>
|
||||||
|
|
||||||
{/* Dismiss Button */}
|
{/* Dismiss Button */}
|
||||||
<div className='flex flex-shrink-0 pt-0.5'>
|
<div className='flex shrink-0 pt-0.5'>
|
||||||
<button
|
<button
|
||||||
type='button'
|
type='button'
|
||||||
className='inline-flex rounded-md text-gray-600 dark:text-gray-600 hover:text-gray-700 dark:hover:text-gray-500 focus:outline-none focus:ring-2 focus:ring-primary-500 focus:ring-offset-2'
|
className='inline-flex rounded-md text-gray-600 hover:text-gray-700 focus:outline-none focus:ring-2 focus:ring-primary-500 focus:ring-offset-2 dark:text-gray-600 dark:hover:text-gray-500'
|
||||||
onClick={dismissToast}
|
onClick={dismissToast}
|
||||||
data-testid='toast-dismiss'
|
data-testid='toast-dismiss'
|
||||||
>
|
>
|
||||||
<span className='sr-only'>Close</span>
|
<span className='sr-only'>Close</span>
|
||||||
<Icon src={require('@tabler/icons/x.svg')} className='w-5 h-5' />
|
<Icon src={require('@tabler/icons/x.svg')} className='h-5 w-5' />
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</HStack>
|
</HStack>
|
||||||
|
|
|
@ -49,7 +49,7 @@ const Widget: React.FC<IWidget> = ({
|
||||||
<WidgetTitle title={title} />
|
<WidgetTitle title={title} />
|
||||||
{action || (onActionClick && (
|
{action || (onActionClick && (
|
||||||
<IconButton
|
<IconButton
|
||||||
className='w-6 h-6 ml-2 text-black dark:text-white rtl:rotate-180'
|
className='ml-2 h-6 w-6 text-black rtl:rotate-180 dark:text-white'
|
||||||
src={actionIcon}
|
src={actionIcon}
|
||||||
onClick={onActionClick}
|
onClick={onActionClick}
|
||||||
title={actionTitle}
|
title={actionTitle}
|
||||||
|
|
|
@ -16,7 +16,7 @@ const UploadProgress: React.FC<IUploadProgress> = ({ progress }) => {
|
||||||
<HStack alignItems='center' space={2}>
|
<HStack alignItems='center' space={2}>
|
||||||
<Icon
|
<Icon
|
||||||
src={require('@tabler/icons/cloud-upload.svg')}
|
src={require('@tabler/icons/cloud-upload.svg')}
|
||||||
className='w-7 h-7 text-gray-500'
|
className='h-7 w-7 text-gray-500'
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Stack space={1}>
|
<Stack space={1}>
|
||||||
|
@ -24,11 +24,11 @@ const UploadProgress: React.FC<IUploadProgress> = ({ progress }) => {
|
||||||
<FormattedMessage id='upload_progress.label' defaultMessage='Uploading…' />
|
<FormattedMessage id='upload_progress.label' defaultMessage='Uploading…' />
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
<div className='w-full h-1.5 rounded-lg bg-gray-200 relative'>
|
<div className='relative h-1.5 w-full rounded-lg bg-gray-200'>
|
||||||
<Motion defaultStyle={{ width: 0 }} style={{ width: spring(progress) }}>
|
<Motion defaultStyle={{ width: 0 }} style={{ width: spring(progress) }}>
|
||||||
{({ width }) =>
|
{({ width }) =>
|
||||||
(<div
|
(<div
|
||||||
className='absolute left-0 top-0 h-1.5 bg-primary-600 rounded-lg'
|
className='absolute left-0 top-0 h-1.5 rounded-lg bg-primary-600'
|
||||||
style={{ width: `${width}%` }}
|
style={{ width: `${width}%` }}
|
||||||
/>)
|
/>)
|
||||||
}
|
}
|
||||||
|
|
|
@ -141,7 +141,7 @@ const Upload: React.FC<IUpload> = ({
|
||||||
|
|
||||||
const uploadIcon = mediaType === 'unknown' && (
|
const uploadIcon = mediaType === 'unknown' && (
|
||||||
<Icon
|
<Icon
|
||||||
className='h-16 w-16 mx-auto my-12 text-gray-800 dark:text-gray-200'
|
className='mx-auto my-12 h-16 w-16 text-gray-800 dark:text-gray-200'
|
||||||
src={MIMETYPE_ICONS[mimeType || ''] || defaultIcon}
|
src={MIMETYPE_ICONS[mimeType || ''] || defaultIcon}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
|
|
@ -38,7 +38,7 @@ const AboutPage: React.FC = () => {
|
||||||
<div>
|
<div>
|
||||||
<FormattedMessage id='about.also_available' defaultMessage='Available in:' />
|
<FormattedMessage id='about.also_available' defaultMessage='Available in:' />
|
||||||
{' '}
|
{' '}
|
||||||
<ul className='p-0 inline list-none'>
|
<ul className='inline list-none p-0'>
|
||||||
<li className="inline after:content-['_·_']">
|
<li className="inline after:content-['_·_']">
|
||||||
<a href='#' onClick={() => setLocale(defaultLocale)}>
|
<a href='#' onClick={() => setLocale(defaultLocale)}>
|
||||||
{/* @ts-ignore */}
|
{/* @ts-ignore */}
|
||||||
|
@ -60,7 +60,7 @@ const AboutPage: React.FC = () => {
|
||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='prose dark:prose-invert mx-auto py-20'>
|
<div className='prose mx-auto py-20 dark:prose-invert'>
|
||||||
<div dangerouslySetInnerHTML={{ __html: pageHtml }} />
|
<div dangerouslySetInnerHTML={{ __html: pageHtml }} />
|
||||||
|
|
||||||
{alsoAvailable}
|
{alsoAvailable}
|
||||||
|
|
|
@ -71,7 +71,7 @@ const MediaItem: React.FC<IMediaItem> = ({ attachment, onOpenMedia }) => {
|
||||||
src={attachment.preview_url}
|
src={attachment.preview_url}
|
||||||
alt={attachment.description}
|
alt={attachment.description}
|
||||||
style={{ objectPosition: `${x}% ${y}%` }}
|
style={{ objectPosition: `${x}% ${y}%` }}
|
||||||
className='w-full h-full rounded-lg overflow-hidden'
|
className='h-full w-full overflow-hidden rounded-lg'
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
} else if (['gifv', 'video'].indexOf(attachment.type) !== -1) {
|
} else if (['gifv', 'video'].indexOf(attachment.type) !== -1) {
|
||||||
|
|
|
@ -17,7 +17,7 @@ const MovedNote: React.FC<IMovedNote> = ({ from, to }) => (
|
||||||
<HStack className='mb-2' alignItems='center' space={1.5}>
|
<HStack className='mb-2' alignItems='center' space={1.5}>
|
||||||
<Icon
|
<Icon
|
||||||
src={require('@tabler/icons/briefcase.svg')}
|
src={require('@tabler/icons/briefcase.svg')}
|
||||||
className='text-primary-600 dark:text-primary-400 flex-none'
|
className='flex-none text-primary-600 dark:text-primary-400'
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div className='truncate'>
|
<div className='truncate'>
|
||||||
|
|
|
@ -102,16 +102,16 @@ const Header: React.FC<IHeader> = ({ account }) => {
|
||||||
|
|
||||||
if (!account) {
|
if (!account) {
|
||||||
return (
|
return (
|
||||||
<div className='-mt-4 -mx-4'>
|
<div className='-mx-4 -mt-4'>
|
||||||
<div>
|
<div>
|
||||||
<div className='relative h-32 w-full lg:h-48 md:rounded-t-xl bg-gray-200 dark:bg-gray-900/50' />
|
<div className='relative h-32 w-full bg-gray-200 dark:bg-gray-900/50 md:rounded-t-xl lg:h-48' />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className='px-4 sm:px-6'>
|
<div className='px-4 sm:px-6'>
|
||||||
<HStack alignItems='bottom' space={5} className='-mt-12'>
|
<HStack alignItems='bottom' space={5} className='-mt-12'>
|
||||||
<div className='flex relative'>
|
<div className='relative flex'>
|
||||||
<div
|
<div
|
||||||
className='h-24 w-24 bg-gray-400 rounded-full ring-4 ring-white dark:ring-gray-800'
|
className='h-24 w-24 rounded-full bg-gray-400 ring-4 ring-white dark:ring-gray-800'
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</HStack>
|
</HStack>
|
||||||
|
@ -586,13 +586,13 @@ const Header: React.FC<IHeader> = ({ account }) => {
|
||||||
const menu = makeMenu();
|
const menu = makeMenu();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='-mt-4 -mx-4'>
|
<div className='-mx-4 -mt-4'>
|
||||||
{(account.moved && typeof account.moved === 'object') && (
|
{(account.moved && typeof account.moved === 'object') && (
|
||||||
<MovedNote from={account} to={account.moved} />
|
<MovedNote from={account} to={account.moved} />
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<div className='relative flex flex-col justify-center h-32 w-full lg:h-48 md:rounded-t-xl bg-gray-200 dark:bg-gray-900/50 overflow-hidden isolate'>
|
<div className='relative isolate flex h-32 w-full flex-col justify-center overflow-hidden bg-gray-200 dark:bg-gray-900/50 md:rounded-t-xl lg:h-48'>
|
||||||
{renderHeader()}
|
{renderHeader()}
|
||||||
|
|
||||||
<div className='absolute top-2 left-2'>
|
<div className='absolute top-2 left-2'>
|
||||||
|
@ -610,12 +610,12 @@ const Header: React.FC<IHeader> = ({ account }) => {
|
||||||
<Avatar
|
<Avatar
|
||||||
src={account.avatar}
|
src={account.avatar}
|
||||||
size={96}
|
size={96}
|
||||||
className='relative h-24 w-24 rounded-full ring-4 ring-white dark:ring-primary-900 bg-white dark:bg-primary-900'
|
className='relative h-24 w-24 rounded-full bg-white ring-4 ring-white dark:bg-primary-900 dark:ring-primary-900'
|
||||||
/>
|
/>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className='mt-6 flex justify-end w-full sm:pb-1'>
|
<div className='mt-6 flex w-full justify-end sm:pb-1'>
|
||||||
<HStack space={2} className='mt-10'>
|
<HStack space={2} className='mt-10'>
|
||||||
<SubscriptionButton account={account} />
|
<SubscriptionButton account={account} />
|
||||||
{renderMessageButton()}
|
{renderMessageButton()}
|
||||||
|
@ -644,7 +644,7 @@ const Header: React.FC<IHeader> = ({ account }) => {
|
||||||
<Comp key={idx} {...itemProps} className='group'>
|
<Comp key={idx} {...itemProps} className='group'>
|
||||||
<HStack space={3} alignItems='center'>
|
<HStack space={3} alignItems='center'>
|
||||||
{menuItem.icon && (
|
{menuItem.icon && (
|
||||||
<SvgIcon src={menuItem.icon} className='h-5 w-5 text-gray-400 flex-none group-hover:text-gray-500' />
|
<SvgIcon src={menuItem.icon} className='h-5 w-5 flex-none text-gray-400 group-hover:text-gray-500' />
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<div className='truncate'>{menuItem.text}</div>
|
<div className='truncate'>{menuItem.text}</div>
|
||||||
|
|
|
@ -21,7 +21,7 @@ const DashCounter: React.FC<IDashCounter> = ({ count, label, to = '#', percent =
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Link
|
<Link
|
||||||
className='bg-gray-200 dark:bg-gray-800 p-4 rounded flex flex-col items-center space-y-2 hover:-translate-y-1 transition-transform cursor-pointer'
|
className='flex cursor-pointer flex-col items-center space-y-2 rounded bg-gray-200 p-4 transition-transform hover:-translate-y-1 dark:bg-gray-800'
|
||||||
to={to}
|
to={to}
|
||||||
>
|
>
|
||||||
<Text align='center' size='2xl' weight='medium'>
|
<Text align='center' size='2xl' weight='medium'>
|
||||||
|
@ -45,7 +45,7 @@ interface IDashCounters {
|
||||||
/** Wrapper container for dash counters. */
|
/** Wrapper container for dash counters. */
|
||||||
const DashCounters: React.FC<IDashCounters> = ({ children }) => {
|
const DashCounters: React.FC<IDashCounters> = ({ children }) => {
|
||||||
return (
|
return (
|
||||||
<div className='grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 gap-2'>
|
<div className='grid grid-cols-1 gap-2 sm:grid-cols-2 lg:grid-cols-3'>
|
||||||
{children}
|
{children}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
|
@ -135,7 +135,7 @@ const Report: React.FC<IReport> = ({ id }) => {
|
||||||
<Link
|
<Link
|
||||||
to={`/@${reporterAcct}`}
|
to={`/@${reporterAcct}`}
|
||||||
title={reporterAcct}
|
title={reporterAcct}
|
||||||
className='text-primary-600 dark:text-accent-blue hover:underline'
|
className='text-primary-600 hover:underline dark:text-accent-blue'
|
||||||
>
|
>
|
||||||
@{reporterAcct}
|
@{reporterAcct}
|
||||||
</Link>
|
</Link>
|
||||||
|
|
|
@ -115,13 +115,13 @@ const Dashboard: React.FC = () => {
|
||||||
<ListItem label={<FormattedMessage id='admin.software.frontend' defaultMessage='Frontend' />}>
|
<ListItem label={<FormattedMessage id='admin.software.frontend' defaultMessage='Frontend' />}>
|
||||||
<a
|
<a
|
||||||
href={sourceCode.ref ? `${sourceCode.url}/tree/${sourceCode.ref}` : sourceCode.url}
|
href={sourceCode.ref ? `${sourceCode.url}/tree/${sourceCode.ref}` : sourceCode.url}
|
||||||
className='flex space-x-1 items-center truncate'
|
className='flex items-center space-x-1 truncate'
|
||||||
target='_blank'
|
target='_blank'
|
||||||
>
|
>
|
||||||
<span>{sourceCode.displayName} {sourceCode.version}</span>
|
<span>{sourceCode.displayName} {sourceCode.version}</span>
|
||||||
|
|
||||||
<Icon
|
<Icon
|
||||||
className='w-4 h-4'
|
className='h-4 w-4'
|
||||||
src={require('@tabler/icons/external-link.svg')}
|
src={require('@tabler/icons/external-link.svg')}
|
||||||
/>
|
/>
|
||||||
</a>
|
</a>
|
||||||
|
|
|
@ -85,7 +85,7 @@ const Ad: React.FC<IAd> = ({ ad }) => {
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
<Icon
|
<Icon
|
||||||
className='w-5 h-5 stroke-accent-500'
|
className='h-5 w-5 stroke-accent-500'
|
||||||
src={require('@tabler/icons/timeline.svg')}
|
src={require('@tabler/icons/timeline.svg')}
|
||||||
/>
|
/>
|
||||||
</HStack>
|
</HStack>
|
||||||
|
|
|
@ -40,11 +40,11 @@ const Search: React.FC = () => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='flex items-center gap-1'>
|
<div className='flex items-center gap-1'>
|
||||||
<label className='flex-grow relative'>
|
<label className='relative grow'>
|
||||||
<span style={{ display: 'none' }}>{intl.formatMessage(messages.search)}</span>
|
<span style={{ display: 'none' }}>{intl.formatMessage(messages.search)}</span>
|
||||||
|
|
||||||
<input
|
<input
|
||||||
className='block w-full sm:text-sm dark:bg-gray-800 dark:text-white dark:placeholder:text-gray-500 focus:ring-primary-500 focus:border-primary-500 rounded-full'
|
className='block w-full rounded-full focus:border-primary-500 focus:ring-primary-500 dark:bg-gray-800 dark:text-white dark:placeholder:text-gray-500 sm:text-sm'
|
||||||
type='text'
|
type='text'
|
||||||
value={value}
|
value={value}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
|
|
|
@ -36,9 +36,9 @@ const AuthLayout = () => {
|
||||||
<LandingGradient />
|
<LandingGradient />
|
||||||
|
|
||||||
<main className='relative h-full sm:flex sm:justify-center'>
|
<main className='relative h-full sm:flex sm:justify-center'>
|
||||||
<div className='w-full h-full flex flex-col sm:max-w-lg md:max-w-2xl lg:max-w-6xl'>
|
<div className='flex h-full w-full flex-col sm:max-w-lg md:max-w-2xl lg:max-w-6xl'>
|
||||||
<header className='flex justify-between relative py-12 px-2 mb-auto'>
|
<header className='relative mb-auto flex justify-between py-12 px-2'>
|
||||||
<div className='relative z-0 flex-1 px-2 lg:flex lg:items-center lg:justify-center lg:absolute lg:inset-0'>
|
<div className='relative z-0 flex-1 px-2 lg:absolute lg:inset-0 lg:flex lg:items-center lg:justify-center'>
|
||||||
<Link to='/' className='cursor-pointer'>
|
<Link to='/' className='cursor-pointer'>
|
||||||
<SiteLogo alt={instance.title} className='h-7' />
|
<SiteLogo alt={instance.title} className='h-7' />
|
||||||
</Link>
|
</Link>
|
||||||
|
@ -57,8 +57,8 @@ const AuthLayout = () => {
|
||||||
)}
|
)}
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<div className='flex flex-col justify-center items-center'>
|
<div className='flex flex-col items-center justify-center'>
|
||||||
<div className='pb-10 sm:mx-auto w-full sm:max-w-lg md:max-w-2xl'>
|
<div className='w-full pb-10 sm:mx-auto sm:max-w-lg md:max-w-2xl'>
|
||||||
<Card variant='rounded' size='xl'>
|
<Card variant='rounded' size='xl'>
|
||||||
<CardBody>
|
<CardBody>
|
||||||
<Switch>
|
<Switch>
|
||||||
|
|
|
@ -109,7 +109,7 @@ const NativeCaptchaField: React.FC<INativeCaptchaField> = ({ captcha, onChange,
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack space={2}>
|
<Stack space={2}>
|
||||||
<div className='flex items-center justify-center bg-white w-full border border-solid border-gray-300 dark:border-gray-600 rounded-md'>
|
<div className='flex w-full items-center justify-center rounded-md border border-solid border-gray-300 bg-white dark:border-gray-600'>
|
||||||
<img alt='captcha' src={captcha.get('url')} onClick={onClick} />
|
<img alt='captcha' src={captcha.get('url')} onClick={onClick} />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
@ -17,7 +17,7 @@ const ConsumersList: React.FC<IConsumersList> = () => {
|
||||||
|
|
||||||
if (providers.size > 0) {
|
if (providers.size > 0) {
|
||||||
return (
|
return (
|
||||||
<Card className='p-4 sm:rounded-xl bg-gray-50 dark:bg-primary-800'>
|
<Card className='bg-gray-50 p-4 dark:bg-primary-800 sm:rounded-xl'>
|
||||||
<Text size='xs' theme='muted'>
|
<Text size='xs' theme='muted'>
|
||||||
<FormattedMessage id='oauth_consumers.title' defaultMessage='Other ways to sign in' />
|
<FormattedMessage id='oauth_consumers.title' defaultMessage='Other ways to sign in' />
|
||||||
</Text>
|
</Text>
|
||||||
|
|
|
@ -27,11 +27,11 @@ const LoginForm: React.FC<ILoginForm> = ({ isLoading, handleSubmit }) => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<div className='pb-4 sm:pb-10 mb-4 border-b border-gray-200 dark:border-gray-800 border-solid -mx-4 sm:-mx-10'>
|
<div className='-mx-4 mb-4 border-b border-solid border-gray-200 pb-4 dark:border-gray-800 sm:-mx-10 sm:pb-10'>
|
||||||
<h1 className='text-center font-bold text-2xl'><FormattedMessage id='login_form.header' defaultMessage='Sign In' /></h1>
|
<h1 className='text-center text-2xl font-bold'><FormattedMessage id='login_form.header' defaultMessage='Sign In' /></h1>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Stack className='sm:pt-10 sm:w-2/3 md:w-1/2 mx-auto' space={5}>
|
<Stack className='mx-auto sm:w-2/3 sm:pt-10 md:w-1/2' space={5}>
|
||||||
<Form onSubmit={handleSubmit}>
|
<Form onSubmit={handleSubmit}>
|
||||||
<FormGroup labelText={intl.formatMessage(messages.username)}>
|
<FormGroup labelText={intl.formatMessage(messages.username)}>
|
||||||
<Input
|
<Input
|
||||||
|
|
|
@ -48,13 +48,13 @@ const OtpAuthForm: React.FC<IOtpAuthForm> = ({ mfa_token }) => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<div className='pb-4 sm:pb-10 mb-4 border-b border-gray-200 dark:border-gray-600 border-solid -mx-4 sm:-mx-10'>
|
<div className='-mx-4 mb-4 border-b border-solid border-gray-200 pb-4 dark:border-gray-600 sm:-mx-10 sm:pb-10'>
|
||||||
<h1 className='text-center font-bold text-2xl'>
|
<h1 className='text-center text-2xl font-bold'>
|
||||||
<FormattedMessage id='login.otp_log_in' defaultMessage='OTP Login' />
|
<FormattedMessage id='login.otp_log_in' defaultMessage='OTP Login' />
|
||||||
</h1>
|
</h1>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className='sm:pt-10 sm:w-2/3 md:w-1/2 mx-auto'>
|
<div className='mx-auto sm:w-2/3 sm:pt-10 md:w-1/2'>
|
||||||
<Form onSubmit={handleSubmit}>
|
<Form onSubmit={handleSubmit}>
|
||||||
<FormGroup
|
<FormGroup
|
||||||
labelText={intl.formatMessage(messages.otpCodeLabel)}
|
labelText={intl.formatMessage(messages.otpCodeLabel)}
|
||||||
|
|
|
@ -59,13 +59,13 @@ const PasswordResetConfirm = () => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<div className='pb-4 sm:pb-10 mb-4 border-b border-gray-200 dark:border-gray-600 border-solid -mx-4 sm:-mx-10'>
|
<div className='-mx-4 mb-4 border-b border-solid border-gray-200 pb-4 dark:border-gray-600 sm:-mx-10 sm:pb-10'>
|
||||||
<h1 className='text-center font-bold text-2xl'>
|
<h1 className='text-center text-2xl font-bold'>
|
||||||
<FormattedMessage id='reset_password.header' defaultMessage='Set New Password' />
|
<FormattedMessage id='reset_password.header' defaultMessage='Set New Password' />
|
||||||
</h1>
|
</h1>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className='sm:pt-10 sm:w-2/3 md:w-1/2 mx-auto'>
|
<div className='mx-auto sm:w-2/3 sm:pt-10 md:w-1/2'>
|
||||||
<Form onSubmit={handleSubmit}>
|
<Form onSubmit={handleSubmit}>
|
||||||
<FormGroup labelText={<FormattedMessage id='reset_password.password.label' defaultMessage='Password' />} errors={renderErrors()}>
|
<FormGroup labelText={<FormattedMessage id='reset_password.password.label' defaultMessage='Password' />} errors={renderErrors()}>
|
||||||
<Input
|
<Input
|
||||||
|
|
|
@ -35,13 +35,13 @@ const PasswordReset = () => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<div className='pb-4 sm:pb-10 mb-4 border-b border-gray-200 dark:border-gray-600 border-solid -mx-4 sm:-mx-10'>
|
<div className='-mx-4 mb-4 border-b border-solid border-gray-200 pb-4 dark:border-gray-600 sm:-mx-10 sm:pb-10'>
|
||||||
<h1 className='text-center font-bold text-2xl'>
|
<h1 className='text-center text-2xl font-bold'>
|
||||||
<FormattedMessage id='password_reset.header' defaultMessage='Reset Password' />
|
<FormattedMessage id='password_reset.header' defaultMessage='Reset Password' />
|
||||||
</h1>
|
</h1>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className='sm:pt-10 sm:w-2/3 md:w-1/2 mx-auto'>
|
<div className='mx-auto sm:w-2/3 sm:pt-10 md:w-1/2'>
|
||||||
<Form onSubmit={handleSubmit}>
|
<Form onSubmit={handleSubmit}>
|
||||||
<FormGroup labelText={intl.formatMessage(messages.nicknameOrEmail)}>
|
<FormGroup labelText={intl.formatMessage(messages.nicknameOrEmail)}>
|
||||||
<Input
|
<Input
|
||||||
|
|
|
@ -41,7 +41,7 @@ const AuthToken: React.FC<IAuthToken> = ({ token, isCurrent }) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='p-4 rounded-lg bg-gray-100 dark:bg-primary-800'>
|
<div className='rounded-lg bg-gray-100 p-4 dark:bg-primary-800'>
|
||||||
<Stack space={2}>
|
<Stack space={2}>
|
||||||
<Stack>
|
<Stack>
|
||||||
<Text size='md' weight='medium'>{token.app_name}</Text>
|
<Text size='md' weight='medium'>{token.app_name}</Text>
|
||||||
|
@ -85,7 +85,7 @@ const AuthTokenList: React.FC = () => {
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const body = tokens ? (
|
const body = tokens ? (
|
||||||
<div className='grid grid-cols-1 sm:grid-cols-2 gap-4'>
|
<div className='grid grid-cols-1 gap-4 sm:grid-cols-2'>
|
||||||
{tokens.map((token) => (
|
{tokens.map((token) => (
|
||||||
<AuthToken key={token.id} token={token} isCurrent={token.id === currentTokenId} />
|
<AuthToken key={token.id} token={token} isCurrent={token.id === currentTokenId} />
|
||||||
))}
|
))}
|
||||||
|
|
|
@ -156,7 +156,7 @@ const ChatComposer = React.forwardRef<HTMLTextAreaElement | null, IChatComposer>
|
||||||
|
|
||||||
<HStack alignItems='stretch' justifyContent='between' space={4}>
|
<HStack alignItems='stretch' justifyContent='between' space={4}>
|
||||||
{features.chatsMedia && (
|
{features.chatsMedia && (
|
||||||
<Stack justifyContent='end' alignItems='center' className='w-10 mb-1.5'>
|
<Stack justifyContent='end' alignItems='center' className='mb-1.5 w-10'>
|
||||||
<UploadButton
|
<UploadButton
|
||||||
onSelectFile={onSelectFile}
|
onSelectFile={onSelectFile}
|
||||||
resetFileKey={resetFileKey}
|
resetFileKey={resetFileKey}
|
||||||
|
@ -205,7 +205,7 @@ const ChatComposer = React.forwardRef<HTMLTextAreaElement | null, IChatComposer>
|
||||||
</Combobox>
|
</Combobox>
|
||||||
</Stack>
|
</Stack>
|
||||||
|
|
||||||
<Stack space={2} justifyContent='end' alignItems='center' className='w-10 mb-1.5'>
|
<Stack space={2} justifyContent='end' alignItems='center' className='mb-1.5 w-10'>
|
||||||
{isOverCharacterLimit ? (
|
{isOverCharacterLimit ? (
|
||||||
<Text size='sm' theme='danger'>{overLimitText}</Text>
|
<Text size='sm' theme='danger'>{overLimitText}</Text>
|
||||||
) : null}
|
) : null}
|
||||||
|
|
|
@ -68,7 +68,7 @@ const ChatListItem: React.FC<IChatListItemInterface> = ({ chat, onClick }) => {
|
||||||
role='button'
|
role='button'
|
||||||
key={chat.id}
|
key={chat.id}
|
||||||
onClick={() => onClick(chat)}
|
onClick={() => onClick(chat)}
|
||||||
className='group px-2 py-3 w-full flex flex-col rounded-lg hover:bg-gray-100 dark:hover:bg-gray-800 focus:shadow-inset-ring'
|
className='group flex w-full flex-col rounded-lg px-2 py-3 hover:bg-gray-100 focus:shadow-inset-ring dark:hover:bg-gray-800'
|
||||||
data-testid='chat-list-item'
|
data-testid='chat-list-item'
|
||||||
>
|
>
|
||||||
<HStack alignItems='center' justifyContent='between' space={2} className='w-full'>
|
<HStack alignItems='center' justifyContent='between' space={2} className='w-full'>
|
||||||
|
@ -76,7 +76,7 @@ const ChatListItem: React.FC<IChatListItemInterface> = ({ chat, onClick }) => {
|
||||||
<Avatar src={chat.account?.avatar} size={40} className='flex-none' />
|
<Avatar src={chat.account?.avatar} size={40} className='flex-none' />
|
||||||
|
|
||||||
<Stack alignItems='start' className='overflow-hidden'>
|
<Stack alignItems='start' className='overflow-hidden'>
|
||||||
<div className='flex items-center space-x-1 flex-grow w-full'>
|
<div className='flex w-full grow items-center space-x-1'>
|
||||||
<Text weight='bold' size='sm' align='left' truncate>{chat.account?.display_name || `@${chat.account.username}`}</Text>
|
<Text weight='bold' size='sm' align='left' truncate>{chat.account?.display_name || `@${chat.account.username}`}</Text>
|
||||||
{chat.account?.verified && <VerificationBadge />}
|
{chat.account?.verified && <VerificationBadge />}
|
||||||
</div>
|
</div>
|
||||||
|
@ -88,7 +88,7 @@ const ChatListItem: React.FC<IChatListItemInterface> = ({ chat, onClick }) => {
|
||||||
weight='medium'
|
weight='medium'
|
||||||
theme='muted'
|
theme='muted'
|
||||||
truncate
|
truncate
|
||||||
className='w-full h-5 pointer-events-none italic'
|
className='pointer-events-none h-5 w-full italic'
|
||||||
data-testid='chat-last-message'
|
data-testid='chat-last-message'
|
||||||
>
|
>
|
||||||
{intl.formatMessage(isBlocked ? messages.blockedYou : messages.blocking)}
|
{intl.formatMessage(isBlocked ? messages.blockedYou : messages.blocking)}
|
||||||
|
@ -102,7 +102,7 @@ const ChatListItem: React.FC<IChatListItemInterface> = ({ chat, onClick }) => {
|
||||||
weight='medium'
|
weight='medium'
|
||||||
theme={chat.last_message.unread ? 'default' : 'muted'}
|
theme={chat.last_message.unread ? 'default' : 'muted'}
|
||||||
truncate
|
truncate
|
||||||
className='w-full h-5 truncate-child pointer-events-none'
|
className='truncate-child pointer-events-none h-5 w-full'
|
||||||
data-testid='chat-last-message'
|
data-testid='chat-last-message'
|
||||||
dangerouslySetInnerHTML={{ __html: chat.last_message?.content }}
|
dangerouslySetInnerHTML={{ __html: chat.last_message?.content }}
|
||||||
/>
|
/>
|
||||||
|
@ -114,12 +114,12 @@ const ChatListItem: React.FC<IChatListItemInterface> = ({ chat, onClick }) => {
|
||||||
|
|
||||||
<HStack alignItems='center' space={2}>
|
<HStack alignItems='center' space={2}>
|
||||||
{features.chatsDelete && (
|
{features.chatsDelete && (
|
||||||
<div className='text-gray-600 hidden group-hover:block hover:text-gray-100'>
|
<div className='hidden text-gray-600 hover:text-gray-100 group-hover:block'>
|
||||||
<DropdownMenuContainer items={menu}>
|
<DropdownMenuContainer items={menu}>
|
||||||
<IconButton
|
<IconButton
|
||||||
src={require('@tabler/icons/dots.svg')}
|
src={require('@tabler/icons/dots.svg')}
|
||||||
title='Settings'
|
title='Settings'
|
||||||
className='text-gray-600 dark:text-gray-600 hover:text-gray-700 dark:hover:text-gray-500'
|
className='text-gray-600 hover:text-gray-700 dark:text-gray-600 dark:hover:text-gray-500'
|
||||||
iconClassName='w-4 h-4'
|
iconClassName='w-4 h-4'
|
||||||
/>
|
/>
|
||||||
</DropdownMenuContainer>
|
</DropdownMenuContainer>
|
||||||
|
@ -130,7 +130,7 @@ const ChatListItem: React.FC<IChatListItemInterface> = ({ chat, onClick }) => {
|
||||||
<>
|
<>
|
||||||
{chat.last_message.unread && (
|
{chat.last_message.unread && (
|
||||||
<div
|
<div
|
||||||
className='w-2 h-2 rounded-full bg-secondary-500'
|
className='h-2 w-2 rounded-full bg-secondary-500'
|
||||||
data-testid='chat-unread-indicator'
|
data-testid='chat-unread-indicator'
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
|
@ -106,8 +106,8 @@ const ChatMessageListIntro = () => {
|
||||||
</Button>
|
</Button>
|
||||||
</HStack>
|
</HStack>
|
||||||
) : (
|
) : (
|
||||||
<HStack justifyContent='center' alignItems='center' space={1} className='flex-shrink-0'>
|
<HStack justifyContent='center' alignItems='center' space={1} className='shrink-0'>
|
||||||
<Icon src={require('@tabler/icons/clock.svg')} className='text-gray-600 w-4 h-4' />
|
<Icon src={require('@tabler/icons/clock.svg')} className='h-4 w-4 text-gray-600' />
|
||||||
{chat.message_expiration && (
|
{chat.message_expiration && (
|
||||||
<Text size='sm' theme='muted'>
|
<Text size='sm' theme='muted'>
|
||||||
{intl.formatMessage(messages.messageLifespan, { day: secondsToDays(chat.message_expiration) })}
|
{intl.formatMessage(messages.messageLifespan, { day: secondsToDays(chat.message_expiration) })}
|
||||||
|
|
|
@ -305,7 +305,7 @@ const ChatMessageList: React.FC<IChatMessageList> = ({ chat }) => {
|
||||||
<IconButton
|
<IconButton
|
||||||
src={require('@tabler/icons/dots.svg')}
|
src={require('@tabler/icons/dots.svg')}
|
||||||
title={intl.formatMessage(messages.more)}
|
title={intl.formatMessage(messages.more)}
|
||||||
className='text-gray-600 dark:text-gray-600 hover:text-gray-700 dark:hover:text-gray-500'
|
className='text-gray-600 hover:text-gray-700 dark:text-gray-600 dark:hover:text-gray-500'
|
||||||
iconClassName='w-4 h-4'
|
iconClassName='w-4 h-4'
|
||||||
/>
|
/>
|
||||||
</DropdownMenuContainer>
|
</DropdownMenuContainer>
|
||||||
|
@ -379,12 +379,12 @@ const ChatMessageList: React.FC<IChatMessageList> = ({ chat }) => {
|
||||||
{(isMyMessage && features.chatsReadReceipts) ? (
|
{(isMyMessage && features.chatsReadReceipts) ? (
|
||||||
<>
|
<>
|
||||||
{isRead ? (
|
{isRead ? (
|
||||||
<span className='rounded-full flex flex-col items-center justify-center p-0.5 bg-primary-500 text-white dark:bg-primary-400 dark:text-primary-900 border border-solid border-primary-500 dark:border-primary-400'>
|
<span className='flex flex-col items-center justify-center rounded-full border border-solid border-primary-500 bg-primary-500 p-0.5 text-white dark:border-primary-400 dark:bg-primary-400 dark:text-primary-900'>
|
||||||
<Icon src={require('@tabler/icons/check.svg')} strokeWidth={3} className='w-2.5 h-2.5' />
|
<Icon src={require('@tabler/icons/check.svg')} strokeWidth={3} className='h-2.5 w-2.5' />
|
||||||
</span>
|
</span>
|
||||||
) : (
|
) : (
|
||||||
<span className='rounded-full flex flex-col items-center justify-center p-0.5 bg-transparent text-primary-500 dark:text-primary-400 border border-solid border-primary-500 dark:border-primary-400'>
|
<span className='flex flex-col items-center justify-center rounded-full border border-solid border-primary-500 bg-transparent p-0.5 text-primary-500 dark:border-primary-400 dark:text-primary-400'>
|
||||||
<Icon src={require('@tabler/icons/check.svg')} strokeWidth={3} className='w-2.5 h-2.5' />
|
<Icon src={require('@tabler/icons/check.svg')} strokeWidth={3} className='h-2.5 w-2.5' />
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
|
@ -419,7 +419,7 @@ const ChatMessageList: React.FC<IChatMessageList> = ({ chat }) => {
|
||||||
|
|
||||||
if (isBlocked) {
|
if (isBlocked) {
|
||||||
return (
|
return (
|
||||||
<Stack alignItems='center' justifyContent='center' className='h-full flex-grow'>
|
<Stack alignItems='center' justifyContent='center' className='h-full grow'>
|
||||||
<Stack alignItems='center' space={2}>
|
<Stack alignItems='center' space={2}>
|
||||||
<Avatar src={chat.account.avatar} size={75} />
|
<Avatar src={chat.account.avatar} size={75} />
|
||||||
<Text align='center'>
|
<Text align='center'>
|
||||||
|
@ -436,7 +436,7 @@ const ChatMessageList: React.FC<IChatMessageList> = ({ chat }) => {
|
||||||
|
|
||||||
if (isError) {
|
if (isError) {
|
||||||
return (
|
return (
|
||||||
<Stack alignItems='center' justifyContent='center' className='h-full flex-grow'>
|
<Stack alignItems='center' justifyContent='center' className='h-full grow'>
|
||||||
<Stack space={4}>
|
<Stack space={4}>
|
||||||
<Stack space={1}>
|
<Stack space={1}>
|
||||||
<Text size='lg' weight='bold' align='center'>
|
<Text size='lg' weight='bold' align='center'>
|
||||||
|
@ -459,7 +459,7 @@ const ChatMessageList: React.FC<IChatMessageList> = ({ chat }) => {
|
||||||
|
|
||||||
if (isLoading) {
|
if (isLoading) {
|
||||||
return (
|
return (
|
||||||
<div className='flex-grow flex flex-col justify-end pb-4'>
|
<div className='flex grow flex-col justify-end pb-4'>
|
||||||
<div className='px-4'>
|
<div className='px-4'>
|
||||||
<PlaceholderChatMessage isMyMessage />
|
<PlaceholderChatMessage isMyMessage />
|
||||||
<PlaceholderChatMessage />
|
<PlaceholderChatMessage />
|
||||||
|
@ -472,8 +472,8 @@ const ChatMessageList: React.FC<IChatMessageList> = ({ chat }) => {
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='h-full flex flex-col flex-grow space-y-6'>
|
<div className='flex h-full grow flex-col space-y-6'>
|
||||||
<div className='flex-grow flex flex-col justify-end'>
|
<div className='flex grow flex-col justify-end'>
|
||||||
<Virtuoso
|
<Virtuoso
|
||||||
ref={node}
|
ref={node}
|
||||||
alignToBottom
|
alignToBottom
|
||||||
|
|
|
@ -60,11 +60,11 @@ const ChatPage: React.FC<IChatPage> = ({ chatId }) => {
|
||||||
<div
|
<div
|
||||||
ref={containerRef}
|
ref={containerRef}
|
||||||
style={{ height }}
|
style={{ height }}
|
||||||
className='h-screen bg-white dark:bg-primary-900 text-gray-900 dark:text-gray-100 shadow-lg dark:shadow-none overflow-hidden sm:rounded-t-xl'
|
className='h-screen overflow-hidden bg-white text-gray-900 shadow-lg dark:bg-primary-900 dark:text-gray-100 dark:shadow-none sm:rounded-t-xl'
|
||||||
>
|
>
|
||||||
{isOnboarded ? (
|
{isOnboarded ? (
|
||||||
<div
|
<div
|
||||||
className='grid grid-cols-9 overflow-hidden h-full dark:divide-x-2 dark:divide-solid dark:divide-gray-800'
|
className='grid h-full grid-cols-9 overflow-hidden dark:divide-x-2 dark:divide-solid dark:divide-gray-800'
|
||||||
data-testid='chat-page'
|
data-testid='chat-page'
|
||||||
>
|
>
|
||||||
<Stack
|
<Stack
|
||||||
|
|
|
@ -16,7 +16,7 @@ const BlankslateEmpty: React.FC<IBlankslate> = () => {
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack space={6} alignItems='center' justifyContent='center' className='p-6 h-full'>
|
<Stack space={6} alignItems='center' justifyContent='center' className='h-full p-6'>
|
||||||
<Stack space={2} className='max-w-sm'>
|
<Stack space={2} className='max-w-sm'>
|
||||||
<Text size='2xl' weight='bold' tag='h2' align='center'>
|
<Text size='2xl' weight='bold' tag='h2' align='center'>
|
||||||
<FormattedMessage
|
<FormattedMessage
|
||||||
|
|
|
@ -13,7 +13,7 @@ const BlankslateWithChats = () => {
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack space={6} alignItems='center' justifyContent='center' className='p-6 h-full'>
|
<Stack space={6} alignItems='center' justifyContent='center' className='h-full p-6'>
|
||||||
<Stack space={2} className='max-w-sm'>
|
<Stack space={2} className='max-w-sm'>
|
||||||
<Text size='2xl' weight='bold' tag='h2' align='center'>
|
<Text size='2xl' weight='bold' tag='h2' align='center'>
|
||||||
<FormattedMessage
|
<FormattedMessage
|
||||||
|
|
|
@ -116,12 +116,12 @@ const ChatPageMain = () => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack className='h-full overflow-hidden'>
|
<Stack className='h-full overflow-hidden'>
|
||||||
<HStack alignItems='center' justifyContent='between' space={2} className='px-4 py-4 w-full'>
|
<HStack alignItems='center' justifyContent='between' space={2} className='w-full p-4'>
|
||||||
<HStack alignItems='center' space={2} className='overflow-hidden'>
|
<HStack alignItems='center' space={2} className='overflow-hidden'>
|
||||||
<HStack alignItems='center'>
|
<HStack alignItems='center'>
|
||||||
<IconButton
|
<IconButton
|
||||||
src={require('@tabler/icons/arrow-left.svg')}
|
src={require('@tabler/icons/arrow-left.svg')}
|
||||||
className='sm:hidden h-7 w-7 mr-2 sm:mr-0'
|
className='mr-2 h-7 w-7 sm:mr-0 sm:hidden'
|
||||||
onClick={() => history.push('/chats')}
|
onClick={() => history.push('/chats')}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
@ -130,8 +130,8 @@ const ChatPageMain = () => {
|
||||||
</Link>
|
</Link>
|
||||||
</HStack>
|
</HStack>
|
||||||
|
|
||||||
<Stack alignItems='start' className='overflow-hidden h-11'>
|
<Stack alignItems='start' className='h-11 overflow-hidden'>
|
||||||
<div className='flex items-center space-x-1 flex-grow w-full'>
|
<div className='flex w-full grow items-center space-x-1'>
|
||||||
<Link to={`/@${chat.account.acct}`}>
|
<Link to={`/@${chat.account.acct}`}>
|
||||||
<Text weight='bold' size='sm' align='left' truncate>
|
<Text weight='bold' size='sm' align='left' truncate>
|
||||||
{chat.account.display_name || `@${chat.account.username}`}
|
{chat.account.display_name || `@${chat.account.username}`}
|
||||||
|
@ -212,8 +212,8 @@ const ChatPageMain = () => {
|
||||||
onSelect={isBlocking ? handleUnblockUser : handleBlockUser}
|
onSelect={isBlocking ? handleUnblockUser : handleBlockUser}
|
||||||
className='!px-0 hover:!bg-transparent'
|
className='!px-0 hover:!bg-transparent'
|
||||||
>
|
>
|
||||||
<div className='w-full flex items-center space-x-2 font-bold text-sm text-primary-500 dark:text-accent-blue'>
|
<div className='flex w-full items-center space-x-2 text-sm font-bold text-primary-500 dark:text-accent-blue'>
|
||||||
<Icon src={require('@tabler/icons/ban.svg')} className='w-5 h-5' />
|
<Icon src={require('@tabler/icons/ban.svg')} className='h-5 w-5' />
|
||||||
<span>{intl.formatMessage(isBlocking ? messages.unblockUser : messages.blockUser, { acct: chat.account.acct })}</span>
|
<span>{intl.formatMessage(isBlocking ? messages.unblockUser : messages.blockUser, { acct: chat.account.acct })}</span>
|
||||||
</div>
|
</div>
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
|
@ -224,8 +224,8 @@ const ChatPageMain = () => {
|
||||||
onSelect={handleLeaveChat}
|
onSelect={handleLeaveChat}
|
||||||
className='!px-0 hover:!bg-transparent'
|
className='!px-0 hover:!bg-transparent'
|
||||||
>
|
>
|
||||||
<div className='w-full flex items-center space-x-2 font-bold text-sm text-danger-600 dark:text-danger-500'>
|
<div className='flex w-full items-center space-x-2 text-sm font-bold text-danger-600 dark:text-danger-500'>
|
||||||
<Icon src={require('@tabler/icons/logout.svg')} className='w-5 h-5' />
|
<Icon src={require('@tabler/icons/logout.svg')} className='h-5 w-5' />
|
||||||
<span>{intl.formatMessage(messages.leaveChat)}</span>
|
<span>{intl.formatMessage(messages.leaveChat)}</span>
|
||||||
</div>
|
</div>
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
|
|
|
@ -20,11 +20,11 @@ const ChatPageNew: React.FC<IChatPageNew> = () => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack className='h-full space-y-4'>
|
<Stack className='h-full space-y-4'>
|
||||||
<Stack className='flex-grow pt-6 px-4 sm:px-6'>
|
<Stack className='grow px-4 pt-6 sm:px-6'>
|
||||||
<HStack alignItems='center'>
|
<HStack alignItems='center'>
|
||||||
<IconButton
|
<IconButton
|
||||||
src={require('@tabler/icons/arrow-left.svg')}
|
src={require('@tabler/icons/arrow-left.svg')}
|
||||||
className='sm:hidden h-7 w-7 mr-2 sm:mr-0'
|
className='mr-2 h-7 w-7 sm:mr-0 sm:hidden'
|
||||||
onClick={() => history.push('/chats')}
|
onClick={() => history.push('/chats')}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
|
|
@ -47,11 +47,11 @@ const ChatPageSettings = () => {
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack className='h-full py-6 px-4 sm:p-6 space-y-8'>
|
<Stack className='h-full space-y-8 py-6 px-4 sm:p-6'>
|
||||||
<HStack alignItems='center'>
|
<HStack alignItems='center'>
|
||||||
<IconButton
|
<IconButton
|
||||||
src={require('@tabler/icons/arrow-left.svg')}
|
src={require('@tabler/icons/arrow-left.svg')}
|
||||||
className='sm:hidden h-7 w-7 mr-2 sm:mr-0'
|
className='mr-2 h-7 w-7 sm:mr-0 sm:hidden'
|
||||||
onClick={() => history.push('/chats')}
|
onClick={() => history.push('/chats')}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
|
|
@ -64,7 +64,7 @@ const ChatPageSidebar = () => {
|
||||||
)}
|
)}
|
||||||
</Stack>
|
</Stack>
|
||||||
|
|
||||||
<Stack className='flex-grow h-full'>
|
<Stack className='h-full grow'>
|
||||||
<ChatList
|
<ChatList
|
||||||
onClickChat={handleClickChat}
|
onClickChat={handleClickChat}
|
||||||
searchValue={debouncedSearch}
|
searchValue={debouncedSearch}
|
||||||
|
|
|
@ -36,9 +36,9 @@ const Welcome = () => {
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack className='py-20 px-4 sm:px-0 h-full overflow-y-auto' data-testid='chats-welcome'>
|
<Stack className='h-full overflow-y-auto py-20 px-4 sm:px-0' data-testid='chats-welcome'>
|
||||||
<div className='w-full sm:w-3/5 xl:w-2/5 mx-auto mb-2.5'>
|
<div className='mx-auto mb-2.5 w-full sm:w-3/5 xl:w-2/5'>
|
||||||
<Text align='center' weight='bold' className='mb-6 text-2xl md:text-3xl leading-8'>
|
<Text align='center' weight='bold' className='mb-6 text-2xl leading-8 md:text-3xl'>
|
||||||
{intl.formatMessage(messages.title, { br: <br /> })}
|
{intl.formatMessage(messages.title, { br: <br /> })}
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
|
@ -47,7 +47,7 @@ const Welcome = () => {
|
||||||
</Text>
|
</Text>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Form onSubmit={handleSubmit} className='space-y-8 w-full sm:w-2/3 lg:w-3/5 sm:mx-auto'>
|
<Form onSubmit={handleSubmit} className='w-full space-y-8 sm:mx-auto sm:w-2/3 lg:w-3/5'>
|
||||||
<Stack space={2}>
|
<Stack space={2}>
|
||||||
<CardTitle title={<FormattedMessage id='chat.page_settings.privacy' defaultMessage='Privacy' />} />
|
<CardTitle title={<FormattedMessage id='chat.page_settings.privacy' defaultMessage='Privacy' />} />
|
||||||
|
|
||||||
|
|
|
@ -20,11 +20,11 @@ const Blankslate = ({ onSearch }: IBlankslate) => {
|
||||||
<Stack
|
<Stack
|
||||||
alignItems='center'
|
alignItems='center'
|
||||||
justifyContent='center'
|
justifyContent='center'
|
||||||
className='h-full flex-grow'
|
className='h-full grow'
|
||||||
data-testid='chat-pane-blankslate'
|
data-testid='chat-pane-blankslate'
|
||||||
>
|
>
|
||||||
<Stack space={4}>
|
<Stack space={4}>
|
||||||
<Stack space={1} className='max-w-[80%] mx-auto'>
|
<Stack space={1} className='mx-auto max-w-[80%]'>
|
||||||
<Text size='lg' weight='bold' align='center'>
|
<Text size='lg' weight='bold' align='center'>
|
||||||
{intl.formatMessage(messages.title)}
|
{intl.formatMessage(messages.title)}
|
||||||
</Text>
|
</Text>
|
||||||
|
|
|
@ -45,7 +45,7 @@ const ChatPane = () => {
|
||||||
const renderBody = () => {
|
const renderBody = () => {
|
||||||
if (hasSearchValue || Number(chats?.length) > 0 || isLoading) {
|
if (hasSearchValue || Number(chats?.length) > 0 || isLoading) {
|
||||||
return (
|
return (
|
||||||
<Stack space={4} className='flex-grow h-full'>
|
<Stack space={4} className='h-full grow'>
|
||||||
{features.chatsSearch && (
|
{features.chatsSearch && (
|
||||||
<div className='px-4'>
|
<div className='px-4'>
|
||||||
<ChatSearchInput
|
<ChatSearchInput
|
||||||
|
|
|
@ -12,7 +12,7 @@ const Blankslate = () => {
|
||||||
const intl = useIntl();
|
const intl = useIntl();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack justifyContent='center' alignItems='center' space={2} className='h-full w-2/3 mx-auto'>
|
<Stack justifyContent='center' alignItems='center' space={2} className='mx-auto h-full w-2/3'>
|
||||||
<Text weight='bold' size='lg' align='center'>
|
<Text weight='bold' size='lg' align='center'>
|
||||||
{intl.formatMessage(messages.title)}
|
{intl.formatMessage(messages.title)}
|
||||||
</Text>
|
</Text>
|
||||||
|
|
|
@ -88,7 +88,7 @@ const ChatSearch = (props: IChatSearch) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack space={4} className='flex-grow h-full'>
|
<Stack space={4} className='h-full grow'>
|
||||||
<div className='px-4'>
|
<div className='px-4'>
|
||||||
<Input
|
<Input
|
||||||
data-testid='search'
|
data-testid='search'
|
||||||
|
@ -111,7 +111,7 @@ const ChatSearch = (props: IChatSearch) => {
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Stack className='flex-grow'>
|
<Stack className='grow'>
|
||||||
{renderBody()}
|
{renderBody()}
|
||||||
</Stack>
|
</Stack>
|
||||||
</Stack>
|
</Stack>
|
||||||
|
|
|
@ -12,7 +12,7 @@ const EmptyResultsBlankslate = () => {
|
||||||
const intl = useIntl();
|
const intl = useIntl();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack justifyContent='center' alignItems='center' space={2} className='h-full w-2/3 mx-auto'>
|
<Stack justifyContent='center' alignItems='center' space={2} className='mx-auto h-full w-2/3'>
|
||||||
<Text weight='bold' size='lg' align='center' data-testid='no-results'>
|
<Text weight='bold' size='lg' align='center' data-testid='no-results'>
|
||||||
{intl.formatMessage(messages.title)}
|
{intl.formatMessage(messages.title)}
|
||||||
</Text>
|
</Text>
|
||||||
|
|
|
@ -29,7 +29,7 @@ const Results = ({ accountSearchResult, onSelect }: IResults) => {
|
||||||
<button
|
<button
|
||||||
key={account.id}
|
key={account.id}
|
||||||
type='button'
|
type='button'
|
||||||
className='px-2 py-3 w-full rounded-lg flex flex-col hover:bg-gray-100 dark:hover:bg-gray-800'
|
className='flex w-full flex-col rounded-lg px-2 py-3 hover:bg-gray-100 dark:hover:bg-gray-800'
|
||||||
onClick={() => onSelect(account.id)}
|
onClick={() => onSelect(account.id)}
|
||||||
data-testid='account'
|
data-testid='account'
|
||||||
>
|
>
|
||||||
|
@ -37,7 +37,7 @@ const Results = ({ accountSearchResult, onSelect }: IResults) => {
|
||||||
<Avatar src={account.avatar} size={40} />
|
<Avatar src={account.avatar} size={40} />
|
||||||
|
|
||||||
<Stack alignItems='start'>
|
<Stack alignItems='start'>
|
||||||
<div className='flex items-center space-x-1 flex-grow'>
|
<div className='flex grow items-center space-x-1'>
|
||||||
<Text weight='bold' size='sm' truncate>{account.display_name}</Text>
|
<Text weight='bold' size='sm' truncate>{account.display_name}</Text>
|
||||||
{account.verified && <VerificationBadge />}
|
{account.verified && <VerificationBadge />}
|
||||||
</div>
|
</div>
|
||||||
|
@ -48,7 +48,7 @@ const Results = ({ accountSearchResult, onSelect }: IResults) => {
|
||||||
), []);
|
), []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='relative flex-grow'>
|
<div className='relative grow'>
|
||||||
<Virtuoso
|
<Virtuoso
|
||||||
data={accounts}
|
data={accounts}
|
||||||
itemContent={(index, chat) => (
|
itemContent={(index, chat) => (
|
||||||
|
|
|
@ -31,9 +31,9 @@ const ChatPaneHeader = (props: IChatPaneHeader) => {
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<HStack {...rest} alignItems='center' justifyContent='between' className='rounded-t-xl h-16 py-3 px-4'>
|
<HStack {...rest} alignItems='center' justifyContent='between' className='h-16 rounded-t-xl py-3 px-4'>
|
||||||
<ButtonComp
|
<ButtonComp
|
||||||
className='flex-grow flex items-center flex-row space-x-1 h-16'
|
className='flex h-16 grow flex-row items-center space-x-1'
|
||||||
data-testid='title'
|
data-testid='title'
|
||||||
{...buttonProps}
|
{...buttonProps}
|
||||||
>
|
>
|
||||||
|
@ -47,7 +47,7 @@ const ChatPaneHeader = (props: IChatPaneHeader) => {
|
||||||
({unreadCount})
|
({unreadCount})
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
<div className='bg-accent-300 w-2.5 h-2.5 rounded-full' />
|
<div className='h-2.5 w-2.5 rounded-full bg-accent-300' />
|
||||||
</HStack>
|
</HStack>
|
||||||
)}
|
)}
|
||||||
</ButtonComp>
|
</ButtonComp>
|
||||||
|
|
|
@ -107,7 +107,7 @@ const ChatSettings = () => {
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Stack space={4} className='w-5/6 mx-auto'>
|
<Stack space={4} className='mx-auto w-5/6'>
|
||||||
<HStack alignItems='center' space={3}>
|
<HStack alignItems='center' space={3}>
|
||||||
<Avatar src={chat.account.avatar_static} size={50} />
|
<Avatar src={chat.account.avatar_static} size={50} />
|
||||||
<Stack>
|
<Stack>
|
||||||
|
@ -135,14 +135,14 @@ const ChatSettings = () => {
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Stack space={5}>
|
<Stack space={5}>
|
||||||
<button onClick={isBlocking ? handleUnblockUser : handleBlockUser} className='w-full flex items-center space-x-2 font-bold text-sm text-primary-600 dark:text-accent-blue'>
|
<button onClick={isBlocking ? handleUnblockUser : handleBlockUser} className='flex w-full items-center space-x-2 text-sm font-bold text-primary-600 dark:text-accent-blue'>
|
||||||
<Icon src={require('@tabler/icons/ban.svg')} className='w-5 h-5' />
|
<Icon src={require('@tabler/icons/ban.svg')} className='h-5 w-5' />
|
||||||
<span>{intl.formatMessage(isBlocking ? messages.unblockUser : messages.blockUser, { acct: chat.account.acct })}</span>
|
<span>{intl.formatMessage(isBlocking ? messages.unblockUser : messages.blockUser, { acct: chat.account.acct })}</span>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
{features.chatsDelete && (
|
{features.chatsDelete && (
|
||||||
<button onClick={handleLeaveChat} className='w-full flex items-center space-x-2 font-bold text-sm text-danger-600'>
|
<button onClick={handleLeaveChat} className='flex w-full items-center space-x-2 text-sm font-bold text-danger-600'>
|
||||||
<Icon src={require('@tabler/icons/logout.svg')} className='w-5 h-5' />
|
<Icon src={require('@tabler/icons/logout.svg')} className='h-5 w-5' />
|
||||||
<span>{intl.formatMessage(messages.leaveChat)}</span>
|
<span>{intl.formatMessage(messages.leaveChat)}</span>
|
||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
|
|
|
@ -87,7 +87,7 @@ const ChatWindow = () => {
|
||||||
|
|
||||||
<Stack alignItems='start'>
|
<Stack alignItems='start'>
|
||||||
<LinkWrapper enabled={isOpen} to={`/@${chat.account.acct}`}>
|
<LinkWrapper enabled={isOpen} to={`/@${chat.account.acct}`}>
|
||||||
<div className='flex items-center space-x-1 flex-grow'>
|
<div className='flex grow items-center space-x-1'>
|
||||||
<Text size='sm' weight='bold' truncate>{chat.account.display_name || `@${chat.account.acct}`}</Text>
|
<Text size='sm' weight='bold' truncate>{chat.account.display_name || `@${chat.account.acct}`}</Text>
|
||||||
{chat.account.verified && <VerificationBadge />}
|
{chat.account.verified && <VerificationBadge />}
|
||||||
</div>
|
</div>
|
||||||
|
@ -113,7 +113,7 @@ const ChatWindow = () => {
|
||||||
onToggle={toggleChatPane}
|
onToggle={toggleChatPane}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Stack className='overflow-hidden flex-grow h-full' space={2}>
|
<Stack className='h-full grow overflow-hidden' space={2}>
|
||||||
<Chat chat={chat} inputRef={inputRef} />
|
<Chat chat={chat} inputRef={inputRef} />
|
||||||
</Stack>
|
</Stack>
|
||||||
</>
|
</>
|
||||||
|
|
|
@ -160,7 +160,7 @@ const Chat: React.FC<ChatInterface> = ({ chat, inputRef, className }) => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack className={classNames('overflow-hidden flex flex-grow', className)} onMouseOver={handleMouseOver}>
|
<Stack className={classNames('overflow-hidden flex flex-grow', className)} onMouseOver={handleMouseOver}>
|
||||||
<div className='flex-grow h-full overflow-hidden flex justify-center'>
|
<div className='flex h-full grow justify-center overflow-hidden'>
|
||||||
<ChatMessageList chat={chat} />
|
<ChatMessageList chat={chat} />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
@ -42,7 +42,7 @@ const DurationSelector = ({ onDurationChange }: IDurationSelector) => {
|
||||||
}, [value]);
|
}, [value]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='grid grid-cols-1 gap-y-2 gap-x-2 sm:grid-cols-3'>
|
<div className='grid grid-cols-1 gap-2 sm:grid-cols-3'>
|
||||||
<div className='sm:col-span-1'>
|
<div className='sm:col-span-1'>
|
||||||
<Select
|
<Select
|
||||||
value={days}
|
value={days}
|
||||||
|
|
|
@ -79,7 +79,7 @@ const Option: React.FC<IOption> = ({
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<AutosuggestInput
|
<AutosuggestInput
|
||||||
className='rounded-md dark:!bg-transparent !bg-transparent'
|
className='rounded-md !bg-transparent dark:!bg-transparent'
|
||||||
placeholder={intl.formatMessage(messages.option_placeholder, { number: index + 1 })}
|
placeholder={intl.formatMessage(messages.option_placeholder, { number: index + 1 })}
|
||||||
maxLength={maxChars}
|
maxLength={maxChars}
|
||||||
value={title}
|
value={title}
|
||||||
|
|
|
@ -33,7 +33,7 @@ const ReplyIndicator: React.FC<IReplyIndicator> = ({ status, hideActions, onCanc
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack space={2} className='p-4 rounded-lg bg-gray-100 dark:bg-gray-800'>
|
<Stack space={2} className='rounded-lg bg-gray-100 p-4 dark:bg-gray-800'>
|
||||||
<AccountContainer
|
<AccountContainer
|
||||||
{...actions}
|
{...actions}
|
||||||
id={status.getIn(['account', 'id']) as string}
|
id={status.getIn(['account', 'id']) as string}
|
||||||
|
@ -44,7 +44,7 @@ const ReplyIndicator: React.FC<IReplyIndicator> = ({ status, hideActions, onCanc
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Text
|
<Text
|
||||||
className='break-words status__content'
|
className='status__content break-words'
|
||||||
size='sm'
|
size='sm'
|
||||||
dangerouslySetInnerHTML={{ __html: status.contentHtml }}
|
dangerouslySetInnerHTML={{ __html: status.contentHtml }}
|
||||||
direction={isRtl(status.search_index) ? 'rtl' : 'ltr'}
|
direction={isRtl(status.search_index) ? 'rtl' : 'ltr'}
|
||||||
|
|
|
@ -236,7 +236,7 @@ const SearchResults = () => {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{filterByAccount ? (
|
{filterByAccount ? (
|
||||||
<HStack className='mb-4 pb-4 px-2 border-solid border-b border-gray-200 dark:border-gray-800' space={2}>
|
<HStack className='mb-4 border-b border-solid border-gray-200 px-2 pb-4 dark:border-gray-800' space={2}>
|
||||||
<IconButton iconClassName='h-5 w-5' src={require('@tabler/icons/x.svg')} onClick={handleUnsetAccount} />
|
<IconButton iconClassName='h-5 w-5' src={require('@tabler/icons/x.svg')} onClick={handleUnsetAccount} />
|
||||||
<Text>
|
<Text>
|
||||||
<FormattedMessage
|
<FormattedMessage
|
||||||
|
|
|
@ -149,7 +149,7 @@ const Search = (props: ISearch) => {
|
||||||
<div
|
<div
|
||||||
role='button'
|
role='button'
|
||||||
tabIndex={0}
|
tabIndex={0}
|
||||||
className='absolute inset-y-0 right-0 rtl:left-0 rtl:right-auto px-3 flex items-center cursor-pointer'
|
className='absolute inset-y-0 right-0 flex cursor-pointer items-center px-3 rtl:left-0 rtl:right-auto'
|
||||||
onClick={handleClear}
|
onClick={handleClear}
|
||||||
>
|
>
|
||||||
<SvgIcon
|
<SvgIcon
|
||||||
|
|
|
@ -62,7 +62,7 @@ const SpoilerInput = React.forwardRef<AutosuggestInput, ISpoilerInput>(({
|
||||||
onSuggestionSelected={onSuggestionSelected}
|
onSuggestionSelected={onSuggestionSelected}
|
||||||
searchTokens={[':']}
|
searchTokens={[':']}
|
||||||
id='cw-spoiler-input'
|
id='cw-spoiler-input'
|
||||||
className='rounded-md dark:!bg-transparent !bg-transparent'
|
className='rounded-md !bg-transparent dark:!bg-transparent'
|
||||||
ref={ref}
|
ref={ref}
|
||||||
autoFocus
|
autoFocus
|
||||||
/>
|
/>
|
||||||
|
|
|
@ -33,7 +33,7 @@ const CryptoAddress: React.FC<ICryptoAddress> = (props): JSX.Element => {
|
||||||
<Stack>
|
<Stack>
|
||||||
<HStack alignItems='center' className='mb-1'>
|
<HStack alignItems='center' className='mb-1'>
|
||||||
<CryptoIcon
|
<CryptoIcon
|
||||||
className='flex items-start justify-center w-6 mr-2.5 rtl:ml-2.5 rtl:mr-0'
|
className='mr-2.5 flex w-6 items-start justify-center rtl:ml-2.5 rtl:mr-0'
|
||||||
ticker={ticker}
|
ticker={ticker}
|
||||||
title={title}
|
title={title}
|
||||||
/>
|
/>
|
||||||
|
@ -41,12 +41,12 @@ const CryptoAddress: React.FC<ICryptoAddress> = (props): JSX.Element => {
|
||||||
<Text weight='bold'>{title || ticker.toUpperCase()}</Text>
|
<Text weight='bold'>{title || ticker.toUpperCase()}</Text>
|
||||||
|
|
||||||
<HStack alignItems='center' className='ml-auto'>
|
<HStack alignItems='center' className='ml-auto'>
|
||||||
<a className='text-gray-500 ml-1 rtl:ml-0 rtl:mr-1' href='#' onClick={handleModalClick}>
|
<a className='ml-1 text-gray-500 rtl:ml-0 rtl:mr-1' href='#' onClick={handleModalClick}>
|
||||||
<Icon src={require('@tabler/icons/qrcode.svg')} size={20} />
|
<Icon src={require('@tabler/icons/qrcode.svg')} size={20} />
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
{explorerUrl && (
|
{explorerUrl && (
|
||||||
<a className='text-gray-500 ml-1 rtl:ml-0 rtl:mr-1' href={explorerUrl} target='_blank'>
|
<a className='ml-1 text-gray-500 rtl:ml-0 rtl:mr-1' href={explorerUrl} target='_blank'>
|
||||||
<Icon src={require('@tabler/icons/external-link.svg')} size={20} />
|
<Icon src={require('@tabler/icons/external-link.svg')} size={20} />
|
||||||
</a>
|
</a>
|
||||||
)}
|
)}
|
||||||
|
|
|
@ -55,7 +55,7 @@ const Developers: React.FC = () => {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Column label={intl.formatMessage(messages.heading)}>
|
<Column label={intl.formatMessage(messages.heading)}>
|
||||||
<div className='grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 gap-2'>
|
<div className='grid grid-cols-1 gap-2 sm:grid-cols-2 lg:grid-cols-3'>
|
||||||
<DashWidget to='/developers/apps/create'>
|
<DashWidget to='/developers/apps/create'>
|
||||||
<SvgIcon src={require('@tabler/icons/apps.svg')} className='text-gray-700 dark:text-gray-600' />
|
<SvgIcon src={require('@tabler/icons/apps.svg')} className='text-gray-700 dark:text-gray-600' />
|
||||||
|
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue