Esta página explica as práticas recomendadas para criar um widget mais avançado para uma melhor experiência do usuário.
Otimizações para atualizar o conteúdo do widget
Atualizar o conteúdo do widget pode ser caro em termos computacionais. Para economizar o consumo de bateria, otimize o tipo, a frequência e o momento da atualização.
Tipos de atualizações de widgets
Há três maneiras de atualizar um widget: uma atualização completa, uma parcial e, no caso de um widget de coleção, uma atualização de dados. Cada um tem custos e ramificações computacionais diferentes.
A seguir, descrevemos cada tipo de atualização e fornecemos snippets de código para cada um.
Atualização completa:chame
AppWidgetManager.updateAppWidget(int, android.widget.RemoteViews)
. para atualizar completamente o widget. Isso substitui o modeloRemoteViews
com um novoRemoteViews
. Essa é a atualização mais cara em termos computacionais.Kotlin
val appWidgetManager = AppWidgetManager.getInstance(context) val remoteViews = RemoteViews(context.getPackageName(), R.layout.widgetlayout).also { setTextViewText(R.id.textview_widget_layout1, "Updated text1") setTextViewText(R.id.textview_widget_layout2, "Updated text2") } appWidgetManager.updateAppWidget(appWidgetId, remoteViews)
Java
AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context); RemoteViews remoteViews = new RemoteViews(context.getPackageName(), R.layout.widgetlayout); remoteViews.setTextViewText(R.id.textview_widget_layout1, "Updated text1"); remoteViews.setTextViewText(R.id.textview_widget_layout2, "Updated text2"); appWidgetManager.updateAppWidget(appWidgetId, remoteViews);
Atualização parcial:chamada
AppWidgetManager.partiallyUpdateAppWidget
para atualizar partes do widget. Isso mescla o novoRemoteViews
com oRemoteViews
fornecido anteriormente. Esse método será ignorado se um widget não receber pelo menos uma atualização completa peloupdateAppWidget(int[], RemoteViews)
.Kotlin
val appWidgetManager = AppWidgetManager.getInstance(context) val remoteViews = RemoteViews(context.getPackageName(), R.layout.widgetlayout).also { setTextViewText(R.id.textview_widget_layout, "Updated text") } appWidgetManager.partiallyUpdateAppWidget(appWidgetId, remoteViews)
Java
AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context); RemoteViews remoteViews = new RemoteViews(context.getPackageName(), R.layout.widgetlayout); remoteViews.setTextViewText(R.id.textview_widget_layout, "Updated text"); appWidgetManager.partiallyUpdateAppWidget(appWidgetId, remoteViews);
Atualização de dados da coleção: chame
AppWidgetManager.notifyAppWidgetViewDataChanged
para invalidar os dados de uma visualização de coleção no widget. Isso acionaRemoteViewsFactory.onDataSetChanged
. Enquanto isso, os dados antigos vão aparecer no widget. É possível realizar tarefas caras de forma síncrona com segurança com esse método.Kotlin
val appWidgetManager = AppWidgetManager.getInstance(context) appWidgetManager.notifyAppWidgetViewDataChanged(appWidgetId, R.id.widget_listview)
Java
AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context); appWidgetManager.notifyAppWidgetViewDataChanged(appWidgetId, R.id.widget_listview);
É possível chamar esses métodos em qualquer lugar do app, desde que ele tenha o
mesmo UID da classe
AppWidgetProvider
correspondente.
Determinar a frequência de atualização de um widget
Os widgets são atualizados periodicamente, dependendo do valor fornecido para o
atributo
updatePeriodMillis
. O widget pode ser atualizado em resposta à interação do usuário,
atualizações ou ambos.
Atualizar periodicamente
É possível controlar a frequência da atualização periódica especificando um valor para
AppWidgetProviderInfo.updatePeriodMillis
no XML appwidget-provider
. Cada
atualização aciona o método AppWidgetProvider.onUpdate()
, que é onde você
inserir o código para atualizar o widget. No entanto, considere as alternativas para
atualizações de broadcast receiver descritas em
seção a seguir se o widget precisar carregar dados de forma assíncrona ou demorar mais
de 10 segundos para atualizar, porque depois de 10 segundos, o sistema considera uma
BroadcastReceiver
não responda.
O updatePeriodMillis
não aceita valores menores que 30 minutos. No entanto, se
Se quiser desativar as atualizações periódicas, especifique 0.
Você pode permitir que os usuários ajustem a frequência das atualizações em uma configuração. Para
Por exemplo, ele pode querer que o mostrador da bolsa seja atualizado a cada 15 minutos ou somente a
vezes por dia. Nesse caso, defina updatePeriodMillis
como 0 e use
WorkManager
.
Atualizar em resposta a uma interação do usuário
Aqui estão algumas maneiras recomendadas de atualizar o widget com base na interação do usuário:
A partir de uma atividade do app:chame diretamente
AppWidgetManager.updateAppWidget
em resposta a uma interação do usuário, como com o toque do usuário.De interações remotas, como uma notificação ou um widget de app: crie um
PendingIntent
e atualize o widget usando oActivity
,Broadcast
ouService
invocado. Você pode escolher sua própria prioridade. Para Por exemplo, se você selecionar umBroadcast
para aPendingIntent
, poderá escolher uma transmissão em primeiro plano para dar PrioridadeBroadcastReceiver
.
Atualizar em resposta a um evento de transmissão
Um exemplo de evento de transmissão que requer a atualização de um widget é quando a o usuário tira uma foto. Nesse caso, você quer atualizar o widget quando uma nova foto for detectada.
É possível programar um job com JobScheduler
e especificar uma transmissão como a
usando o
JobInfo.Builder.addTriggerContentUri
.
Também é possível registrar um BroadcastReceiver
para a transmissão, por exemplo,
detectando
ACTION_LOCALE_CHANGED
.
No entanto, como isso consome recursos do dispositivo, use com cuidado e ouça
apenas para a transmissão específica. Com a introdução das limitações
de transmissão no Android
7.0 (nível 24 da API) e no Android 8.0 (nível 26 da API), os apps não podem registrar transmissões
implícitas nos manifestos, com algumas
exceções.
Considerações ao atualizar um widget de um BroadcastReceiver
Se o widget for atualizado com base em uma BroadcastReceiver
, incluindo
AppWidgetProvider
, lembre-se das seguintes considerações sobre o
duração e prioridade de uma atualização de widget.
Duração da atualização
Como regra geral, o sistema permite que os broadcast receivers, que normalmente são executados no linha de execução principal, execute por até 10 segundos antes de considerar que não são responsivas e para acionar um aplicativo não Erro de resposta (ANR). Se demorar mais para atualizar o widget, considere as seguintes alternativas:
Programe uma tarefa usando
WorkManager
.Dê mais tempo ao receptor com o método
goAsync
. Isso permite que os receptores sejam executados por 30 segundos.
Consulte Considerações de segurança e práticas recomendadas para mais informações.
Prioridade da atualização
Por padrão, as transmissões, incluindo as feitas usando
AppWidgetProvider.onUpdate
, são executadas como processos em segundo plano. Isso significa
que recursos do sistema sobrecarregados podem causar um atraso na invocação do
broadcast receiver. Para priorizar a transmissão, torne-a um processo em primeiro plano.
Por exemplo, adicione o
Intent.FLAG_RECEIVER_FOREGROUND
flag para o Intent
transmitida ao PendingIntent.getBroadcast
quando o usuário
toca em determinada parte do widget.
Crie visualizações precisas que incluam itens dinâmicos
Esta seção explica a abordagem recomendada para mostrar vários itens em
uma visualização de widget com uma visualização
de coleção, ou seja, um widget que usa um
ListView
, GridView
ou StackView
.
Caso seu widget use uma dessas visualizações, criar uma visualização escalonável diretamente fornecendo o widget real layout prejudica a quando a visualização do widget não exibe itens. Isso ocorre porque dados de visualização de coleção são definidos dinamicamente no tempo de execução e são semelhantes à como mostrado na figura 1.
Para que as visualizações de widgets com visualizações de coleção sejam exibidas corretamente no widget
seletor, recomendamos manter um arquivo de layout separado, designado apenas para o
prévia. Esse arquivo de layout separado inclui o layout real do widget e uma
visualização de coleção de marcador de posição com itens falsos. Por exemplo, é possível imitar um
ListView
fornecendo um marcador de posição LinearLayout
com vários itens de lista
falsos.
Para ilustrar um exemplo de ListView
, comece com um arquivo de layout separado:
// res/layout/widget_preview.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@drawable/widget_background"
android:orientation="vertical">
// Include the actual widget layout that contains ListView.
<include
layout="@layout/widget_view"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
// The number of fake items you include depends on the values you provide
// for minHeight or targetCellHeight in the AppWidgetProviderInfo
// definition.
<TextView android:text="@string/fake_item1"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginVertical="?attr/appWidgetInternalPadding" />
<TextView android:text="@string/fake_item2"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginVertical="?attr/appWidgetInternalPadding" />
</LinearLayout>
Especifique o arquivo de layout da visualização ao fornecer o atributo previewLayout
do
os metadados AppWidgetProviderInfo
. Você ainda especifica o layout real do widget
para o atributo initialLayout
e usa o layout real do widget ao
criar uma RemoteViews
no momento da execução.
<appwidget-provider
previewLayout="@layout/widget_previe"
initialLayout="@layout/widget_view" />
Itens de lista complexos
O exemplo na seção anterior apresenta itens de lista falsos, porque a lista
itens são objetos TextView
. Pode ser
mais complexo para fornecer itens falsos se os itens forem layouts complexos.
Considere um item de lista definido em widget_list_item.xml
e que consiste em
dois objetos TextView
:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView android:id="@id/title"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="@string/fake_title" />
<TextView android:id="@id/content"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="@string/fake_content" />
</LinearLayout>
Para fornecer itens de lista falsos, você pode incluir o layout várias vezes, mas isso faz com que cada item de lista seja idêntico. Para fornecer itens de lista exclusivos, siga estas etapas:
Crie um conjunto de atributos para os valores de texto:
<resources> <attr name="widgetTitle" format="string" /> <attr name="widgetContent" format="string" /> </resources>
Use estes atributos para definir o texto:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="wrap_content"> <TextView android:id="@id/title" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="?widgetTitle" /> <TextView android:id="@id/content" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="?widgetContent" /> </LinearLayout>
Crie quantos estilos forem necessários para a visualização. Redefina os valores em cada estilo:
<resources> <style name="Theme.Widget.ListItem"> <item name="widgetTitle"></item> <item name="widgetContent"></item> </style> <style name="Theme.Widget.ListItem.Preview1"> <item name="widgetTitle">Fake Title 1</item> <item name="widgetContent">Fake content 1</item> </style> <style name="Theme.Widget.ListItem.Preview2"> <item name="widgetTitle">Fake title 2</item> <item name="widgetContent">Fake content 2</item> </style> </resources>
Aplique os estilos aos itens falsos no layout de visualização:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="wrap_content" ...> <include layout="@layout/widget_view" ... /> <include layout="@layout/widget_list_item" android:theme="@style/Theme.Widget.ListItem.Preview1" /> <include layout="@layout/widget_list_item" android:theme="@style/Theme.Widget.ListItem.Preview2" /> </LinearLayout>