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.
@@ -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
- # Language to multilingual voice mapping
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
- # Get recommended voice
95
- local recommended_voice="${LANGUAGE_VOICES[$lang]}"
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 multilingual voice: $recommended_voice"
173
+ echo "📢 Recommended voice for $provider TTS: $recommended_voice"
99
174
  echo ""
100
- echo "TTS will now speak in $lang using multilingual voices."
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
- local recommended_voice="${LANGUAGE_VOICES[$lang]}"
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> Set language"
198
- echo " language-manager.sh get Get current language"
199
- echo " language-manager.sh code Get language code only"
200
- echo " language-manager.sh check-voice <name> Check if voice is multilingual"
201
- echo " language-manager.sh best-voice Get best voice for current language"
202
- echo " language-manager.sh list List all supported languages"
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}" 2>/dev/null || aplay "${TEMP_FILE}" 2>/dev/null || mpg123 "${TEMP_FILE}" 2>/dev/null) &
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" 2>/dev/null || aplay "$TEMP_FILE" 2>/dev/null || paplay "$TEMP_FILE" 2>/dev/null) &
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
- # Route to appropriate provider implementation
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.2",
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": [