From 7eee8fe4b5797c0318b6e4b97143cda06c6fe895 Mon Sep 17 00:00:00 2001 From: dudu0506 Date: Fri, 10 Jan 2025 10:30:07 +0800 Subject: [PATCH 1/3] feat: show more on reply content --- src/components/Compose/DraftList.tsx | 5 +- src/components/Compose/Editor.tsx | 4 +- src/components/Posts/PostBodyContent.tsx | 46 +----------- src/components/Posts/PostBodyReplyContent.tsx | 74 +++++++++++++++++++ src/store/useComposeStore.ts | 11 +++ 5 files changed, 95 insertions(+), 45 deletions(-) create mode 100644 src/components/Posts/PostBodyReplyContent.tsx diff --git a/src/components/Compose/DraftList.tsx b/src/components/Compose/DraftList.tsx index ab9b4d57a9..c6926518ca 100644 --- a/src/components/Compose/DraftList.tsx +++ b/src/components/Compose/DraftList.tsx @@ -144,7 +144,7 @@ const DraftListItem = memo(function DraftListItem({ draft, h export const DraftList = memo(function DraftList() { const currentProfileAll = useCurrentProfileAll(); const { drafts, removeDraft } = useComposeDraftStateStore(); - const { updateChars, apply, currentDraftId, clear } = useComposeStateStore(); + const { updateChars, apply, focused, currentDraftId, clear } = useComposeStateStore(); const { updateScheduleTime } = useComposeScheduleStateStore(); const setEditorContent = useSetEditorContent(); @@ -191,6 +191,7 @@ export const DraftList = memo(function DraftList() { ); apply({ ...draft, + focused, posts: draft.posts.map((x) => ({ ...x, ...(full @@ -211,7 +212,7 @@ export const DraftList = memo(function DraftList() { if (draft.scheduleTime) updateScheduleTime(draft.scheduleTime); router.history.push('/'); }, - [apply, router, setEditorContent, updateChars, updateScheduleTime, currentProfileAll], + [apply, router, setEditorContent, updateChars, updateScheduleTime, currentProfileAll, focused], ); if (!drafts.length) { diff --git a/src/components/Compose/Editor.tsx b/src/components/Compose/Editor.tsx index 5edafefaf3..d4c64452d9 100644 --- a/src/components/Compose/Editor.tsx +++ b/src/components/Compose/Editor.tsx @@ -47,7 +47,7 @@ interface EditorProps { } export const Editor = memo(function Editor({ post, replying }: EditorProps) { - const { type, posts, updateChars, loadComponentsFromChars } = useComposeStateStore(); + const { type, posts, updateChars, updateFocused, loadComponentsFromChars } = useComposeStateStore(); const [, startTransition] = useTransition(); const { chars } = post; @@ -103,6 +103,8 @@ export const Editor = memo(function Editor({ post, replying }: EditorProps) { contentEditable={ updateFocused(true)} + onBlur={() => updateFocused(false)} className="flex-1 flex-shrink-0 cursor-text resize-none appearance-none border-none bg-transparent p-0 pb-2 text-left text-[16px] leading-6 text-main outline-none outline-0 focus:ring-0" /> } diff --git a/src/components/Posts/PostBodyContent.tsx b/src/components/Posts/PostBodyContent.tsx index 680185963a..5557ea2571 100644 --- a/src/components/Posts/PostBodyContent.tsx +++ b/src/components/Posts/PostBodyContent.tsx @@ -1,11 +1,10 @@ 'use client'; -import { t } from '@lingui/core/macro'; -import { Select, Trans } from '@lingui/react/macro'; +import { Trans } from '@lingui/react/macro'; import { useForkRef } from '@mui/material'; import { compact } from 'lodash-es'; import { usePathname, useRouter } from 'next/navigation.js'; -import { forwardRef, type HTMLProps, useMemo, useState } from 'react'; +import { forwardRef, useMemo, useState } from 'react'; import { useAsync } from 'react-use'; import { TwitterArticleBody } from '@/components/Article/TwitterArticleBody.js'; @@ -17,6 +16,7 @@ import { PollCard } from '@/components/Poll/PollCard.js'; import { Attachments } from '@/components/Posts/Attachment.js'; import { CollapsedContent } from '@/components/Posts/CollapsedContent.js'; import { ContentTranslator } from '@/components/Posts/ContentTranslator.js'; +import { PostBodyReplyContent } from '@/components/Posts/PostBodyReplyContent.js'; import { PostLinks } from '@/components/Posts/PostLinks.js'; import { Quote } from '@/components/Posts/Quote.js'; import { RedPacketInspector } from '@/components/RedPacket/RedPacketInspector.js'; @@ -24,11 +24,9 @@ import { IS_APPLE, IS_SAFARI } from '@/constants/bowser.js'; import { PageRoute, Source } from '@/constants/enum.js'; import { EMPTY_LIST, RP_HASH_TAG } from '@/constants/index.js'; import { classNames } from '@/helpers/classNames.js'; -import { formatUrl } from '@/helpers/formatUrl.js'; import { getEncryptedPayloadFromImageAttachment, getEncryptedPayloadFromText } from '@/helpers/getEncryptedPayload.js'; import { getPostUrl } from '@/helpers/getPostUrl.js'; import { isRoutePathname } from '@/helpers/isRoutePathname.js'; -import { isValidUrl } from '@/helpers/isValidUrl.js'; import { resolveOembedUrl } from '@/helpers/resolveOembedUrl.js'; import { resolvePostArticleUrl } from '@/helpers/resolvePostArticleUrl.js'; import { trimify } from '@/helpers/trimify.js'; @@ -50,12 +48,6 @@ export interface PostBodyContentProps { showTranslate?: boolean; } -const overrideComponents = { - a: function Anchor({ title }: HTMLProps) { - return {title && isValidUrl(title) ? formatUrl(title, 30) : title}; - }, -}; - function canSkipWaitingForPayload(post: Post) { const content = post.metadata.content?.content; @@ -181,37 +173,7 @@ export const PostBodyContent = forwardRef( ); } - if (isReply) { - return ( -
- - {liteRawContent} - -
- {post.metadata.content?.asset?.type ? ( - + ) : null} + {post.quoteOn ? {t`[Quote]`} : null} +
+
+ ); +}); diff --git a/src/store/useComposeStore.ts b/src/store/useComposeStore.ts index 27b634bb60..07a024b0bd 100644 --- a/src/store/useComposeStore.ts +++ b/src/store/useComposeStore.ts @@ -80,6 +80,9 @@ export interface ComposeBaseState { cursor: Cursor; // tracking the current applied draft id currentDraftId?: string; + + // editor focus state + focused: boolean; } interface ComposeState extends ComposeBaseState { @@ -134,6 +137,8 @@ interface ComposeState extends ComposeBaseState { // reset the editor apply: (state: ComposeBaseState) => void; clear: () => void; + + updateFocused: (focused: boolean) => void; } export function createInitPostState(): Record { @@ -183,6 +188,7 @@ const useComposeStateBase = create( immer((set, get) => ({ type: 'compose', cursor: initialPostCursor, + focused: false, posts: [createInitSinglePostState(initialPostCursor)], computed: { get nextAvailablePost() { @@ -623,11 +629,16 @@ const useComposeStateBase = create( type: state.type, cursor: id, currentDraftId: undefined, + focused: false, posts: [createInitSinglePostState(id)], } satisfies ComposeBaseState; Object.assign(state, nextState); }), + updateFocused: (focused) => + set((state) => { + state.focused = focused; + }), })), ); From 2bd691fe27404d421b9e6398704b1ff68e78e1fa Mon Sep 17 00:00:00 2001 From: guanbinrui Date: Mon, 13 Jan 2025 10:35:37 +0800 Subject: [PATCH 2/3] refactor: performance --- src/components/Posts/PostBodyReplyContent.tsx | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/src/components/Posts/PostBodyReplyContent.tsx b/src/components/Posts/PostBodyReplyContent.tsx index 84aa24d329..d6bf9565d6 100644 --- a/src/components/Posts/PostBodyReplyContent.tsx +++ b/src/components/Posts/PostBodyReplyContent.tsx @@ -28,10 +28,6 @@ export const PostBodyReplyContent = memo(function Pos const [multiple, setMultiple] = useState(1); const { focused } = useComposeStateStore(); - const toggleShowMore = useCallback(() => { - setMultiple((prev) => (overflow ? prev + 1 : 1)); - }, [overflow]); - useEffect(() => { if (focused) { setMultiple(1); @@ -53,7 +49,10 @@ export const PostBodyReplyContent = memo(function Pos
{overflow || multiple > 1 ? ( - + setMultiple((prev) => (overflow ? prev + 1 : 1))} + > {overflow ? Show more : Show less} ) : null} From 4269a60846402e5612b082a7e2ba805563652b66 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" Date: Mon, 13 Jan 2025 02:37:52 +0000 Subject: [PATCH 3/3] chore: code maintenance [bot] --- src/components/Posts/PostBodyReplyContent.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/components/Posts/PostBodyReplyContent.tsx b/src/components/Posts/PostBodyReplyContent.tsx index d6bf9565d6..a3e5b3ea42 100644 --- a/src/components/Posts/PostBodyReplyContent.tsx +++ b/src/components/Posts/PostBodyReplyContent.tsx @@ -1,6 +1,6 @@ import { t } from '@lingui/core/macro'; import { Select, Trans } from '@lingui/react/macro'; -import { type HTMLProps, memo, useCallback, useEffect, useState } from 'react'; +import { type HTMLProps, memo, useEffect, useState } from 'react'; import { ClickableButton } from '@/components/ClickableButton.js'; import { NakedMarkup } from '@/components/Markup/NakedMarkup.js';