[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:
Echo
2026-02-06 15:53:34 +01:00
committed by Claire
parent 8caaffe435
commit 69f0c52bb5
11 changed files with 336 additions and 40 deletions

View File

@@ -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 => ({

View File

@@ -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}

View File

@@ -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

View File

@@ -14,6 +14,7 @@ export interface StatusProps {
muted?: boolean;
hidden?: boolean;
unread?: boolean;
featured?: boolean;
showThread?: boolean;
showActions?: boolean;
isQuotedPost?: boolean;

View File

@@ -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 (

View File

@@ -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');

View File

@@ -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) {

View File

@@ -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} />
</>
);
};

View File

@@ -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>
);
};

View File

@@ -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>
);
};

View File

@@ -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;
}
}