blob: d64082d4e11446ac11fa45b7a8609104950546f3 [file] [log] [blame]
/*
* Copyright (C) 2015 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.messaging.ui;
import android.app.Activity;
import android.app.Fragment;
import android.app.PendingIntent;
import android.appwidget.AppWidgetManager;
import android.content.ActivityNotFoundException;
import android.content.ClipData;
import android.content.ComponentName;
import android.content.ContentValues;
import android.content.Context;
import android.content.Intent;
import android.graphics.Point;
import android.graphics.Rect;
import android.net.Uri;
import android.os.Bundle;
import android.provider.ContactsContract.Contacts;
import android.provider.ContactsContract.Intents;
import android.provider.MediaStore;
import android.provider.Telephony;
import androidx.annotation.Nullable;
import androidx.core.app.TaskStackBuilder;
import androidx.localbroadcastmanager.content.LocalBroadcastManager;
import android.text.TextUtils;
import com.android.ex.photo.Intents.PhotoViewIntentBuilder;
import com.android.messaging.R;
import com.android.messaging.datamodel.ConversationImagePartsView;
import com.android.messaging.datamodel.MediaScratchFileProvider;
import com.android.messaging.datamodel.MessagingContentProvider;
import com.android.messaging.datamodel.data.MessageData;
import com.android.messaging.datamodel.data.MessagePartData;
import com.android.messaging.datamodel.data.ParticipantData;
import com.android.messaging.receiver.NotificationReceiver;
import com.android.messaging.sms.MmsSmsUtils;
import com.android.messaging.ui.appsettings.ApnEditorActivity;
import com.android.messaging.ui.appsettings.ApnSettingsActivity;
import com.android.messaging.ui.appsettings.ApplicationSettingsActivity;
import com.android.messaging.ui.appsettings.PerSubscriptionSettingsActivity;
import com.android.messaging.ui.appsettings.SettingsActivity;
import com.android.messaging.ui.attachmentchooser.AttachmentChooserActivity;
import com.android.messaging.ui.conversation.ConversationActivity;
import com.android.messaging.ui.conversation.LaunchConversationActivity;
import com.android.messaging.ui.conversationlist.ArchivedConversationListActivity;
import com.android.messaging.ui.conversationlist.ConversationListActivity;
import com.android.messaging.ui.conversationlist.ForwardMessageActivity;
import com.android.messaging.ui.conversationsettings.PeopleAndOptionsActivity;
import com.android.messaging.ui.debug.DebugMmsConfigActivity;
import com.android.messaging.ui.photoviewer.BuglePhotoViewActivity;
import com.android.messaging.util.Assert;
import com.android.messaging.util.ContentType;
import com.android.messaging.util.ConversationIdSet;
import com.android.messaging.util.LogUtil;
import com.android.messaging.util.UiUtils;
import com.android.messaging.util.UriUtil;
/**
* A central repository of Intents used to start activities.
*/
public class UIIntentsImpl extends UIIntents {
private static final String CELL_BROADCAST_LIST_ACTIVITY =
"com.android.cellbroadcastreceiver.CellBroadcastListActivity";
private static final String CALL_TARGET_CLICK_KEY = "touchPoint";
private static final String CALL_TARGET_CLICK_EXTRA_KEY =
"android.telecom.extra.OUTGOING_CALL_EXTRAS";
private static final String MEDIA_SCANNER_CLASS =
"com.android.providers.media.MediaScannerService";
private static final String MEDIA_SCANNER_PACKAGE = "com.android.providers.media";
private static final String MEDIA_SCANNER_SCAN_ACTION = "android.media.IMediaScannerService";
/**
* Get an intent which takes you to a conversation
*/
private Intent getConversationActivityIntent(final Context context,
final String conversationId, final MessageData draft,
final boolean withCustomTransition) {
final Intent intent = new Intent(context, ConversationActivity.class);
// Always try to reuse the same ConversationActivity in the current task so that we don't
// have two conversation activities in the back stack.
intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
// Otherwise we're starting a new conversation
if (conversationId != null) {
intent.putExtra(UI_INTENT_EXTRA_CONVERSATION_ID, conversationId);
}
if (draft != null) {
intent.putExtra(UI_INTENT_EXTRA_DRAFT_DATA, draft);
// If draft attachments came from an external content provider via a share intent, we
// need to propagate the URI permissions through to ConversationActivity. This requires
// putting the URIs into the ClipData (setData also works, but accepts only one URI).
ClipData clipData = null;
for (final MessagePartData partData : draft.getParts()) {
if (partData.isAttachment()) {
final Uri uri = partData.getContentUri();
if (clipData == null) {
clipData = ClipData.newRawUri("Attachments", uri);
} else {
clipData.addItem(new ClipData.Item(uri));
}
}
}
if (clipData != null) {
intent.setClipData(clipData);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
}
}
if (withCustomTransition) {
intent.putExtra(UI_INTENT_EXTRA_WITH_CUSTOM_TRANSITION, true);
}
if (!(context instanceof Activity)) {
// If the caller supplies an application context, and not an activity context, we must
// include this flag
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
}
return intent;
}
@Override
public void launchPermissionCheckActivity(final Context context) {
final Intent intent = new Intent(context, PermissionCheckActivity.class);
context.startActivity(intent);
}
/**
* Get an intent which takes you to the conversation list
*/
private Intent getConversationListActivityIntent(final Context context) {
return new Intent(context, ConversationListActivity.class);
}
@Override
public void launchConversationListActivity(final Context context) {
final Intent intent = getConversationListActivityIntent(context);
context.startActivity(intent);
}
/**
* Get an intent which shows the low storage warning activity.
*/
private Intent getSmsStorageLowWarningActivityIntent(final Context context) {
return new Intent(context, SmsStorageLowWarningActivity.class);
}
@Override
public void launchConversationActivity(final Context context,
final String conversationId, final MessageData draft, final Bundle activityOptions,
final boolean withCustomTransition) {
Assert.isTrue(!withCustomTransition || activityOptions != null);
final Intent intent = getConversationActivityIntent(context, conversationId, draft,
withCustomTransition);
context.startActivity(intent, activityOptions);
}
@Override
public void launchConversationActivityNewTask(
final Context context, final String conversationId) {
final Intent intent = getConversationActivityIntent(context, conversationId, null,
false /* withCustomTransition */);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
}
@Override
public void launchConversationActivityWithParentStack(final Context context,
final String conversationId, final String smsBody) {
final MessageData messageData = TextUtils.isEmpty(smsBody)
? null
: MessageData.createDraftSmsMessage(conversationId, null, smsBody);
TaskStackBuilder.create(context)
.addNextIntentWithParentStack(
getConversationActivityIntent(context, conversationId, messageData,
false /* withCustomTransition */))
.startActivities();
}
@Override
public void launchCreateNewConversationActivity(final Context context,
final MessageData draft) {
final Intent intent = getConversationActivityIntent(context, null, draft,
false /* withCustomTransition */);
context.startActivity(intent);
}
@Override
public void launchDebugMmsConfigActivity(final Context context) {
context.startActivity(new Intent(context, DebugMmsConfigActivity.class));
}
@Override
public void launchAddContactActivity(final Context context, final String destination) {
final Intent intent = new Intent(Intent.ACTION_INSERT_OR_EDIT);
final String destinationType = MmsSmsUtils.isEmailAddress(destination) ?
Intents.Insert.EMAIL : Intents.Insert.PHONE;
intent.setType(Contacts.CONTENT_ITEM_TYPE);
intent.putExtra(destinationType, destination);
startExternalActivity(context, intent);
}
@Override
public void launchSettingsActivity(final Context context) {
final Intent intent = new Intent(context, SettingsActivity.class);
context.startActivity(intent);
}
@Override
public void launchArchivedConversationsActivity(final Context context) {
final Intent intent = new Intent(context, ArchivedConversationListActivity.class);
context.startActivity(intent);
}
@Override
public void launchBlockedParticipantsActivity(final Context context) {
final Intent intent = new Intent(context, BlockedParticipantsActivity.class);
context.startActivity(intent);
}
@Override
public void launchDocumentImagePicker(final Fragment fragment) {
final Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
intent.putExtra(Intent.EXTRA_MIME_TYPES, MessagePartData.ACCEPTABLE_GALLERY_MEDIA_TYPES);
intent.addCategory(Intent.CATEGORY_OPENABLE);
intent.setType(ContentType.ANY_TYPE);
fragment.startActivityForResult(intent, REQUEST_PICK_MEDIA_FROM_DOCUMENT_PICKER);
}
@Override
public void launchContactCardPicker(final Fragment fragment) {
final Intent intent = new Intent(Intent.ACTION_PICK);
intent.setType(Contacts.CONTENT_TYPE);
try {
fragment.startActivityForResult(intent, REQUEST_PICK_CONTACT_CARD);
} catch (final ActivityNotFoundException ex) {
LogUtil.w(LogUtil.BUGLE_TAG, "Couldn't find activity:", ex);
UiUtils.showToastAtBottom(R.string.activity_not_found_message);
}
}
@Override
public void launchPeopleAndOptionsActivity(final Activity activity,
final String conversationId) {
final Intent intent = new Intent(activity, PeopleAndOptionsActivity.class);
intent.putExtra(UI_INTENT_EXTRA_CONVERSATION_ID, conversationId);
activity.startActivityForResult(intent, 0);
}
@Override
public void launchPhoneCallActivity(final Context context, final String phoneNumber,
final Point clickPosition) {
final Intent intent = new Intent(Intent.ACTION_CALL,
Uri.parse(UriUtil.SCHEME_TEL + phoneNumber));
final Bundle extras = new Bundle();
extras.putParcelable(CALL_TARGET_CLICK_KEY, clickPosition);
intent.putExtra(CALL_TARGET_CLICK_EXTRA_KEY, extras);
startExternalActivity(context, intent);
}
@Override
public void launchClassZeroActivity(final Context context, final ContentValues messageValues) {
final Intent classZeroIntent = new Intent(context, ClassZeroActivity.class)
.putExtra(UI_INTENT_EXTRA_MESSAGE_VALUES, messageValues)
.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_MULTIPLE_TASK);
context.startActivity(classZeroIntent);
}
@Override
public void launchForwardMessageActivity(final Context context, final MessageData message) {
final Intent forwardMessageIntent = new Intent(context, ForwardMessageActivity.class)
.putExtra(UI_INTENT_EXTRA_DRAFT_DATA, message);
context.startActivity(forwardMessageIntent);
}
@Override
public void launchVCardDetailActivity(final Context context, final Uri vcardUri) {
final Intent vcardDetailIntent = new Intent(context, VCardDetailActivity.class)
.putExtra(UI_INTENT_EXTRA_VCARD_URI, vcardUri);
context.startActivity(vcardDetailIntent);
}
@Override
public void launchSaveVCardToContactsActivity(final Context context, final Uri vcardUri) {
Assert.isTrue(MediaScratchFileProvider.isMediaScratchSpaceUri(vcardUri));
final Intent intent = new Intent();
intent.setAction(Intent.ACTION_VIEW);
intent.setDataAndType(vcardUri, ContentType.TEXT_VCARD.toLowerCase());
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
startExternalActivity(context, intent);
}
@Override
public void launchAttachmentChooserActivity(final Activity activity,
final String conversationId, final int requestCode) {
final Intent intent = new Intent(activity, AttachmentChooserActivity.class);
intent.putExtra(UI_INTENT_EXTRA_CONVERSATION_ID, conversationId);
activity.startActivityForResult(intent, requestCode);
}
@Override
public void launchFullScreenVideoViewer(final Context context, final Uri videoUri) {
final Intent intent = new Intent(Intent.ACTION_VIEW);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
// So we don't see "surrounding" images in Gallery
intent.putExtra("SingleItemOnly", true);
intent.setDataAndType(videoUri, ContentType.VIDEO_UNSPECIFIED);
startExternalActivity(context, intent);
}
@Override
public void launchFullScreenPhotoViewer(final Activity activity, final Uri initialPhoto,
final Rect initialPhotoBounds, final Uri photosUri) {
final PhotoViewIntentBuilder builder =
com.android.ex.photo.Intents.newPhotoViewIntentBuilder(
activity, BuglePhotoViewActivity.class);
builder.setPhotosUri(photosUri.toString());
builder.setInitialPhotoUri(initialPhoto.toString());
builder.setProjection(ConversationImagePartsView.PhotoViewQuery.PROJECTION);
// Set the location of the imageView so that the photoviewer can animate from that location
// to full screen.
builder.setScaleAnimation(initialPhotoBounds.left, initialPhotoBounds.top,
initialPhotoBounds.width(), initialPhotoBounds.height());
builder.setDisplayThumbsFullScreen(false);
builder.setMaxInitialScale(8);
activity.startActivity(builder.build());
activity.overridePendingTransition(0, 0);
}
@Override
public void launchApplicationSettingsActivity(final Context context, final boolean topLevel) {
final Intent intent = new Intent(context, ApplicationSettingsActivity.class);
intent.putExtra(UI_INTENT_EXTRA_TOP_LEVEL_SETTINGS, topLevel);
context.startActivity(intent);
}
@Override
public void launchPerSubscriptionSettingsActivity(final Context context, final int subId,
final String settingTitle) {
final Intent intent = getPerSubscriptionSettingsIntent(context, subId, settingTitle);
context.startActivity(intent);
}
@Override
public Intent getViewUrlIntent(final String url) {
final Uri uri = Uri.parse(url);
return new Intent(Intent.ACTION_VIEW, uri);
}
@Override
public void broadcastConversationSelfIdChange(final Context context,
final String conversationId, final String conversationSelfId) {
final Intent intent = new Intent(CONVERSATION_SELF_ID_CHANGE_BROADCAST_ACTION);
intent.putExtra(UI_INTENT_EXTRA_CONVERSATION_ID, conversationId);
intent.putExtra(UI_INTENT_EXTRA_CONVERSATION_SELF_ID, conversationSelfId);
LocalBroadcastManager.getInstance(context).sendBroadcast(intent);
}
@Override
public PendingIntent getPendingIntentForConversationListActivity(final Context context) {
final Intent intent = getConversationListActivityIntent(context);
return getPendingIntentWithParentStack(context, intent, 0);
}
@Override
public PendingIntent getPendingIntentForConversationActivity(final Context context,
final String conversationId, final MessageData draft) {
final Intent intent = getConversationActivityIntent(context, conversationId, draft,
false /* withCustomTransition */);
// Ensure that the platform doesn't reuse PendingIntents across conversations
intent.setData(MessagingContentProvider.buildConversationMetadataUri(conversationId));
return getPendingIntentWithParentStack(context, intent, 0);
}
@Override
public Intent getIntentForConversationActivity(final Context context,
final String conversationId, final MessageData draft) {
final Intent intent = getConversationActivityIntent(context, conversationId, draft,
false /* withCustomTransition */);
return intent;
}
@Override
public PendingIntent getPendingIntentForSendingMessageToConversation(final Context context,
final String conversationId, final String selfId, final boolean requiresMms,
final int requestCode) {
final Intent intent = new Intent(context, RemoteInputEntrypointActivity.class);
intent.setAction(Intent.ACTION_SENDTO);
// Ensure that the platform doesn't reuse PendingIntents across conversations
intent.setData(MessagingContentProvider.buildConversationMetadataUri(conversationId));
intent.putExtra(UIIntents.UI_INTENT_EXTRA_CONVERSATION_ID, conversationId);
intent.putExtra(UIIntents.UI_INTENT_EXTRA_SELF_ID, selfId);
intent.putExtra(UIIntents.UI_INTENT_EXTRA_REQUIRES_MMS, requiresMms);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
return getPendingIntentWithParentStack(context, intent, requestCode);
}
@Override
public PendingIntent getPendingIntentForClearingNotifications(final Context context,
final int updateTargets, final ConversationIdSet conversationIdSet,
final int requestCode) {
final Intent intent = new Intent(context, NotificationReceiver.class);
intent.setAction(ACTION_RESET_NOTIFICATIONS);
intent.putExtra(UI_INTENT_EXTRA_NOTIFICATIONS_UPDATE, updateTargets);
if (conversationIdSet != null) {
intent.putExtra(UI_INTENT_EXTRA_CONVERSATION_ID_SET,
conversationIdSet.getDelimitedString());
}
return PendingIntent.getBroadcast(context,
requestCode, intent,
PendingIntent.FLAG_UPDATE_CURRENT);
}
/**
* Gets a PendingIntent associated with an Intent to start an Activity. All notifications
* that starts an Activity must use this method to get a PendingIntent, which achieves two
* goals:
* 1. The target activities will be created, with any existing ones destroyed. This ensures
* we don't end up with multiple instances of ConversationListActivity, for example.
* 2. The target activity, when launched, will have its backstack correctly constructed so
* back navigation will work correctly.
*/
private static PendingIntent getPendingIntentWithParentStack(final Context context,
final Intent intent, final int requestCode) {
final TaskStackBuilder stackBuilder = TaskStackBuilder.create(context);
// Adds the back stack for the Intent (plus the Intent itself)
stackBuilder.addNextIntentWithParentStack(intent);
final PendingIntent resultPendingIntent =
stackBuilder.getPendingIntent(requestCode, PendingIntent.FLAG_UPDATE_CURRENT);
return resultPendingIntent;
}
@Override
public PendingIntent getPendingIntentForLowStorageNotifications(final Context context) {
final TaskStackBuilder taskStackBuilder = TaskStackBuilder.create(context);
final Intent conversationListIntent = getConversationListActivityIntent(context);
taskStackBuilder.addNextIntent(conversationListIntent);
taskStackBuilder.addNextIntentWithParentStack(
getSmsStorageLowWarningActivityIntent(context));
return taskStackBuilder.getPendingIntent(
0, PendingIntent.FLAG_UPDATE_CURRENT);
}
@Override
public PendingIntent getPendingIntentForSecondaryUserNewMessageNotification(
final Context context) {
return getPendingIntentForConversationListActivity(context);
}
@Override
public Intent getWirelessAlertsIntent() {
final Intent intent = new Intent(Intent.ACTION_MAIN);
intent.setComponent(new ComponentName(CMAS_COMPONENT, CELL_BROADCAST_LIST_ACTIVITY));
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
return intent;
}
@Override
public Intent getApnEditorIntent(final Context context, final String rowId, final int subId) {
final Intent intent = new Intent(context, ApnEditorActivity.class);
intent.putExtra(UI_INTENT_EXTRA_APN_ROW_ID, rowId);
intent.putExtra(UI_INTENT_EXTRA_SUB_ID, subId);
return intent;
}
@Override
public Intent getApnSettingsIntent(final Context context, final int subId) {
final Intent intent = new Intent(context, ApnSettingsActivity.class)
.putExtra(UI_INTENT_EXTRA_SUB_ID, subId);
return intent;
}
@Override
public Intent getAdvancedSettingsIntent(final Context context) {
return getPerSubscriptionSettingsIntent(context, ParticipantData.DEFAULT_SELF_SUB_ID, null);
}
@Override
public Intent getChangeDefaultSmsAppIntent(final Activity activity) {
final Intent intent = new Intent(Telephony.Sms.Intents.ACTION_CHANGE_DEFAULT);
intent.putExtra(Telephony.Sms.Intents.EXTRA_PACKAGE_NAME, activity.getPackageName());
return intent;
}
@Override
public void launchBrowserForUrl(final Context context, final String url) {
final Intent intent = getViewUrlIntent(url);
startExternalActivity(context, intent);
}
/**
* Provides a safe way to handle external activities which may not exist.
*/
private void startExternalActivity(final Context context, final Intent intent) {
try {
context.startActivity(intent);
} catch (final ActivityNotFoundException ex) {
LogUtil.w(LogUtil.BUGLE_TAG, "Couldn't find activity:", ex);
UiUtils.showToastAtBottom(R.string.activity_not_found_message);
}
}
private Intent getPerSubscriptionSettingsIntent(final Context context, final int subId,
@Nullable final String settingTitle) {
return new Intent(context, PerSubscriptionSettingsActivity.class)
.putExtra(UI_INTENT_EXTRA_SUB_ID, subId)
.putExtra(UI_INTENT_EXTRA_PER_SUBSCRIPTION_SETTING_TITLE, settingTitle);
}
@Override
public Intent getLaunchConversationActivityIntent(final Context context) {
final Intent intent = new Intent(context, LaunchConversationActivity.class);
intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_NO_HISTORY);
return intent;
}
@Override
public void kickMediaScanner(final Context context, final String volume) {
final Intent intent = new Intent(MEDIA_SCANNER_SCAN_ACTION)
.putExtra(MediaStore.MEDIA_SCANNER_VOLUME, volume)
.setClassName(MEDIA_SCANNER_PACKAGE, MEDIA_SCANNER_CLASS);
context.startService(intent);
}
@Override
public PendingIntent getWidgetPendingIntentForConversationActivity(final Context context,
final String conversationId, final int requestCode) {
final Intent intent = getConversationActivityIntent(context, null, null,
false /* withCustomTransition */);
if (conversationId != null) {
intent.putExtra(UI_INTENT_EXTRA_CONVERSATION_ID, conversationId);
// Set the action to something unique to this conversation so if someone calls this
// function again on a different conversation, they'll get a new PendingIntent instead
// of the old one.
intent.setAction(ACTION_WIDGET_CONVERSATION + conversationId);
}
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
return getPendingIntentWithParentStack(context, intent, requestCode);
}
@Override
public PendingIntent getWidgetPendingIntentForConversationListActivity(
final Context context) {
final Intent intent = getConversationListActivityIntent(context);
return getPendingIntentWithParentStack(context, intent, 0);
}
@Override
public PendingIntent getWidgetPendingIntentForConfigurationActivity(final Context context,
final int appWidgetId) {
final Intent configureIntent = new Intent(context, WidgetPickConversationActivity.class);
configureIntent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, appWidgetId);
configureIntent.setAction(AppWidgetManager.ACTION_APPWIDGET_CONFIGURE);
configureIntent.setData(Uri.parse(configureIntent.toUri(Intent.URI_INTENT_SCHEME)));
configureIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_NO_HISTORY);
return getPendingIntentWithParentStack(context, configureIntent, 0);
}
}