mirror of
https://github.com/docker/docs.git
synced 2026-06-19 07:35:16 +00:00
ee71c80562
Replaces @material-symbols/svg-400 (5.5 MB) with heroicons (488 KB) — an 11x reduction in icon asset size. Heroicons uses a single consistent distribution format (24px solid SVGs, fill="currentColor") eliminating the need for the separate utils/svg.html partial, the icon-svg-stroke CSS utility, and the dual-path resolution logic in icon.html. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> Signed-off-by: David Karlsson <35727626+dvdksn@users.noreply.github.com>
766 lines
28 KiB
HTML
766 lines
28 KiB
HTML
<!-- Gordon AI Chat Panel -->
|
|
<script>
|
|
window.GORDON_BASE_URL = {{ if eq hugo.Environment "production" -}}
|
|
'https://ai-backend-service.docker.com'
|
|
{{- else if getenv "HUGO_GORDON_URL" -}}
|
|
'{{ getenv "HUGO_GORDON_URL" }}'
|
|
{{- else -}}
|
|
'https://ai-backend-service-stage.docker.com'
|
|
{{- end }};
|
|
</script>
|
|
<div
|
|
x-data="{
|
|
isLoading: false,
|
|
error: null,
|
|
messages: $persist([]).using(sessionStorage).as('gordon-messages'),
|
|
currentQuestion: '',
|
|
threadId: $persist(null).using(sessionStorage).as('gordon-threadId'),
|
|
includePageContext: $persist(true).using(sessionStorage).as('gordon-includePageContext'),
|
|
maxTurnsPerThread: 10,
|
|
|
|
init() {
|
|
// Clean up any streaming messages that might be persisted
|
|
this.messages = this.messages.filter(m => !m.isStreaming)
|
|
|
|
// Watch for store changes to focus input
|
|
this.$watch('$store.gordon.isOpen', (isOpen) => {
|
|
if (isOpen) {
|
|
this.$nextTick(() => {
|
|
this.$refs.input?.focus()
|
|
})
|
|
}
|
|
})
|
|
|
|
// Watch for query from store and populate input
|
|
this.$watch('$store.gordon.query', (query) => {
|
|
if (query) {
|
|
this.currentQuestion = query
|
|
const shouldAutoSubmit = this.$store.gordon.autoSubmit
|
|
this.$nextTick(() => {
|
|
if (shouldAutoSubmit) {
|
|
this.askQuestion()
|
|
} else {
|
|
this.$refs.input?.focus()
|
|
this.$refs.input?.select()
|
|
}
|
|
})
|
|
// Clear the store query and autoSubmit flag after using them
|
|
this.$store.gordon.query = ''
|
|
this.$store.gordon.autoSubmit = false
|
|
}
|
|
})
|
|
},
|
|
|
|
getTurnCount() {
|
|
return this.messages.filter(m => m.role === 'user').length
|
|
},
|
|
|
|
getRemainingTurns() {
|
|
return this.maxTurnsPerThread - this.getTurnCount()
|
|
},
|
|
|
|
isThreadLimitReached() {
|
|
return this.getTurnCount() >= this.maxTurnsPerThread
|
|
},
|
|
|
|
shouldShowCountdown() {
|
|
const remaining = this.getRemainingTurns()
|
|
return remaining > 0 && remaining <= 3
|
|
},
|
|
|
|
async askQuestion() {
|
|
const question = this.currentQuestion.trim()
|
|
if (!question || this.isLoading || this.isThreadLimitReached()) {
|
|
return
|
|
}
|
|
|
|
// Add user message to UI
|
|
this.messages.push({
|
|
role: 'user',
|
|
content: question
|
|
})
|
|
|
|
this.currentQuestion = ''
|
|
// Reset textarea height
|
|
this.$nextTick(() => {
|
|
if (this.$refs.input) {
|
|
this.$refs.input.style.height = 'auto'
|
|
}
|
|
})
|
|
this.isLoading = true
|
|
this.error = null
|
|
|
|
// Add placeholder for assistant response
|
|
const responseIndex = this.messages.length
|
|
this.messages.push({
|
|
role: 'assistant',
|
|
content: '',
|
|
isStreaming: true,
|
|
questionAnswerId: null,
|
|
feedback: null,
|
|
copied: false
|
|
})
|
|
|
|
this.$nextTick(() => {
|
|
this.$refs.messagesContainer?.scrollTo({
|
|
top: this.$refs.messagesContainer.scrollHeight,
|
|
behavior: 'smooth'
|
|
})
|
|
})
|
|
|
|
try {
|
|
await this.streamGordonResponse(responseIndex)
|
|
} catch (err) {
|
|
// Only set error if messages weren't cleared
|
|
if (this.messages.length > 0) {
|
|
if (err.message === 'RATE_LIMIT_EXCEEDED') {
|
|
this.error = 'You\'ve exceeded your question quota for the day. Please come back tomorrow.'
|
|
} else {
|
|
this.error = 'Failed to get response. Please try again.'
|
|
}
|
|
}
|
|
console.error('Gordon API error:', err)
|
|
// Only try to remove message if it still exists
|
|
if (this.messages[responseIndex]) {
|
|
this.messages.splice(responseIndex, 1)
|
|
}
|
|
} finally {
|
|
this.isLoading = false
|
|
}
|
|
},
|
|
|
|
getSessionId() {
|
|
let sessionId = sessionStorage.getItem('gordon-session-id')
|
|
if (!sessionId) {
|
|
sessionId = `docs-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`
|
|
sessionStorage.setItem('gordon-session-id', sessionId)
|
|
}
|
|
return sessionId
|
|
},
|
|
|
|
async streamGordonResponse(responseIndex) {
|
|
|
|
// Build API request from messages, excluding the streaming placeholder
|
|
// The placeholder is at responseIndex, so we take everything before it
|
|
const conversationMessages = this.messages.slice(0, responseIndex).map((msg, i) => {
|
|
const message = {
|
|
role: msg.role,
|
|
content: msg.content
|
|
}
|
|
|
|
// Add copilot_references to the last message (most recent user question)
|
|
if (i === responseIndex - 1) {
|
|
message.copilot_references = [
|
|
{
|
|
data: {
|
|
origin: 'docs-website',
|
|
email: 'docs@docker.com',
|
|
uuid: this.getSessionId(),
|
|
action: 'AskGordon',
|
|
...(this.includePageContext && {
|
|
page_url: window.location.href,
|
|
{{ with .Title }}page_title: {{ . | jsonify }}{{ end }}
|
|
})
|
|
}
|
|
}
|
|
]
|
|
}
|
|
|
|
return message
|
|
})
|
|
|
|
const isNewConversation = !this.threadId
|
|
const payload = {
|
|
messages: conversationMessages,
|
|
...(this.threadId && { thread_uuid: this.threadId })
|
|
}
|
|
|
|
const response = await fetch(window.GORDON_BASE_URL + '/public/ask', {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify(payload)
|
|
})
|
|
|
|
if (!response.ok) {
|
|
if (response.status === 429) {
|
|
throw new Error('RATE_LIMIT_EXCEEDED')
|
|
}
|
|
throw new Error(`HTTP ${response.status}: ${response.statusText}`)
|
|
}
|
|
|
|
const reader = response.body.getReader()
|
|
const decoder = new TextDecoder()
|
|
|
|
while (true) {
|
|
const { done, value } = await reader.read()
|
|
if (done) break
|
|
|
|
const chunk = decoder.decode(value, { stream: true })
|
|
const lines = chunk.split('\n')
|
|
|
|
for (const line of lines) {
|
|
if (!line.trim() || !line.startsWith('data: ')) continue
|
|
|
|
const data = line.slice(6)
|
|
if (data === '[DONE]') continue
|
|
|
|
try {
|
|
const parsed = JSON.parse(data)
|
|
|
|
// Capture thread_id for new conversations
|
|
if (parsed.thread_id) {
|
|
if (isNewConversation) {
|
|
this.threadId = parsed.thread_id // $persist auto-saves to sessionStorage
|
|
} else if (parsed.thread_id !== this.threadId) {
|
|
console.warn('Backend returned unexpected thread_id:', parsed.thread_id)
|
|
}
|
|
continue
|
|
}
|
|
|
|
// Capture question_answer_id for feedback
|
|
if (parsed.question_answer_id) {
|
|
this.messages[responseIndex].questionAnswerId = parsed.question_answer_id
|
|
continue
|
|
}
|
|
|
|
if (parsed.choices && parsed.choices[0]?.delta?.content) {
|
|
const content = parsed.choices[0].delta.content
|
|
this.messages[responseIndex].content += content
|
|
|
|
this.$nextTick(() => {
|
|
const container = this.$refs.messagesContainer
|
|
if (container) {
|
|
const isNearBottom = container.scrollHeight - container.scrollTop - container.clientHeight < 100
|
|
if (isNearBottom) {
|
|
container.scrollTop = container.scrollHeight
|
|
}
|
|
}
|
|
})
|
|
}
|
|
} catch (e) {
|
|
console.error('Failed to parse SSE data:', e)
|
|
}
|
|
}
|
|
}
|
|
|
|
this.messages[responseIndex].isStreaming = false
|
|
},
|
|
|
|
clearChat() {
|
|
if (confirm('Clear all messages?')) {
|
|
this.messages = [] // $persist auto-saves empty array
|
|
this.threadId = null // $persist auto-removes from sessionStorage
|
|
this.error = null
|
|
}
|
|
},
|
|
|
|
formatMessageContent(content) {
|
|
return this.$markdown(content)
|
|
},
|
|
|
|
async submitFeedback(messageIndex, feedbackType) {
|
|
const message = this.messages[messageIndex]
|
|
|
|
if (message.feedback === feedbackType) {
|
|
return
|
|
}
|
|
|
|
// Clear any previous error
|
|
message.feedbackError = null
|
|
|
|
if (!message.questionAnswerId) {
|
|
message.feedbackError = 'Unable to submit feedback'
|
|
console.error('No question_answer_id available for feedback')
|
|
return
|
|
}
|
|
|
|
try {
|
|
const response = await fetch(window.GORDON_BASE_URL + '/feedback', {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify({
|
|
id: message.questionAnswerId,
|
|
feedback: feedbackType
|
|
})
|
|
})
|
|
|
|
if (response.ok) {
|
|
message.feedback = feedbackType // $persist auto-saves
|
|
} else {
|
|
message.feedbackError = 'Unable to submit feedback'
|
|
console.error('Failed to submit feedback:', response.status, response.statusText)
|
|
}
|
|
} catch (err) {
|
|
message.feedbackError = 'Unable to submit feedback'
|
|
console.error('Error submitting feedback:', err)
|
|
}
|
|
},
|
|
|
|
async copyAnswer(messageIndex) {
|
|
const message = this.messages[messageIndex]
|
|
if (!message.content) return
|
|
|
|
try {
|
|
await navigator.clipboard.writeText(message.content)
|
|
message.copied = true
|
|
this.$nextTick(() => {
|
|
setTimeout(() => {
|
|
message.copied = false
|
|
}, 2000)
|
|
})
|
|
} catch (err) {
|
|
console.error('Failed to copy:', err)
|
|
}
|
|
}
|
|
}"
|
|
x-cloak
|
|
@keydown.escape.window="$store.gordon.close()"
|
|
>
|
|
<!-- Overlay backdrop -->
|
|
<div
|
|
x-show="$store.gordon.isOpen"
|
|
x-transition.opacity.duration.300ms
|
|
@click="$store.gordon.close()"
|
|
class="fixed inset-0 z-40 bg-black/50"
|
|
></div>
|
|
|
|
<!-- Chat panel sliding in from right -->
|
|
<div
|
|
id="gordon-chat"
|
|
x-show="$store.gordon.isOpen"
|
|
x-transition:enter="transition ease-out duration-200"
|
|
x-transition:enter-start="translate-x-full"
|
|
x-transition:enter-end="translate-x-0"
|
|
x-transition:leave="transition ease-in duration-200"
|
|
x-transition:leave-start="translate-x-0"
|
|
x-transition:leave-end="translate-x-full"
|
|
class="fixed top-0 right-0 z-50 flex h-screen w-full flex-col overflow-hidden rounded-none bg-white shadow-2xl transition-all duration-200 md:w-[min(80ch,90vw)] dark:bg-gray-950"
|
|
>
|
|
<!-- Header -->
|
|
<div
|
|
class="z-10 flex items-center justify-between bg-blue-500 px-6 py-3 shadow-lg/30 dark:shadow-lg/60"
|
|
>
|
|
<div class="flex items-center gap-3">
|
|
{{ partialCached "icon" "images/gordon-logo.svg" "images/gordon-logo.svg" }}
|
|
</div>
|
|
<div class="flex items-center gap-2">
|
|
<div data-tooltip-wrapper class="relative">
|
|
<button
|
|
@click="clearChat()"
|
|
aria-label="Start a new chat"
|
|
class="cursor-pointer rounded p-2 text-white transition-colors enabled:hover:bg-white/15 disabled:cursor-not-allowed disabled:opacity-50"
|
|
:disabled="messages.length === 0"
|
|
>
|
|
<span class="icon-svg">
|
|
{{ partialCached "icon" "arrow-path" "arrow-path" }}
|
|
</span>
|
|
</button>
|
|
<div
|
|
id="gordon-tooltip"
|
|
data-tooltip-body
|
|
class="absolute top-0 left-0 hidden rounded-sm bg-gray-900 p-2 text-sm whitespace-nowrap text-white"
|
|
role="tooltip"
|
|
x-show="messages.length > 0"
|
|
>
|
|
Start a new chat
|
|
<div
|
|
data-tooltip-arrow
|
|
class="absolute h-2 w-2 rotate-45 bg-gray-900"
|
|
></div>
|
|
</div>
|
|
</div>
|
|
<button
|
|
@click="$store.gordon.close()"
|
|
title="Close chat drawer"
|
|
class="cursor-pointer rounded p-2 text-white transition-colors hover:bg-white/15"
|
|
aria-label="Close chat"
|
|
>
|
|
<span class="icon-svg">
|
|
{{ partialCached "icon" "x-mark" "x-mark" }}
|
|
</span>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Messages container -->
|
|
<div
|
|
x-ref="messagesContainer"
|
|
class="flex-1 space-y-4 p-6"
|
|
style="scrollbar-width: none; &::-webkit-scrollbar: { display: none; }"
|
|
:class="{ 'overflow-y-auto': messages.length > 0 }"
|
|
>
|
|
<!-- Welcome message when empty -->
|
|
<template x-if="messages.length === 0">
|
|
<div
|
|
class="flex h-full flex-col items-center justify-start overflow-y-auto px-2 pt-4 pb-2 text-center"
|
|
>
|
|
<div class="mx-auto mb-2 w-4/5 max-w-xs" aria-hidden="true">
|
|
<div
|
|
class="motion-safe:animate-[robotFloat_5s_ease-in-out_infinite]"
|
|
>
|
|
{{ partialCached "icon" "images/gordon-robot.svg" "images/gordon-robot.svg" }}
|
|
</div>
|
|
</div>
|
|
<div class="flex w-full flex-col items-center justify-start gap-3">
|
|
<h3 class="text-xl font-semibold text-gray-900 dark:text-white">
|
|
What can I help you with?
|
|
</h3>
|
|
<p class="max-w-sm text-lg text-gray-600 dark:text-gray-400">
|
|
I'm Gordon, your AI assistant for Docker and documentation
|
|
questions.
|
|
</p>
|
|
<div class="w-full text-left">
|
|
<p
|
|
class="text-sm font-bold text-gray-600 uppercase dark:text-gray-400"
|
|
>
|
|
Try asking
|
|
</p>
|
|
</div>
|
|
{{- $gordonSuggestions := slice
|
|
(dict
|
|
"title" "Get started with Docker"
|
|
"question" "Help me get started with Docker. What should I do first?"
|
|
"icon" "rocket-launch")
|
|
(dict
|
|
"title" "Docker Hardened Images"
|
|
"question" "How do Docker Hardened Images work?"
|
|
"icon" "icons/dhi.svg")
|
|
(dict
|
|
"title" "MCP Toolkit"
|
|
"question" "What is MCP Toolkit?"
|
|
"icon" "icons/toolkit.svg")
|
|
(dict
|
|
"title" "Create an org"
|
|
"question" "How do I create an org?"
|
|
"icon" "icons/headset.svg")
|
|
-}}
|
|
<div class="grid w-full grid-cols-2 gap-2">
|
|
{{- range $gordonSuggestions }}
|
|
<button
|
|
@click="currentQuestion = {{ .question | jsonify }}; askQuestion()"
|
|
class="cursor-pointer rounded-xl border border-gray-200 bg-white p-3 transition hover:scale-[1.02] hover:border-blue-500 dark:border-gray-700 dark:bg-gray-900 dark:hover:border-blue-500"
|
|
>
|
|
<div class="flex items-center gap-3">
|
|
<span
|
|
class="icon-svg shrink-0 rounded-lg bg-blue-100 p-1 text-blue-500 dark:bg-gray-700 dark:text-blue-400"
|
|
>
|
|
{{ partialCached "icon" .icon .icon }}
|
|
</span>
|
|
<p class="text-sm font-medium">{{ .title }}</p>
|
|
</div>
|
|
</button>
|
|
{{- end }}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<!-- Messages -->
|
|
<div x-show="messages.length > 0" class="relative flex flex-col gap-4">
|
|
<template x-for="(message, index) in messages" :key="index">
|
|
<div
|
|
:class="message.role === 'user' ? 'flex justify-end' : 'flex justify-start'"
|
|
>
|
|
<div class="flex max-w-full flex-col gap-2">
|
|
<div
|
|
:class="message.role === 'user' ? 'bg-blue-500 dark:bg-blue-800 text-white' : 'max-w-none bg-gray-100 dark:bg-gray-800'"
|
|
class="prose prose-sm dark:prose-invert rounded-lg px-4"
|
|
>
|
|
<template x-if="!message.content && message.isStreaming">
|
|
<div class="flex gap-1 py-3">
|
|
<span
|
|
class="inline-block h-2 w-2 animate-bounce rounded-full bg-current"
|
|
style="animation-delay: 0ms"
|
|
></span>
|
|
<span
|
|
class="inline-block h-2 w-2 animate-bounce rounded-full bg-current"
|
|
style="animation-delay: 150ms"
|
|
></span>
|
|
<span
|
|
class="inline-block h-2 w-2 animate-bounce rounded-full bg-current"
|
|
style="animation-delay: 300ms"
|
|
></span>
|
|
</div>
|
|
</template>
|
|
<template x-if="message.content">
|
|
<div x-html="formatMessageContent(message.content)"></div>
|
|
</template>
|
|
</div>
|
|
<!-- Feedback buttons for assistant messages -->
|
|
<template
|
|
x-if="message.role === 'assistant' && !message.isStreaming"
|
|
>
|
|
<div class="flex flex-col gap-2">
|
|
<div class="flex items-center gap-3 text-xs">
|
|
<!-- "Was this helpful?" prompt with gordon icon -->
|
|
<div
|
|
class="flex items-center gap-1.5 text-gray-600 dark:text-gray-400"
|
|
>
|
|
<span class="icon-svg icon-sm">
|
|
{{ partialCached "icon" "gordon" "gordon" }}
|
|
</span>
|
|
<span class="font-medium">Was this helpful?</span>
|
|
</div>
|
|
|
|
<!-- Feedback buttons group -->
|
|
<div class="flex items-center gap-2">
|
|
<!-- Thumbs up - Helpful -->
|
|
<button
|
|
@click="submitFeedback(index, 'positive')"
|
|
:class="message.feedback === 'positive'
|
|
? 'bg-green-100 text-green-700 ring ring-green-500 dark:bg-green-900/50 dark:text-green-400 dark:ring-green-600'
|
|
: 'bg-gray-100 text-gray-600 hover:bg-green-50 hover:text-green-600 hover:ring hover:ring-green dark:bg-gray-800 dark:text-gray-200 dark:hover:bg-green-900/30 dark:hover:text-green-400 dark:hover:ring-green'"
|
|
class="flex cursor-pointer items-center gap-1.5 rounded-full px-3 py-1.5 transition-all duration-200"
|
|
title="Helpful"
|
|
>
|
|
<span class="icon-svg icon-sm transition-transform">
|
|
{{ partialCached "icon" "hand-thumb-up" "hand-thumb-up" }}
|
|
</span>
|
|
<span class="hidden font-medium sm:inline"
|
|
>Helpful</span
|
|
>
|
|
</button>
|
|
|
|
<!-- Thumbs down - Not quite -->
|
|
<button
|
|
@click="submitFeedback(index, 'negative')"
|
|
:class="message.feedback === 'negative'
|
|
? 'bg-red-100 text-red-700 ring ring-red-500 dark:bg-red-900/50 dark:text-red-400 dark:ring-red-600'
|
|
: 'bg-gray-100 text-gray-600 hover:bg-red-50 hover:text-red-600 hover:ring hover:ring-red dark:bg-gray-800 dark:text-gray-200 dark:hover:bg-red-900/30 dark:hover:text-red-400 dark:hover:ring-red'"
|
|
class="flex cursor-pointer items-center gap-1.5 rounded-full px-3 py-1.5 transition-all duration-200"
|
|
title="Not quite"
|
|
>
|
|
<span class="icon-svg icon-sm transition-transform">
|
|
{{ partialCached "icon" "hand-thumb-down" "hand-thumb-down" }}
|
|
</span>
|
|
<span class="hidden font-medium sm:inline"
|
|
>Not quite</span
|
|
>
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Separator -->
|
|
<div class="h-4 w-px bg-gray-300 dark:bg-gray-600"></div>
|
|
|
|
<!-- Copy button -->
|
|
<button
|
|
@click="copyAnswer(index)"
|
|
class="hover:ring-blue flex cursor-pointer items-center gap-1.5 rounded-full bg-gray-100 px-3 py-1.5 text-gray-600 transition-all duration-200 hover:bg-blue-50 hover:text-blue-700 hover:ring dark:bg-gray-800 dark:text-gray-200 dark:hover:bg-blue-900/30 dark:hover:text-blue-300 dark:hover:ring-blue-400"
|
|
:title="message.copied ? 'Copied!' : 'Copy answer'"
|
|
>
|
|
<span
|
|
x-show="message.copied !== true"
|
|
class="icon-svg icon-sm"
|
|
>
|
|
{{ partialCached "icon" "document-duplicate" "document-duplicate" }}
|
|
</span>
|
|
<span
|
|
x-show="message.copied === true"
|
|
class="icon-svg icon-sm"
|
|
>
|
|
{{ partialCached "icon" "check-circle" "check-circle" }}
|
|
</span>
|
|
<span class="hidden font-medium sm:inline">Copy</span>
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Error message -->
|
|
<template x-if="message.feedbackError">
|
|
<div
|
|
x-transition:enter="transition ease-out duration-200"
|
|
x-transition:enter-start="opacity-0 translate-y-1"
|
|
x-transition:enter-end="opacity-100 translate-y-0"
|
|
class="flex items-center gap-1.5 text-xs text-red-600 dark:text-red-400"
|
|
>
|
|
<span x-text="message.feedbackError"></span>
|
|
</div>
|
|
</template>
|
|
</div>
|
|
</template>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
</div>
|
|
|
|
<!-- Error message -->
|
|
<template x-if="error">
|
|
<div
|
|
class="rounded-lg border border-red-200 bg-red-50 p-4 text-red-800 dark:border-red-800 dark:bg-red-900/20 dark:text-red-400"
|
|
>
|
|
<p class="text-sm" x-text="error"></p>
|
|
</div>
|
|
</template>
|
|
|
|
<!-- Countdown warning when approaching limit -->
|
|
<template x-if="shouldShowCountdown()">
|
|
<div
|
|
class="rounded-lg border border-yellow-200 bg-yellow-50 p-3 text-yellow-800 dark:border-yellow-800 dark:bg-yellow-900/20 dark:text-yellow-400"
|
|
>
|
|
<p class="text-sm">
|
|
<span x-text="getRemainingTurns()"></span>
|
|
<span
|
|
x-text="getRemainingTurns() === 1 ? 'question' : 'questions'"
|
|
></span>
|
|
remaining in this thread.
|
|
</p>
|
|
</div>
|
|
</template>
|
|
|
|
<!-- Thread limit warning -->
|
|
<template x-if="isThreadLimitReached()">
|
|
<div
|
|
class="rounded-lg border border-blue-200 bg-blue-50 p-4 text-blue-800 dark:border-blue-800 dark:bg-blue-900/20 dark:text-blue-400"
|
|
>
|
|
<p class="mb-3 text-sm">
|
|
You've reached the maximum of
|
|
<span x-text="maxTurnsPerThread"></span> questions per thread. For
|
|
better answer quality, start a new thread.
|
|
</p>
|
|
<button
|
|
@click="clearChat()"
|
|
class="cursor-pointer rounded-lg bg-blue-600 px-4 py-2 text-sm font-semibold text-white transition-colors hover:bg-blue-700 dark:bg-blue-500 dark:hover:bg-blue-600"
|
|
>
|
|
Start a new thread
|
|
</button>
|
|
</div>
|
|
</template>
|
|
</div>
|
|
|
|
<!-- Input area -->
|
|
<form @submit.prevent="askQuestion()" class="space-y-2 p-4 pt-0">
|
|
<div class="relative w-full">
|
|
<div class="relative flex-1 self-stretch">
|
|
<textarea
|
|
id="question"
|
|
x-ref="input"
|
|
x-model="currentQuestion"
|
|
@input="$el.style.height = 'auto'; $el.style.height = Math.min(160, $el.scrollHeight) + 'px'"
|
|
@keydown.enter="if (!$event.shiftKey) { $event.preventDefault(); askQuestion() }"
|
|
placeholder="Ask a question about Docker..."
|
|
rows="3"
|
|
:disabled="isLoading || isThreadLimitReached()"
|
|
class="block min-h-[3rem] w-full resize-none rounded-lg border border-gray-300 bg-white p-3 leading-normal focus:border-blue-500 focus:ring-2 focus:ring-blue-500/20 focus:outline-none enabled:hover:border-white disabled:cursor-not-allowed disabled:opacity-50 dark:border-gray-600 dark:bg-gray-900 dark:text-white dark:focus:border-blue-400"
|
|
></textarea>
|
|
</div>
|
|
<button
|
|
type="submit"
|
|
:disabled="!currentQuestion.trim() || isLoading || isThreadLimitReached()"
|
|
:title="isLoading ? 'Sending...' : (isThreadLimitReached() ? 'Thread limit reached. Start a new thread.' : 'Send question')"
|
|
class="absolute top-2 right-2 cursor-pointer rounded p-2 text-blue-500 transition-colors enabled:hover:bg-blue-500/15 disabled:cursor-not-allowed disabled:text-gray-500"
|
|
>
|
|
<template x-if="!isLoading">
|
|
<span class="icon-svg">
|
|
{{ partialCached "icon" "paper-airplane" "paper-airplane" }}
|
|
</span>
|
|
</template>
|
|
<template x-if="isLoading">
|
|
<span class="icon-svg animate-spin">
|
|
{{ partialCached "icon" "arrow-path" "arrow-path" }}
|
|
</span>
|
|
</template>
|
|
</button>
|
|
</div>
|
|
<div
|
|
class="flex items-center justify-between"
|
|
x-data="{ showTooltip: false }"
|
|
>
|
|
<div class="relative">
|
|
<!-- Tooltip -->
|
|
<div
|
|
x-show="showTooltip"
|
|
x-transition:enter="transition ease-out duration-100"
|
|
x-transition:enter-start="opacity-0 scale-95"
|
|
x-transition:enter-end="opacity-100 scale-100"
|
|
x-transition:leave="transition ease-in duration-75"
|
|
x-transition:leave-start="opacity-100 scale-100"
|
|
x-transition:leave-end="opacity-0 scale-95"
|
|
class="absolute bottom-full left-0 mb-2 w-56 rounded-lg bg-gray-900 p-2.5 text-xs text-white shadow-lg dark:bg-gray-700"
|
|
style="display: none;"
|
|
>
|
|
<div class="relative">
|
|
<p>
|
|
When enabled, Gordon considers the current page you're viewing
|
|
to provide more relevant answers.
|
|
</p>
|
|
<!-- Arrow -->
|
|
<div
|
|
class="absolute -bottom-3 left-4 h-2 w-2 rotate-45 bg-gray-900 dark:bg-gray-700"
|
|
></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<a
|
|
href="https://github.com/docker/docs/issues/23966"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
class="text-xs text-gray-500 underline hover:text-blue-600 dark:text-gray-400 dark:hover:text-blue-400"
|
|
>
|
|
Share feedback
|
|
</a>
|
|
</div>
|
|
</form>
|
|
|
|
<!-- Disclaimer -->
|
|
<div
|
|
class="bg-blue-50 px-4 py-3 text-xs text-gray-600 dark:bg-gray-800 dark:text-gray-400"
|
|
>
|
|
Answers are generated based on the documentation.
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Styles for Gordon chat -->
|
|
<style>
|
|
/* Robot floating animation */
|
|
@keyframes robotFloat {
|
|
0%,
|
|
100% {
|
|
transform: translateY(0px) rotate(0deg);
|
|
}
|
|
50% {
|
|
transform: translateY(-15px) rotate(1deg);
|
|
}
|
|
}
|
|
|
|
/* Code block styles */
|
|
#gordon-chat pre {
|
|
background: #0d1117;
|
|
border-radius: 0.25rem;
|
|
padding: 0;
|
|
margin: 0.5rem 0;
|
|
overflow-x: auto;
|
|
white-space: pre;
|
|
}
|
|
|
|
#gordon-chat pre code {
|
|
background: #0d1117;
|
|
color: #c9d1d9;
|
|
padding: 1rem;
|
|
display: block;
|
|
font-family: "Roboto Mono", monospace;
|
|
font-size: 0.875rem;
|
|
line-height: 1.5;
|
|
white-space: pre;
|
|
overflow-x: auto;
|
|
}
|
|
|
|
#gordon-chat pre code * {
|
|
white-space: pre;
|
|
}
|
|
|
|
/* Inline code styling (not in pre blocks) */
|
|
#gordon-chat .prose code.not-prose {
|
|
background-color: rgb(229 231 235);
|
|
color: rgb(17 24 39);
|
|
padding: 0.2em 0.4em;
|
|
border-radius: 0.25rem;
|
|
font-family: "Roboto Mono", monospace;
|
|
font-size: 0.875em;
|
|
}
|
|
|
|
.dark #gordon-chat .prose code.not-prose {
|
|
background-color: rgb(55 65 81);
|
|
color: rgb(229 231 235);
|
|
}
|
|
</style>
|