Rendi disponibili le app per la TV

Android TV utilizza l'interfaccia di ricerca di Android per recuperare i dati sui contenuti dalle app installate e mostrare risultati di ricerca all'utente. I dati sui contenuti della tua app possono essere inclusi in questi risultati per offrire all'utente accesso immediato ai contenuti della tua app.

La tua app deve fornire ad Android TV i campi di dati da cui Android TV può generare risultati di ricerca suggeriti man mano che l'utente inserisce caratteri nella finestra di dialogo di ricerca. A questo scopo, l'app deve implementare un fornitore di contenuti che offra i suggerimenti insieme a un file di configurazione searchable.xml che descriva il fornitore di contenuti e altre informazioni fondamentali per Android TV. Devi anche avere un'attività che gestisca l'intent che viene attivato quando l'utente seleziona un risultato di ricerca suggerito. Per ulteriori dettagli, consulta l'articolo Aggiungere suggerimenti di ricerca personalizzati. Questa guida illustra i punti principali specifici delle app Android TV.

Prima di leggere questa guida, assicurati di acquisire familiarità con i concetti descritti nella guida all'API Search. Inoltre, consulta la sezione Aggiungere la funzionalità di ricerca.

Il codice di esempio in questa guida proviene dall' app di esempio Leanback.

Identifica colonne

SearchManager descrive i campi di dati previsti rappresentandoli come colonne di un database locale. Indipendentemente dal formato dei dati, devi mappare i campi di dati a queste colonne, in genere nella classe che accede ai dati dei contenuti. Per informazioni sulla creazione di una classe che mappa i dati esistenti ai campi obbligatori, consulta Creare una tabella di suggerimenti.

La classe SearchManager include diverse colonne per Android TV. Alcune delle colonne più importanti sono descritte nella seguente tabella.

Valore Descrizione
SUGGEST_COLUMN_TEXT_1 Il nome dei tuoi contenuti (obbligatorio)
SUGGEST_COLUMN_TEXT_2 Una descrizione testuale dei contenuti
SUGGEST_COLUMN_RESULT_CARD_IMAGE Un'immagine, un poster o una copertina per i tuoi contenuti
SUGGEST_COLUMN_CONTENT_TYPE Il tipo MIME del file multimediale
SUGGEST_COLUMN_VIDEO_WIDTH La larghezza di risoluzione dei contenuti multimediali.
SUGGEST_COLUMN_VIDEO_HEIGHT L'altezza della risoluzione dei contenuti multimediali.
SUGGEST_COLUMN_PRODUCTION_YEAR L'anno di produzione dei tuoi contenuti (obbligatorio)
SUGGEST_COLUMN_DURATION La durata in millisecondi dei contenuti multimediali (obbligatoria)

Il framework di ricerca richiede le seguenti colonne:

Quando i valori di queste colonne dei contenuti corrispondono ai valori relativi agli stessi contenuti di altri fornitori trovati dai server di Google, il sistema fornisce un link diretto alla tua app nella visualizzazione dei dettagli dei contenuti, insieme ai link alle app di altri fornitori. Questo argomento è spiegato più approfonditamente nella sezione Link diretto alla tua app nella schermata dei dettagli.

La classe del database dell'applicazione potrebbe definire le colonne come segue:

Kotlin

class VideoDatabase {
    companion object {
        // The columns we'll include in the video database table
        val KEY_NAME = SearchManager.SUGGEST_COLUMN_TEXT_1
        val KEY_DESCRIPTION = SearchManager.SUGGEST_COLUMN_TEXT_2
        val KEY_ICON = SearchManager.SUGGEST_COLUMN_RESULT_CARD_IMAGE
        val KEY_DATA_TYPE = SearchManager.SUGGEST_COLUMN_CONTENT_TYPE
        val KEY_IS_LIVE = SearchManager.SUGGEST_COLUMN_IS_LIVE
        val KEY_VIDEO_WIDTH = SearchManager.SUGGEST_COLUMN_VIDEO_WIDTH
        val KEY_VIDEO_HEIGHT = SearchManager.SUGGEST_COLUMN_VIDEO_HEIGHT
        val KEY_AUDIO_CHANNEL_CONFIG = SearchManager.SUGGEST_COLUMN_AUDIO_CHANNEL_CONFIG
        val KEY_PURCHASE_PRICE = SearchManager.SUGGEST_COLUMN_PURCHASE_PRICE
        val KEY_RENTAL_PRICE = SearchManager.SUGGEST_COLUMN_RENTAL_PRICE
        val KEY_RATING_STYLE = SearchManager.SUGGEST_COLUMN_RATING_STYLE
        val KEY_RATING_SCORE = SearchManager.SUGGEST_COLUMN_RATING_SCORE
        val KEY_PRODUCTION_YEAR = SearchManager.SUGGEST_COLUMN_PRODUCTION_YEAR
        val KEY_COLUMN_DURATION = SearchManager.SUGGEST_COLUMN_DURATION
        val KEY_ACTION = SearchManager.SUGGEST_COLUMN_INTENT_ACTION
        ...
    }
    ...
}

Java

public class VideoDatabase {
    // The columns we'll include in the video database table
    public static final String KEY_NAME = SearchManager.SUGGEST_COLUMN_TEXT_1;
    public static final String KEY_DESCRIPTION = SearchManager.SUGGEST_COLUMN_TEXT_2;
    public static final String KEY_ICON = SearchManager.SUGGEST_COLUMN_RESULT_CARD_IMAGE;
    public static final String KEY_DATA_TYPE = SearchManager.SUGGEST_COLUMN_CONTENT_TYPE;
    public static final String KEY_IS_LIVE = SearchManager.SUGGEST_COLUMN_IS_LIVE;
    public static final String KEY_VIDEO_WIDTH = SearchManager.SUGGEST_COLUMN_VIDEO_WIDTH;
    public static final String KEY_VIDEO_HEIGHT = SearchManager.SUGGEST_COLUMN_VIDEO_HEIGHT;
    public static final String KEY_AUDIO_CHANNEL_CONFIG =
            SearchManager.SUGGEST_COLUMN_AUDIO_CHANNEL_CONFIG;
    public static final String KEY_PURCHASE_PRICE = SearchManager.SUGGEST_COLUMN_PURCHASE_PRICE;
    public static final String KEY_RENTAL_PRICE = SearchManager.SUGGEST_COLUMN_RENTAL_PRICE;
    public static final String KEY_RATING_STYLE = SearchManager.SUGGEST_COLUMN_RATING_STYLE;
    public static final String KEY_RATING_SCORE = SearchManager.SUGGEST_COLUMN_RATING_SCORE;
    public static final String KEY_PRODUCTION_YEAR = SearchManager.SUGGEST_COLUMN_PRODUCTION_YEAR;
    public static final String KEY_COLUMN_DURATION = SearchManager.SUGGEST_COLUMN_DURATION;
    public static final String KEY_ACTION = SearchManager.SUGGEST_COLUMN_INTENT_ACTION;
...

Quando crei la mappa dalle colonne SearchManager ai campi dei dati, devi anche specificare il valore _ID per assegnare a ogni riga un ID univoco.

Kotlin


companion object {
    ....
    private fun buildColumnMap(): Map<String, String> {
        return mapOf(
          KEY_NAME to KEY_NAME,
          KEY_DESCRIPTION to KEY_DESCRIPTION,
          KEY_ICON to KEY_ICON,
          KEY_DATA_TYPE to KEY_DATA_TYPE,
          KEY_IS_LIVE to KEY_IS_LIVE,
          KEY_VIDEO_WIDTH to KEY_VIDEO_WIDTH,
          KEY_VIDEO_HEIGHT to KEY_VIDEO_HEIGHT,
          KEY_AUDIO_CHANNEL_CONFIG to KEY_AUDIO_CHANNEL_CONFIG,
          KEY_PURCHASE_PRICE to KEY_PURCHASE_PRICE,
          KEY_RENTAL_PRICE to KEY_RENTAL_PRICE,
          KEY_RATING_STYLE to KEY_RATING_STYLE,
          KEY_RATING_SCORE to KEY_RATING_SCORE,
          KEY_PRODUCTION_YEAR to KEY_PRODUCTION_YEAR,
          KEY_COLUMN_DURATION to KEY_COLUMN_DURATION,
          KEY_ACTION to KEY_ACTION,
          BaseColumns._ID to ("rowid AS " + BaseColumns._ID),
          SearchManager.SUGGEST_COLUMN_INTENT_DATA_ID to ("rowid AS " + SearchManager.SUGGEST_COLUMN_INTENT_DATA_ID),
          SearchManager.SUGGEST_COLUMN_SHORTCUT_ID to ("rowid AS " + SearchManager.SUGGEST_COLUMN_SHORTCUT_ID)
        )
    }
}

Java

...
  private static HashMap<String, String> buildColumnMap() {
    HashMap<String, String> map = new HashMap<String, String>();
    map.put(KEY_NAME, KEY_NAME);
    map.put(KEY_DESCRIPTION, KEY_DESCRIPTION);
    map.put(KEY_ICON, KEY_ICON);
    map.put(KEY_DATA_TYPE, KEY_DATA_TYPE);
    map.put(KEY_IS_LIVE, KEY_IS_LIVE);
    map.put(KEY_VIDEO_WIDTH, KEY_VIDEO_WIDTH);
    map.put(KEY_VIDEO_HEIGHT, KEY_VIDEO_HEIGHT);
    map.put(KEY_AUDIO_CHANNEL_CONFIG, KEY_AUDIO_CHANNEL_CONFIG);
    map.put(KEY_PURCHASE_PRICE, KEY_PURCHASE_PRICE);
    map.put(KEY_RENTAL_PRICE, KEY_RENTAL_PRICE);
    map.put(KEY_RATING_STYLE, KEY_RATING_STYLE);
    map.put(KEY_RATING_SCORE, KEY_RATING_SCORE);
    map.put(KEY_PRODUCTION_YEAR, KEY_PRODUCTION_YEAR);
    map.put(KEY_COLUMN_DURATION, KEY_COLUMN_DURATION);
    map.put(KEY_ACTION, KEY_ACTION);
    map.put(BaseColumns._ID, "rowid AS " +
            BaseColumns._ID);
    map.put(SearchManager.SUGGEST_COLUMN_INTENT_DATA_ID, "rowid AS " +
            SearchManager.SUGGEST_COLUMN_INTENT_DATA_ID);
    map.put(SearchManager.SUGGEST_COLUMN_SHORTCUT_ID, "rowid AS " +
            SearchManager.SUGGEST_COLUMN_SHORTCUT_ID);
    return map;
  }
...

Nell'esempio precedente, nota la mappatura al campo SUGGEST_COLUMN_INTENT_DATA_ID. Questa è la parte dell'URI che rimanda al contenuto univoco dei dati in questa riga, l'ultima parte dell'URI, che descrive dove sono archiviati i contenuti. La prima parte dell'URI, quando è comune a tutte le righe della tabella, viene impostata nel file searchable.xml come attributo android:searchSuggestIntentData, come descritto nella sezione Gestire i suggerimenti di ricerca.

Se la prima parte dell'URI è diversa per ogni riga nella tabella, mappa quel valore con il campo SUGGEST_COLUMN_INTENT_DATA. Quando l'utente seleziona questi contenuti, l'intent che si attiva fornisce i dati sull'intent provenienti dalla combinazione di SUGGEST_COLUMN_INTENT_DATA_ID e dell'attributo android:searchSuggestIntentData o del valore del campo SUGGEST_COLUMN_INTENT_DATA.

Fornisci dati sui suggerimenti di ricerca

Implementa un fornitore di contenuti per visualizzare i suggerimenti sui termini di ricerca nella finestra di dialogo di ricerca di Android TV. Il sistema invia una query al fornitore di contenuti per ricevere suggerimenti richiamando il metodo query() ogni volta che viene digitata una lettera. Nell'implementazione di query(), il fornitore di contenuti cerca i dati dei suggerimenti e restituisce un Cursor che rimanda alle righe designate per i suggerimenti.

Kotlin

fun query(uri: Uri, projection: Array<String>, selection: String, selectionArgs: Array<String>,
        sortOrder: String): Cursor {
    // Use the UriMatcher to see what kind of query we have and format the db query accordingly
    when (URI_MATCHER.match(uri)) {
        SEARCH_SUGGEST -> {
            Log.d(TAG, "search suggest: ${selectionArgs[0]} URI: $uri")
            if (selectionArgs == null) {
                throw IllegalArgumentException(
                        "selectionArgs must be provided for the Uri: $uri")
            }
            return getSuggestions(selectionArgs[0])
        }
        else -> throw IllegalArgumentException("Unknown Uri: $uri")
    }
}

private fun getSuggestions(query: String): Cursor {
    val columns = arrayOf<String>(
            BaseColumns._ID,
            VideoDatabase.KEY_NAME,
            VideoDatabase.KEY_DESCRIPTION,
            VideoDatabase.KEY_ICON,
            VideoDatabase.KEY_DATA_TYPE,
            VideoDatabase.KEY_IS_LIVE,
            VideoDatabase.KEY_VIDEO_WIDTH,
            VideoDatabase.KEY_VIDEO_HEIGHT,
            VideoDatabase.KEY_AUDIO_CHANNEL_CONFIG,
            VideoDatabase.KEY_PURCHASE_PRICE,
            VideoDatabase.KEY_RENTAL_PRICE,
            VideoDatabase.KEY_RATING_STYLE,
            VideoDatabase.KEY_RATING_SCORE,
            VideoDatabase.KEY_PRODUCTION_YEAR,
            VideoDatabase.KEY_COLUMN_DURATION,
            VideoDatabase.KEY_ACTION,
            SearchManager.SUGGEST_COLUMN_INTENT_DATA_ID
    )
    return videoDatabase.getWordMatch(query.toLowerCase(), columns)
}

Java

@Override
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs,
        String sortOrder) {
    // Use the UriMatcher to see what kind of query we have and format the db query accordingly
    switch (URI_MATCHER.match(uri)) {
        case SEARCH_SUGGEST:
            Log.d(TAG, "search suggest: " + selectionArgs[0] + " URI: " + uri);
            if (selectionArgs == null) {
                throw new IllegalArgumentException(
                        "selectionArgs must be provided for the Uri: " + uri);
            }
            return getSuggestions(selectionArgs[0]);
        default:
            throw new IllegalArgumentException("Unknown Uri: " + uri);
    }
}

private Cursor getSuggestions(String query) {
    query = query.toLowerCase();
    String[] columns = new String[]{
        BaseColumns._ID,
        VideoDatabase.KEY_NAME,
        VideoDatabase.KEY_DESCRIPTION,
        VideoDatabase.KEY_ICON,
        VideoDatabase.KEY_DATA_TYPE,
        VideoDatabase.KEY_IS_LIVE,
        VideoDatabase.KEY_VIDEO_WIDTH,
        VideoDatabase.KEY_VIDEO_HEIGHT,
        VideoDatabase.KEY_AUDIO_CHANNEL_CONFIG,
        VideoDatabase.KEY_PURCHASE_PRICE,
        VideoDatabase.KEY_RENTAL_PRICE,
        VideoDatabase.KEY_RATING_STYLE,
        VideoDatabase.KEY_RATING_SCORE,
        VideoDatabase.KEY_PRODUCTION_YEAR,
        VideoDatabase.KEY_COLUMN_DURATION,
        VideoDatabase.KEY_ACTION,
        SearchManager.SUGGEST_COLUMN_INTENT_DATA_ID
    };
    return videoDatabase.getWordMatch(query, columns);
}
...

Il fornitore di contenuti riceve un trattamento speciale nel file manifest. Anziché essere contrassegnata come attività, viene descritta come <provider>. Il provider include l'attributo android:authorities per indicare al sistema lo spazio dei nomi del fornitore di contenuti. Inoltre, devi impostare il relativo attributo android:exported su "true" in modo che la ricerca globale di Android possa utilizzare i risultati restituiti.

<provider android:name="com.example.android.tvleanback.VideoContentProvider"
    android:authorities="com.example.android.tvleanback"
    android:exported="true" />

Gestire i suggerimenti di ricerca

L'app deve includere un file res/xml/searchable.xml per configurare le impostazioni dei suggerimenti di ricerca.

Nel file res/xml/searchable.xml, includi l'attributo android:searchSuggestAuthority per indicare al sistema lo spazio dei nomi del tuo fornitore di contenuti. Deve corrispondere al valore della stringa specificato nell'attributo android:authorities dell'elemento <provider> nel file AndroidManifest.xml.

Includi anche un'etichetta, ovvero il nome dell'applicazione. Le impostazioni di ricerca di sistema utilizzano questa etichetta quando enumerano le app disponibili per la ricerca.

Il file searchable.xml deve includere anche android:searchSuggestIntentAction con il valore "android.intent.action.VIEW" per definire l'azione intent per fornire un suggerimento personalizzato. È diversa dall'azione per intenzione per fornire un termine di ricerca, come descritto nella sezione seguente. Per conoscere altri modi per dichiarare l'azione intent per i suggerimenti, consulta Dichiarazione dell'azione intent.

Insieme all'azione intent, la tua app deve fornire i dati sull'intent, che specifichi con l'attributo android:searchSuggestIntentData. Questa è la prima parte dell'URI che rimanda ai contenuti, che descrive la porzione dell'URI comune a tutte le righe della tabella di mappatura per i contenuti. La porzione dell'URI univoca per ogni riga viene stabilita con il campo SUGGEST_COLUMN_INTENT_DATA_ID, come descritto nella sezione Identifica colonne. Per altri modi per dichiarare i dati sugli intent per i suggerimenti, consulta Dichiarazione dei dati sull'intent.

L'attributo android:searchSuggestSelection=" ?" specifica il valore trasmesso come parametro selection del metodo query(). Il valore del punto interrogativo (?) viene sostituito con il testo della query.

Infine, devi includere anche l'attributo android:includeInGlobalSearch con il valore "true". Ecco un esempio di file searchable.xml:

<searchable xmlns:android="http://schemas.android.com/apk/res/android"
    android:label="@string/search_label"
    android:hint="@string/search_hint"
    android:searchSettingsDescription="@string/settings_description"
    android:searchSuggestAuthority="com.example.android.tvleanback"
    android:searchSuggestIntentAction="android.intent.action.VIEW"
    android:searchSuggestIntentData="content://com.example.android.tvleanback/video_database_leanback"
    android:searchSuggestSelection=" ?"
    android:searchSuggestThreshold="1"
    android:includeInGlobalSearch="true">
</searchable>

Gestire i termini di ricerca

Non appena la finestra di dialogo di ricerca contiene una parola che corrisponde al valore in una delle colonne dell'app, come descritto nella sezione Identifica colonne, il sistema attiva l'intent ACTION_SEARCH. L'attività nell'app che gestisce questo intent cerca nel repository le colonne contenenti la parola specificata nei relativi valori e restituisce un elenco di contenuti con quelle colonne. Nel file AndroidManifest.xml, indichi l'attività che gestisce l'intent ACTION_SEARCH come mostrato nell'esempio seguente:

...
  <activity
      android:name="com.example.android.tvleanback.DetailsActivity"
      android:exported="true">

      <!-- Receives the search request. -->
      <intent-filter>
          <action android:name="android.intent.action.SEARCH" />
          <!-- No category needed, because the Intent will specify this class component -->
      </intent-filter>

      <!-- Points to searchable meta data. -->
      <meta-data android:name="android.app.searchable"
          android:resource="@xml/searchable" />
  </activity>
...
  <!-- Provides search suggestions for keywords against video meta data. -->
  <provider android:name="com.example.android.tvleanback.VideoContentProvider"
      android:authorities="com.example.android.tvleanback"
      android:exported="true" />
...

L'attività deve inoltre descrivere la configurazione disponibile per la ricerca con un riferimento al file searchable.xml. Affinché tu possa utilizzare la finestra di dialogo di ricerca globale, il manifest deve descrivere l'attività che deve ricevere le query di ricerca. Il file manifest deve inoltre descrivere l'elemento <provider> , esattamente come viene descritto nel file searchable.xml.

Link diretto alla tua app nella schermata dei dettagli

Se hai impostato la configurazione della ricerca come descritto nella sezione Gestione dei suggerimenti di ricerca e mappato i campi SUGGEST_COLUMN_TEXT_1, SUGGEST_COLUMN_PRODUCTION_YEAR e SUGGEST_COLUMN_DURATION come descritto nella sezione Identifica colonne, nella schermata dei dettagli viene visualizzato un link diretto a un'azione di visualizzazione dei tuoi contenuti che viene avviata quando l'utente seleziona un risultato di ricerca:

Link diretto nella schermata dei dettagli

Quando l'utente seleziona il link della tua app, identificato dal pulsante **Disponibile su** nella schermata dei dettagli, il sistema avvia l'attività che gestisce ACTION_VIEW impostato come android:searchSuggestIntentAction con il valore "android.intent.action.VIEW" nel file searchable.xml.

Puoi anche configurare un intent personalizzato per avviare la tua attività. come dimostrato nell' app di esempio Leanback. Tieni presente che l'app di esempio avvia il proprio LeanbackDetailsFragment per mostrare i dettagli dei contenuti multimediali selezionati; nelle tue app, avvia immediatamente l'attività che riproduce i contenuti multimediali per salvare all'utente altri clic o due.

Comportamento di ricerca

La ricerca è disponibile in Android TV dalla schermata Home e dall'interno dell'app. I risultati di ricerca sono diversi per questi due casi.

Eseguire ricerche dalla schermata Home

Quando l'utente esegue una ricerca dalla schermata Home, il primo risultato viene visualizzato nella scheda di un'entità. Se ci sono app in grado di riprodurre i contenuti, nella parte inferiore della scheda viene visualizzato un link a ognuna:

Riproduzione dei risultati di ricerca TV

Non puoi inserire in modo programmatico un'app nella scheda dell'entità. Per essere incluse come opzione di riproduzione, i risultati di ricerca di un'app devono corrispondere a titolo, anno e durata dei contenuti cercati.

Sotto la scheda potrebbero essere disponibili altri risultati di ricerca. Per vederle, l'utente deve premere il telecomando e scorrere verso il basso. I risultati per ogni app vengono visualizzati in una riga separata. Non puoi controllare l'ordinamento delle righe. Sono elencate per prime le app che supportano le azioni di visualizzazione.

Risultati di ricerca TV

Cerca dall'app

L'utente può anche avviare una ricerca dall'app avviando il microfono dal telecomando o dal controller del controller. I risultati di ricerca vengono visualizzati in un'unica riga sopra i contenuti dell'app. L'app genera risultati di ricerca utilizzando il proprio provider di ricerca globale.

Risultati di ricerca in-app per TV

Scopri di più

Per scoprire di più sulla ricerca di un'app TV, leggi Integrare le funzionalità di ricerca di Android nell'app e Aggiungere la funzionalità di ricerca.

Per maggiori informazioni su come personalizzare l'esperienza di ricerca in-app con un SearchFragment, leggi Eseguire ricerche nelle app TV.