mirror of
https://github.com/glitch-soc/mastodon.git
synced 2026-03-29 03:00:33 +02:00
[Glitch] Profile redesign: Pinned posts
Port 2e30044a37 to glitch-soc
Co-authored-by: diondiondion <mail@diondiondion.com>
Signed-off-by: Claire <claire.github-309c@sitedethib.com>
This commit is contained in:
@@ -28,10 +28,12 @@ export const TIMELINE_INSERT = 'TIMELINE_INSERT';
|
||||
// When adding new special markers here, make sure to update TIMELINE_NON_STATUS_MARKERS in actions/timelines_typed.js
|
||||
export const TIMELINE_SUGGESTIONS = 'inline-follow-suggestions';
|
||||
export const TIMELINE_GAP = null;
|
||||
export const TIMELINE_PINNED_VIEW_ALL = 'pinned-view-all';
|
||||
|
||||
export const TIMELINE_NON_STATUS_MARKERS = [
|
||||
TIMELINE_GAP,
|
||||
TIMELINE_SUGGESTIONS,
|
||||
TIMELINE_PINNED_VIEW_ALL,
|
||||
];
|
||||
|
||||
export const loadPending = timeline => ({
|
||||
|
||||
@@ -110,6 +110,7 @@ class Status extends ImmutablePureComponent {
|
||||
muted: PropTypes.bool,
|
||||
hidden: PropTypes.bool,
|
||||
unread: PropTypes.bool,
|
||||
featured: PropTypes.bool,
|
||||
showActions: PropTypes.bool,
|
||||
prepend: PropTypes.string,
|
||||
withDismiss: PropTypes.bool,
|
||||
@@ -686,7 +687,7 @@ class Status extends ImmutablePureComponent {
|
||||
const {statusContentProps, hashtagBar} = getHashtagBarForStatus(status);
|
||||
|
||||
const header = this.props.headerRenderFn
|
||||
? this.props.headerRenderFn({ status, account, avatarSize, messages, onHeaderClick: this.handleHeaderClick, statusProps: this.props })
|
||||
? this.props.headerRenderFn({ status, account, avatarSize, messages, onHeaderClick: this.handleHeaderClick, featured })
|
||||
: (
|
||||
<StatusHeader
|
||||
status={status}
|
||||
|
||||
@@ -2,6 +2,8 @@ import type { FC, HTMLAttributes, MouseEventHandler, ReactNode } from 'react';
|
||||
|
||||
import { defineMessage, useIntl } from 'react-intl';
|
||||
|
||||
import classNames from 'classnames';
|
||||
|
||||
import { isStatusVisibility } from '@/flavours/glitch/api_types/statuses';
|
||||
import type { Account } from '@/flavours/glitch/models/account';
|
||||
import type { Status } from '@/flavours/glitch/models/status';
|
||||
@@ -12,8 +14,6 @@ import type { DisplayNameProps } from '../display_name';
|
||||
import { LinkedDisplayName } from '../display_name';
|
||||
import { VisibilityIcon } from '../visibility_icon';
|
||||
|
||||
import type { StatusProps } from './types';
|
||||
|
||||
export interface StatusHeaderProps {
|
||||
status: Status;
|
||||
account?: Account;
|
||||
@@ -22,17 +22,17 @@ export interface StatusHeaderProps {
|
||||
wrapperProps?: HTMLAttributes<HTMLDivElement>;
|
||||
displayNameProps?: DisplayNameProps;
|
||||
onHeaderClick?: MouseEventHandler<HTMLDivElement>;
|
||||
className?: string;
|
||||
featured?: boolean;
|
||||
}
|
||||
|
||||
export type StatusHeaderRenderFn = (
|
||||
args: StatusHeaderProps,
|
||||
statusProps?: StatusProps,
|
||||
) => ReactNode;
|
||||
export type StatusHeaderRenderFn = (args: StatusHeaderProps) => ReactNode;
|
||||
|
||||
export const StatusHeader: FC<StatusHeaderProps> = ({
|
||||
status,
|
||||
account,
|
||||
children,
|
||||
className,
|
||||
avatarSize = 48,
|
||||
wrapperProps,
|
||||
onHeaderClick,
|
||||
@@ -45,7 +45,7 @@ export const StatusHeader: FC<StatusHeaderProps> = ({
|
||||
onClick={onHeaderClick}
|
||||
onAuxClick={onHeaderClick}
|
||||
{...wrapperProps}
|
||||
className='status__info'
|
||||
className={classNames('status__info', className)}
|
||||
/* eslint-enable jsx-a11y/no-static-element-interactions, jsx-a11y/click-events-have-key-events */
|
||||
>
|
||||
<StatusDisplayName
|
||||
|
||||
@@ -14,6 +14,7 @@ export interface StatusProps {
|
||||
muted?: boolean;
|
||||
hidden?: boolean;
|
||||
unread?: boolean;
|
||||
featured?: boolean;
|
||||
showThread?: boolean;
|
||||
showActions?: boolean;
|
||||
isQuotedPost?: boolean;
|
||||
|
||||
@@ -5,9 +5,10 @@ import ImmutablePureComponent from 'react-immutable-pure-component';
|
||||
|
||||
import { debounce } from 'lodash';
|
||||
|
||||
import { TIMELINE_GAP, TIMELINE_SUGGESTIONS } from 'flavours/glitch/actions/timelines';
|
||||
import { TIMELINE_GAP, TIMELINE_PINNED_VIEW_ALL, TIMELINE_SUGGESTIONS } from 'flavours/glitch/actions/timelines';
|
||||
import { RegenerationIndicator } from 'flavours/glitch/components/regeneration_indicator';
|
||||
import { InlineFollowSuggestions } from 'flavours/glitch/features/home_timeline/components/inline_follow_suggestions';
|
||||
import { PinnedShowAllButton } from '@/flavours/glitch/features/account_timeline/v2/pinned_statuses';
|
||||
|
||||
import { StatusQuoteManager } from '../components/status_quoted';
|
||||
|
||||
@@ -35,6 +36,7 @@ export default class StatusList extends ImmutablePureComponent {
|
||||
timelineId: PropTypes.string.isRequired,
|
||||
lastId: PropTypes.string,
|
||||
bindToDocument: PropTypes.bool,
|
||||
statusProps: PropTypes.object,
|
||||
regex: PropTypes.string,
|
||||
};
|
||||
|
||||
@@ -52,7 +54,7 @@ export default class StatusList extends ImmutablePureComponent {
|
||||
};
|
||||
|
||||
render () {
|
||||
const { statusIds, featuredStatusIds, onLoadMore, timelineId, ...other } = this.props;
|
||||
const { statusIds, featuredStatusIds, onLoadMore, timelineId, statusProps, ...other } = this.props;
|
||||
const { isLoading, isPartial } = other;
|
||||
|
||||
if (isPartial) {
|
||||
@@ -83,6 +85,7 @@ export default class StatusList extends ImmutablePureComponent {
|
||||
contextType={timelineId}
|
||||
scrollKey={this.props.scrollKey}
|
||||
withCounters={this.props.withCounters}
|
||||
{...statusProps}
|
||||
/>
|
||||
);
|
||||
}
|
||||
@@ -90,16 +93,21 @@ export default class StatusList extends ImmutablePureComponent {
|
||||
) : null;
|
||||
|
||||
if (scrollableContent && featuredStatusIds) {
|
||||
scrollableContent = featuredStatusIds.map(statusId => (
|
||||
<StatusQuoteManager
|
||||
key={`f-${statusId}`}
|
||||
id={statusId}
|
||||
featured
|
||||
contextType={timelineId}
|
||||
scrollKey={this.props.scrollKey}
|
||||
withCounters={this.props.withCounters}
|
||||
/>
|
||||
)).concat(scrollableContent);
|
||||
scrollableContent = featuredStatusIds.map(statusId => {
|
||||
if (statusId === TIMELINE_PINNED_VIEW_ALL) {
|
||||
return <PinnedShowAllButton key={TIMELINE_PINNED_VIEW_ALL} />
|
||||
}
|
||||
return (
|
||||
<StatusQuoteManager
|
||||
key={`f-${statusId}`}
|
||||
id={statusId}
|
||||
featured
|
||||
contextType={timelineId}
|
||||
scrollKey={this.props.scrollKey}
|
||||
withCounters={this.props.withCounters}
|
||||
{...statusProps} />
|
||||
);
|
||||
}).concat(scrollableContent);
|
||||
}
|
||||
|
||||
return (
|
||||
|
||||
@@ -14,9 +14,11 @@ import {
|
||||
GroupBadge,
|
||||
MutedBadge,
|
||||
} from '@/flavours/glitch/components/badge';
|
||||
import { Icon } from '@/flavours/glitch/components/icon';
|
||||
import { useAccount } from '@/flavours/glitch/hooks/useAccount';
|
||||
import type { AccountRole } from '@/flavours/glitch/models/account';
|
||||
import { useAppDispatch, useAppSelector } from '@/flavours/glitch/store';
|
||||
import IconPinned from '@/images/icons/icon_pinned.svg?react';
|
||||
|
||||
import { isRedesignEnabled } from '../common';
|
||||
|
||||
@@ -119,6 +121,16 @@ export const AccountBadges: FC<{ accountId: string }> = ({ accountId }) => {
|
||||
return <div className={'account__header__badges'}>{badges}</div>;
|
||||
};
|
||||
|
||||
export const PinnedBadge: FC = () => (
|
||||
<Badge
|
||||
className={classes.badge}
|
||||
icon={<Icon id='pinned' icon={IconPinned} />}
|
||||
label={
|
||||
<FormattedMessage id='account.timeline.pinned' defaultMessage='Pinned' />
|
||||
}
|
||||
/>
|
||||
);
|
||||
|
||||
function isAdminBadge(role: AccountRole) {
|
||||
const name = role.name.toLowerCase();
|
||||
return isRedesignEnabled() && (name === 'admin' || name === 'owner');
|
||||
|
||||
@@ -296,6 +296,11 @@ svg.badgeIcon {
|
||||
text-decoration: none;
|
||||
color: var(--color-text-primary);
|
||||
border-radius: 0;
|
||||
transition: color 0.2s ease-in-out;
|
||||
|
||||
&:not([aria-current='page']):is(:hover, :focus) {
|
||||
color: var(--color-text-brand-soft);
|
||||
}
|
||||
}
|
||||
|
||||
:global(.active) {
|
||||
|
||||
@@ -3,6 +3,7 @@ import type { FC } from 'react';
|
||||
|
||||
import { FormattedMessage } from 'react-intl';
|
||||
|
||||
import classNames from 'classnames';
|
||||
import { useParams } from 'react-router';
|
||||
|
||||
import { List as ImmutableList } from 'immutable';
|
||||
@@ -13,7 +14,6 @@ import {
|
||||
} from '@/flavours/glitch/actions/timelines_typed';
|
||||
import { Column } from '@/flavours/glitch/components/column';
|
||||
import { ColumnBackButton } from '@/flavours/glitch/components/column_back_button';
|
||||
import { FeaturedCarousel } from '@/flavours/glitch/components/featured_carousel';
|
||||
import { LoadingIndicator } from '@/flavours/glitch/components/loading_indicator';
|
||||
import { RemoteHint } from '@/flavours/glitch/components/remote_hint';
|
||||
import StatusList from '@/flavours/glitch/components/status_list';
|
||||
@@ -29,6 +29,12 @@ import { useFilters } from '../hooks/useFilters';
|
||||
|
||||
import { FeaturedTags } from './featured_tags';
|
||||
import { AccountFilters } from './filters';
|
||||
import {
|
||||
PinnedStatusProvider,
|
||||
renderPinnedStatusHeader,
|
||||
usePinnedStatusIds,
|
||||
} from './pinned_statuses';
|
||||
import classes from './styles.module.scss';
|
||||
|
||||
const emptyList = ImmutableList<string>();
|
||||
|
||||
@@ -50,11 +56,13 @@ const AccountTimelineV2: FC<{ multiColumn: boolean }> = ({ multiColumn }) => {
|
||||
|
||||
// Add this key to remount the timeline when accountId changes.
|
||||
return (
|
||||
<InnerTimeline
|
||||
accountId={accountId}
|
||||
key={accountId}
|
||||
multiColumn={multiColumn}
|
||||
/>
|
||||
<PinnedStatusProvider>
|
||||
<InnerTimeline
|
||||
accountId={accountId}
|
||||
key={accountId}
|
||||
multiColumn={multiColumn}
|
||||
/>
|
||||
</PinnedStatusProvider>
|
||||
);
|
||||
};
|
||||
|
||||
@@ -74,11 +82,14 @@ const InnerTimeline: FC<{ accountId: string; multiColumn: boolean }> = ({
|
||||
|
||||
const timeline = useAppSelector((state) => selectTimelineByKey(state, key));
|
||||
const { blockedBy, hidden, suspended } = useAccountVisibility(accountId);
|
||||
const forceEmptyState = blockedBy || hidden || suspended;
|
||||
|
||||
const dispatch = useAppDispatch();
|
||||
useEffect(() => {
|
||||
if (!timeline && !!accountId) {
|
||||
dispatch(expandTimelineByKey({ key }));
|
||||
if (accountId) {
|
||||
if (!timeline) {
|
||||
dispatch(expandTimelineByKey({ key }));
|
||||
}
|
||||
}
|
||||
}, [accountId, dispatch, key, timeline]);
|
||||
|
||||
@@ -91,7 +102,10 @@ const InnerTimeline: FC<{ accountId: string; multiColumn: boolean }> = ({
|
||||
[accountId, dispatch, key],
|
||||
);
|
||||
|
||||
const forceEmptyState = blockedBy || hidden || suspended;
|
||||
const { isLoading: isPinnedLoading, statusIds: pinnedStatusIds } =
|
||||
usePinnedStatusIds({ accountId, tagged, forceEmptyState });
|
||||
|
||||
const isLoading = !!timeline?.isLoading || isPinnedLoading;
|
||||
|
||||
return (
|
||||
<Column bindToDocument={!multiColumn}>
|
||||
@@ -99,25 +113,22 @@ const InnerTimeline: FC<{ accountId: string; multiColumn: boolean }> = ({
|
||||
|
||||
<StatusList
|
||||
alwaysPrepend
|
||||
prepend={
|
||||
<Prepend
|
||||
accountId={accountId}
|
||||
tagged={tagged}
|
||||
forceEmpty={forceEmptyState}
|
||||
/>
|
||||
}
|
||||
prepend={<Prepend accountId={accountId} forceEmpty={forceEmptyState} />}
|
||||
append={<RemoteHint accountId={accountId} />}
|
||||
scrollKey='account_timeline'
|
||||
// We want to have this component when timeline is undefined (loading),
|
||||
// because if we don't the prepended component will re-render with every filter change.
|
||||
statusIds={forceEmptyState ? emptyList : (timeline?.items ?? emptyList)}
|
||||
isLoading={!!timeline?.isLoading}
|
||||
featuredStatusIds={pinnedStatusIds}
|
||||
isLoading={isLoading}
|
||||
hasMore={!forceEmptyState && !!timeline?.hasMore}
|
||||
onLoadMore={handleLoadMore}
|
||||
emptyMessage={<EmptyMessage accountId={accountId} />}
|
||||
bindToDocument={!multiColumn}
|
||||
timelineId='account'
|
||||
withCounters
|
||||
className={classNames(classes.statusWrapper)}
|
||||
statusProps={{ headerRenderFn: renderPinnedStatusHeader }}
|
||||
/>
|
||||
</Column>
|
||||
);
|
||||
@@ -125,9 +136,8 @@ const InnerTimeline: FC<{ accountId: string; multiColumn: boolean }> = ({
|
||||
|
||||
const Prepend: FC<{
|
||||
accountId: string;
|
||||
tagged?: string;
|
||||
forceEmpty: boolean;
|
||||
}> = ({ forceEmpty, accountId, tagged }) => {
|
||||
}> = ({ forceEmpty, accountId }) => {
|
||||
if (forceEmpty) {
|
||||
return <AccountHeader accountId={accountId} hideTabs />;
|
||||
}
|
||||
@@ -137,7 +147,6 @@ const Prepend: FC<{
|
||||
<AccountHeader accountId={accountId} hideTabs />
|
||||
<AccountFilters />
|
||||
<FeaturedTags accountId={accountId} />
|
||||
<FeaturedCarousel accountId={accountId} tagged={tagged} />
|
||||
</>
|
||||
);
|
||||
};
|
||||
|
||||
@@ -0,0 +1,146 @@
|
||||
import type { FC, ReactNode } from 'react';
|
||||
import {
|
||||
createContext,
|
||||
useCallback,
|
||||
useContext,
|
||||
useEffect,
|
||||
useMemo,
|
||||
useState,
|
||||
} from 'react';
|
||||
|
||||
import { FormattedMessage } from 'react-intl';
|
||||
|
||||
import classNames from 'classnames';
|
||||
|
||||
import { TIMELINE_PINNED_VIEW_ALL } from '@/flavours/glitch/actions/timelines';
|
||||
import {
|
||||
expandTimelineByKey,
|
||||
timelineKey,
|
||||
} from '@/flavours/glitch/actions/timelines_typed';
|
||||
import { Button } from '@/flavours/glitch/components/button';
|
||||
import { Icon } from '@/flavours/glitch/components/icon';
|
||||
import { StatusHeader } from '@/flavours/glitch/components/status/header';
|
||||
import type { StatusHeaderRenderFn } from '@/flavours/glitch/components/status/header';
|
||||
import { selectTimelineByKey } from '@/flavours/glitch/selectors/timelines';
|
||||
import { useAppDispatch, useAppSelector } from '@/flavours/glitch/store';
|
||||
import IconPinned from '@/images/icons/icon_pinned.svg?react';
|
||||
|
||||
import { isRedesignEnabled } from '../common';
|
||||
import { PinnedBadge } from '../components/badges';
|
||||
|
||||
import classes from './styles.module.scss';
|
||||
|
||||
const PinnedStatusContext = createContext<{
|
||||
showAllPinned: boolean;
|
||||
onShowAllPinned: () => void;
|
||||
}>({
|
||||
showAllPinned: false,
|
||||
onShowAllPinned: () => {
|
||||
throw new Error('No onShowAllPinned provided');
|
||||
},
|
||||
});
|
||||
|
||||
export const PinnedStatusProvider: FC<{ children: ReactNode }> = ({
|
||||
children,
|
||||
}) => {
|
||||
const [showAllPinned, setShowAllPinned] = useState(false);
|
||||
const handleShowAllPinned = useCallback(() => {
|
||||
setShowAllPinned(true);
|
||||
}, []);
|
||||
|
||||
// Memoize so the context doesn't change every render.
|
||||
const value = useMemo(
|
||||
() => ({
|
||||
showAllPinned,
|
||||
onShowAllPinned: handleShowAllPinned,
|
||||
}),
|
||||
[handleShowAllPinned, showAllPinned],
|
||||
);
|
||||
|
||||
return (
|
||||
<PinnedStatusContext.Provider value={value}>
|
||||
{children}
|
||||
</PinnedStatusContext.Provider>
|
||||
);
|
||||
};
|
||||
|
||||
export function usePinnedStatusIds({
|
||||
accountId,
|
||||
tagged,
|
||||
forceEmptyState = false,
|
||||
}: {
|
||||
accountId: string;
|
||||
tagged?: string;
|
||||
forceEmptyState?: boolean;
|
||||
}) {
|
||||
const pinnedKey = timelineKey({
|
||||
type: 'account',
|
||||
userId: accountId,
|
||||
tagged,
|
||||
pinned: true,
|
||||
});
|
||||
|
||||
const dispatch = useAppDispatch();
|
||||
useEffect(() => {
|
||||
dispatch(expandTimelineByKey({ key: pinnedKey }));
|
||||
}, [dispatch, pinnedKey]);
|
||||
|
||||
const pinnedTimeline = useAppSelector((state) =>
|
||||
selectTimelineByKey(state, pinnedKey),
|
||||
);
|
||||
|
||||
const { showAllPinned } = useContext(PinnedStatusContext);
|
||||
|
||||
const pinnedTimelineItems = pinnedTimeline?.items; // Make a const to avoid the React Compiler complaining.
|
||||
const pinnedStatusIds = useMemo(() => {
|
||||
if (!pinnedTimelineItems || forceEmptyState) {
|
||||
return undefined;
|
||||
}
|
||||
|
||||
if (pinnedTimelineItems.size <= 1 || showAllPinned) {
|
||||
return pinnedTimelineItems;
|
||||
}
|
||||
return pinnedTimelineItems.slice(0, 1).push(TIMELINE_PINNED_VIEW_ALL);
|
||||
}, [forceEmptyState, pinnedTimelineItems, showAllPinned]);
|
||||
|
||||
return {
|
||||
statusIds: pinnedStatusIds,
|
||||
isLoading: !!pinnedTimeline?.isLoading,
|
||||
showAllPinned,
|
||||
};
|
||||
}
|
||||
|
||||
export const renderPinnedStatusHeader: StatusHeaderRenderFn = ({
|
||||
featured,
|
||||
...args
|
||||
}) => {
|
||||
if (!featured) {
|
||||
return <StatusHeader {...args} />;
|
||||
}
|
||||
return (
|
||||
<StatusHeader {...args} className={classes.pinnedStatusHeader}>
|
||||
<PinnedBadge />
|
||||
</StatusHeader>
|
||||
);
|
||||
};
|
||||
|
||||
export const PinnedShowAllButton: FC = () => {
|
||||
const { onShowAllPinned } = useContext(PinnedStatusContext);
|
||||
|
||||
if (!isRedesignEnabled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return (
|
||||
<Button
|
||||
onClick={onShowAllPinned}
|
||||
className={classNames(classes.pinnedViewAllButton, 'focusable')}
|
||||
>
|
||||
<Icon id='pinned' icon={IconPinned} />
|
||||
<FormattedMessage
|
||||
id='account.timeline.pinned.view_all'
|
||||
defaultMessage='View all pinned posts'
|
||||
/>
|
||||
</Button>
|
||||
);
|
||||
};
|
||||
@@ -0,0 +1,52 @@
|
||||
import type { FC } from 'react';
|
||||
|
||||
import { Link } from 'react-router-dom';
|
||||
|
||||
import { RelativeTimestamp } from '@/flavours/glitch/components/relative_timestamp';
|
||||
import type { StatusHeaderProps } from '@/flavours/glitch/components/status/header';
|
||||
import {
|
||||
StatusDisplayName,
|
||||
StatusEditedAt,
|
||||
StatusVisibility,
|
||||
} from '@/flavours/glitch/components/status/header';
|
||||
import type { Account } from '@/flavours/glitch/models/account';
|
||||
|
||||
export const AccountStatusHeader: FC<StatusHeaderProps> = ({
|
||||
status,
|
||||
account,
|
||||
children,
|
||||
avatarSize = 48,
|
||||
wrapperProps,
|
||||
onHeaderClick,
|
||||
}) => {
|
||||
const statusAccount = status.get('account') as Account | undefined;
|
||||
const editedAt = status.get('edited_at') as string;
|
||||
|
||||
return (
|
||||
/* eslint-disable jsx-a11y/no-static-element-interactions, jsx-a11y/click-events-have-key-events */
|
||||
<div
|
||||
onClick={onHeaderClick}
|
||||
onAuxClick={onHeaderClick}
|
||||
{...wrapperProps}
|
||||
className='status__info'
|
||||
/* eslint-enable jsx-a11y/no-static-element-interactions, jsx-a11y/click-events-have-key-events */
|
||||
>
|
||||
<Link
|
||||
to={`/@${statusAccount?.acct}/${status.get('id') as string}`}
|
||||
className='status__relative-time'
|
||||
>
|
||||
<StatusVisibility visibility={status.get('visibility')} />
|
||||
<RelativeTimestamp timestamp={status.get('created_at') as string} />
|
||||
{editedAt && <StatusEditedAt editedAt={editedAt} />}
|
||||
</Link>
|
||||
|
||||
<StatusDisplayName
|
||||
statusAccount={statusAccount}
|
||||
friendAccount={account}
|
||||
avatarSize={avatarSize}
|
||||
/>
|
||||
|
||||
{children}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
@@ -10,6 +10,12 @@
|
||||
font-weight: 500;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
transition: color 0.2s ease-in-out;
|
||||
|
||||
&:hover,
|
||||
&:focus {
|
||||
color: var(--color-text-brand-soft);
|
||||
}
|
||||
}
|
||||
|
||||
.filterSelectIcon {
|
||||
@@ -57,3 +63,57 @@
|
||||
overflow: visible;
|
||||
max-width: none !important;
|
||||
}
|
||||
|
||||
.statusWrapper {
|
||||
:global(.status) {
|
||||
padding-left: 24px;
|
||||
padding-right: 24px;
|
||||
}
|
||||
|
||||
&:has(.pinnedViewAllButton) :global(.status):has(.pinnedStatusHeader) {
|
||||
border-bottom: none;
|
||||
}
|
||||
|
||||
article:has(.pinnedViewAllButton) {
|
||||
border-bottom: 1px solid var(--color-border-primary);
|
||||
}
|
||||
}
|
||||
|
||||
.pinnedViewAllButton {
|
||||
background-color: var(--color-bg-primary);
|
||||
border-radius: 8px;
|
||||
border: 1px solid var(--color-border-primary);
|
||||
box-sizing: border-box;
|
||||
color: var(--color-text-primary);
|
||||
line-height: normal;
|
||||
margin: 12px 24px;
|
||||
padding: 8px;
|
||||
transition: border-color 0.2s ease-in-out;
|
||||
width: calc(100% - 48px);
|
||||
|
||||
&:hover,
|
||||
&:focus {
|
||||
background-color: inherit;
|
||||
border-color: var(--color-bg-brand-base-hover);
|
||||
}
|
||||
}
|
||||
|
||||
.pinnedStatusHeader {
|
||||
display: grid;
|
||||
grid-template-columns: max-content auto;
|
||||
grid-template-rows: 1fr 1fr;
|
||||
gap: 4px;
|
||||
|
||||
> :global(.status__relative-time) {
|
||||
grid-column: 2;
|
||||
height: auto;
|
||||
}
|
||||
|
||||
> :global(.status__display-name) {
|
||||
grid-row: span 2;
|
||||
}
|
||||
|
||||
> :global(.account-role) {
|
||||
justify-self: end;
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user