feat: nervous

This commit is contained in:
Soulter
2026-01-18 17:07:19 +08:00
parent 06fa7be63e
commit fa4df28c22
2 changed files with 84 additions and 7 deletions
+9 -2
View File
@@ -4,6 +4,8 @@
<v-btn icon="mdi-close" @click="handleClose" flat variant="text" />
<v-btn :icon="isCodeMode ? 'mdi-code-tags-check' : 'mdi-code-tags'" @click="toggleCodeMode" flat
variant="text" :color="isCodeMode ? 'primary' : ''" />
<v-btn :icon="isNervousMode ? 'mdi-emoticon-confused' : 'mdi-emoticon-confused-outline'"
@click="toggleNervousMode" flat variant="text" :color="isNervousMode ? 'primary' : ''" />
</div>
<div class="live-mode-content">
@@ -11,8 +13,8 @@
<!-- 爆炸效果层 -->
<div v-if="isExploding" class="explosion-wave"></div>
<SiriOrb :energy="orbEnergy" :mode="isActive ? orbMode : 'idle'" :is-dark="isDark" :code-mode="isCodeMode"
class="siri-orb" />
<SiriOrb :energy="orbEnergy" :mode="isActive ? orbMode : 'idle'" :is-dark="isDark"
:code-mode="isCodeMode" :nervous-mode="isNervousMode" class="siri-orb" />
</div>
<div class="status-text">
{{ statusText }}
@@ -68,6 +70,7 @@ const vadRecording = useVADRecording();
const isActive = ref(false); // Live Mode 是否激活
const isExploding = ref(false); // 是否正在展示爆炸动画
const isCodeMode = ref(false); // 是否开启代码模式
const isNervousMode = ref(false); // 是否开启紧张模式
// 使用 VAD 提供的 isSpeaking 状态
const isSpeaking = computed(() => vadRecording.isSpeaking.value);
const isListening = ref(false); // 是否在监听
@@ -523,6 +526,10 @@ function toggleCodeMode() {
isCodeMode.value = !isCodeMode.value;
}
function toggleNervousMode() {
isNervousMode.value = !isNervousMode.value;
}
// 监听用户打断
watch(isSpeaking, (newVal) => {
if (newVal && isPlaying.value) {
+75 -5
View File
@@ -3,20 +3,34 @@
<div class="live-orb">
</div>
<div class="eyes-container">
<div class="eye" :class="{ 'blink': isBlinking }">
<div class="eye" :class="{ 'blink': isBlinking, 'nervous': nervousMode }">
<!-- Nervous Mode > -->
<div v-if="nervousMode" class="nervous-eye-content">
<svg viewBox="0 0 30 60" width="100%" height="100%">
<path d="M 0 10 L 30 30 L 0 50" fill="none" stroke="#7d80e4" stroke-width="8" />
</svg>
</div>
<!-- Code Mode Layer -->
<transition name="fade">
<div v-if="codeMode" class="code-rain-container">
<div v-if="codeMode && !nervousMode" class="code-rain-container">
<div v-for="(col, i) in codeColumns" :key="i" class="code-column" :style="col.style">
{{ col.content }}
</div>
</div>
</transition>
</div>
<div class="eye" :class="{ 'blink': isBlinking }">
<div class="eye" :class="{ 'blink': isBlinking, 'nervous': nervousMode }">
<!-- Nervous Mode < -->
<div v-if="nervousMode" class="nervous-eye-content">
<svg viewBox="0 0 30 60" width="100%" height="100%">
<path d="M 30 10 L 0 30 L 30 50" fill="none" stroke="#7d80e4" stroke-width="8" />
</svg>
</div>
<!-- Code Mode Layer -->
<transition name="fade">
<div v-if="codeMode" class="code-rain-container">
<div v-if="codeMode && !nervousMode" class="code-rain-container">
<div v-for="(col, i) in codeColumns" :key="i" class="code-column" :style="col.style">
{{ col.content }}
</div>
@@ -24,6 +38,15 @@
</transition>
</div>
</div>
<!-- Hair Accessory Star -->
<div class="accessory-star">
<svg viewBox="0 0 24 24" width="100%" height="100%">
<path d="M12 2l2.4 7.2h7.6l-6 4.8 2.4 7.2-6-4.8-6 4.8 2.4-7.2-6-4.8h7.6z"
fill="rgba(125, 128, 228, 0.4)" stroke="rgba(180, 182, 255, 0.6)" stroke-width="3"
stroke-linejoin="round" />
</svg>
</div>
</div>
</template>
@@ -35,6 +58,7 @@ const props = defineProps<{
mode: 'idle' | 'listening' | 'speaking' | 'processing';
isDark?: boolean;
codeMode?: boolean;
nervousMode?: boolean;
}>();
// 内部状态
@@ -197,6 +221,7 @@ const scheduleBlink = () => {
};
const triggerBlink = () => {
if (props.nervousMode) return;
isBlinking.value = true;
setTimeout(() => {
isBlinking.value = false;
@@ -353,7 +378,7 @@ const styleVars = computed(() => {
.eyes-container {
position: absolute;
display: flex;
gap: 55px;
gap: 60px;
z-index: 5;
/* Center it */
top: 42%;
@@ -378,6 +403,22 @@ const styleVars = computed(() => {
transform: scaleY(0.1);
}
.eye.nervous {
background-color: transparent;
display: flex;
align-items: center;
justify-content: center;
box-shadow: none;
}
.nervous-eye-content {
width: 100%;
height: 100%;
display: flex;
align-items: center;
justify-content: center;
}
.code-rain-container {
position: absolute;
top: 0;
@@ -421,4 +462,33 @@ const styleVars = computed(() => {
.fade-leave-to {
opacity: 0;
}
.accessory-star {
position: absolute;
width: 15px;
height: 15px;
top: 20%;
right: 20%;
transform: rotate(5deg);
z-index: -100;
opacity: 0.8;
filter: drop-shadow(0 0 5px rgba(180, 182, 255, 0.4));
animation: starFloat 4s ease-in-out infinite;
pointer-events: none;
mix-blend-mode: screen;
}
@keyframes starFloat {
0%,
100% {
transform: rotate(5deg) translateY(0) scale(1);
opacity: 0.3;
}
50% {
transform: rotate(10deg) translateY(-3px) scale(1.05);
opacity: 0.5;
}
}
</style>