Что не так с моим преобразованием текста в речь в моем таймере обратного отсчета?

Привет всем, я пытаюсь включить преобразование текста в речь в моем CountDownTimer. Я хотел бы, чтобы через определенное время он сказал: «Осталось x секунд». Я только начал использовать TextToSpeech, и я не совсем уверен, что делаю.

    package com.android.countdown;


import java.util.Locale;
import android.app.Activity;
import android.os.Bundle;
import android.os.CountDownTimer;
import android.view.View;
import android.speech.tts.TextToSpeech;
import android.widget.Button;
import android.widget.TextView;
import android.view.View.OnClickListener;

public class countdown extends Activity implements TextToSpeech.OnInitListener{
 CountDownTimer Counter1;
 CountDownTimer Counter2;
 CountDownTimer Counter3;
 int Interval = 1;
 TextToSpeech tts;

    public String formatTime(long millis) {
          String output = "0:00";
          long seconds = millis / 1000;
          long minutes = seconds / 60;

          seconds = seconds % 60;
          minutes = minutes % 60;


          String secondsD = String.valueOf(seconds);
          String minutesD = String.valueOf(minutes);

          if (seconds < 10)
            secondsD = "0" + seconds;
          if (minutes < 10)
            minutesD = "0" + minutes;

          output = minutesD + " : " + secondsD;
          return output;
        }

   @Override
     public void onCreate(Bundle savedInstanceState) {
         super.onCreate(savedInstanceState);
         setContentView(R.layout.main);




//Declare Start/Stop timer
 Button btnstart = (Button)findViewById(R.id.btnstart);
 Button btnstop = (Button)findViewById(R.id.btnstop);

//Text field to show time left
 final TextView mCounter1TextField=(TextView)findViewById(R.id.counter1);
 final TextView mCounter2TextField = (TextView)findViewById(R.id.counter2);
 final TextView mCounter3TextField=(TextView)findViewById(R.id.counter3);



//Counter 1
Counter1 = new CountDownTimer(20000 , Interval) {
public void onTick(long millisUntilFinished){
    mCounter1TextField.setText("Seconds left: " + formatTime(millisUntilFinished));
    if (millisUntilFinished == 10000) {
     instantiate();
    }


}

public void onFinish() {
    Counter1.start();  
}
};

//Counter 2
Counter2 = new CountDownTimer(80000 , Interval) {
 public void onTick(long millisUntilFinished) {
     mCounter2TextField.setText("Seconds left: " + formatTime(millisUntilFinished));

 }

 public void onFinish() {
     mCounter2TextField.setText("Finished!");
     Counter2.start();
 }
 };

//Counter 3
Counter3 = new CountDownTimer(3000 , Interval) {
  public void onTick(long millisUntilFinished) {
      mCounter3TextField.setText("Seconds left: " + formatTime(millisUntilFinished));
  }

  public void onFinish() {
    mCounter3TextField.setText("Finished!");
    Counter3.start();
  }
  };


//Start Button
btnstart.setOnClickListener(new OnClickListener() {
 public void onClick(View v) {
  Counter1.start();
  Counter2.start();
  Counter3.start();
   }
});

//Stop Button
btnstop.setOnClickListener(new OnClickListener() {
 public void onClick(View v) {
  Counter1.cancel();
  Counter2.cancel();
  Counter3.cancel();
  if (tts != null) {
  tts.stop();
        tts.shutdown();
  }
    }
});
}

   public void instantiate() {
      tts = new TextToSpeech(this, this);
      tts.setLanguage(Locale.US);
         tts.speak("You have 10 seconds remaining", TextToSpeech.QUEUE_ADD, null);

   }

 @Override
 public void onInit(int status) {

 }

    @Override
    public void onDestroy() {
        // Don't forget to shutdown!
        if (tts != null) {
            tts.stop();
            tts.shutdown();
        }

        super.onDestroy();
    }


}

person MJ93    schedule 25.01.2011    source источник
comment
Если мой ответ неверен, вы должны опубликовать трассировку стека, которая более подробно объясняет полученную вами ошибку.   -  person McStretch    schedule 25.01.2011
comment
Также обратите внимание, что только строка tts=new TextToSpeech(.. является условной, если она истинна - у вас нет фигурных скобок вокруг двух других строк. Две другие ниже всегда выполняются. При первом запуске OnTick tts будет нулевым, и вы' получим исключение NullPointer. Во всяком случае, как сказал McStretch, это не подходящее место для его создания.   -  person NickT    schedule 25.01.2011
comment
@NickT: Хороший улов, я даже не заметил, что если бы был там. Я объединил ваш ответ со своим и упомянул вас.   -  person McStretch    schedule 25.01.2011


Ответы (2)


Ваш второй аргумент в tts = new TextToSpeech(this, this) не реализует TextToSpeech.OnInitListener.

Вам нужно, чтобы countdown или другой класс реализовывал TextToSpeech.OnInitListener:

public class countdown extends Activity implements TextToSpeech.OnInitListener {

Затем реализуйте onInit() в указанном классе:

void onInit(int status){
   // implementation 
}

И, наконец, передайте класс, реализующий OnInitListener, в конструктор TextToSpeech:

// The second 'this' will be replaced with another class if you 
// decide to use a class other than countdown to implement the interface.
tts = new TextToSpeech(this, this);

Ознакомьтесь с TextToSpeechActivity.java учебник для полного рабочего примера.

ИЗМЕНИТЬ

Как упоминал NickT, вам также необходимо добавить фигурные скобки в оператор if в onTick:

if (millisUntilFinished == 10000) {
    tts = new TextToSpeech(this, this);
    tts.setLanguage(Locale.US);
    tts.speak("You have 10 seconds remaining", TextToSpeech.QUEUE_ADD, null);
}

В противном случае вы всегда будете выполнять setLanguage и speak, что даст вам NullPointerException, если только millisUntilFinished == 10000 не истинно.


http://developer.android.com/reference/android/speech/tts/TextToSpeech.OnInitListener.html

person McStretch    schedule 25.01.2011
comment
Я добавил tts = new TextToSpeech(this, this); на мой public void OnInit() { потому что, если я оставлю его там, где находится if (millisUntilFinished == 100000), я все равно получу ту же ошибку. Однако странно то, что он не будет произносить текст, когда останется 10000 миллисекунд. Но если я поставлю if (millisUntilFinished ‹ 11000 && millisUntilFinished › 9000), то это сработает, но голос будет просто повторяться, пока я не закрою эмулятор. - person MJ93; 25.01.2011
comment
Итак, сейчас у меня есть tts = new TextToSpeech(this, this); под моим if (millisUntilFinished) == 10000 { процедура. Он говорит мне: конструктор new TextToSpeech(new CountDownTimer(){}, new CountDownTimer(){} не определен. - person MJ93; 26.01.2011
comment
О, извините, я не понял, что onTick был вложен в этот экземпляр объекта. Вот где интервал действительно помогает в вопросе: P. Вам потребуется реализовать TextToSpeech.OnInitListener в классе, а затем создать экземпляр объекта этого типа и передать его в качестве второго аргумента конструктору. - person McStretch; 26.01.2011
comment
Я внедрил TextToSpeech.OnInitListener в свой класс обратного отсчета. Но я не уверен, как создать второй аргумент конструктора. - person MJ93; 26.01.2011
comment
Я только что загрузил весь свой новый код, чтобы облегчить нам обоим задачу, ха-ха. Проверьте мой оригинальный пост. Я сделал оператор if 'instantiate()'. Проблема сейчас в том, что через 10 секунд речи не будет. - person MJ93; 26.01.2011

Другой подход, который не требует реализации TextToSpeech.OnInitListener в вашей деятельности, что означает более чистый код (на мой взгляд), это:

tts = new TextToSpeech(this, new TextToSpeech.OnInitListener() {
    @Override
    public void onInit(int status){
        if(status == TextToSpeech.SUCCESS) {
            Log.d("myapp", "TextToSpeech enabled");
        }
    }
});

Если вы хотите «попросить» данные tts. Я делаю это так:

// I have this code inside onCreate(), but you can call it elsewhere.
Intent checkIntent = new Intent();
checkIntent.setAction(TextToSpeech.Engine.ACTION_CHECK_TTS_DATA);
startActivityForResult(checkIntent, 6);

// Then, in the activity add this code:
@Override
protected void onActivityResult(final int requestCode, final int resultCode, final Intent intent) {
    if (requestCode == 6) {
        if (resultCode == TextToSpeech.Engine.CHECK_VOICE_DATA_PASS) {
            tts = new TextToSpeech(this, new TextToSpeech.OnInitListener() {
                  @Override
                  public void onInit(int status){
                      if(status == TextToSpeech.SUCCESS) {
                          Log.d("myapp", "TextToSpeech prepared");
                      }
                  }
              });
        }
    }
}

Почти такой же, как TextToSpeech.isLanguageAvailable(). В ближайшее время собираюсь менять.

person Jorge Fuentes González    schedule 08.06.2013