参考
概要
- スマホと Wear が接続されていると、Notification が Wear にも表示(同期)される。
- Wear では通知はカードとして表示され、このカードが表示されるところを context stream という。
- これまでの通知でももちろん Wear に表示されるが、Wear 用に Notification を拡張することができる。
Notification を作る
Notification の作成には
NotificationCompat.Builder を使う。これで作っておけば、システムがかってにスマホと Wear で通知の見た目を変えてくれる。
通知の発行には
NotificationManagerCompat を使う(NotificationManager ではなく)。
■ SmallIcon だけの Notification
int notificationId = 001;
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif);
// Notification を発行
NotificationManagerCompat notificationManager =
NotificationManagerCompat.from(this);
notificationManager.notify(notificationId, notificationBuilder.build());
スマホ
Wear(左: ホーム画面、右: タップした状態)
data:image/s3,"s3://crabby-images/dc7a9/dc7a983009ced8c4e2a45cb509018849c19dd114" alt=""
Wearのカードの右上に表示されるアイコンは setSmallIcon() で指定したものではなく、アプリアイコンになる。
タップしたときの背景色はどこから来てるのかよくわからない。アプリアイコンのカラーパレット?
■ タイトルとメッセージありの Notification
int notificationId = 001;
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
// Notification を発行
NotificationManagerCompat notificationManager =
NotificationManagerCompat.from(this);
notificationManager.notify(notificationId, notificationBuilder.build());
スマホ
Wear(左: ホーム画面、右: タップした状態)
data:image/s3,"s3://crabby-images/72b28/72b288b53d594fa929a0fda3de2bd71da70bb73a" alt=""
■ Content Intent ありの Notification
int notificationId = 001;
// Content Intent 用の PendingIntent を作成
Intent intent = new Intent(this, MainActivity.class);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, 0);
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent);
// Notification を発行
NotificationManagerCompat notificationManager =
NotificationManagerCompat.from(this);
notificationManager.notify(notificationId, notificationBuilder.build());
スマホ
Wear(左: ホーム画面、中央: タップした状態、右: タップしたあと左にスワイプ)
data:image/s3,"s3://crabby-images/20610/206104aebcdef590a8b7cdb32c20ad39ddf546e9" alt=""
data:image/s3,"s3://crabby-images/e2f41/e2f41217a56d1a5ce731effc4440e256483c16c2" alt=""
setContentIntent() で PendingIntent を指定すると、Open on phone(携帯で開く)が追加され、それをタップすると指定した PendingIntent が実行される。
■ InboxStyle の Notification
Intent intent = new Intent(this, MainActivity.class);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, 0);
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.setStyle(new NotificationCompat.InboxStyle()
.addLine("1行目")
.addLine("2行目")
.setContentTitle("インボックス タイトル")
.setSummaryText("+3 more"));
// Notification を発行
NotificationManagerCompat notificationManager =
NotificationManagerCompat.from(this);
notificationManager.notify(notificationId, notificationBuilder.build());
スマホ
Wear(左: ホーム画面、中央: タップした状態、右: タップしたあと左にスワイプ)
data:image/s3,"s3://crabby-images/6e865/6e865e5658be9cf5d92863f15bcec7de8434e69d" alt=""
data:image/s3,"s3://crabby-images/9f9b5/9f9b5a1b9a1c3897e8364150079d8d490bbef68e" alt=""
■ BigTextStyle の Notification
Intent intent = new Intent(this, MainActivity.class);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, 0);
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.setStyle(new NotificationCompat.BigTextStyle()
.bigText(getString(R.string.long_text)
.setContentTitle("ビッグテキスト")
.setSummaryText("サマリー"));
// Notification を発行
NotificationManagerCompat notificationManager =
NotificationManagerCompat.from(this);
notificationManager.notify(notificationId, notificationBuilder.build());
スマホ(上: 閉じてる状態、下: 開いた状態)
Wear(上: ホーム画面、中: タップした状態、下: 中でタップすると全文が見れる)
data:image/s3,"s3://crabby-images/2bbcc/2bbcc20ab52e84174224aa90b1a7c706679448b1" alt=""
■ BigPictureStyle の Notification
Intent intent = new Intent(this, MainActivity.class);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, 0);
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.setStyle(new NotificationCompat.BigPictureStyle()
.bigPicture(bitmap)
.setContentTitle("ビッグピクチャー")
.setSummaryText("サマリー"));
// Notification を発行
NotificationManagerCompat notificationManager =
NotificationManagerCompat.from(this);
notificationManager.notify(notificationId, notificationBuilder.build());
スマホ(上: 閉じてる状態、下: 開いた状態)
Wear(上: ホーム画面、下: タップした状態)
data:image/s3,"s3://crabby-images/0f9c7/0f9c779d76ae28ba146ed6c9acf2ffcadb90ae4a" alt=""
data:image/s3,"s3://crabby-images/3aceb/3acebdc1fb1772169c84154d7f9718193463b354" alt=""
bitPicture() で指定した画像が背景になる。カードがなく背景にセットされた画像を見れるページが追加される。
タイトルには BigPictureStyle.setContentTitle() が表示されるが、その下は setSummaryText() ではなく setContentText() が表示される。
Action ボタンを追加する
Gmail の Notification の Archive に相当するもの。
*アイコンがぼけているように見えますが、実際ぼけています。Open on phone と比較すると明らかにぼけています。
NotificationCompat.Builder の addAction() で、画像、テキスト、タップされたときの PendingIntent を指定する。
Intent intent = new Intent(this, MainActivity.class);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, 0);
// Action 用の PendingIntent を作成
Intent mapIntent = new Intent(Intent.ACTION_VIEW);
Uri geoUri = Uri.parse("geo:0,0?q=" + Uri.encode(location));
mapIntent.setData(geoUri);
PendingIntent mapPendingIntent = PendingIntent.getActivity(this, 0, mapIntent, 0);
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.addAction(R.drawable.ic_map, "Map", mapPendingIntent);
// Notification を発行
NotificationManagerCompat notificationManager =
NotificationManagerCompat.from(this);
notificationManager.notify(notificationId, notificationBuilder.build());
addAction() の第1引数に指定するアイコンは ActionBar のアイコンと同じサイズにする。
つまり、 32dp x 32dp(内枠 24dp x 24dp)。
Wear 側で表示される画像は、スマホの解像度に一致したリソース。スマホが xxhdpi の場合、hdpi のリソースがあっても、xxhdpi のリソースが Wear に表示される。
スマホ
Wear(上: ホーム画面、下: タップした状態)
data:image/s3,"s3://crabby-images/d70b1/d70b13fef139918d742bdf783bbdfa160f35b3ce" alt=""
data:image/s3,"s3://crabby-images/7d4a4/7d4a4166f8b6355d30bcdd6d68c46db2acd763b8" alt=""
Wear だけのアクションを追加
Wear にだけ表示されるアクションを追加できる。スマホの Notification には表示されない。
WearableExtender.addAction() を使う。
このメソッドを Action を追加すると、Wearable では NotificationCompatBuilder.addAction() で指定された Action は表示されない。
そのため、別の機能を WearableExtender.addAction() で与えるというよりは、スマホ用に NotificationCompatBuilder.addAction() で指定した機能を、Wear では Wear 用に適したものに置き換えたい場合に使う。
例えば、Gmail の返信機能は、スマホでは返信画面を開くだけだが、Wear では音声入力の結果を返信できるようにしている。
Intent intent = new Intent(this, MainActivity.class);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, 0);
// Action 用の PendingIntent を作成
Intent mapIntent = new Intent(Intent.ACTION_VIEW);
Uri geoUri = Uri.parse("geo:0,0?q=" + Uri.encode(location));
mapIntent.setData(geoUri);
PendingIntent mapPendingIntent = PendingIntent.getActivity(this, 0, mapIntent, 0);
// Wear 用 Action を作成
NotificationCompat.Action action =
new NotificationCompat.Action.Builder(R.drawable.ic_map_for_wear,
"Map for Wear", mapPendingIntent)
.build();
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.addAction(R.drawable.ic_map, "Map for phone", mapPendingIntent)
.extend(new NotificationCompat.WearableExtender().addAction(action));
// Notification を発行
NotificationManagerCompat notificationManager =
NotificationManagerCompat.from(this);
notificationManager.notify(notificationId, notificationBuilder.build());
Wear 用のアイコンには Action Bar の2倍、つまり 64dp x 64dp(内枠 48dp x 48dp)の画像を指定する。こうするとNotificationCompat.Builder.addAction() で Wear に表示していたときと違って画像がぼけない。
スマホ
Wear(上: ホーム画面、下: タップした状態)
data:image/s3,"s3://crabby-images/71338/71338aac7bbbffaf55cc0f172a1cbb13045670ff" alt=""
data:image/s3,"s3://crabby-images/2ee27/2ee2751af4949c5a4b4a2ec689ef81507cefd36b" alt=""
背景画像を指定する
■ setLargeIcon() を使う
setLargeIcon() を使うと、カードをタップしたときの背景に利用される。
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.setLargeIcon(bitmap);
BigPictureStyle を使った場合、setLargeIcon() を指定していても Wear の背景は BigPictureStyle.bigPicture() で指定した画像になる。
スマホ
Wear(左: ホーム画面、右: タップした状態)
data:image/s3,"s3://crabby-images/54373/5437341c5e62807499629d63c665fe3a568d4c3e" alt=""
data:image/s3,"s3://crabby-images/77ee8/77ee8fa8d28cd047e503702267bcec866075ea6c" alt=""
■ WearableExtender.setBackground() を使う
スマホの Notification で LargeIcon を使いたくない場合は、NotificationCompat.WearableExtender の setBackground() を使う。
NotificationCompat.WearableExtender wearableExtender =
new NotificationCompat.WearableExtender()
.setBackground(BitmapFactory.decodeResource(
getResources(), R.drawable.sample));
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.extend(wearableExtender);
スマホ
Wear(左: ホーム画面、右: タップした状態)
data:image/s3,"s3://crabby-images/a9b32/a9b3201999734bbe7e0f4a0fbafabaa30e8746be" alt=""
data:image/s3,"s3://crabby-images/08a34/08a34d7fdba25601ed6eb68347197e533005de37" alt=""
Wear のカードでアプリアイコンを表示しない
カードの右上に表示されるアプリアイコンを消すには、
NotificationCompat.WearableExtender.setHintHideIcon() を使う。
NotificationCompat.WearableExtender wearableExtender =
new NotificationCompat.WearableExtender()
.setHintHideIcon(true);
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.extend(wearableExtender);
Wear(左: ホーム画面、右: タップした状態)
data:image/s3,"s3://crabby-images/6ec68/6ec685563eee532b0644e693db9844872db4d944" alt=""
data:image/s3,"s3://crabby-images/dfa57/dfa57f1ad5fe96536dfa7908a4a6ea0d2c8cab18" alt=""
Wear のカード内にアイコンを表示する
NotificationCompat.WearableExtender.setContentIcon() を使う。
NotificationCompat.WearableExtender wearableExtender =
new NotificationCompat.WearableExtender()
.setContentIcon(R.drawable.ic_droid);
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.extend(wearableExtender);
ここでは、アプリアイコンサイズの画像を指定してみた。
ambient 時は白の tint がかかる。
Wear(上: ホーム画面、下: タップした状態)
data:image/s3,"s3://crabby-images/0fb8b/0fb8b54be61147e955c106aefc414bc7e7f2672c" alt=""
data:image/s3,"s3://crabby-images/a351f/a351fd254ed2133448b309792c42bc8d0369e909" alt=""
Wear のカード内にアイコンを左に表示する
NotificationCompat.WearableExtender. setContentIconGravity() で GravityCompat.START を指定する。
指定できるのは START と END で、デフォルトは END。
NotificationCompat.WearableExtender wearableExtender =
new NotificationCompat.WearableExtender()
.setContentIcon(R.drawable.ic_droid)
.setContentIconGravity(GravityCompat.START);
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.extend(wearableExtender);
Wear(上: ホーム画面、下: タップした状態)
data:image/s3,"s3://crabby-images/b32aa/b32aa63b68d65f847464a186b63233d5222899af" alt=""
カードの位置を変える
setGravity() を使う。指定できるのは Gravity.TOP、Gravity.CENTER_VERTICAL、Gravity.BOTTOM。デフォルトは BOTTOM。
NotificationCompat.WearableExtender wearableExtender =
new NotificationCompat.WearableExtender()
.setGravity(Gravity.TOP);
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.extend(wearableExtender);
Wear(左: ホーム画面、右: タップした状態)
Gravity.TOP
data:image/s3,"s3://crabby-images/23108/231089af7a4777881429fc94b830a532b73202d8" alt=""
data:image/s3,"s3://crabby-images/87034/87034ed1f6071d53198f57d1793f9ab9ef7749ce" alt=""
Gravity.CENTER_VERTICAL
data:image/s3,"s3://crabby-images/a3616/a3616022750715fd6d6c75797e5da8062eccc439" alt=""
data:image/s3,"s3://crabby-images/26181/26181bb8fa3c6381846338fdb1847923623db405" alt=""
オリジナルのカードを表示する
これは Wear アプリから Notification を発行するときにだけ使えます。
setDisplayIntent() で、カードをタップしたときに表示する Activity を持った PendingIntent を指定します。
Wear 上のコードなので、NotificationCompat ではなく Notification を使っています。
// この MainActivity は Wear アプリの MainActivity
Intent intent = new Intent(this, MainActivity.class);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, 0);
// カードをタップしたときに表示される Activity
// ここでは ImageView 1つだけのレイアウト
Intent displayIntent = new Intent(this, DisplayActivity.class);
PendingIntent displayPendingIntent = PendingIntent.getActivity(this,
0, displayIntent, PendingIntent.FLAG_UPDATE_CURRENT);
NotificationCompat.WearableExtender wearableExtender =
new NotificationCompat.WearableExtender()
.setDisplayIntent(displayPendingIntent);
Notification.Builder notificationBuilder =
new Notification.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.extend(wearableExtender);
NotificationManager notificationManager =
(NotificationManager) getSystemService(NOTIFICATION_SERVICE);
notificationManager.notify(notificationId, notificationBuilder.build());
Wear から Notification を発行する場合、setSmallIcon() で指定したアイコンがカードの右上に表示される。
setContentIntent() で指定した PendingIntent は Open(開く)という Action になる。
Wear(左: ホーム画面、右: タップした状態)
data:image/s3,"s3://crabby-images/ab567/ab567a7f59de88c0b6d8aa57174d1463ea466792" alt=""
data:image/s3,"s3://crabby-images/52131/521316a4084c7fcfc201bd7c01ddcb34e4261dd1" alt=""
オリジナルのカードの大きさを変える
setCustomSizePreset()を使う。
指定できるのは、SIZE_DEFAULT、SIZE_FULL_SCREEN、SIZE_LEARGE、SIZE_MEDIUM、SIZE_SMALL、SIZE_XSMALL。
NotificationCompat.WearableExtender wearableExtender =
new NotificationCompat.WearableExtender()
.setDisplayIntent(displayPendingIntent)
.setCustomSizePreset(NotificationCompat.WearableExtender.SIZE_FULL_SCREEN);
Notification.Builder notificationBuilder =
new Notification.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.extend(wearableExtender);
Wear(左: ホーム画面、右: タップした状態)
SIZE_FULL_SCREEN
data:image/s3,"s3://crabby-images/72f70/72f70367d899061ae4b7d3cf794129fd3f62952b" alt=""
data:image/s3,"s3://crabby-images/40c9a/40c9a7274a66632adaf157e12944548b277c110c" alt=""
SIZE_LARGE
data:image/s3,"s3://crabby-images/ba8ae/ba8aeef597601ae433405635f8b5474c1a1da176" alt=""
data:image/s3,"s3://crabby-images/97540/97540f06b77c49b147d9a27ddbcbfab613ed4a14" alt=""
SIZE_MEDIUM
data:image/s3,"s3://crabby-images/3b13e/3b13e30ea11cb3a6be0099ec745b6faff5e88f3a" alt=""
data:image/s3,"s3://crabby-images/73d14/73d14bed84b91807a28b60bcf33e599180760b0a" alt=""
SIZE_SMALL
data:image/s3,"s3://crabby-images/c7187/c71878b2e1fab906a37b9e80acf52c392c2ae5a9" alt=""
data:image/s3,"s3://crabby-images/a2c16/a2c161113bbec6b228dd004c305ec49c8b0b1d0e" alt=""
SIZE_XSMALL
data:image/s3,"s3://crabby-images/33aa0/33aa0dc39d3455814f00146780b404af731c82e8" alt=""
data:image/s3,"s3://crabby-images/81813/81813127e7e1a748ec60e9f367c069a9874a05ce" alt=""
Notification から音声入力を使う
Notification に返信するなどテキストを入力するアクションがある場合、Wear にはキーボードがない代わりに
RemoteInput を使って、音声入力を使うことができる。
*エミュレータには音声入力がないので、AVDの設定で Hardware keyboard present を有効にしておくと、代わりにキーボードで入力できる。
RemoteInput は
RemoteInput.Builder を使って作る。コントラクタには音声入力結果に紐づけるキー(文字列)を渡す。
private static final String EXTRA_VOICE_REPLY = "extra_voice_reply";
...
String replyLabel = getResources().getString(R.string.reply_label);
RemoteInput remoteInput = new RemoteInput.Builder(EXTRA_VOICE_REPLY)
.setLabel(replyLabel)
.build();
setLabel() でラベルを指定すると、上部の青色部分に表示される。
複数の項目を入力させたい場合などのに便利。
例えば、"Ok google, remind me" というと、About what? と When? を別々に入力する画面になる。
RemoteInput を Notification に組み込むには、NotificationCompat.Action.Builder の addRemoteInput() を使う。
// 音声入力の結果を受けとるための PendingIntent を作る
Intent replyIntent = new Intent(this, ReplyActivity.class);
PendingIntent replyPendingIntent =
PendingIntent.getActivity(this, 0, replyIntent,
PendingIntent.FLAG_UPDATE_CURRENT);
// RemoteInput 用の Action を作る
NotificationCompat.Action action =
new NotificationCompat.Action.Builder(R.drawable.ic_reply,
"Reply message", replyPendingIntent)
.addRemoteInput(remoteInput)
.build();
// WearableExtender に addAction() で RemoteInput の Action を追加
NotificationCompat.WearableExtender wearableExtender =
new NotificationCompat.WearableExtender()
.addAction(action);
Notification.Builder notificationBuilder =
new Notification.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("タイトル")
.setContentText("メッセージ")
.setContentIntent(pendingIntent)
.extend(wearableExtender);
音声入力が正しく終ると、NotificationCompat.Action.Builder の第3引数に指定した PendingIntent が実行される。
PendingIntent に指定された Activity もしくは Service では、getIntent() で取得した Intent を
RemoteInput.getResultsFromIntent() に渡して Bundle を取得し、RemoteInput に指定したキーで文字列を取り出す。
/**
* Activity.getIntent() を渡す
*/
private CharSequence getMessageText(Intent intent) {
Bundle remoteInput = RemoteInput.getResultsFromIntent(intent);
if (remoteInput != null) {
return remoteInput.getCharSequence(EXTRA_VOICE_REPLY);
}
return null;
}
* 音声入力の結果は ClipData として保存されているため、Intent.getExtras() を使わずに getResultsFromIntent() を使うこと。
Wear(上: ホーム画面、中: タップした状態、下: reply actionをタップ&入力した状態)
data:image/s3,"s3://crabby-images/02d19/02d191d332cdb7f65d66f945d25a797677308e2f" alt=""
data:image/s3,"s3://crabby-images/a6c77/a6c770d37fabec538bf3a89763e783a6af60e54e" alt=""
data:image/s3,"s3://crabby-images/4bc1f/4bc1f619a6801538b9b18f6e3d463df24d900c21" alt=""
setLabel() を指定しない場合。
data:image/s3,"s3://crabby-images/d5b59/d5b59605b90a50b16c239f9b100a8200bffc53ef" alt=""
■ 音声入力時に選択肢を与える
選択肢をあらかじめ用意しておくこともできる。選択肢は5つまで。setChoices() で文字列の配列を渡して指定する。
<?xml version="1.0" encoding="utf-8"?>
<resources>
<string-array name="reply_choices">
<item>Yes</item>
<item>No</item>
<item>Maybe</item>
</string-array>
</resources>
public static final EXTRA_VOICE_REPLY = "extra_voice_reply";
...
String replyLabel = getResources().getString(R.string.reply_label);
// 選択肢
String[] replyChoices = getResources().getStringArray(R.array.reply_choices);
RemoteInput remoteInput = new RemoteInput.Builder(EXTRA_VOICE_REPLY)
.setLabel(replyLabel)
.setChoices(replyChoices)
.build();
data:image/s3,"s3://crabby-images/1afaa/1afaa01ee65ce4fe735043e5adbfa80b735386b8" alt=""
data:image/s3,"s3://crabby-images/02749/02749b8e36ed2f2b4a6255575c64ec981cbf4e99" alt=""
■ 選択肢だけから選択させる
音声入力を使わず、選択肢だけから選択させることもできる。
そのためには、
setAllowFreeFormInput() で false を指定する。
この場合、setChoices() で選択肢を指定しておかないと IllegalArgumentException になる。
public static final EXTRA_VOICE_REPLY = "extra_voice_reply";
...
String replyLabel = getResources().getString(R.string.reply_label);
// 選択肢
String[] replyChoices = getResources().getStringArray(R.array.reply_choices);
RemoteInput remoteInput = new RemoteInput.Builder(EXTRA_VOICE_REPLY)
.setLabel(replyLabel)
.setChoices(replyChoices)
.setAllowFreeFormInput(false)
.build();
Page を追加する
追加の情報を表示したい場合などに、Page を追加することができる。
例えば、ハングアウトのカードでは、メインのカードに最新のメッセージが表示され、次のページに最近の投稿メッセージが表示される。
Google Now の天気カードでは、メインのカードに今日の天気、次のページに明日から四日間の天気が表示される。
data:image/s3,"s3://crabby-images/28f38/28f38b31ccaade43001dd330eaf5afc9bc7c2c8f" alt=""
Page を追加するには NotificationCompat.WearableExtender の
addPage() か
addPages() を使う。
// メインの Notification の NotificationCompat.Builder を作る
NotificationCompat.Builder notificationBuilder =
new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notif)
.setContentTitle("Page 1")
.setContentText("吾輩は猫である")
.setContentIntent(pendingIntent);
// 2ページ目の Notification を作る
Notification secondPageNotification =
new NotificationCompat.Builder(this)
.setContentTitle("Page 2")
.setContentText(getString(R.string.long_text))
.build();
// addPage() で2ページ目を追加し、extend() で1ページ目を拡張する
Notification twoPageNotification =
new NotificationCompat.WearableExtender()
.addPage(secondPageNotification)
.extend(notificationBuilder)
.build();
// Notification を発行
NotificationManagerCompat notificationManager
= NotificationManagerCompat.from(this);
notificationManager.notify(notificationId, twoPageNotification);
スマホ
Wear(上: ホーム画面、中: タップした状態)
data:image/s3,"s3://crabby-images/1886d/1886dff235c3e55209f7e6bdcc2432be19cf2d41" alt=""
data:image/s3,"s3://crabby-images/da032/da032154790eb5b49725521d682802076b4759e0" alt=""
Notification をまとめる
同じような通知はまとめて表示することが推奨されているが、Wearでは個別の通知の中がみれないのは不便。
そこで、1つのカードにグループ化し、カードをタップするとそれぞれの Notification が個別のカードに別れるよう作ることができる。
Gmailのカードはまさにこの形になっている。
カードをグループ化するには
NotificationCompat.Builder.setGroup() で同じ文字列を指定する。
final static String GROUP_KEY_EMAILS = "group_key_emails";
// Build the notification, setting the group appropriately
Notification notif = new NotificationCompat.Builder(mContext)
.setContentTitle("New mail from " + sender1)
.setContentText(subject1)
.setSmallIcon(R.drawable.new_mail);
.setGroup(GROUP_KEY_EMAILS)
.build();
// Issue the notification
NotificationManagerCompat notificationManager =
NotificationManagerCompat.from(this);
notificationManager.notify(notificationId1, notif);
setGroup() に同じ文字列を指定し、notify() で指定する ID は別にする。
Notification notif2 = new NotificationCompat.Builder(mContext)
.setContentTitle("New mail from " + sender2)
.setContentText(subject2)
.setSmallIcon(R.drawable.new_mail);
.setGroup(GROUP_KEY_EMAILS)
.build();
notificationManager.notify(notificationId2, notif2);
Wear(上: ホーム画面、中: タップした状態、下: 中をタップした状態)
data:image/s3,"s3://crabby-images/bd7c6/bd7c6240e23b5e901350f31a1be3d90a7be7c209" alt=""
グループ化されたカードがスタックされる順番は、新しく発行されたものが上になる(デフォルト)。
setSortKey() を使うと、順番を任意に指定することができる。
setGroup() を指定した Notification はスマホでは表示されないため、Summary Notification を用意する。
Summary Notification では setGroup() に同じ文字列を指定し、setGroupSummry() に true を指定する。この Summary Notification は Wear では表示されない。
Notification summaryNotification = new NotificationCompat.Builder(mContext)
.setContentTitle("2 new messages")
.setSmallIcon(R.drawable.ic_notif)
.setStyle(new NotificationCompat.InboxStyle()
.addLine(line1)
.addLine(line2)
.setBigContentTitle("2 new messages")
.setSummaryText(summary))
.setGroup(GROUP_KEY_EMAILS)
.setGroupSummary(true)
.build();
notificationManager.notify(notificationId3, summaryNotification);
スマホ
Summary Notification は Wear に直接表示されることはないが、
NotificationCompat.WearableExtender を使ってスタック全体の背景やスタック全体に対する Action を指定できる。
Bitmap background = BitmapFactory.decodeResource(getResources(),
R.drawable.ic_background);
NotificationCompat.WearableExtender wearableExtender =
new NotificationCompat.WearableExtender()
.setBackground(background);
// extend() で背景を指定
Notification summaryNotificationWithBackground =
new NotificationCompat.Builder(mContext)
.setContentTitle("2 new messages")
...
.extend(wearableExtender)
.setGroup(GROUP_KEY_EMAILS)
.setGroupSummary(true)
.build();
Wear
data:image/s3,"s3://crabby-images/d20af/d20afe49e8ab4f4ef0ced6cb819d453228d74ee3" alt=""
data:image/s3,"s3://crabby-images/f92dc/f92dc114b7b064c1c2a3419f2ad43e7ce5238474" alt=""