Виджет, который вызывает приложение распознавания речи
Я пытаюсь создать виджет, содержащий один ImageView, который при нажатии запускает приложение распознавания речи. Я никогда не работал с виджетами и ожидающими намерений, поэтому я смущен: как создать ожидающее намерение начать работу распознавания речи?
Я пробовал что-то вроде этого, но это, конечно же, терпит неудачу:
Intent intent = new Intent();
Intent voiceIntent = new Intent(RecognizerIntent.ACTION_RECOGNIZE_SPEECH);
voiceIntent.putExtra(RecognizerIntent.EXTRA_LANGUAGE_MODEL,
RecognizerIntent.LANGUAGE_MODEL_FREE_FORM);
voiceIntent.putExtra(RecognizerIntent.EXTRA_PROMPT,
"Speech recognition demo");
voiceIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.putExtra(RecognizerIntent.EXTRA_RESULTS_PENDINGINTENT, voiceIntent);
PendingIntent pendingIntent = PendingIntent.getActivity(context, 0,
intent, 0);
RemoteViews views = new RemoteViews(context.getPackageName(),
R.layout.main);
views.setOnClickPendingIntent(R.id.button, pendingIntent);
Ответы
Ответ 1
Я понял! Мне нужны были два обычных намерения, завернутые в два ожидающих намерения, например:
// this intent points to activity that should handle results
Intent activityIntent = new Intent(context, ResultsActivity.class);
// this intent wraps results activity intent
PendingIntent resultsPendingIntent = PendingIntent.getActivity(context, 0, activityIntent, 0);
// this intent calls the speech recognition
Intent voiceIntent = new Intent(RecognizerIntent.ACTION_RECOGNIZE_SPEECH);
voiceIntent.putExtra(RecognizerIntent.EXTRA_LANGUAGE_MODEL, RecognizerIntent.LANGUAGE_MODEL_FREE_FORM);
voiceIntent.putExtra(RecognizerIntent.EXTRA_PROMPT, "Speech recognition demo");
voiceIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
voiceIntent.putExtra(RecognizerIntent.EXTRA_RESULTS_PENDINGINTENT, resultsPendingIntent);
// this intent wraps voice recognition intent
PendingIntent pendingIntent = PendingIntent.getActivity(context, 0, voiceIntent, 0);
rv.setOnClickPendingIntent(R.id.btn, pendingIntent);
Ответ 2
Я тоже сталкиваюсь с той же проблемой.
Извините, что у меня недостаточно репутации для комментариев.
Нет необходимости использовать прозрачную активность для отправки намерения распознавания.
Как ответ zorglub76
Intent voiceIntent = new Intent(RecognizerIntent.ACTION_RECOGNIZE_SPEECH);
voiceIntent.putExtra(RecognizerIntent.EXTRA_LANGUAGE_MODEL,RecognizerIntent.LANGUAGE_MODEL_FREE_FORM);
voiceIntent.putExtra(RecognizerIntent.EXTRA_PROMPT, "Speech recognition demo");
voiceIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
voiceIntent.putExtra(RecognizerIntent.EXTRA_RESULTS_PENDINGINTENT, resultsPendingIntent);
Результат распознавания будет только лишним resultingPendingIntent
Итак, все, что вам нужно сделать, это:
В ResultsActivity.onCreate()
ArrayList<String> voiceResults = this.getIntent().getExtras().getStringArrayList(RecognizerIntent.EXTRA_RESULTS);
Позаботьтесь о NullPointerException
, и вы получите результат из ArrayList!
Ответ 3
Я хотел создать Google как виджет. Я попробовал решение zorglub76, но я не смог получить результат...
Я решил это, создав фиктивную прозрачную активность, которая обрабатывает распознавание голоса от конца до конца.
Он работает следующим образом: Widget- > VoiceRecognitionStarterActivity- > RecognizerIntent- > VoiceRecognitionStarterActivity.onActivityResult.
Мой класс виджетов:
public class MyWidgetProvider extends AppWidgetProvider {
@Override
public void onUpdate(Context context, AppWidgetManager appWidgetManager,int[] appWidgetIds) {
// Get all ids
ComponentName thisWidget = new ComponentName(context, MyWidgetProvider.class);
int[] allWidgetIds = appWidgetManager.getAppWidgetIds(thisWidget);
for (int widgetId : allWidgetIds) {
RemoteViews remoteViews = new RemoteViews(context.getPackageName(), R.layout.widget_layout);
Intent activityIntent = new Intent(context, VoiceRecognitionStarterActivity.class);
PendingIntent pendingIntent = PendingIntent.getActivity(context, 0, activityIntent, 0);
remoteViews.setOnClickPendingIntent(R.id.mic_image, pendingIntent);
activityIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(context.getString(R.string.search_url)));
pendingIntent = PendingIntent.getActivity(context, 0, activityIntent, 0);
remoteViews.setOnClickPendingIntent(R.id.search_box_image, pendingIntent);
appWidgetManager.updateAppWidget(widgetId, remoteViews);
}
}
}
Моя прозрачная активность:
public class VoiceRecognitionStarterActivity extends Activity
{
private static final String TAG = "VoiceRecognitionStarterActivity";
private int SPEECH_REQUEST_CODE = 1;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
sendRecognizeIntent();
}
private void sendRecognizeIntent()
{
Intent intent = new Intent(RecognizerIntent.ACTION_RECOGNIZE_SPEECH);
intent.putExtra(RecognizerIntent.EXTRA_LANGUAGE_MODEL, RecognizerIntent.LANGUAGE_MODEL_FREE_FORM);
intent.putExtra(RecognizerIntent.EXTRA_PROMPT, "Speak to search");
intent.putExtra(RecognizerIntent.EXTRA_MAX_RESULTS, 10);
startActivityForResult(intent, SPEECH_REQUEST_CODE);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data)
{
if (requestCode == SPEECH_REQUEST_CODE)
{
if (resultCode == RESULT_OK) {
Log.d(TAG, "result ok");
Intent searchIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(getString(R.string.search_url)));
startActivity(searchIntent);
finish();
} else {
Log.d(TAG, "result NOT ok");
finish();
}
}
super.onActivityResult(requestCode, resultCode, data);
}
}
Чтобы сделать процесс прозрачным, см. this post
Ответ 4
Это полностью функционально и основано на виджетах ListView в Android SDK. Это не особенно для виджета, но я уверен, что вы можете изменить его, чтобы он работал для виджета.
Создайте действие под названием SearchActivity:
// CustomSearch (View) & ISearch (Interface) are objects that I created and are irrelevant
public class SearchActivity extends AppCompatActivity implements ISearch
{
// Variables
private CustomSearch mSearchView;
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_search);
mSearchView = (CustomSearch)findViewById(R.id.search);
mSearchView.setPendingComponentName(getComponentName());
mSearchView.setSearchListener(this);
}
@Override
protected void onNewIntent(Intent intent)
{
if (Intent.ACTION_SEARCH.equals(intent.getAction()))
{
String query = intent.getStringExtra(SearchManager.QUERY);
Log.i("SEARCH >", "You said: " + query);
}
}
}
Добавить активность в AndroidManifest.xml
<activity
android:name=".activities.SearchActivity"
android:label="@string/app_name"
android:theme="@style/CustomTheme.NoActionBar">
<intent-filter>
<action android:name="android.intent.action.SEARCH"/>
</intent-filter>
</activity>
В вашем пользовательском Widget/View:
buttonVoice.setOnClickListener(new View.OnClickListener()
{
@Override
public void onClick(View v)
{
// Get activity from either SearchableInfo or ComponentName
ComponentName searchActivity = mComponentName;
// Wrap component in intent
Intent queryIntent = new Intent(Intent.ACTION_SEARCH);
queryIntent.setComponent(searchActivity);
// Wrap query intent in pending intent
PendingIntent pending = PendingIntent.getActivity(getContext(), 0, queryIntent, PendingIntent.FLAG_ONE_SHOT);
// Create bundle now because if we wrap it in pending intent, it becomes immutable
Bundle queryExtras = new Bundle();
// Create voice intent
Intent voiceIntent = new Intent(RecognizerIntent.ACTION_RECOGNIZER_SPEECH);
voiceIntent.putExtra(RecognizerIntent.EXTRA_LANGUAGE_MODEL, RecognizerIntent.ACTION_RECOGNIZE_SPEECH);
voiceIntent.putExtra(RecognizerIntent.EXTRA_PROMPT, "Speak");
voiceIntent.putExtra(RecognizerIntent.EXTRA_CALLING_PACKAGE, searchActivity
voiceIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
// Wrap the pending intent & bundle inside the voice intent
voiceIntent.putExtra(RecognizerIntent.EXTRA_RESULTS_PENDINGINTENT, pending);
voiceIntent.putExtra(RecognizerIntent.EXTRA_RESULTS_PENDINGINTENT_BUNDLE, queryExtras);
// Start the voice search
getContext().startActivity(voiceIntent);
}
}