2025-01-06 10:32:44 -05:00
|
|
|
import { useMemo } from 'react';
|
2024-05-14 11:00:01 -04:00
|
|
|
import { useRecoilValue } from 'recoil';
|
|
|
|
|
import { useAuthContext, useLocalize } from '~/hooks';
|
2025-01-09 15:40:10 -05:00
|
|
|
import type { TMessageProps, TMessageIcon } from '~/common';
|
🚀 feat: Shared Links (#2772)
* ✨ feat(types): add necessary types for shared link feature
* ✨ feat: add shared links functions to data service
Added functions for retrieving, creating, updating, and deleting shared links and shared messages.
* ✨ feat: Add useGetSharedMessages hook to fetch shared messages by shareId
Adds a new hook `useGetSharedMessages` which fetches shared messages based on the provided shareId.
* ✨ feat: Add share schema and data access functions to API models
* ✨ feat: Add share endpoint to API
The GET /api/share/${shareId} is exposed to the public, so authentication is not required. Other paths require authentication.
* ♻️ refactor(utils): generalize react-query cache manipulation functions
Introduces generic functions for manipulating react-query cache entries, marking a refinement in how query cache data is managed. It aims to enhance the flexibility and reusability of the cache interaction patterns within our application.
- Replaced specific index names with more generic terms in queries.ts, enhancing consistency across data handling functions.
- Introduced new utility functions in collection.ts for adding, updating, and deleting data entries in an InfiniteData<TCollection>. These utility functions (`addData`, `updateData`, `deleteData`, `findPage`) are designed to be re-usable across different data types and collections.
- Adapted existing conversation utility functions in convos.ts to leverage these new generic utilities.
* ✨ feat(shared-link): add functions to manipulate shared link cache list
implemented new utility functions to handle additions, updates, and deletions in the shared link cache list.
* ✨ feat: Add mutations and queries for shared links
* ✨ feat(shared-link): add `Share` button to conversation list
- Added a share button in each conversation in the conversation list.
- Implemented functionality where clicking the share button triggers a POST request to the API.
- The API checks if a share link was already created for the conversation today; if so, it returns the existing link.
- If no link was created for today, the API will create a new share link and return it.
- Each click on the share button results in a new API request, following the specification similar to ChatGPT's share link feature.
* ♻️ refactor(hooks): generalize useNavScrolling for broader use
- Modified `useNavScrolling` to accept a generic type parameter `TData`, allowing it to be used with different data structures besides `ConversationListResponse`.
- Updated instances in `Nav.tsx` and `ArchivedChatsTable.tsx` to explicitly specify `ConversationListResponse` as the type argument when invoking `useNavScrolling`.
* ✨ feat(settings): add shared links listing table with delete functionality in settings
- Integrated a delete button for each shared link in the table, allowing users to remove links as needed.
* ♻️ refactor(components): separate `EndpointIcon` from `Icon` component for standalone use
* ♻️ refactor: update useGetSharedMessages to return TSharedLink
- Modified the useGetSharedMessages hook to return not only a list of TMessage but also the TSharedLink itself.
- This change was necessary to support displaying the title and date in the Shared Message UI, which requires data from TSharedLink.
* ✨ feat(shared link): add UI for displaying shared conversations without authentication
- Implemented a new UI component to display shared conversations, designed to be accessible without requiring authentication.
- Reused components from the authenticated Messages module where possible. Copied and adapted components that could not be directly reused to fit the non-authenticated context.
* 🔧 chore: Add translations
Translate labels only. Messages remain in English as they are possibly subject to change.
* ♻️ refactor: add icon and tooltip props to EditMenuButton component
* moved icon and popover to arguments so that EditMenuButton can be reused.
* modified so that when a ShareButton is closed, the parent DropdownMenu is also closed.
* ♻️irefactor: added DropdownMenu for Export and Share
* ♻️ refactor: renamed component names more intuitive
* More accurate naming of the dropdown menu.
* When the export button is closed, the parent dropdown menu is also closed.
* 🌍 chore: updated translations
* 🐞 Fix: OpenID Profile Image Download (#2757)
* Add fetch requirement
Fixes - error: [openidStrategy] downloadImage: Error downloading image at URL "https://graph.microsoft.com/v1.0/me/photo/$value": TypeError: response.buffer is not a function
* Update openidStrategy.js
---------
Co-authored-by: Danny Avila <danacordially@gmail.com>
* 🚑 fix(export): Issue exporting Conversation with Assistants (#2769)
* 🚑 fix(export): use content as text if content is present in the message
If the endpoint is assistants, the text of the message goes into content, not message.text.
* refactor(ExportModel): TypeScript, remove unused code
---------
Co-authored-by: Yuichi Ohneda <ohneda@gmail.com>
* 📤style: export button icon (#2752)
* refactor(ShareDialog): logic and styling
* refactor(ExportAndShareMenu): imports order and icon update
* chore: imports
* chore: imports/render logic
* feat: message branching
* refactor: add optional config to useGetStartupConfig
* refactor: disable endpoints query
* chore: fix search view styling gradient in light mode
* style: ShareView gradient styling
* refactor(Share): use select queries
* style: shared link table buttons
* localization and dark text styling
* style: fix clipboard button layout shift app-wide and add localization for copy code
* support assistants message content in shared links, add useCopyToClipboard, add copy buttons to Search Messages and Shared Link Messages
* add localizations
* comparisons
---------
Co-authored-by: Yuichi Ohneda <ohneda@gmail.com>
Co-authored-by: bsu3338 <bsu3338@users.noreply.github.com>
Co-authored-by: Fuegovic <32828263+fuegovic@users.noreply.github.com>
2024-05-17 18:13:32 -04:00
|
|
|
import MinimalHoverButtons from '~/components/Chat/Messages/MinimalHoverButtons';
|
2024-05-14 11:00:01 -04:00
|
|
|
import Icon from '~/components/Chat/Messages/MessageIcon';
|
|
|
|
|
import SearchContent from './Content/SearchContent';
|
|
|
|
|
import SearchButtons from './SearchButtons';
|
|
|
|
|
import SubRow from './SubRow';
|
|
|
|
|
import { cn } from '~/utils';
|
|
|
|
|
import store from '~/store';
|
|
|
|
|
|
2025-04-15 10:04:00 +02:00
|
|
|
const MessageAvatar = ({ iconData }: { iconData: TMessageIcon }) => (
|
|
|
|
|
<div className="relative flex flex-shrink-0 flex-col items-end">
|
|
|
|
|
<div className="pt-0.5">
|
|
|
|
|
<div className="flex h-6 w-6 items-center justify-center overflow-hidden rounded-full">
|
|
|
|
|
<Icon iconData={iconData} />
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const MessageBody = ({ message, messageLabel, fontSize }) => (
|
|
|
|
|
<div
|
|
|
|
|
className={cn('relative flex w-11/12 flex-col', message.isCreatedByUser ? '' : 'agent-turn')}
|
|
|
|
|
>
|
|
|
|
|
<div className={cn('select-none font-semibold', fontSize)}>{messageLabel}</div>
|
|
|
|
|
<SearchContent message={message} />
|
|
|
|
|
<SubRow classes="text-xs">
|
|
|
|
|
<MinimalHoverButtons message={message} />
|
|
|
|
|
<SearchButtons message={message} />
|
|
|
|
|
</SubRow>
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
export default function SearchMessage({ message }: Pick<TMessageProps, 'message'>) {
|
2024-05-14 11:00:01 -04:00
|
|
|
const UsernameDisplay = useRecoilValue<boolean>(store.UsernameDisplay);
|
2024-08-07 14:23:33 -04:00
|
|
|
const fontSize = useRecoilValue(store.fontSize);
|
2024-05-14 11:00:01 -04:00
|
|
|
const { user } = useAuthContext();
|
|
|
|
|
const localize = useLocalize();
|
|
|
|
|
|
2025-01-09 15:40:10 -05:00
|
|
|
const iconData: TMessageIcon = useMemo(
|
|
|
|
|
() => ({
|
2025-04-15 10:04:00 +02:00
|
|
|
endpoint: message?.endpoint ?? '',
|
|
|
|
|
model: message?.model ?? '',
|
2025-01-09 15:40:10 -05:00
|
|
|
iconURL: message?.iconURL ?? '',
|
2025-04-15 10:04:00 +02:00
|
|
|
isCreatedByUser: message?.isCreatedByUser ?? false,
|
2025-01-09 15:40:10 -05:00
|
|
|
}),
|
2025-04-15 10:04:00 +02:00
|
|
|
[message?.endpoint, message?.model, message?.iconURL, message?.isCreatedByUser],
|
2025-01-06 10:32:44 -05:00
|
|
|
);
|
|
|
|
|
|
2025-04-15 10:04:00 +02:00
|
|
|
const messageLabel = useMemo(() => {
|
|
|
|
|
if (message?.isCreatedByUser) {
|
|
|
|
|
return UsernameDisplay
|
|
|
|
|
? (user?.name ?? '') || (user?.username ?? '')
|
|
|
|
|
: localize('com_user_message');
|
|
|
|
|
}
|
|
|
|
|
return message?.sender ?? '';
|
|
|
|
|
}, [
|
|
|
|
|
message?.isCreatedByUser,
|
|
|
|
|
message?.sender,
|
|
|
|
|
UsernameDisplay,
|
|
|
|
|
user?.name,
|
|
|
|
|
user?.username,
|
|
|
|
|
localize,
|
|
|
|
|
]);
|
|
|
|
|
|
2024-05-14 11:00:01 -04:00
|
|
|
if (!message) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (
|
2025-04-15 10:04:00 +02:00
|
|
|
<div className="text-token-text-primary w-full bg-transparent">
|
|
|
|
|
<div className="m-auto p-4 py-2 md:gap-6">
|
|
|
|
|
<div className="final-completion group mx-auto flex flex-1 gap-3 md:max-w-3xl md:px-5 lg:max-w-[40rem] lg:px-1 xl:max-w-[48rem] xl:px-5">
|
|
|
|
|
<MessageAvatar iconData={iconData} />
|
|
|
|
|
<MessageBody message={message} messageLabel={messageLabel} fontSize={fontSize} />
|
2024-05-14 11:00:01 -04:00
|
|
|
</div>
|
|
|
|
|
</div>
|
2025-04-15 10:04:00 +02:00
|
|
|
</div>
|
2024-05-14 11:00:01 -04:00
|
|
|
);
|
|
|
|
|
}
|