Add --cap-drop=ALL, --security-opt=no-new-privileges:true to all non-privileged containers. Per-app capability grants for apps needing CHOWN/SETUID/SETGID. Read-only root filesystem with tmpfs for compatible apps (searxng, grafana, uptime-kuma, filebrowser, photoprism, vaultwarden). Add Fedimint "Create a Community" goal with 4-step wizard. Fix deploy script cp -rf for audio directory. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
159 lines
4.8 KiB
Vue
159 lines
4.8 KiB
Vue
<template>
|
|
<div class="chat-fullscreen">
|
|
<!-- Close button + connection indicator (desktop: top-right pill) -->
|
|
<div class="chat-mode-pill hidden md:flex">
|
|
<button class="chat-close-btn" @click="closeChat">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
|
|
</svg>
|
|
<span class="text-xs font-medium">Close</span>
|
|
</button>
|
|
<div
|
|
v-if="aiuiConnected"
|
|
class="w-2 h-2 rounded-full bg-green-400 ml-2 shadow-[0_0_6px_rgba(74,222,128,0.5)]"
|
|
title="AIUI connected"
|
|
/>
|
|
</div>
|
|
|
|
<!-- Loading indicator while iframe loads -->
|
|
<Transition name="fade">
|
|
<div v-if="aiuiUrl && !aiuiConnected" class="chat-loading">
|
|
<div class="glass-card p-8 flex flex-col items-center gap-4">
|
|
<div class="chat-loading-spinner" />
|
|
<p class="text-sm text-white/60">Loading AI assistant...</p>
|
|
</div>
|
|
</div>
|
|
</Transition>
|
|
|
|
<!-- AIUI iframe — on mobile, leave room for close bar + tab bar at bottom -->
|
|
<iframe
|
|
v-if="aiuiUrl"
|
|
ref="aiuiFrame"
|
|
:src="aiuiUrl"
|
|
class="chat-iframe chat-iframe-mobile"
|
|
sandbox="allow-scripts allow-same-origin allow-forms"
|
|
allow="microphone"
|
|
style="background: transparent"
|
|
/>
|
|
|
|
<!-- Fallback when no AIUI URL configured -->
|
|
<div v-else class="chat-placeholder">
|
|
<div class="chat-placeholder-inner">
|
|
<div class="chat-placeholder-icon">
|
|
<svg class="w-8 h-8 text-white/40" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 12h.01M12 12h.01M16 12h.01M21 12c0 4.418-4.03 8-9 8a9.863 9.863 0 01-4.255-.949L3 20l1.395-3.72C3.512 15.042 3 13.574 3 12c0-4.418 4.03-8 9-8s9 3.582 9 8z" />
|
|
</svg>
|
|
</div>
|
|
<h2 class="text-2xl font-semibold text-white mb-2">AI Assistant</h2>
|
|
<p class="text-white/60 mb-4 leading-relaxed">
|
|
AIUI is not connected. Configure the AIUI URL in your environment settings.
|
|
</p>
|
|
<p class="text-xs text-white/30">
|
|
Set <code class="text-white/50">VITE_AIUI_URL</code> or deploy the AIUI container.
|
|
</p>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Mobile close bar: sits above the tab bar, below the AIUI content -->
|
|
<div class="md:hidden mobile-back-btn flex items-center justify-center">
|
|
<button
|
|
class="w-full glass-button px-6 py-2.5 rounded-lg font-medium flex items-center justify-center gap-2 text-sm"
|
|
style="background: rgba(0, 0, 0, 0.5); backdrop-filter: blur(18px); -webkit-backdrop-filter: blur(18px);"
|
|
@click="closeChat"
|
|
>
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
|
|
</svg>
|
|
Close Chat
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, computed, onMounted, onBeforeUnmount } from 'vue'
|
|
import { useRouter } from 'vue-router'
|
|
import { ContextBroker } from '@/services/contextBroker'
|
|
|
|
const router = useRouter()
|
|
const aiuiFrame = ref<HTMLIFrameElement | null>(null)
|
|
const aiuiConnected = ref(false)
|
|
let broker: ContextBroker | null = null
|
|
|
|
const aiuiUrl = computed(() => {
|
|
const envUrl = import.meta.env.VITE_AIUI_URL
|
|
if (envUrl) return `${envUrl}?embedded=true`
|
|
if (import.meta.env.PROD) return '/aiui/?embedded=true'
|
|
return ''
|
|
})
|
|
|
|
function closeChat() {
|
|
if (window.history.length > 1) {
|
|
router.back()
|
|
} else {
|
|
router.push('/dashboard')
|
|
}
|
|
}
|
|
|
|
function onAiuiMessage(event: MessageEvent) {
|
|
if (!aiuiUrl.value) return
|
|
// Validate origin — only accept messages from AIUI
|
|
try {
|
|
const expected = new URL(aiuiUrl.value, window.location.origin).origin
|
|
if (event.origin !== expected) return
|
|
} catch { return }
|
|
const msg = event.data
|
|
if (msg && msg.type === 'ready') {
|
|
aiuiConnected.value = true
|
|
}
|
|
}
|
|
|
|
onMounted(() => {
|
|
window.addEventListener('message', onAiuiMessage)
|
|
if (aiuiUrl.value) {
|
|
broker = new ContextBroker(aiuiFrame, aiuiUrl.value)
|
|
broker.start()
|
|
}
|
|
})
|
|
|
|
onBeforeUnmount(() => {
|
|
window.removeEventListener('message', onAiuiMessage)
|
|
broker?.stop()
|
|
broker = null
|
|
})
|
|
</script>
|
|
|
|
<style scoped>
|
|
.chat-loading {
|
|
position: absolute;
|
|
inset: 0;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
z-index: 10;
|
|
}
|
|
|
|
.chat-loading-spinner {
|
|
width: 32px;
|
|
height: 32px;
|
|
border: 3px solid rgba(255, 255, 255, 0.1);
|
|
border-top-color: #fb923c;
|
|
border-radius: 50%;
|
|
animation: spin 0.8s linear infinite;
|
|
}
|
|
|
|
@keyframes spin {
|
|
to { transform: rotate(360deg); }
|
|
}
|
|
|
|
.fade-enter-active,
|
|
.fade-leave-active {
|
|
transition: opacity 0.3s ease;
|
|
}
|
|
|
|
.fade-enter-from,
|
|
.fade-leave-to {
|
|
opacity: 0;
|
|
}
|
|
</style>
|