Поиск

Облако тэгов

Android | Андроид | ОС Андроид | КПК | Коммуникатор | Андроид-смартфон | Motorola | Samsung | HTC | LG | программы | Acer | OS Android | планшет | HTC Hero | устройство под управлением Андроид | Dell | операционная система Андроид | Android 2.3 | Gigabyte | Android 2.2 | Google | Froyo | Gingerbread | Android SDK | Galaxy | Garmin-Asus | Dopod | HTC Tattoo | описание | Sony Ericsson | плеер | Android 3.0 Honeycomb | Nexus S | HTC Click | HTC Magic | Super AMOLED | Motorolla | OS Android 2.2 | i7500 | OS Android 2.1 | NVIDIA Tegra 2 | eLocity A7 | Galaxy S | Android KitKat | Lenovo | Android 2.1 | QUMO Go | Андроид КитКат | Андроид KitKat | Alcatel | Планшет | Android Market | Skype | Gsmart Rola | ОС Андроид 4.4 | Android 4.0 Ice Cream Sandwich | Android 3.2 Honeycomb | Android 3.1 Honeycomb | Android 1.1 Bender | Android 1.5 Cupcake | Android 2.0.1 Eclair | Android 2.0 Eclair | Android 1.6 Donut | Android 2.3.3 Gingerbread | Android 4.4 | Android 2.1 Eclair | программы для HTC | OS Android 4.4 | nVidia Tegra | Андроид 4.4 | Android 2.3 Gingerbread | Android 2.2 Froyo | Android 1.0 | update | HTC Legend | Moto Morrison | PC | LiveCD | виджеты | Lenovo Ophone | Samsung InstinctQ M900 | Nexus | GW880 | Sygic | программирование | Donut | андроид-нетбук | Agora | Yuhua | Android-market | flash | android virtual devace | Rachel | Xperia X3 | I7500 | Google Nexus One | Lenovo LePhone | конвертеры | калькуляторы | офис | Milestone | office | ридеры | Nvidia Tegra 2 | android | андроид | HTC Droid Eris | HTC Desire | Android-смартфон | i1 | Google Maps | Navitel | GT-I8520 | Kyocera Zio M6000 | Zeus | HighScreen | GW620 | Pabasonic |

Устройства на базе ОС Андроид

 
Создание виджетов для Google Android.
16.07.2009 15:08

В состав пакета для разработчиков Android 1.5 входит AppWidget framework, предназначенный для создания виджетов, которые пользователь может размещать на рабочем столе своего смартфона. С помощью виджетов можно отобразить в удобной для пользователя форме информацию из различных приложений.

Например, можно создать виджет, который будет отображать список невыполненных дел или выдавать на экран информацию об играющей в фоновом режиме музыке.

Widget

Когда пользователь перетаскивает виждет на домашний экран, он фактически резервирует место, на котором будет отображаться контент из Вашего приложения. Кроме того, пользователь может взаимодействовать с Вашим приложением через виджет, например, приостанавливать проигрывание музыки. Если у Вас есть сервис, работающий в фоновом режиме, Вы можете обновлять виджет по собственному графику, или использовать стандартный механизм из AppWidget framework.

Виджет представляет собой BroadcastReceiver скрещенный с XML описанием параметров виждета. AppWidget framework связывается с Вашим виджетом через радиовещание, когда требуется обновление. Обновления строятся и посылаются с помощью RemoteViews, включающий контент, для отображения на домашней странице.

Вы без особого труда можете создать виджет для своего приложения. Давайте создадим виджет для Android приложения "Слово дня" (исходники). Я не буду рассматривать здесь это приложение, а сосредоточусь на самом виджете.

Прежде всего нужно создать XML описание виджета, указав там область домашнего экрана, которую Вы хотели бы зарезервировать для своего виджета, начальный слой для отображения, а также частоту обновления информации. Домашний экран Android разбит на ячейки, так что указанные вами размеры округляются до размеров ячеек. Это может сбить с толку. Возможно вам поможет вот эта формула:

Минимальный размер в dip=(Число ячеек * 74dip)-2dip

В нашем примере мы хотим создать виджет, который будет занимать 2 ячейки в ширину и 1 в высоту. Это значит, что минимальный размер должен составлять 146dip x 72dip. Мы хотим, чтобы информация обновлялась раз в день, то есть через каждые 86,400,000 миллисекунд. XML metadata имеет вид:


xmlns:android="http://schemas.android.com/apk/res/android"
android:minWidth="146dip"
android:minHeight="72dip"
android:initialLayout="@layout/widget_message"
android:updatePeriodMillis="86400000"
/>

Теперь давайте сопоставим эти XML данные и BroadcastReceiver в AndroidManifes

<!-- Broadcast Receiver that will process AppWidget updates -->
<receiver android:name=".WordWidget" android:label="@string/widget_name">
<intent-filter>
<action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
</intent-filter>
<meta-data android:name="android.appwidget.provider" android:resource="@xml/widget_word" />
</receiver>

<!-- Service to perform web API queries -->
<service android:name=".WordWidget$UpdateService" />

Давайте напишем код для BroadcastReceiver, который будет управлять AppWidget запросами. Чтобы помочь виджету управлять различными радиовещательными событиями, разработчики написали класс AppWidgetProvider. Важно отметить, что мы запустим фоновый сервис, чтобы выполнять обновления. Это связано с тем, что BroadcastReceivers - потомок Application Not Responding (ANR) таймера, который может предложить пользователю закрыть приложение, если запрос выполняется слишком долго. Выполнение веб запроса может занимать несколько секунд, поэтому мы используем именно сервисы, чтобы избежать ANR таймаутов.

/**
* Определяем простой виджет, который отображает слово дня. Для обновлений
* мы порождаем фоновой сервис {@link Service} для выполнения API запросов.
*/
public class WordWidget extends AppWidgetProvider {
@Override
public void onUpdate(Context context, AppWidgetManager appWidgetManager,
int[] appWidgetIds) {
// Чтобы предотвратить любые ANR таймацты, мы выполняем обновление в сервисе.
context.startService(new Intent(context, UpdateService.class));
}

public static class UpdateService extends Service {
@Override
public void onStart(Intent intent, int startId) {
// Выполняем сегодняшнее обновление виджета.
RemoteViews updateViews = buildUpdate(this);

// Помещаем обновление этого выджета на домашний экран
ComponentName thisWidget = new ComponentName(this, WordWidget.class);
AppWidgetManager manager = AppWidgetManager.getInstance(this);
manager.updateAppWidget(thisWidget, updateViews);
}

/**
* Строим обновление виджета, чтобы показать текущее слово.
* Заблокируем пока не получим online ответ.
*/
public RemoteViews buildUpdate(Context context) {
// Берем название месяца из ресурсов
Resources res = context.getResources();
String[] monthNames = res.getStringArray(R.array.month_names);

// ищем текущий месяц и день
Time today = new Time();
today.setToNow();

// Заголовок страницы в вмде "Wiktionary:Word of the day/March 21"
String pageName = res.getString(R.string.template_wotd_title,
monthNames[today.month], today.monthDay);
RemoteViews updateViews = null;
String pageContent = "";

try {
// Пытаемся послать запрос Wiktionary API для получения слова дня
SimpleWikiHelper.prepareUserAgent(context);
pageContent = SimpleWikiHelper.getPageContent(pageName, false);
} catch (ApiException e) {
Log.e("WordWidget", "Couldn't contact API", e);
} catch (ParseException e) {
Log.e("WordWidget", "Couldn't parse API response", e);
}

// Используем регулярное выражение для парсинга слов и их описания
Pattern pattern = Pattern.compile(SimpleWikiHelper.WORD_OF_DAY_REGEX);
Matcher matcher = pattern.matcher(pageContent);
if (matcher.find()) {
// Выполняем обновление контента виджета
updateViews = new RemoteViews(context.getPackageName(), R.layout.widget_word);

String wordTitle = matcher.group(1);
updateViews.setTextViewText(R.id.word_title, wordTitle);
updateViews.setTextViewText(R.id.word_type, matcher.group(2));
updateViews.setTextViewText(R.id.definition, matcher.group(3).trim());

// Когда пользователь кликает на виджет, запускается страница Wiktionary.
String definePage = res.getString(R.string.template_define_url,
Uri.encode(wordTitle));
Intent defineIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(definePage));
PendingIntent pendingIntent = PendingIntent.getActivity(context,
0 /* no requestCode */, defineIntent, 0 /* no flags */);
updateViews.setOnClickPendingIntent(R.id.widget, pendingIntent);

} else {
// Слово дня не найдено, показываем ошибку
updateViews = new RemoteViews(context.getPackageName(), R.layout.widget_message);
CharSequence errorMessage = context.getText(R.string.widget_error);
updateViews.setTextViewText(R.id.message, errorMessage);
}
return updateViews;
}

@Override
public IBinder onBind(Intent intent) {
// Мы не хотим привязываться к этому сервису
return null;
}
}
}

Все. Мы написали виджет для приложения Wiktionary "Слово дня". Когда требуется обновление, мы с помощью online API получаем новые данные. AppWidget framework автоматически запрашивает у нас обновления, например в момент добавления виджета на экран или согласно своему расписанию - раз в день.

Вообще виджеты стоит создавать для отображения не очень часто обновляемых данных (не чаще чем раз в час). Старайтесь обновлять информацию как можно реже и предусмотрите возможность ручного обновления.

Созданные таким образом виджеты можно размещать на любом домашнем экране, поддерживающим AppWidget framework.


По материалам сайта www.mobilab.ru

Обновлено 17.07.2009 09:30