set flag NoAWT. set flag Android. import android.app.*; import android.content.*; import android.os.Bundle; import android.view.View; import android.view.View.OnClickListener; import android.speech.RecognitionListener; import android.speech.RecognizerIntent; import android.speech.SpeechRecognizer; import android.widget.Button; import android.widget.TextView; import android.util.Log; import android.media.*; sS earURL = "http://botcompany.de:8083"; static SpeechRecognizer sr; static final String TAG = "MyStt3Activity"; static S language = "en-US"; static int extraResults = 1; static bool fastSynthesis = false; // use device voice (faster) static bool noSpeech = false; // disable speech output static bool manualMode = false; // Start recognition on click only sbool muteAlways, printErrors; sbool listenAfterTalk = true; // it works now static volatile bool listening, recognitionCancelled; static new L<S> history; static Lock speechLock = lock(); static boolean showPicture = true; // global switch. if false, you get the normal text display instead of the smiley static L<S> emotions = litlist( "#1001283", "happy", "#1001284", "unhappy", "#1001285", "oh", "#1001286", "bored", "#1001287", "wide mouth"); static float statusFontSize = /*25*/17; static float userFontSize = 25; static float myFontSize = 25; static int borderColor = 0xFFFFFFFF; static int waitColor = /*0xFFFF0000*/ 0xFFFFFF00; // yellow static S lastEmotionImage, lastEmotion /*, onScreenText = ""*/; static ImageView emoView; static TextView statusView, userTextView, myTextView, lblInputView; static LinearLayout leftFlag, rightFlag; sS statusText; static EditText inputView; sbool recogOnActivate = true; // start listening when app activated sbool hideTitleBar; static CatConnector connector; static LS postponed = syncList(); // output lines postponed because we're listening to user svoid androidCatMain { if (androidIsAdminMode()) { print("Going admin."); androidUnsetAdminMode(); removeMyselfAsInjection(); aShowStartScreen(); ret; } if (hideTitleBar) aHideTitleBar(); androidSay_keepEngine = true; if (muteAlways) androidMuteAudio(); S hello = null; /*if (args.length != 0 && args[0].equals("nogfx")) setOpt(getMainClass(), "showPicture", false);*/ try { history.add("*"); //hello = callStaticAnswerMethod("*", history); if (hello == null) hello = german() ? "hallo" : "hello"; } catch (Throwable e) { e.printStackTrace(); return; } if (!androidIsAdminMode()) aClearConsole(); listening = true; // so user can cancel early //if (!noSpeech) say(hello); justASec(); // show interface callOptMC('happy); connector = new CatConnectorImpl(androidAssistant_token()); connector.startEar(vf<S> say); // setLanguage(language); aAddMenuItems("Switch to manual mode", "Switch to auto mode"); // init engine? if (german()) androidSayInGerman(""); else androidSayInEnglish(""); if (recognitionCancelled) recognitionCancelled = false; else androidUI(f newRecognizer); noMainDone(); } static void newRecognizer() { //print("listening"); listening = true; sr = SpeechRecognizer.createSpeechRecognizer(androidActivity()); sr.setRecognitionListener(new listener()); recog(); } static class listener implements RecognitionListener { public void onReadyForSpeech(Bundle params) { if (recognitionCancelled) { recognitionCancelled = false; sr.stopListening(); ret; } callOptMC('setBorderAndStatus, 0xFF66FF66, german() ? "JETZT SPRECHEN!" : "TALK NOW!"); callOptMC('oh); //showText(german() ? "SAG WAS." : "TALK NOW."); } public void onBeginningOfSpeech() { //showText("User talks"); //callOptMC('oh); if (!manualMode && !muteAlways) androidMuteAudio(); // Mute now, so we don't hear the end-of-speech sound } public void onRmsChanged(float rmsdB) {} public void onBufferReceived(byte[] buffer) {} public void onEndOfSpeech() { ping(); //showText("onEndOfSpeech"); callOptMC('setBorderAndStatus, aWhite(), baseStatus()); } public void onError(int error) { ping(); listening = false; if (printErrors) if (error == 6) // timeout print("speech timeout"); else print("error " + error); // screw the errors! try { sr.destroy(); } catch (Throwable e) { print(e); } if (!manualMode) newRecognizer(); else callOptMC('setBorderAndStatus, aWhite(), baseStatus()); callOpt(getMainClass(), "happy"); } public void onResults(Bundle results) { ping(); listening = false; //showText("onResults"); ArrayList<S> data = results.getStringArrayList(SpeechRecognizer.RESULTS_RECOGNITION); fS s = data.get(0); onInput(s, false); } public void onPartialResults(Bundle partialResults) { print("onPartialResults"); } public void onEvent(int eventType, Bundle params) { print("onEvent" + eventType); } } svoid onInput(S _s, final bool typed) { fS s = trim(_s); if (empty(s)) ret; thread "onInput" { connector.sendLine(s, typed); aSetText(userTextView, lang("You: ", "Du: ") + s); aPutViewBehindOtherView(userTextView, myTextView); showText( (typed ? lang("You typed: ", "Du schrubst: ") : lang("I heard: ", "Ich habe geh\u00f6rt: ")) + quote(s)); // TODO: fix the java strings umlaut problem final boolean goodbye = match3("goodbye", s) || match3("bye", s) || match3("tsch\u00fcss", s) || match3("tsch\u00fcss ...", s); // get answer history.add(s); handleCommand(s); S answer; try { answer = goodbye ? "tsch\u00fcss" : callStaticAnswerMethod(s, history); } catch (Throwable e) { e.printStackTrace(); appendToFile(getProgramFile("errors.txt"), getTheStackTrace(e)); answer = "Fehler"; } if (answer != null) print(answer); androidUI(r { if (goodbye) { print("\nGOODBYE!"); sr.destroy(); callOpt(getMainClass(), "disappear"); } else { sr.stopListening(); listening = false; //newRecognizer(); // always make a new one - gives endless errors if (!manualMode) recog(); } }); } // end of thread } svoid recog() { if (sr == null) ret with newRecognizer(); print("recog"); listening = true; justASec(); Intent intent = new Intent(RecognizerIntent.ACTION_RECOGNIZE_SPEECH); intent.putExtra(RecognizerIntent.EXTRA_LANGUAGE_MODEL,RecognizerIntent.LANGUAGE_MODEL_FREE_FORM); intent.putExtra(RecognizerIntent.EXTRA_LANGUAGE, language); intent.putExtra(RecognizerIntent.EXTRA_CALLING_PACKAGE,"voice.recognition.test"); intent.putExtra(RecognizerIntent.EXTRA_MAX_RESULTS, extraResults); // customize recognition time settings callOpt(mc(), "recogTimeSettings", intent); sr.startListening(intent); print("started listening"); } svoid say(fS s) { ping(); lock speechLock; showText(">> " + s); aSetText(myTextView, lang("Me: ", "Ich: ") + s); aPutViewBehindOtherView(myTextView, userTextView); if (manualMode && listening) { print("User is speaking, postponing speech output."); postponed.add(s); ret; } androidUI_wait(f stopListening); history.add(s); if (!noSpeech) { androidUnmuteAudio(); callOptMC('oh); if (fastSynthesis) { if (german()) androidSayInGerman(s); else androidSayInEnglish(s); } else { File mp3 = cereproc_silent(german() ? "Leopold" : "Jack", s); androidPlayMp3(androidActivity(), mp3); } callOptMC('happy); if (listenAfterTalk) //androidUI_noWait(r startListening); // works, but freezes UI androidUI_noWait(r newRecognizer); } if (muteAlways) androidMuteAudio(); } static void stopListening() { listening = false; if (sr != null) sr.stopListening(); } svoid startListening() { if (listening) ret; listening = true; recog(); } sbool german() { ret swic(language, "de"); } svoid switchToManual { stopListening(); manualMode = true; androidUnmuteAudio(); } svoid switchToAuto { manualMode = false; startListening(); androidUnmuteAudio(); } svoid onMenuCommand(S s) { //print("menu cmd: " + s); handleCommand(s); } // spoken, typed or through menu svoid handleCommand(S s) { if "stop listening|Switch to manual mode" switchToManual(); if "Switch to auto mode" switchToAuto(); if "german|deutsch" setLanguage("de-DE"); if "english|englisch" setLanguage("en-US"); } svoid cleanMeUp_leo { if (sr != null) { sr.destroy(); sr = null; } } sS baseStatus() { ret !manualMode ? "" : german() ? "KLICKEN ZUM SPRECHEN" : "CLICK ON ME IF YOU WANT TO TALK"; } sS lang(S en, S de) { ret german() ? de : en; } svoid setLanguage(S l) { language = l; setCurrentBotLanguage(l); aSetText(lblInputView, inputViewLabel()); androidUI_noWait(r newRecognizer); } svoid justASec { callOptMC('setBorderAndStatus, waitColor, german() ? "BITTE WARTEN" : "JUST A SEC"); // (initializing speech recognizer) } // Don't use - freezes UI svoid _cancelRecognition { //print("Cancelling recognition " + listening + " " + (sr != null)); recognitionCancelled = true; //stopListening(); //listening = false; //if (sr != null) sr.cancel(); //callOptMC('setBorderAndStatus, aWhite(), baseStatus()); } static void showText(S text) { print(text); /*if (neq(onScreenText, text) && lastEmotion != null) { onScreenText = text; emo_show(); }*/ } static void emo(S emotion) { if (!showPicture) return; int i; for (i = 0; i < emotions.size(); i += 2) if (emotions.get(i+1).equalsIgnoreCase(emotion)) break; if (i >= emotions.size()) { print("Emotion not found: " + emotion); // use the last one from the list as default i -= 2; } lastEmotionImage = emotions.get(i); lastEmotion = emotions.get(i+1); emo_show(); } static void emo_show() { if (!showPicture) return; androidUI { Runnable onClick = r { if (!manualMode) ret; //if (listening) if (borderColor != -1) { //androidLater(500, r { stopListening(); }); //showText("stop"); stopListening(); } else { //showText ("start"); newRecognizer(); } }; if (statusView == null) { // init UI statusView = aFontSize(statusFontSize, aSetForeground(aBlack(), aCenteredTextView())); inputView = aSingleLineEditText(); aOnEnter(inputView, r { onInput(aGetText(inputView), true) }); //aOnChange(inputView, f cancelRecognition); // freezes!? //aOnChange(inputView, r { listening = false }); //aOnChange(inputView, f cancelRecognition); lblInputView = aFontSize(20, aBottomTextView(inputViewLabel())); userTextView = aFontSize(userFontSize, aSetForeground(0xFF000055, aTextViewWithWindowFocusChangeNotify(voidfunc(Bool windowFocused) { if (windowFocused && recogOnActivate) newRecognizer(); }))); myTextView = aFontSize(myFontSize, aSetForeground(0xFF005500, aRightAlignedTextView()); androidShow(aVerticalLinearLayout( statusView, aWestCenterAndEast/*_alignTop2*/( leftFlag = aVerticalLinearLayout(androidClickableImage(#1101639, 0 /*transparent*/, r { setLanguage("en-US") })), emoView = androidClickableImage(lastEmotionImage, aWhite(), onClick), rightFlag = aVerticalLinearLayout(androidClickableImage(#1101638, 0 /*transparent*/, r { setLanguage("de-DE") })) ), userTextView, myTextView, androidPrintLogScrollView(), aWestAndCenter(lblInputView, inputView))); } if (statusText == null) statusText = baseStatus(); aSetText(statusView, statusText); aSetImageFromSnippet(emoView, lastEmotionImage); aSetBackground(emoView, borderColor); aSetBackground(eq(language, "en-US") ? borderColor : 0xFFCCCCCC, leftFlag); aSetBackground(eq(language, "de-DE") ? borderColor : 0xFFCCCCCC, rightFlag); /*doEvery(1000, new Runnable { S text = ""; public void run() { S s = aGetText(inputView); if (eq(s, text)) ret; text = s; cancelRecognition(); } });*/ } } static void setBorderAndStatus(int color, S status) { if (color != borderColor || neq(status, statusText)) { borderColor = color; statusText = status; if (lastEmotion != null) emo(lastEmotion); if (!(manualMode && listening) && nempty(postponed)) for (S s : getAndClearList(postponed)) say(s); } } static void setBorderColor(int color) { if (color != borderColor) { borderColor = color; if (lastEmotion != null) emo(lastEmotion); } } static void happy() { emo("happy"); } static void unhappy() { emo("unhappy"); } static void oh() { emo("oh"); } static void bored() { emo("bored"); } static void wideMouth() { emo("wide mouth"); } static void disappear() { if (!showPicture) ret; happy(); androidLater(1000, r { androidShowFullScreenColor(0xFFFFFFFF); androidLater(1000, r { System.exit(0); // totally unrecommended but works }); }); } sS inputViewLabel() { ret lang("Or type here:", "Oder hier tippen:"); } sinterface CatConnector { void sendLine(S line, bool typed); AutoCloseable startEar(VF1<S> onLine); } sclass CatConnectorImpl implements CatConnector { S token; *() {} *(S *token) {} public void sendLine(S line, bool typed) { postPage(earURL + "/heard", +token, +line, typed := typed ? "1" : null, +language); } public AutoCloseable startEar(final VF1<S> onLine) { thread "Long-Poll" { repeat with sleep 1 { pcall { // TODO: interrupt connection on cleanup S action = postPageSilently(earURL + "/poll", +token); for (S s : lines(action)) pcallF(onLine, s); } } } null; } }