1 /*
2 * Copyright 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.lnotifications;
18 
19 import android.app.Activity;
20 import android.app.Notification;
21 import android.app.NotificationManager;
22 import android.content.Context;
23 import android.content.Intent;
24 import android.database.Cursor;
25 import android.graphics.Bitmap;
26 import android.graphics.drawable.BitmapDrawable;
27 import android.graphics.drawable.Drawable;
28 import android.net.Uri;
29 import android.os.Bundle;
30 import android.provider.ContactsContract;
31 import android.provider.MediaStore;
32 import android.support.v4.app.Fragment;
33 import android.util.Log;
34 import android.view.LayoutInflater;
35 import android.view.View;
36 import android.view.ViewGroup;
37 import android.widget.ArrayAdapter;
38 import android.widget.Button;
39 import android.widget.ImageView;
40 import android.widget.Spinner;
41 import android.widget.TextView;
42 import android.widget.Toast;
43 
44 import java.io.IOException;
45 
46 /**
47  * Fragment that demonstrates how to attach metadata introduced in Android L, such as
48  * priority data, notification category and person data.
49  */
50 public class OtherMetadataFragment extends Fragment {
51 
52     private static final String TAG = OtherMetadataFragment.class.getSimpleName();
53 
54     /**
55      * Request code used for picking a contact.
56      */
57     public static final int REQUEST_CODE_PICK_CONTACT = 1;
58 
59     /**
60      * Incremental int used for ID for notifications so that each notification will be
61      * treated differently.
62      */
63     private int mIncrementalNotificationId = 0;
64 
65     private NotificationManager mNotificationManager;
66 
67     /**
68      * Button to show a notification.
69      */
70     private Button mShowNotificationButton;
71 
72     /**
73      *  Spinner that holds possible categories used for a notification as
74      *  {@link Notification.Builder#setCategory(String)}.
75      */
76     private Spinner mCategorySpinner;
77 
78     /**
79      * Spinner that holds possible priorities used for a notification as
80      * {@link Notification.Builder#setPriority(int)}.
81      */
82     private Spinner mPrioritySpinner;
83 
84     /**
85      * Holds a URI for the person to be attached to the notification.
86      */
87     private Uri mContactUri;
88 
89     /**
90      * Use this factory method to create a new instance of
91      * this fragment using the provided parameters.
92      *
93      * @return A new instance of fragment NotificationFragment.
94      */
newInstance()95     public static OtherMetadataFragment newInstance() {
96         OtherMetadataFragment fragment = new OtherMetadataFragment();
97         fragment.setRetainInstance(true);
98         return fragment;
99     }
100 
OtherMetadataFragment()101     public OtherMetadataFragment() {
102         // Required empty public constructor
103     }
104 
105     @Override
onCreate(Bundle savedInstanceState)106     public void onCreate(Bundle savedInstanceState) {
107         super.onCreate(savedInstanceState);
108         mNotificationManager = (NotificationManager) getActivity().getSystemService(Context
109                 .NOTIFICATION_SERVICE);
110     }
111 
112     @Override
onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)113     public View onCreateView(LayoutInflater inflater, ViewGroup container,
114                              Bundle savedInstanceState) {
115         // Inflate the layout for this fragment
116         return inflater.inflate(R.layout.fragment_other_metadata, container, false);
117     }
118 
119     @Override
onViewCreated(View view, Bundle savedInstanceState)120     public void onViewCreated(View view, Bundle savedInstanceState) {
121         super.onViewCreated(view, savedInstanceState);
122         mShowNotificationButton = (Button) view.findViewById(R.id.show_notification_button);
123         mShowNotificationButton.setOnClickListener(new View.OnClickListener() {
124             @Override
125             public void onClick(View view) {
126                 Priority selectedPriority = (Priority) mPrioritySpinner.getSelectedItem();
127                 Category selectedCategory = (Category) mCategorySpinner.getSelectedItem();
128                 showNotificationClicked(selectedPriority, selectedCategory, mContactUri);
129             }
130         });
131 
132         mCategorySpinner = (Spinner) view.findViewById(R.id.category_spinner);
133         ArrayAdapter<Category> categoryArrayAdapter = new ArrayAdapter<Category>(getActivity(),
134                 android.R.layout.simple_spinner_item, Category.values());
135         categoryArrayAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
136         mCategorySpinner.setAdapter(categoryArrayAdapter);
137 
138         mPrioritySpinner = (Spinner) view.findViewById(R.id.priority_spinner);
139         ArrayAdapter<Priority> priorityArrayAdapter = new ArrayAdapter<Priority>(getActivity(),
140                 android.R.layout.simple_spinner_item, Priority.values());
141         priorityArrayAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
142         mPrioritySpinner.setAdapter(priorityArrayAdapter);
143 
144         view.findViewById(R.id.attach_person).setOnClickListener(new View.OnClickListener() {
145             @Override
146             public void onClick(View view) {
147                 findContact();
148             }
149         });
150 
151         view.findViewById(R.id.contact_entry).setVisibility(View.GONE);
152     }
153 
154     @Override
onActivityResult(int requestCode, int resultCode, Intent data)155     public void onActivityResult(int requestCode, int resultCode, Intent data) {
156         super.onActivityResult(requestCode, resultCode, data);
157         switch (requestCode) {
158             case REQUEST_CODE_PICK_CONTACT:
159                 if (resultCode == Activity.RESULT_OK) {
160                     Uri contactUri = data.getData();
161                     mContactUri = contactUri;
162                     updateContactEntryFromUri(contactUri);
163                 }
164                 break;
165         }
166     }
167 
168     /**
169      * Invoked when {@link #mShowNotificationButton} is clicked.
170      * Creates a new notification and sets metadata passed as arguments.
171      *
172      * @param priority   The priority metadata.
173      * @param category   The category metadata.
174      * @param contactUri The URI to be added to the new notification as metadata.
175      *
176      * @return A Notification instance.
177      */
createNotification(Priority priority, Category category, Uri contactUri)178     private Notification createNotification(Priority priority, Category category, Uri contactUri) {
179         Notification.Builder notificationBuilder = new Notification.Builder(getActivity())
180                 .setContentTitle("Notification with other metadata")
181                 .setSmallIcon(R.drawable.ic_launcher_notification)
182                 .setPriority(priority.value)
183                 .setCategory(category.value)
184                 .setContentText(String.format("Category %s, Priority %s", category.value,
185                         priority.name()));
186         if (contactUri != null) {
187             notificationBuilder.addPerson(contactUri.toString());
188             Bitmap photoBitmap = loadBitmapFromContactUri(contactUri);
189             if (photoBitmap != null) {
190                 notificationBuilder.setLargeIcon(photoBitmap);
191             }
192         }
193         return notificationBuilder.build();
194     }
195 
196     /**
197      * Invoked when {@link #mShowNotificationButton} is clicked.
198      * Creates a new notification and sets metadata passed as arguments.
199      *
200      * @param priority   The priority metadata.
201      * @param category   The category metadata.
202      * @param contactUri The URI to be added to the new notification as metadata.
203      */
showNotificationClicked(Priority priority, Category category, Uri contactUri)204     private void showNotificationClicked(Priority priority, Category category, Uri contactUri) {
205         // Assigns a unique (incremented) notification ID in order to treat each notification as a
206         // different one. This helps demonstrate how a priority flag affects ordering.
207         mIncrementalNotificationId++;
208         mNotificationManager.notify(mIncrementalNotificationId, createNotification(priority,
209                 category, contactUri));
210         Toast.makeText(getActivity(), "Show Notification clicked", Toast.LENGTH_SHORT).show();
211     }
212 
findContact()213     private void findContact() {
214         Intent intent = new Intent(Intent.ACTION_PICK, ContactsContract.Contacts.CONTENT_URI);
215         startActivityForResult(intent, REQUEST_CODE_PICK_CONTACT);
216     }
217 
218     /**
219      * Returns a {@link Bitmap} from the Uri specified as the argument.
220      *
221      * @param contactUri The Uri from which the result Bitmap is created.
222      * @return The {@link Bitmap} instance retrieved from the contactUri.
223      */
loadBitmapFromContactUri(Uri contactUri)224     private Bitmap loadBitmapFromContactUri(Uri contactUri) {
225         if (contactUri == null) {
226             return null;
227         }
228         Bitmap result = null;
229         Cursor cursor = getActivity().getContentResolver().query(contactUri, null, null, null,
230                 null);
231         if (cursor != null && cursor.moveToFirst()) {
232             int idx = cursor.getColumnIndex(ContactsContract.Contacts.PHOTO_ID);
233             String hasPhoto = cursor.getString(idx);
234             Uri photoUri = Uri.withAppendedPath(contactUri, ContactsContract.Contacts.Photo
235                     .CONTENT_DIRECTORY);
236             if (hasPhoto != null) {
237                 try {
238                     result = MediaStore.Images.Media.getBitmap(getActivity().getContentResolver()
239                             , photoUri);
240                 } catch (IOException e) {
241                     Log.e(TAG, String.format("Failed to load resource. Uri %s", photoUri), e);
242                 }
243             } else {
244                 Drawable defaultContactDrawable = getActivity().getResources().getDrawable(R
245                         .drawable.ic_contact_picture);
246                 result = ((BitmapDrawable) defaultContactDrawable).getBitmap();
247             }
248         }
249         return result;
250     }
251 
252     /**
253      * Updates the Contact information on the screen when a contact is picked.
254      *
255      * @param contactUri The Uri from which the contact is retrieved.
256      */
updateContactEntryFromUri(Uri contactUri)257     private void updateContactEntryFromUri(Uri contactUri) {
258         Cursor cursor = getActivity().getContentResolver().query(contactUri, null, null, null,
259                 null);
260         if (cursor != null && cursor.moveToFirst()) {
261             int idx = cursor.getColumnIndex(ContactsContract.Contacts.DISPLAY_NAME);
262             String name = cursor.getString(idx);
263             idx = cursor.getColumnIndex(ContactsContract.Contacts.PHOTO_ID);
264             String hasPhoto = cursor.getString(idx);
265 
266             Uri photoUri = Uri.withAppendedPath(contactUri, ContactsContract.Contacts.Photo
267                     .CONTENT_DIRECTORY);
268             ImageView contactPhoto = (ImageView) getActivity().findViewById(R.id.contact_photo);
269             if (hasPhoto != null) {
270                 contactPhoto.setImageURI(photoUri);
271             } else {
272                 Drawable defaultContactDrawable = getActivity().getResources().getDrawable(R
273                         .drawable.ic_contact_picture);
274                 contactPhoto.setImageDrawable(defaultContactDrawable);
275             }
276             TextView contactName = (TextView) getActivity().findViewById(R.id.contact_name);
277             contactName.setText(name);
278 
279             getActivity().findViewById(R.id.contact_entry).setVisibility(View.VISIBLE);
280             getActivity().findViewById(R.id.attach_person).setVisibility(View.GONE);
281             getActivity().findViewById(R.id.click_to_change).setOnClickListener(new View.OnClickListener() {
282                 @Override
283                 public void onClick(View view) {
284                     findContact();
285                 }
286             });
287             Log.i(TAG, String.format("Contact updated. Name %s, PhotoUri %s", name, photoUri));
288         }
289     }
290 
291     /**
292      * Enum indicating possible categories in {@link Notification} used from
293      * {@link #mCategorySpinner}.
294      */
295     private enum Category {
296         ALARM("alarm"),
297         CALL("call"),
298         EMAIL("email"),
299         ERROR("err"),
300         EVENT("event"),
301         MESSAGE("msg"),
302         PROGRESS("progress"),
303         PROMO("promo"),
304         RECOMMENDATION("recommendation"),
305         SERVICE("service"),
306         SOCIAL("social"),
307         STATUS("status"),
308         SYSTEM("sys"),
309         TRANSPORT("transport");
310 
311         private final String value;
312 
Category(String value)313         Category(String value) {
314             this.value = value;
315         }
316 
317         @Override
toString()318         public String toString() {
319             return value;
320         }
321     }
322 
323     /**
324      * Enum indicating possible priorities in {@link Notification} used from
325      * {@link #mPrioritySpinner}.
326      */
327     private enum Priority {
328         DEFAULT(Notification.PRIORITY_DEFAULT),
329         MAX(Notification.PRIORITY_MAX),
330         HIGH(Notification.PRIORITY_HIGH),
331         LOW(Notification.PRIORITY_LOW),
332         MIN(Notification.PRIORITY_MIN);
333 
334         private final int value;
335 
Priority(int value)336         Priority(int value) {
337             this.value = value;
338         }
339     }
340 }
341