• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2014 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.example.android.wearable.synchronizednotifications;
18 
19 import android.app.PendingIntent;
20 import android.content.Intent;
21 import android.support.v4.app.Fragment;
22 import android.app.Activity;
23 import android.net.Uri;
24 import android.os.Bundle;
25 import android.support.v4.app.NotificationCompat;
26 import android.support.v4.app.NotificationManagerCompat;
27 import android.util.Log;
28 import android.view.LayoutInflater;
29 import android.view.MenuItem;
30 import android.view.View;
31 import android.view.ViewGroup;
32 import android.widget.Toast;
33 
34 import com.example.android.wearable.synchronizednotifications.common.Constants;
35 import com.google.android.gms.common.ConnectionResult;
36 import com.google.android.gms.common.api.GoogleApiClient;
37 import com.google.android.gms.common.api.ResultCallback;
38 import com.google.android.gms.wearable.DataApi;
39 import com.google.android.gms.wearable.PutDataMapRequest;
40 import com.google.android.gms.wearable.PutDataRequest;
41 import com.google.android.gms.wearable.Wearable;
42 
43 import java.text.SimpleDateFormat;
44 import java.util.Date;
45 import java.util.Locale;
46 
47 
48 /**
49  * A simple fragment that presents three buttons that would trigger three different combinations of
50  * notifications on the handset and the watch:
51  * <ul>
52  * <li>The first button builds a simple local-only notification on the handset.</li>
53  * <li>The second one creates a wearable-only notification by putting a data item in the shared data
54  * store and having a {@link com.google.android.gms.wearable.WearableListenerService} listen for
55  * that on the wearable</li>
56  * <li>The third one creates a local notification and a wearable notification by combining the above
57  * two. It, however, demonstrates how one can set things up so that the dismissal of one
58  * notification results in the dismissal of the other one.</li>
59  * </ul>
60  */
61 public class SynchronizedNotificationsFragment extends Fragment
62         implements GoogleApiClient.ConnectionCallbacks, GoogleApiClient.OnConnectionFailedListener {
63 
64     private static final String TAG = "SynchronizedNotificationsFragment";
65     private GoogleApiClient mGoogleApiClient;
66 
67     @Override
onCreate(Bundle savedInstanceState)68     public void onCreate(Bundle savedInstanceState) {
69         super.onCreate(savedInstanceState);
70         mGoogleApiClient = new GoogleApiClient.Builder(this.getActivity())
71                 .addApi(Wearable.API)
72                 .addConnectionCallbacks(this)
73                 .addOnConnectionFailedListener(this)
74                 .build();
75         setHasOptionsMenu(true);
76     }
77 
78     @Override
onOptionsItemSelected(MenuItem item)79     public boolean onOptionsItemSelected(MenuItem item) {
80         switch (item.getItemId()) {
81             case R.id.btn_phone_only:
82                 buildLocalOnlyNotification(getString(R.string.phone_only), now(),
83                         Constants.PHONE_ONLY_ID, false);
84                 return true;
85             case R.id.btn_wear_only:
86                 buildWearableOnlyNotification(getString(R.string.wear_only), now(),
87                         Constants.WATCH_ONLY_PATH);
88                 return true;
89             case R.id.btn_different:
90                 buildMirroredNotifications(getString(R.string.phone_both), getString(R.string.watch_both), now());
91                 return true;
92         }
93         return false;
94     }
95 
96     /**
97      * Builds a local-only notification for the handset. This is achieved by using
98      * <code>setLocalOnly(true)</code>. If <code>withDismissal</code> is set to <code>true</code>, a
99      * {@link android.app.PendingIntent} will be added to handle the dismissal of notification to
100      * be able to remove the mirrored notification on the wearable.
101      */
buildLocalOnlyNotification(String title, String content, int notificationId, boolean withDismissal)102     private void buildLocalOnlyNotification(String title, String content, int notificationId,
103                                             boolean withDismissal) {
104         NotificationCompat.Builder builder = new NotificationCompat.Builder(this.getActivity());
105         builder.setContentTitle(title)
106                 .setContentText(content)
107                 .setLocalOnly(true)
108                 .setSmallIcon(R.drawable.ic_launcher);
109 
110         if (withDismissal) {
111             Intent dismissIntent = new Intent(Constants.ACTION_DISMISS);
112             dismissIntent.putExtra(Constants.KEY_NOTIFICATION_ID, Constants.BOTH_ID);
113             PendingIntent pendingIntent = PendingIntent
114                     .getService(this.getActivity(), 0, dismissIntent, PendingIntent.FLAG_UPDATE_CURRENT);
115             builder.setDeleteIntent(pendingIntent);
116         }
117         NotificationManagerCompat.from(this.getActivity()).notify(notificationId, builder.build());
118     }
119 
120     /**
121      * Builds a DataItem that on the wearable will be interpreted as a request to show a
122      * notification. The result will be a notification that only shows up on the wearable.
123      */
buildWearableOnlyNotification(String title, String content, String path)124     private void buildWearableOnlyNotification(String title, String content, String path) {
125         if (mGoogleApiClient.isConnected()) {
126             PutDataMapRequest putDataMapRequest = PutDataMapRequest.create(path);
127             putDataMapRequest.getDataMap().putString(Constants.KEY_CONTENT, content);
128             putDataMapRequest.getDataMap().putString(Constants.KEY_TITLE, title);
129             PutDataRequest request = putDataMapRequest.asPutDataRequest();
130             Wearable.DataApi.putDataItem(mGoogleApiClient, request)
131                     .setResultCallback(new ResultCallback<DataApi.DataItemResult>() {
132                         @Override
133                         public void onResult(DataApi.DataItemResult dataItemResult) {
134                             if (!dataItemResult.getStatus().isSuccess()) {
135                                 Log.e(TAG, "buildWatchOnlyNotification(): Failed to set the data, "
136                                         + "status: " + dataItemResult.getStatus().getStatusCode());
137                             }
138                         }
139                     });
140         } else {
141             Log.e(TAG, "buildWearableOnlyNotification(): no Google API Client connection");
142         }
143     }
144 
145     /**
146      * Builds a local notification and sets a DataItem that will be interpreted by the wearable as
147      * a request to build a notification on the wearable as as well. The two notifications show
148      * different messages.
149      * Dismissing either of the notifications will result in dismissal of the other; this is
150      * achieved by creating a {@link android.app.PendingIntent} that results in removal of
151      * the DataItem that created the watch notification. The deletion of the DataItem is observed on
152      * both sides, using WearableListenerService callbacks, and is interpreted on each side as a
153      * request to dismiss the corresponding notification.
154      */
buildMirroredNotifications(String phoneTitle, String watchTitle, String content)155     private void buildMirroredNotifications(String phoneTitle, String watchTitle, String content) {
156         if (mGoogleApiClient.isConnected()) {
157             // Wearable notification
158             buildWearableOnlyNotification(watchTitle, content, Constants.BOTH_PATH);
159 
160             // Local notification, with a pending intent for dismissal
161             buildLocalOnlyNotification(phoneTitle, content, Constants.BOTH_ID, true);
162         }
163     }
164 
165     @Override
onStart()166     public void onStart() {
167         super.onStart();
168         mGoogleApiClient.connect();
169     }
170 
171     @Override
onStop()172     public void onStop() {
173         mGoogleApiClient.disconnect();
174         super.onStop();
175     }
176 
177     @Override
onConnected(Bundle bundle)178     public void onConnected(Bundle bundle) {
179     }
180 
181     @Override
onConnectionSuspended(int i)182     public void onConnectionSuspended(int i) {
183     }
184 
185     @Override
onConnectionFailed(ConnectionResult connectionResult)186     public void onConnectionFailed(ConnectionResult connectionResult) {
187         Log.e(TAG, "Failed to connect to Google API Client");
188     }
189 
190     /**
191      * Returns a string built from the current time
192      */
now()193     private String now() {
194         SimpleDateFormat sdf = new SimpleDateFormat("HH:mm:ss", Locale.getDefault());
195         return sdf.format(new Date());
196     }
197 
198 }
199