agentvibes 2.0.17-beta.2 → 2.0.17-beta.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude/commands/agent-vibes/language.md +23 -0
- package/.claude/commands/agent-vibes/learn.md +67 -0
- package/.claude/commands/agent-vibes/target-voice.md +26 -0
- package/.claude/commands/agent-vibes/target.md +30 -0
- package/.claude/hooks/language-manager.sh +167 -13
- package/.claude/hooks/learn-manager.sh +443 -0
- package/.claude/hooks/play-tts-elevenlabs.sh +3 -2
- package/.claude/hooks/play-tts-piper.sh +3 -2
- package/.claude/hooks/play-tts.sh +7 -4
- package/package.json +1 -1
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
---
|
|
2
|
+
description: Set your main/native language for learning mode
|
|
3
|
+
---
|
|
4
|
+
|
|
5
|
+
Set your main/native language. This is the language you already know and will hear first when learning mode is enabled.
|
|
6
|
+
|
|
7
|
+
Usage:
|
|
8
|
+
```
|
|
9
|
+
/agent-vibes:language english
|
|
10
|
+
/agent-vibes:language spanish
|
|
11
|
+
/agent-vibes:language french
|
|
12
|
+
```
|
|
13
|
+
|
|
14
|
+
The main language uses your currently selected voice. When learning mode is ON, TTS will speak in your main language FIRST, then translate to your target language.
|
|
15
|
+
|
|
16
|
+
Default: english
|
|
17
|
+
|
|
18
|
+
Supported languages: english, spanish, french, german, italian, portuguese, chinese, japanese, korean, hindi, arabic, polish, dutch, turkish, swedish, russian, and 15+ more.
|
|
19
|
+
|
|
20
|
+
After setting your main language:
|
|
21
|
+
1. Set your target language with `/agent-vibes:target <language>`
|
|
22
|
+
2. Set target voice with `/agent-vibes:target-voice <voice>`
|
|
23
|
+
3. Enable learning mode with `/agent-vibes:learn`
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
---
|
|
2
|
+
description: Enable or disable language learning mode
|
|
3
|
+
---
|
|
4
|
+
|
|
5
|
+
Turn language learning mode ON or OFF. When enabled, Claude will speak acknowledgments and completions in BOTH your main language and target language.
|
|
6
|
+
|
|
7
|
+
Usage:
|
|
8
|
+
```
|
|
9
|
+
/agent-vibes:learn # Turn ON
|
|
10
|
+
/agent-vibes:learn off # Turn OFF
|
|
11
|
+
/agent-vibes:learn status # Show current setup
|
|
12
|
+
```
|
|
13
|
+
|
|
14
|
+
## How Learning Mode Works:
|
|
15
|
+
|
|
16
|
+
When learning mode is **ON**:
|
|
17
|
+
1. **First**: Speak in your main language (using your current voice)
|
|
18
|
+
2. **Then**: Speak the SAME message translated to your target language (using target voice)
|
|
19
|
+
|
|
20
|
+
Example:
|
|
21
|
+
```
|
|
22
|
+
Main language (English, Aria): "I'll check that for you"
|
|
23
|
+
Target language (Spanish, Antoni): "Lo verificaré para ti"
|
|
24
|
+
```
|
|
25
|
+
|
|
26
|
+
## Setup Steps:
|
|
27
|
+
|
|
28
|
+
1. Set your main language:
|
|
29
|
+
```
|
|
30
|
+
/agent-vibes:language english
|
|
31
|
+
```
|
|
32
|
+
|
|
33
|
+
2. Set your target language:
|
|
34
|
+
```
|
|
35
|
+
/agent-vibes:target spanish
|
|
36
|
+
```
|
|
37
|
+
|
|
38
|
+
3. Set target voice (recommended):
|
|
39
|
+
```
|
|
40
|
+
/agent-vibes:target-voice Antoni
|
|
41
|
+
```
|
|
42
|
+
|
|
43
|
+
4. Enable learning mode:
|
|
44
|
+
```
|
|
45
|
+
/agent-vibes:learn
|
|
46
|
+
```
|
|
47
|
+
|
|
48
|
+
5. Check your setup:
|
|
49
|
+
```
|
|
50
|
+
/agent-vibes:learn status
|
|
51
|
+
```
|
|
52
|
+
|
|
53
|
+
## Notes:
|
|
54
|
+
|
|
55
|
+
- Translations are **direct translations** of what was said in the main language
|
|
56
|
+
- Same **personality/sentiment** applies to both languages
|
|
57
|
+
- Works with all AgentVibes features (BMAD, personalities, etc.)
|
|
58
|
+
- Requires multilingual voices for target language (Antoni, Rachel, Domi, Bella, etc.)
|
|
59
|
+
- Small pause (0.5s) between main and target language announcements
|
|
60
|
+
|
|
61
|
+
## Disable Learning Mode:
|
|
62
|
+
|
|
63
|
+
```
|
|
64
|
+
/agent-vibes:learn off
|
|
65
|
+
```
|
|
66
|
+
|
|
67
|
+
This returns to normal single-language TTS mode.
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
---
|
|
2
|
+
description: Set the voice for your target language
|
|
3
|
+
---
|
|
4
|
+
|
|
5
|
+
Set which voice to use when speaking your target language. This should typically be a multilingual voice that supports your target language.
|
|
6
|
+
|
|
7
|
+
Usage:
|
|
8
|
+
```
|
|
9
|
+
/agent-vibes:target-voice Antoni
|
|
10
|
+
/agent-vibes:target-voice Rachel
|
|
11
|
+
/agent-vibes:target-voice Domi
|
|
12
|
+
```
|
|
13
|
+
|
|
14
|
+
Recommended multilingual voices:
|
|
15
|
+
- **Antoni** - Best for Spanish, Portuguese
|
|
16
|
+
- **Rachel** - Best for French, English
|
|
17
|
+
- **Domi** - Best for German, European languages
|
|
18
|
+
- **Bella** - Best for Italian, Romance languages
|
|
19
|
+
- **Charlotte** - European languages
|
|
20
|
+
- **Matilda** - Latin languages
|
|
21
|
+
|
|
22
|
+
These voices support 30+ languages using ElevenLabs' Multilingual v2 model.
|
|
23
|
+
|
|
24
|
+
After setting your target voice:
|
|
25
|
+
- Enable learning mode with `/agent-vibes:learn`
|
|
26
|
+
- Check your setup with `/agent-vibes:learn status`
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
---
|
|
2
|
+
description: Set the language you want to learn
|
|
3
|
+
---
|
|
4
|
+
|
|
5
|
+
Set the target language you want to learn. When learning mode is enabled, TTS will speak in your main language FIRST, then speak the translation in your target language.
|
|
6
|
+
|
|
7
|
+
Usage:
|
|
8
|
+
```
|
|
9
|
+
/agent-vibes:target spanish
|
|
10
|
+
/agent-vibes:target french
|
|
11
|
+
/agent-vibes:target german
|
|
12
|
+
```
|
|
13
|
+
|
|
14
|
+
Recommended voices by target language:
|
|
15
|
+
- Spanish → Antoni (ElevenLabs) / es_ES-davefx-medium (Piper)
|
|
16
|
+
- French → Rachel (ElevenLabs) / fr_FR-siwis-medium (Piper)
|
|
17
|
+
- German → Domi (ElevenLabs) / de_DE-thorsten-medium (Piper)
|
|
18
|
+
- Italian → Bella (ElevenLabs) / it_IT-riccardo-x_low (Piper)
|
|
19
|
+
- Portuguese → Matilda (ElevenLabs) / pt_BR-faber-medium (Piper)
|
|
20
|
+
- Chinese → Amy (ElevenLabs) / zh_CN-huayan-medium (Piper)
|
|
21
|
+
- Japanese → Antoni (ElevenLabs) / ja_JP-hikari-medium (Piper)
|
|
22
|
+
- Other languages → Antoni (ElevenLabs) / check available Piper voices
|
|
23
|
+
|
|
24
|
+
**Note:** The system will automatically suggest the correct voice based on your active TTS provider. After setting your target language, the suggestion will match whether you're using ElevenLabs or Piper.
|
|
25
|
+
|
|
26
|
+
After setting your target language:
|
|
27
|
+
1. Set the voice for target language with `/agent-vibes:target-voice <voice>`
|
|
28
|
+
2. Enable learning mode with `/agent-vibes:learn`
|
|
29
|
+
|
|
30
|
+
Supported languages: spanish, french, german, italian, portuguese, chinese, japanese, korean, hindi, arabic, polish, dutch, turkish, swedish, russian, and 15+ more.
|
|
@@ -29,7 +29,68 @@ fi
|
|
|
29
29
|
LANGUAGE_FILE="$CLAUDE_DIR/tts-language.txt"
|
|
30
30
|
mkdir -p "$CLAUDE_DIR"
|
|
31
31
|
|
|
32
|
-
#
|
|
32
|
+
# Source provider manager to detect active provider
|
|
33
|
+
source "$SCRIPT_DIR/provider-manager.sh" 2>/dev/null || true
|
|
34
|
+
|
|
35
|
+
# Language to ElevenLabs multilingual voice mapping
|
|
36
|
+
declare -A ELEVENLABS_VOICES=(
|
|
37
|
+
["spanish"]="Antoni"
|
|
38
|
+
["french"]="Rachel"
|
|
39
|
+
["german"]="Domi"
|
|
40
|
+
["italian"]="Bella"
|
|
41
|
+
["portuguese"]="Matilda"
|
|
42
|
+
["chinese"]="Antoni"
|
|
43
|
+
["japanese"]="Antoni"
|
|
44
|
+
["korean"]="Antoni"
|
|
45
|
+
["russian"]="Domi"
|
|
46
|
+
["polish"]="Antoni"
|
|
47
|
+
["dutch"]="Rachel"
|
|
48
|
+
["turkish"]="Antoni"
|
|
49
|
+
["arabic"]="Antoni"
|
|
50
|
+
["hindi"]="Antoni"
|
|
51
|
+
["swedish"]="Rachel"
|
|
52
|
+
["danish"]="Rachel"
|
|
53
|
+
["norwegian"]="Rachel"
|
|
54
|
+
["finnish"]="Rachel"
|
|
55
|
+
["czech"]="Domi"
|
|
56
|
+
["romanian"]="Rachel"
|
|
57
|
+
["ukrainian"]="Domi"
|
|
58
|
+
["greek"]="Antoni"
|
|
59
|
+
["bulgarian"]="Domi"
|
|
60
|
+
["croatian"]="Domi"
|
|
61
|
+
["slovak"]="Domi"
|
|
62
|
+
)
|
|
63
|
+
|
|
64
|
+
# Language to Piper voice model mapping
|
|
65
|
+
declare -A PIPER_VOICES=(
|
|
66
|
+
["spanish"]="es_ES-davefx-medium"
|
|
67
|
+
["french"]="fr_FR-siwis-medium"
|
|
68
|
+
["german"]="de_DE-thorsten-medium"
|
|
69
|
+
["italian"]="it_IT-riccardo-x_low"
|
|
70
|
+
["portuguese"]="pt_BR-faber-medium"
|
|
71
|
+
["chinese"]="zh_CN-huayan-medium"
|
|
72
|
+
["japanese"]="ja_JP-hikari-medium"
|
|
73
|
+
["korean"]="ko_KR-eunyoung-medium"
|
|
74
|
+
["russian"]="ru_RU-dmitri-medium"
|
|
75
|
+
["polish"]="pl_PL-darkman-medium"
|
|
76
|
+
["dutch"]="nl_NL-rdh-medium"
|
|
77
|
+
["turkish"]="tr_TR-dfki-medium"
|
|
78
|
+
["arabic"]="ar_JO-kareem-medium"
|
|
79
|
+
["hindi"]="hi_IN-amitabh-medium"
|
|
80
|
+
["swedish"]="sv_SE-nst-medium"
|
|
81
|
+
["danish"]="da_DK-talesyntese-medium"
|
|
82
|
+
["norwegian"]="no_NO-talesyntese-medium"
|
|
83
|
+
["finnish"]="fi_FI-harri-medium"
|
|
84
|
+
["czech"]="cs_CZ-jirka-medium"
|
|
85
|
+
["romanian"]="ro_RO-mihai-medium"
|
|
86
|
+
["ukrainian"]="uk_UA-lada-x_low"
|
|
87
|
+
["greek"]="el_GR-rapunzelina-low"
|
|
88
|
+
["bulgarian"]="bg_BG-valentin-medium"
|
|
89
|
+
["croatian"]="hr_HR-gorana-medium"
|
|
90
|
+
["slovak"]="sk_SK-lili-medium"
|
|
91
|
+
)
|
|
92
|
+
|
|
93
|
+
# Backward compatibility: Keep LANGUAGE_VOICES for existing code
|
|
33
94
|
declare -A LANGUAGE_VOICES=(
|
|
34
95
|
["spanish"]="Antoni"
|
|
35
96
|
["french"]="Rachel"
|
|
@@ -91,13 +152,27 @@ set_language() {
|
|
|
91
152
|
# Save language
|
|
92
153
|
echo "$lang" > "$LANGUAGE_FILE"
|
|
93
154
|
|
|
94
|
-
#
|
|
95
|
-
local
|
|
155
|
+
# Detect active provider and get recommended voice
|
|
156
|
+
local provider=""
|
|
157
|
+
if [[ -f "$CLAUDE_DIR/tts-provider.txt" ]]; then
|
|
158
|
+
provider=$(cat "$CLAUDE_DIR/tts-provider.txt")
|
|
159
|
+
elif [[ -f "$HOME/.claude/tts-provider.txt" ]]; then
|
|
160
|
+
provider=$(cat "$HOME/.claude/tts-provider.txt")
|
|
161
|
+
else
|
|
162
|
+
provider="elevenlabs"
|
|
163
|
+
fi
|
|
164
|
+
|
|
165
|
+
local recommended_voice=$(get_voice_for_language "$lang" "$provider")
|
|
166
|
+
|
|
167
|
+
# Fallback to old mapping if provider-aware function returns empty
|
|
168
|
+
if [[ -z "$recommended_voice" ]]; then
|
|
169
|
+
recommended_voice="${LANGUAGE_VOICES[$lang]}"
|
|
170
|
+
fi
|
|
96
171
|
|
|
97
172
|
echo "✓ Language set to: $lang"
|
|
98
|
-
echo "📢 Recommended
|
|
173
|
+
echo "📢 Recommended voice for $provider TTS: $recommended_voice"
|
|
99
174
|
echo ""
|
|
100
|
-
echo "TTS will now speak in $lang
|
|
175
|
+
echo "TTS will now speak in $lang."
|
|
101
176
|
echo "Switch voice with: /agent-vibes:switch \"$recommended_voice\""
|
|
102
177
|
}
|
|
103
178
|
|
|
@@ -105,9 +180,26 @@ set_language() {
|
|
|
105
180
|
get_language() {
|
|
106
181
|
if [[ -f "$LANGUAGE_FILE" ]]; then
|
|
107
182
|
local lang=$(cat "$LANGUAGE_FILE")
|
|
108
|
-
|
|
183
|
+
|
|
184
|
+
# Detect active provider
|
|
185
|
+
local provider=""
|
|
186
|
+
if [[ -f "$CLAUDE_DIR/tts-provider.txt" ]]; then
|
|
187
|
+
provider=$(cat "$CLAUDE_DIR/tts-provider.txt")
|
|
188
|
+
elif [[ -f "$HOME/.claude/tts-provider.txt" ]]; then
|
|
189
|
+
provider=$(cat "$HOME/.claude/tts-provider.txt")
|
|
190
|
+
else
|
|
191
|
+
provider="elevenlabs"
|
|
192
|
+
fi
|
|
193
|
+
|
|
194
|
+
local recommended_voice=$(get_voice_for_language "$lang" "$provider")
|
|
195
|
+
|
|
196
|
+
# Fallback to old mapping
|
|
197
|
+
if [[ -z "$recommended_voice" ]]; then
|
|
198
|
+
recommended_voice="${LANGUAGE_VOICES[$lang]}"
|
|
199
|
+
fi
|
|
200
|
+
|
|
109
201
|
echo "Current language: $lang"
|
|
110
|
-
echo "Recommended voice: $recommended_voice"
|
|
202
|
+
echo "Recommended voice ($provider): $recommended_voice"
|
|
111
203
|
else
|
|
112
204
|
echo "Current language: english (default)"
|
|
113
205
|
echo "No multilingual voice required"
|
|
@@ -153,6 +245,60 @@ get_best_voice_for_language() {
|
|
|
153
245
|
echo "${LANGUAGE_VOICES[$lang]}"
|
|
154
246
|
}
|
|
155
247
|
|
|
248
|
+
# Function to get voice for a specific language and provider
|
|
249
|
+
# Usage: get_voice_for_language <language> [provider]
|
|
250
|
+
# Provider: "elevenlabs" or "piper" (auto-detected if not provided)
|
|
251
|
+
get_voice_for_language() {
|
|
252
|
+
local language="$1"
|
|
253
|
+
local provider="${2:-}"
|
|
254
|
+
|
|
255
|
+
# Convert to lowercase
|
|
256
|
+
language=$(echo "$language" | tr '[:upper:]' '[:lower:]')
|
|
257
|
+
|
|
258
|
+
# Auto-detect provider if not specified
|
|
259
|
+
if [[ -z "$provider" ]]; then
|
|
260
|
+
if command -v get_active_provider &>/dev/null; then
|
|
261
|
+
provider=$(get_active_provider 2>/dev/null)
|
|
262
|
+
else
|
|
263
|
+
# Fallback to checking provider file directly
|
|
264
|
+
# Try current directory first, then search up the tree
|
|
265
|
+
local search_dir="$PWD"
|
|
266
|
+
local found=false
|
|
267
|
+
|
|
268
|
+
while [[ "$search_dir" != "/" ]]; do
|
|
269
|
+
if [[ -f "$search_dir/.claude/tts-provider.txt" ]]; then
|
|
270
|
+
provider=$(cat "$search_dir/.claude/tts-provider.txt")
|
|
271
|
+
found=true
|
|
272
|
+
break
|
|
273
|
+
fi
|
|
274
|
+
search_dir=$(dirname "$search_dir")
|
|
275
|
+
done
|
|
276
|
+
|
|
277
|
+
# If not found in project tree, check global
|
|
278
|
+
if [[ "$found" = false ]]; then
|
|
279
|
+
if [[ -f "$HOME/.claude/tts-provider.txt" ]]; then
|
|
280
|
+
provider=$(cat "$HOME/.claude/tts-provider.txt")
|
|
281
|
+
else
|
|
282
|
+
provider="elevenlabs" # Default
|
|
283
|
+
fi
|
|
284
|
+
fi
|
|
285
|
+
fi
|
|
286
|
+
fi
|
|
287
|
+
|
|
288
|
+
# Return appropriate voice based on provider
|
|
289
|
+
case "$provider" in
|
|
290
|
+
piper)
|
|
291
|
+
echo "${PIPER_VOICES[$language]:-}"
|
|
292
|
+
;;
|
|
293
|
+
elevenlabs)
|
|
294
|
+
echo "${ELEVENLABS_VOICES[$language]:-}"
|
|
295
|
+
;;
|
|
296
|
+
*)
|
|
297
|
+
echo "${ELEVENLABS_VOICES[$language]:-}"
|
|
298
|
+
;;
|
|
299
|
+
esac
|
|
300
|
+
}
|
|
301
|
+
|
|
156
302
|
# Main command handler - only run if script is executed directly, not sourced
|
|
157
303
|
if [[ "${BASH_SOURCE[0]}" == "${0}" ]]; then
|
|
158
304
|
case "${1:-}" in
|
|
@@ -183,6 +329,13 @@ if [[ "${BASH_SOURCE[0]}" == "${0}" ]]; then
|
|
|
183
329
|
best-voice)
|
|
184
330
|
get_best_voice_for_language
|
|
185
331
|
;;
|
|
332
|
+
voice-for-language)
|
|
333
|
+
if [[ -z "$2" ]]; then
|
|
334
|
+
echo "Usage: language-manager.sh voice-for-language <language> [provider]"
|
|
335
|
+
exit 1
|
|
336
|
+
fi
|
|
337
|
+
get_voice_for_language "$2" "$3"
|
|
338
|
+
;;
|
|
186
339
|
list)
|
|
187
340
|
echo "Supported languages and recommended voices:"
|
|
188
341
|
echo ""
|
|
@@ -194,12 +347,13 @@ if [[ "${BASH_SOURCE[0]}" == "${0}" ]]; then
|
|
|
194
347
|
echo "AgentVibes Language Manager"
|
|
195
348
|
echo ""
|
|
196
349
|
echo "Usage:"
|
|
197
|
-
echo " language-manager.sh set <language>
|
|
198
|
-
echo " language-manager.sh get
|
|
199
|
-
echo " language-manager.sh code
|
|
200
|
-
echo " language-manager.sh check-voice <name>
|
|
201
|
-
echo " language-manager.sh best-voice
|
|
202
|
-
echo " language-manager.sh
|
|
350
|
+
echo " language-manager.sh set <language> Set language"
|
|
351
|
+
echo " language-manager.sh get Get current language"
|
|
352
|
+
echo " language-manager.sh code Get language code only"
|
|
353
|
+
echo " language-manager.sh check-voice <name> Check if voice is multilingual"
|
|
354
|
+
echo " language-manager.sh best-voice Get best voice for current language"
|
|
355
|
+
echo " language-manager.sh voice-for-language <lang> [prov] Get voice for language & provider"
|
|
356
|
+
echo " language-manager.sh list List all supported languages"
|
|
203
357
|
exit 1
|
|
204
358
|
;;
|
|
205
359
|
esac
|
|
@@ -0,0 +1,443 @@
|
|
|
1
|
+
#!/bin/bash
|
|
2
|
+
# Language Learning Mode Manager for AgentVibes
|
|
3
|
+
# Handles dual-language TTS for language learning
|
|
4
|
+
|
|
5
|
+
set -e
|
|
6
|
+
|
|
7
|
+
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
8
|
+
PROJECT_DIR="$SCRIPT_DIR/../.."
|
|
9
|
+
|
|
10
|
+
# Configuration files (project-local first, then global fallback)
|
|
11
|
+
MAIN_LANG_FILE="$PROJECT_DIR/.claude/tts-main-language.txt"
|
|
12
|
+
TARGET_LANG_FILE="$PROJECT_DIR/.claude/tts-target-language.txt"
|
|
13
|
+
TARGET_VOICE_FILE="$PROJECT_DIR/.claude/tts-target-voice.txt"
|
|
14
|
+
LEARN_MODE_FILE="$PROJECT_DIR/.claude/tts-learn-mode.txt"
|
|
15
|
+
|
|
16
|
+
GLOBAL_MAIN_LANG_FILE="$HOME/.claude/tts-main-language.txt"
|
|
17
|
+
GLOBAL_TARGET_LANG_FILE="$HOME/.claude/tts-target-language.txt"
|
|
18
|
+
GLOBAL_TARGET_VOICE_FILE="$HOME/.claude/tts-target-voice.txt"
|
|
19
|
+
GLOBAL_LEARN_MODE_FILE="$HOME/.claude/tts-learn-mode.txt"
|
|
20
|
+
|
|
21
|
+
# Colors
|
|
22
|
+
GREEN='\033[0;32m'
|
|
23
|
+
YELLOW='\033[1;33m'
|
|
24
|
+
BLUE='\033[0;34m'
|
|
25
|
+
NC='\033[0m'
|
|
26
|
+
|
|
27
|
+
# Get main language
|
|
28
|
+
get_main_language() {
|
|
29
|
+
if [[ -f "$MAIN_LANG_FILE" ]]; then
|
|
30
|
+
cat "$MAIN_LANG_FILE"
|
|
31
|
+
elif [[ -f "$GLOBAL_MAIN_LANG_FILE" ]]; then
|
|
32
|
+
cat "$GLOBAL_MAIN_LANG_FILE"
|
|
33
|
+
else
|
|
34
|
+
echo "english"
|
|
35
|
+
fi
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
# Set main language
|
|
39
|
+
set_main_language() {
|
|
40
|
+
local language="$1"
|
|
41
|
+
if [[ -z "$language" ]]; then
|
|
42
|
+
echo -e "${YELLOW}Usage: learn-manager.sh set-main-language <language>${NC}"
|
|
43
|
+
exit 1
|
|
44
|
+
fi
|
|
45
|
+
|
|
46
|
+
mkdir -p "$PROJECT_DIR/.claude"
|
|
47
|
+
echo "$language" > "$MAIN_LANG_FILE"
|
|
48
|
+
echo -e "${GREEN}✓${NC} Main language set to: $language"
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
# Get target language
|
|
52
|
+
get_target_language() {
|
|
53
|
+
if [[ -f "$TARGET_LANG_FILE" ]]; then
|
|
54
|
+
cat "$TARGET_LANG_FILE"
|
|
55
|
+
elif [[ -f "$GLOBAL_TARGET_LANG_FILE" ]]; then
|
|
56
|
+
cat "$GLOBAL_TARGET_LANG_FILE"
|
|
57
|
+
else
|
|
58
|
+
echo ""
|
|
59
|
+
fi
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
# Get greeting message for a language
|
|
63
|
+
get_greeting_for_language() {
|
|
64
|
+
local language="$1"
|
|
65
|
+
|
|
66
|
+
case "${language,,}" in
|
|
67
|
+
spanish|español)
|
|
68
|
+
echo "¡Hola! Soy tu profesor de español. ¡Vamos a aprender juntos!"
|
|
69
|
+
;;
|
|
70
|
+
french|français)
|
|
71
|
+
echo "Bonjour! Je suis votre professeur de français. Apprenons ensemble!"
|
|
72
|
+
;;
|
|
73
|
+
german|deutsch)
|
|
74
|
+
echo "Hallo! Ich bin dein Deutschlehrer. Lass uns zusammen lernen!"
|
|
75
|
+
;;
|
|
76
|
+
italian|italiano)
|
|
77
|
+
echo "Ciao! Sono il tuo insegnante di italiano. Impariamo insieme!"
|
|
78
|
+
;;
|
|
79
|
+
portuguese|português)
|
|
80
|
+
echo "Olá! Sou seu professor de português. Vamos aprender juntos!"
|
|
81
|
+
;;
|
|
82
|
+
chinese|中文|mandarin)
|
|
83
|
+
echo "你好!我是你的中文老师。让我们一起学习吧!"
|
|
84
|
+
;;
|
|
85
|
+
japanese|日本語)
|
|
86
|
+
echo "こんにちは!私はあなたの日本語の先生です。一緒に勉強しましょう!"
|
|
87
|
+
;;
|
|
88
|
+
korean|한국어)
|
|
89
|
+
echo "안녕하세요! 저는 당신의 한국어 선생님입니다. 함께 배워봅시다!"
|
|
90
|
+
;;
|
|
91
|
+
russian|русский)
|
|
92
|
+
echo "Здравствуйте! Я ваш учитель русского языка. Давайте учиться вместе!"
|
|
93
|
+
;;
|
|
94
|
+
arabic|العربية)
|
|
95
|
+
echo "مرحبا! أنا معلمك للغة العربية. دعونا نتعلم معا!"
|
|
96
|
+
;;
|
|
97
|
+
hindi|हिन्दी)
|
|
98
|
+
echo "नमस्ते! मैं आपका हिंदी शिक्षक हूं। आइए साथ में सीखें!"
|
|
99
|
+
;;
|
|
100
|
+
dutch|nederlands)
|
|
101
|
+
echo "Hallo! Ik ben je Nederlandse leraar. Laten we samen leren!"
|
|
102
|
+
;;
|
|
103
|
+
polish|polski)
|
|
104
|
+
echo "Cześć! Jestem twoim nauczycielem polskiego. Uczmy się razem!"
|
|
105
|
+
;;
|
|
106
|
+
turkish|türkçe)
|
|
107
|
+
echo "Merhaba! Ben Türkçe öğretmeninizim. Birlikte öğrenelim!"
|
|
108
|
+
;;
|
|
109
|
+
swedish|svenska)
|
|
110
|
+
echo "Hej! Jag är din svenskalärare. Låt oss lära tillsammans!"
|
|
111
|
+
;;
|
|
112
|
+
*)
|
|
113
|
+
echo "Hello! I am your language teacher. Let's learn together!"
|
|
114
|
+
;;
|
|
115
|
+
esac
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
# Set target language
|
|
119
|
+
set_target_language() {
|
|
120
|
+
local language="$1"
|
|
121
|
+
if [[ -z "$language" ]]; then
|
|
122
|
+
echo -e "${YELLOW}Usage: learn-manager.sh set-target-language <language>${NC}"
|
|
123
|
+
exit 1
|
|
124
|
+
fi
|
|
125
|
+
|
|
126
|
+
mkdir -p "$PROJECT_DIR/.claude"
|
|
127
|
+
echo "$language" > "$TARGET_LANG_FILE"
|
|
128
|
+
echo -e "${GREEN}✓${NC} Target language set to: $language"
|
|
129
|
+
|
|
130
|
+
# Automatically set the recommended voice for this language
|
|
131
|
+
local recommended_voice=$(get_recommended_voice_for_language "$language")
|
|
132
|
+
if [[ -n "$recommended_voice" ]]; then
|
|
133
|
+
echo "$recommended_voice" > "$TARGET_VOICE_FILE"
|
|
134
|
+
echo -e "${GREEN}✓${NC} Target voice automatically set to: ${YELLOW}$recommended_voice${NC}"
|
|
135
|
+
|
|
136
|
+
# Detect provider for display
|
|
137
|
+
local provider=""
|
|
138
|
+
if [[ -f "$PROJECT_DIR/.claude/tts-provider.txt" ]]; then
|
|
139
|
+
provider=$(cat "$PROJECT_DIR/.claude/tts-provider.txt")
|
|
140
|
+
elif [[ -f "$HOME/.claude/tts-provider.txt" ]]; then
|
|
141
|
+
provider=$(cat "$HOME/.claude/tts-provider.txt")
|
|
142
|
+
else
|
|
143
|
+
provider="elevenlabs"
|
|
144
|
+
fi
|
|
145
|
+
echo -e " (for ${GREEN}$provider${NC} TTS)"
|
|
146
|
+
echo ""
|
|
147
|
+
|
|
148
|
+
# Greet user in the target language with the target voice
|
|
149
|
+
local greeting=$(get_greeting_for_language "$language")
|
|
150
|
+
echo -e "${BLUE}🎓${NC} Your language teacher says:"
|
|
151
|
+
|
|
152
|
+
# Check if we're using Piper and if the voice is available
|
|
153
|
+
if [[ "$provider" == "piper" ]]; then
|
|
154
|
+
# Quick check: does the voice file exist?
|
|
155
|
+
local voice_dir="${HOME}/.claude/piper-voices"
|
|
156
|
+
if [[ -f "${voice_dir}/${recommended_voice}.onnx" ]]; then
|
|
157
|
+
# Voice exists, play greeting in background
|
|
158
|
+
nohup "$SCRIPT_DIR/play-tts.sh" "$greeting" "$recommended_voice" >/dev/null 2>&1 &
|
|
159
|
+
else
|
|
160
|
+
echo -e "${YELLOW} (Voice not yet downloaded - greeting will play after first download)${NC}"
|
|
161
|
+
fi
|
|
162
|
+
else
|
|
163
|
+
# ElevenLabs - just play it in background
|
|
164
|
+
nohup "$SCRIPT_DIR/play-tts.sh" "$greeting" "$recommended_voice" >/dev/null 2>&1 &
|
|
165
|
+
fi
|
|
166
|
+
else
|
|
167
|
+
# Fallback to suggestion if auto-set failed
|
|
168
|
+
suggest_voice_for_language "$language"
|
|
169
|
+
fi
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
# Get recommended voice for a language (returns voice string, no output)
|
|
173
|
+
get_recommended_voice_for_language() {
|
|
174
|
+
local language="$1"
|
|
175
|
+
local recommended_voice=""
|
|
176
|
+
local provider=""
|
|
177
|
+
|
|
178
|
+
# Detect active provider
|
|
179
|
+
if [[ -f "$PROJECT_DIR/.claude/tts-provider.txt" ]]; then
|
|
180
|
+
provider=$(cat "$PROJECT_DIR/.claude/tts-provider.txt")
|
|
181
|
+
elif [[ -f "$HOME/.claude/tts-provider.txt" ]]; then
|
|
182
|
+
provider=$(cat "$HOME/.claude/tts-provider.txt")
|
|
183
|
+
else
|
|
184
|
+
provider="elevenlabs" # Default
|
|
185
|
+
fi
|
|
186
|
+
|
|
187
|
+
# Source language manager and get provider-specific voice
|
|
188
|
+
if [[ -f "$SCRIPT_DIR/language-manager.sh" ]]; then
|
|
189
|
+
source "$SCRIPT_DIR/language-manager.sh" 2>/dev/null
|
|
190
|
+
recommended_voice=$(get_voice_for_language "$language" "$provider" 2>/dev/null)
|
|
191
|
+
fi
|
|
192
|
+
|
|
193
|
+
# Fallback to hardcoded suggestions if function failed
|
|
194
|
+
if [[ -z "$recommended_voice" ]]; then
|
|
195
|
+
case "${language,,}" in
|
|
196
|
+
spanish|español)
|
|
197
|
+
recommended_voice=$([ "$provider" = "piper" ] && echo "es_ES-davefx-medium" || echo "Antoni")
|
|
198
|
+
;;
|
|
199
|
+
french|français)
|
|
200
|
+
recommended_voice=$([ "$provider" = "piper" ] && echo "fr_FR-siwis-medium" || echo "Rachel")
|
|
201
|
+
;;
|
|
202
|
+
german|deutsch)
|
|
203
|
+
recommended_voice=$([ "$provider" = "piper" ] && echo "de_DE-thorsten-medium" || echo "Domi")
|
|
204
|
+
;;
|
|
205
|
+
italian|italiano)
|
|
206
|
+
recommended_voice=$([ "$provider" = "piper" ] && echo "it_IT-riccardo-x_low" || echo "Bella")
|
|
207
|
+
;;
|
|
208
|
+
portuguese|português)
|
|
209
|
+
recommended_voice=$([ "$provider" = "piper" ] && echo "pt_BR-faber-medium" || echo "Matilda")
|
|
210
|
+
;;
|
|
211
|
+
chinese|中文|mandarin)
|
|
212
|
+
recommended_voice=$([ "$provider" = "piper" ] && echo "zh_CN-huayan-medium" || echo "Amy")
|
|
213
|
+
;;
|
|
214
|
+
*)
|
|
215
|
+
recommended_voice=$([ "$provider" = "piper" ] && echo "en_US-lessac-medium" || echo "Antoni")
|
|
216
|
+
;;
|
|
217
|
+
esac
|
|
218
|
+
fi
|
|
219
|
+
|
|
220
|
+
echo "$recommended_voice"
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
# Suggest voice based on target language (displays suggestion message)
|
|
224
|
+
suggest_voice_for_language() {
|
|
225
|
+
local language="$1"
|
|
226
|
+
local suggested_voice=$(get_recommended_voice_for_language "$language")
|
|
227
|
+
|
|
228
|
+
# Detect provider for display
|
|
229
|
+
local provider=""
|
|
230
|
+
if [[ -f "$PROJECT_DIR/.claude/tts-provider.txt" ]]; then
|
|
231
|
+
provider=$(cat "$PROJECT_DIR/.claude/tts-provider.txt")
|
|
232
|
+
elif [[ -f "$HOME/.claude/tts-provider.txt" ]]; then
|
|
233
|
+
provider=$(cat "$HOME/.claude/tts-provider.txt")
|
|
234
|
+
else
|
|
235
|
+
provider="elevenlabs"
|
|
236
|
+
fi
|
|
237
|
+
|
|
238
|
+
echo ""
|
|
239
|
+
echo -e "${BLUE}💡 Tip:${NC} For $language (using ${GREEN}$provider${NC} TTS), we recommend: ${YELLOW}$suggested_voice${NC}"
|
|
240
|
+
echo -e " Set it with: ${YELLOW}/agent-vibes:target-voice $suggested_voice${NC}"
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
# Get target voice
|
|
244
|
+
get_target_voice() {
|
|
245
|
+
if [[ -f "$TARGET_VOICE_FILE" ]]; then
|
|
246
|
+
cat "$TARGET_VOICE_FILE"
|
|
247
|
+
elif [[ -f "$GLOBAL_TARGET_VOICE_FILE" ]]; then
|
|
248
|
+
cat "$GLOBAL_TARGET_VOICE_FILE"
|
|
249
|
+
else
|
|
250
|
+
echo ""
|
|
251
|
+
fi
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
# Set target voice
|
|
255
|
+
set_target_voice() {
|
|
256
|
+
local voice="$1"
|
|
257
|
+
if [[ -z "$voice" ]]; then
|
|
258
|
+
echo -e "${YELLOW}Usage: learn-manager.sh set-target-voice <voice>${NC}"
|
|
259
|
+
exit 1
|
|
260
|
+
fi
|
|
261
|
+
|
|
262
|
+
mkdir -p "$PROJECT_DIR/.claude"
|
|
263
|
+
echo "$voice" > "$TARGET_VOICE_FILE"
|
|
264
|
+
echo -e "${GREEN}✓${NC} Target voice set to: $voice"
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
# Check if learning mode is enabled
|
|
268
|
+
is_learn_mode_enabled() {
|
|
269
|
+
if [[ -f "$LEARN_MODE_FILE" ]]; then
|
|
270
|
+
local mode=$(cat "$LEARN_MODE_FILE")
|
|
271
|
+
[[ "$mode" == "ON" ]]
|
|
272
|
+
elif [[ -f "$GLOBAL_LEARN_MODE_FILE" ]]; then
|
|
273
|
+
local mode=$(cat "$GLOBAL_LEARN_MODE_FILE")
|
|
274
|
+
[[ "$mode" == "ON" ]]
|
|
275
|
+
else
|
|
276
|
+
return 1
|
|
277
|
+
fi
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
# Enable learning mode
|
|
281
|
+
enable_learn_mode() {
|
|
282
|
+
mkdir -p "$PROJECT_DIR/.claude"
|
|
283
|
+
echo "ON" > "$LEARN_MODE_FILE"
|
|
284
|
+
echo -e "${GREEN}✓${NC} Language learning mode: ${GREEN}ENABLED${NC}"
|
|
285
|
+
echo ""
|
|
286
|
+
|
|
287
|
+
# Auto-set target voice if target language is set but voice is not
|
|
288
|
+
local target_lang=$(get_target_language)
|
|
289
|
+
local target_voice=$(get_target_voice)
|
|
290
|
+
local voice_was_set=false
|
|
291
|
+
|
|
292
|
+
if [[ -n "$target_lang" ]] && [[ -z "$target_voice" ]]; then
|
|
293
|
+
echo -e "${BLUE}ℹ${NC} Auto-configuring voice for $target_lang..."
|
|
294
|
+
local recommended_voice=$(get_recommended_voice_for_language "$target_lang")
|
|
295
|
+
if [[ -n "$recommended_voice" ]]; then
|
|
296
|
+
echo "$recommended_voice" > "$TARGET_VOICE_FILE"
|
|
297
|
+
target_voice="$recommended_voice"
|
|
298
|
+
echo -e "${GREEN}✓${NC} Target voice automatically set to: ${YELLOW}$recommended_voice${NC}"
|
|
299
|
+
|
|
300
|
+
# Detect provider for display
|
|
301
|
+
local provider=""
|
|
302
|
+
if [[ -f "$PROJECT_DIR/.claude/tts-provider.txt" ]]; then
|
|
303
|
+
provider=$(cat "$PROJECT_DIR/.claude/tts-provider.txt")
|
|
304
|
+
elif [[ -f "$HOME/.claude/tts-provider.txt" ]]; then
|
|
305
|
+
provider=$(cat "$HOME/.claude/tts-provider.txt")
|
|
306
|
+
else
|
|
307
|
+
provider="elevenlabs"
|
|
308
|
+
fi
|
|
309
|
+
echo -e " (for ${GREEN}$provider${NC} TTS)"
|
|
310
|
+
echo ""
|
|
311
|
+
voice_was_set=true
|
|
312
|
+
fi
|
|
313
|
+
fi
|
|
314
|
+
|
|
315
|
+
show_status
|
|
316
|
+
|
|
317
|
+
# Greet user with language teacher if everything is configured
|
|
318
|
+
if [[ -n "$target_lang" ]] && [[ -n "$target_voice" ]]; then
|
|
319
|
+
echo ""
|
|
320
|
+
local greeting=$(get_greeting_for_language "$target_lang")
|
|
321
|
+
echo -e "${BLUE}🎓${NC} Your language teacher says:"
|
|
322
|
+
|
|
323
|
+
# Detect provider
|
|
324
|
+
local provider=""
|
|
325
|
+
if [[ -f "$PROJECT_DIR/.claude/tts-provider.txt" ]]; then
|
|
326
|
+
provider=$(cat "$PROJECT_DIR/.claude/tts-provider.txt")
|
|
327
|
+
elif [[ -f "$HOME/.claude/tts-provider.txt" ]]; then
|
|
328
|
+
provider=$(cat "$HOME/.claude/tts-provider.txt")
|
|
329
|
+
else
|
|
330
|
+
provider="elevenlabs"
|
|
331
|
+
fi
|
|
332
|
+
|
|
333
|
+
# Check if we're using Piper and if the voice is available
|
|
334
|
+
if [[ "$provider" == "piper" ]]; then
|
|
335
|
+
# Quick check: does the voice file exist?
|
|
336
|
+
local voice_dir="${HOME}/.claude/piper-voices"
|
|
337
|
+
if [[ -f "${voice_dir}/${target_voice}.onnx" ]]; then
|
|
338
|
+
# Voice exists, play greeting in background
|
|
339
|
+
nohup "$SCRIPT_DIR/play-tts.sh" "$greeting" "$target_voice" >/dev/null 2>&1 &
|
|
340
|
+
else
|
|
341
|
+
echo -e "${YELLOW} (Voice not yet downloaded - greeting will play after first download)${NC}"
|
|
342
|
+
fi
|
|
343
|
+
else
|
|
344
|
+
# ElevenLabs - just play it in background
|
|
345
|
+
nohup "$SCRIPT_DIR/play-tts.sh" "$greeting" "$target_voice" >/dev/null 2>&1 &
|
|
346
|
+
fi
|
|
347
|
+
fi
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
# Disable learning mode
|
|
351
|
+
disable_learn_mode() {
|
|
352
|
+
mkdir -p "$PROJECT_DIR/.claude"
|
|
353
|
+
echo "OFF" > "$LEARN_MODE_FILE"
|
|
354
|
+
echo -e "${GREEN}✓${NC} Language learning mode: ${YELLOW}DISABLED${NC}"
|
|
355
|
+
}
|
|
356
|
+
|
|
357
|
+
# Show learning mode status
|
|
358
|
+
show_status() {
|
|
359
|
+
local main_lang=$(get_main_language)
|
|
360
|
+
local target_lang=$(get_target_language)
|
|
361
|
+
local target_voice=$(get_target_voice)
|
|
362
|
+
local learn_mode="OFF"
|
|
363
|
+
|
|
364
|
+
if is_learn_mode_enabled; then
|
|
365
|
+
learn_mode="ON"
|
|
366
|
+
fi
|
|
367
|
+
|
|
368
|
+
echo -e "${BLUE}━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━${NC}"
|
|
369
|
+
echo -e "${BLUE} Language Learning Mode Status${NC}"
|
|
370
|
+
echo -e "${BLUE}━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━${NC}"
|
|
371
|
+
echo ""
|
|
372
|
+
echo -e " ${BLUE}Learning Mode:${NC} $(if [[ "$learn_mode" == "ON" ]]; then echo -e "${GREEN}ENABLED${NC}"; else echo -e "${YELLOW}DISABLED${NC}"; fi)"
|
|
373
|
+
echo -e " ${BLUE}Main Language:${NC} $main_lang"
|
|
374
|
+
echo -e " ${BLUE}Target Language:${NC} ${target_lang:-"(not set)"}"
|
|
375
|
+
echo -e " ${BLUE}Target Voice:${NC} ${target_voice:-"(not set)"}"
|
|
376
|
+
echo ""
|
|
377
|
+
|
|
378
|
+
if [[ "$learn_mode" == "ON" ]]; then
|
|
379
|
+
if [[ -z "$target_lang" ]]; then
|
|
380
|
+
echo -e " ${YELLOW}⚠${NC} Please set a target language: ${YELLOW}/agent-vibes:target <language>${NC}"
|
|
381
|
+
fi
|
|
382
|
+
if [[ -z "$target_voice" ]]; then
|
|
383
|
+
echo -e " ${YELLOW}⚠${NC} Please set a target voice: ${YELLOW}/agent-vibes:target-voice <voice>${NC}"
|
|
384
|
+
fi
|
|
385
|
+
|
|
386
|
+
if [[ -n "$target_lang" ]] && [[ -n "$target_voice" ]]; then
|
|
387
|
+
echo -e " ${GREEN}✓${NC} All set! TTS will speak in both languages."
|
|
388
|
+
echo ""
|
|
389
|
+
echo -e " ${BLUE}How it works:${NC}"
|
|
390
|
+
echo -e " 1. First: Speak in ${BLUE}$main_lang${NC} (your current voice)"
|
|
391
|
+
echo -e " 2. Then: Speak in ${BLUE}$target_lang${NC} ($target_voice voice)"
|
|
392
|
+
fi
|
|
393
|
+
else
|
|
394
|
+
echo -e " ${BLUE}💡 Tip:${NC} Enable learning mode with: ${YELLOW}/agent-vibes:learn${NC}"
|
|
395
|
+
fi
|
|
396
|
+
|
|
397
|
+
echo ""
|
|
398
|
+
echo -e "${BLUE}━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━${NC}"
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
# Main command handler
|
|
402
|
+
case "${1:-}" in
|
|
403
|
+
get-main-language)
|
|
404
|
+
get_main_language
|
|
405
|
+
;;
|
|
406
|
+
set-main-language)
|
|
407
|
+
set_main_language "$2"
|
|
408
|
+
;;
|
|
409
|
+
get-target-language)
|
|
410
|
+
get_target_language
|
|
411
|
+
;;
|
|
412
|
+
set-target-language)
|
|
413
|
+
set_target_language "$2"
|
|
414
|
+
;;
|
|
415
|
+
get-target-voice)
|
|
416
|
+
get_target_voice
|
|
417
|
+
;;
|
|
418
|
+
set-target-voice)
|
|
419
|
+
set_target_voice "$2"
|
|
420
|
+
;;
|
|
421
|
+
is-enabled)
|
|
422
|
+
if is_learn_mode_enabled; then
|
|
423
|
+
echo "ON"
|
|
424
|
+
exit 0
|
|
425
|
+
else
|
|
426
|
+
echo "OFF"
|
|
427
|
+
exit 1
|
|
428
|
+
fi
|
|
429
|
+
;;
|
|
430
|
+
enable)
|
|
431
|
+
enable_learn_mode
|
|
432
|
+
;;
|
|
433
|
+
disable)
|
|
434
|
+
disable_learn_mode
|
|
435
|
+
;;
|
|
436
|
+
status)
|
|
437
|
+
show_status
|
|
438
|
+
;;
|
|
439
|
+
*)
|
|
440
|
+
echo "Usage: learn-manager.sh {get-main-language|set-main-language|get-target-language|set-target-language|get-target-voice|set-target-voice|is-enabled|enable|disable|status}"
|
|
441
|
+
exit 1
|
|
442
|
+
;;
|
|
443
|
+
esac
|
|
@@ -216,8 +216,9 @@ if [ -f "${TEMP_FILE}" ]; then
|
|
|
216
216
|
# @param Uses global: $TEMP_FILE
|
|
217
217
|
# @sideeffects Plays audio in background
|
|
218
218
|
# @edgecases Falls through players until one works
|
|
219
|
-
# Play audio (WSL/Linux) in background to avoid blocking
|
|
220
|
-
(paplay "${TEMP_FILE}"
|
|
219
|
+
# Play audio (WSL/Linux) in background to avoid blocking, fully detached
|
|
220
|
+
(paplay "${TEMP_FILE}" || aplay "${TEMP_FILE}" || mpg123 "${TEMP_FILE}") >/dev/null 2>&1 &
|
|
221
|
+
disown
|
|
221
222
|
|
|
222
223
|
# Keep temp files for later review - cleaned up weekly by cron
|
|
223
224
|
echo "🎵 Saved to: ${TEMP_FILE}"
|
|
@@ -186,8 +186,9 @@ fi
|
|
|
186
186
|
# @why Support multiple audio players
|
|
187
187
|
# @param Uses global: $TEMP_FILE
|
|
188
188
|
# @sideeffects Plays audio in background
|
|
189
|
-
# Play audio (WSL/Linux) in background
|
|
190
|
-
(mpv "$TEMP_FILE"
|
|
189
|
+
# Play audio (WSL/Linux) in background, fully detached
|
|
190
|
+
(mpv "$TEMP_FILE" || aplay "$TEMP_FILE" || paplay "$TEMP_FILE") >/dev/null 2>&1 &
|
|
191
|
+
disown
|
|
191
192
|
|
|
192
193
|
echo "🎵 Saved to: $TEMP_FILE"
|
|
193
194
|
echo "🎤 Voice used: $VOICE_MODEL (Piper TTS)"
|
|
@@ -1,12 +1,12 @@
|
|
|
1
1
|
#!/bin/bash
|
|
2
2
|
#
|
|
3
|
-
# @fileoverview TTS Provider Router
|
|
3
|
+
# @fileoverview TTS Provider Router with Language Learning Support
|
|
4
4
|
# @context Routes TTS requests to active provider (ElevenLabs or Piper)
|
|
5
5
|
# @architecture Provider abstraction layer - single entry point for all TTS
|
|
6
|
-
# @dependencies provider-manager.sh, play-tts-elevenlabs.sh, play-tts-piper.sh
|
|
6
|
+
# @dependencies provider-manager.sh, play-tts-elevenlabs.sh, play-tts-piper.sh, learn-manager.sh
|
|
7
7
|
# @entrypoints Called by hooks, slash commands, and personality-manager.sh
|
|
8
8
|
# @patterns Provider pattern - delegates to provider-specific implementations
|
|
9
|
-
# @related provider-manager.sh, play-tts-elevenlabs.sh, play-tts-piper.sh
|
|
9
|
+
# @related provider-manager.sh, play-tts-elevenlabs.sh, play-tts-piper.sh, learn-manager.sh
|
|
10
10
|
#
|
|
11
11
|
|
|
12
12
|
# Fix locale warnings
|
|
@@ -27,7 +27,10 @@ ACTIVE_PROVIDER=$(get_active_provider)
|
|
|
27
27
|
# Show GitHub star reminder (once per day)
|
|
28
28
|
"$SCRIPT_DIR/github-star-reminder.sh" 2>/dev/null || true
|
|
29
29
|
|
|
30
|
-
#
|
|
30
|
+
# Normal single-language mode - route to appropriate provider implementation
|
|
31
|
+
# Note: For learning mode, the output style will call this script TWICE:
|
|
32
|
+
# 1. First call with main language text and current voice
|
|
33
|
+
# 2. Second call with translated text and target voice
|
|
31
34
|
case "$ACTIVE_PROVIDER" in
|
|
32
35
|
elevenlabs)
|
|
33
36
|
exec "$SCRIPT_DIR/play-tts-elevenlabs.sh" "$TEXT" "$VOICE_OVERRIDE"
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"$schema": "https://json.schemastore.org/package.json",
|
|
3
3
|
"name": "agentvibes",
|
|
4
|
-
"version": "2.0.17-beta.
|
|
4
|
+
"version": "2.0.17-beta.3",
|
|
5
5
|
"description": "Now your AI Agents can finally talk back! Professional TTS voice for Claude Code and Claude Desktop (via MCP) with multi-provider support.",
|
|
6
6
|
"homepage": "https://agentvibes.org",
|
|
7
7
|
"keywords": [
|