116 lines
6.7 KiB
HTML
116 lines
6.7 KiB
HTML
<!-- Desktop Chat Section -->
|
|
<div class="border-t border-[var(--border-primary)] flex flex-col" :class="{'flex-1 overflow-hidden': isChatMaximized}">
|
|
<!-- Chat Toggle -->
|
|
<div class="bg-[var(--bg-tertiary)] px-4 py-3 hover:bg-[var(--bg-accent-hover)] transition-colors flex items-center">
|
|
<button @click="() => { showChat = !showChat; if (!showChat) isChatMaximized = false; }"
|
|
class="flex-1 text-left flex items-center">
|
|
<span class="font-medium flex items-center">
|
|
<i class="fas fa-comments mr-2"></i>
|
|
${ t('chat.chatWithTranscription') }
|
|
</span>
|
|
</button>
|
|
<div class="flex items-center gap-2">
|
|
<button v-if="chatMessages.length > 0"
|
|
@click="downloadChat"
|
|
class="p-1 text-[var(--text-muted)] hover:text-[var(--text-primary)] transition-colors"
|
|
:title="t('buttons.downloadChat')">
|
|
<i class="fas fa-download text-sm"></i>
|
|
</button>
|
|
<button @click="toggleChatMaximize"
|
|
class="p-1 text-[var(--text-muted)] hover:text-[var(--text-primary)] transition-colors"
|
|
:title="isChatMaximized ? t('tooltips.restoreChat') : t('tooltips.maximizeChat')">
|
|
<i :class="['fas text-sm', isChatMaximized ? 'fa-compress' : 'fa-expand']"></i>
|
|
</button>
|
|
<button @click="() => { showChat = !showChat; if (!showChat) isChatMaximized = false; }"
|
|
class="p-1 text-[var(--text-muted)] hover:text-[var(--text-primary)]">
|
|
<i :class="['fas transition-transform duration-200', showChat ? 'fa-chevron-down' : 'fa-chevron-up']"></i>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Chat Content -->
|
|
<div v-if="showChat" class="flex flex-col relative" :class="isChatMaximized ? 'flex-1 overflow-hidden' : 'max-h-96'">
|
|
<!-- Clear button - fixed at top right -->
|
|
<button v-if="chatMessages.length > 0"
|
|
@click="clearChat"
|
|
class="absolute top-2 right-2 p-1 text-[var(--text-muted)] opacity-60 hover:opacity-100 hover:text-[var(--text-danger)] rounded transition-all duration-200 z-20"
|
|
title="Clear chat">
|
|
<i class="fas fa-times text-xs"></i>
|
|
</button>
|
|
|
|
<!-- Chat Messages -->
|
|
<div ref="chatMessagesRef"
|
|
class="flex-1 overflow-y-auto p-4 space-y-4 bg-[var(--bg-secondary)]">
|
|
<div v-if="chatMessages.length === 0" class="text-center py-8">
|
|
<i class="fas fa-robot text-3xl text-[var(--text-muted)] mb-3"></i>
|
|
<p class="text-[var(--text-muted)]" v-text="t('help.askAboutTranscription')"></p>
|
|
</div>
|
|
|
|
<div v-for="(message, index) in chatMessages"
|
|
:key="index"
|
|
:class="[
|
|
'message relative group',
|
|
message.role === 'user' ? 'user-message ml-auto' : 'ai-message',
|
|
message.role === 'assistant' ? 'pr-10' : ''
|
|
]">
|
|
<!-- Copy button for assistant messages -->
|
|
<button v-if="message.role === 'assistant'"
|
|
@click="copyMessage(message.content, $event)"
|
|
class="absolute top-2 right-2 p-1 text-[var(--text-muted)] hover:text-[var(--text-primary)] hover:bg-[var(--bg-tertiary)] rounded transition-all duration-200"
|
|
:title="t('buttons.copyMessage')">
|
|
<i class="fas fa-copy text-sm"></i>
|
|
</button>
|
|
<!-- Show thinking content if available -->
|
|
<div v-if="message.thinking && message.role === 'assistant'" class="mb-2">
|
|
<button @click="message.thinkingExpanded = !message.thinkingExpanded"
|
|
class="text-xs text-[var(--text-muted)] hover:text-[var(--text-secondary)] flex items-center gap-1">
|
|
<i :class="['fas', message.thinkingExpanded ? 'fa-chevron-down' : 'fa-chevron-right']"></i>
|
|
<span v-text="t('help.modelReasoning')"></span>
|
|
<span class="text-[var(--text-muted)]">(${message.thinking.split('\n').length} lines)</span>
|
|
</button>
|
|
<div v-if="message.thinkingExpanded"
|
|
class="mt-2 p-3 bg-[var(--bg-tertiary)] border border-[var(--border-secondary)] rounded-lg text-xs text-[var(--text-muted)] max-h-64 overflow-y-auto">
|
|
<pre class="whitespace-pre-wrap font-mono">${message.thinking}</pre>
|
|
</div>
|
|
</div>
|
|
<!-- Main message content -->
|
|
<div v-if="message.html" v-html="message.html"></div>
|
|
<div v-else class="whitespace-pre-wrap">${message.content}</div>
|
|
</div>
|
|
|
|
<div v-if="isChatLoading" class="ai-message">
|
|
<i class="fas fa-spinner fa-spin mr-2"></i>
|
|
${ t('chat.thinking') }
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Chat Input -->
|
|
<div class="border-t border-[var(--border-primary)] p-4 bg-[var(--bg-tertiary)]">
|
|
<div class="flex gap-2">
|
|
<textarea v-model="chatInput"
|
|
ref="chatInputRef"
|
|
@keydown="handleChatKeydown"
|
|
:disabled="selectedRecording.status !== 'COMPLETED' || processedTranscription.isError"
|
|
:placeholder="t('chat.placeholderWithHint')"
|
|
class="flex-1 px-3 py-2 bg-[var(--bg-input)] border border-[var(--border-secondary)] rounded-lg resize-none focus:outline-none focus:ring-2 focus:ring-[var(--ring-focus)] text-sm"
|
|
rows="2">
|
|
</textarea>
|
|
<button @click="sendChatMessage"
|
|
:disabled="!chatInput.trim() || isChatLoading || selectedRecording.status !== 'COMPLETED' || processedTranscription.isError"
|
|
class="px-4 py-2 bg-[var(--bg-button)] text-[var(--text-button)] rounded-lg hover:bg-[var(--bg-button-hover)] transition-colors disabled:opacity-50 disabled:cursor-not-allowed">
|
|
<i class="fas fa-paper-plane"></i>
|
|
</button>
|
|
</div>
|
|
<p v-if="processedTranscription.isError"
|
|
class="text-xs text-amber-500 mt-2">
|
|
<i class="fas fa-exclamation-triangle mr-1"></i>
|
|
${ t('chat.cannotChatTranscriptionFailed') }
|
|
</p>
|
|
<p v-else-if="selectedRecording.status !== 'COMPLETED'"
|
|
class="text-xs text-[var(--text-muted)] mt-2">
|
|
${ t('chat.availableAfterTranscription') }
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|