@@ -18,7 +18,8 @@ import { useTTS } from "@/hooks/useTTS"
18
18
import { tagColors } from "@/utils/color"
19
19
import { removeModelSuffix } from "@/db/models"
20
20
import { GenerationInfo } from "./GenerationInfo"
21
-
21
+ import { parseReasoning } from "@/libs/reasoning"
22
+ import { humanizeMilliseconds } from "@/utils/humanize-miliseconds"
22
23
type Props = {
23
24
message : string
24
25
message_type ?: string
@@ -40,6 +41,8 @@ type Props = {
40
41
onSourceClick ?: ( source : any ) => void
41
42
isTTSEnabled ?: boolean
42
43
generationInfo ?: any
44
+ isStreaming : boolean
45
+ reasoningTimeTaken ?: number
43
46
}
44
47
45
48
export const PlaygroundMessage = ( props : Props ) => {
@@ -48,7 +51,6 @@ export const PlaygroundMessage = (props: Props) => {
48
51
49
52
const { t } = useTranslation ( "common" )
50
53
const { cancel, isSpeaking, speak } = useTTS ( )
51
-
52
54
return (
53
55
< div className = "group w-full text-gray-800 dark:text-gray-100" >
54
56
< div className = "text-base md:max-w-2xl lg:max-w-xl xl:max-w-3xl flex lg:px-0 m-auto w-full" >
@@ -94,7 +96,40 @@ export const PlaygroundMessage = (props: Props) => {
94
96
< div className = "flex flex-grow flex-col" >
95
97
{ ! editMode ? (
96
98
props . isBot ? (
97
- < Markdown message = { props . message } />
99
+ < >
100
+ { parseReasoning ( props . message ) . map ( ( e , i ) => {
101
+ if ( e . type === "reasoning" ) {
102
+ return (
103
+ < Collapse
104
+ key = { i }
105
+ className = "border-none !mb-3"
106
+ items = { [
107
+ {
108
+ key : "reasoning" ,
109
+ label :
110
+ props . isStreaming && e ?. reasoning_running ? (
111
+ < div className = "flex items-center gap-2" >
112
+ < span className = "italic" >
113
+ { t ( "reasoning.thinking" ) }
114
+ </ span >
115
+ </ div >
116
+ ) : (
117
+ t ( "reasoning.thought" , {
118
+ time : humanizeMilliseconds (
119
+ props . reasoningTimeTaken
120
+ )
121
+ } )
122
+ ) ,
123
+ children : < Markdown message = { e . content } />
124
+ }
125
+ ] }
126
+ />
127
+ )
128
+ }
129
+
130
+ return < Markdown key = { i } message = { e . content } />
131
+ } ) }
132
+ </ >
98
133
) : (
99
134
< p
100
135
className = { `prose dark:prose-invert whitespace-pre-line prose-p:leading-relaxed prose-pre:p-0 dark:prose-dark ${
@@ -220,8 +255,8 @@ export const PlaygroundMessage = (props: Props) => {
220
255
}
221
256
title = { t ( "generationInfo" ) } >
222
257
< button
223
- aria-label = { t ( "generationInfo" ) }
224
- className = "flex items-center justify-center w-6 h-6 rounded-full bg-gray-100 dark:bg-gray-800 hover:bg-gray-200 dark:hover:bg-gray-700 transition-colors duration-200 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500" >
258
+ aria-label = { t ( "generationInfo" ) }
259
+ className = "flex items-center justify-center w-6 h-6 rounded-full bg-gray-100 dark:bg-gray-800 hover:bg-gray-200 dark:hover:bg-gray-700 transition-colors duration-200 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500" >
225
260
< InfoIcon className = "w-3 h-3 text-gray-400 group-hover:text-gray-500" />
226
261
</ button >
227
262
</ Popover >
@@ -231,7 +266,7 @@ export const PlaygroundMessage = (props: Props) => {
231
266
props . currentMessageIndex === props . totalMessages - 1 && (
232
267
< Tooltip title = { t ( "regenerate" ) } >
233
268
< button
234
- aria-label = { t ( "regenerate" ) }
269
+ aria-label = { t ( "regenerate" ) }
235
270
onClick = { props . onRengerate }
236
271
className = "flex items-center justify-center w-6 h-6 rounded-full bg-gray-100 dark:bg-gray-800 hover:bg-gray-200 dark:hover:bg-gray-700 transition-colors duration-200 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500" >
237
272
< RotateCcw className = "w-3 h-3 text-gray-400 group-hover:text-gray-500" />
0 commit comments