| /* |
| * Copyright (C) 2018 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.systemui.screenrecord; |
| |
| import android.app.Notification; |
| import android.app.NotificationChannel; |
| import android.app.NotificationManager; |
| import android.app.PendingIntent; |
| import android.app.Service; |
| import android.content.ContentResolver; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.content.res.Resources; |
| import android.graphics.Bitmap; |
| import android.graphics.drawable.Icon; |
| import android.media.MediaRecorder; |
| import android.net.Uri; |
| import android.os.Bundle; |
| import android.os.IBinder; |
| import android.os.RemoteException; |
| import android.provider.Settings; |
| import android.util.Log; |
| import android.widget.Toast; |
| |
| import com.android.systemui.R; |
| import com.android.systemui.dagger.qualifiers.LongRunning; |
| |
| import java.io.IOException; |
| import java.util.concurrent.Executor; |
| |
| import javax.inject.Inject; |
| |
| /** |
| * A service which records the device screen and optionally microphone input. |
| */ |
| public class RecordingService extends Service implements MediaRecorder.OnInfoListener { |
| public static final int REQUEST_CODE = 2; |
| |
| private static final int NOTIFICATION_RECORDING_ID = 4274; |
| private static final int NOTIFICATION_PROCESSING_ID = 4275; |
| private static final int NOTIFICATION_VIEW_ID = 4273; |
| private static final String TAG = "RecordingService"; |
| private static final String CHANNEL_ID = "screen_record"; |
| private static final String EXTRA_RESULT_CODE = "extra_resultCode"; |
| private static final String EXTRA_DATA = "extra_data"; |
| private static final String EXTRA_PATH = "extra_path"; |
| private static final String EXTRA_AUDIO_SOURCE = "extra_useAudio"; |
| private static final String EXTRA_SHOW_TAPS = "extra_showTaps"; |
| |
| private static final String ACTION_START = "com.android.systemui.screenrecord.START"; |
| private static final String ACTION_STOP = "com.android.systemui.screenrecord.STOP"; |
| private static final String ACTION_SHARE = "com.android.systemui.screenrecord.SHARE"; |
| private static final String ACTION_DELETE = "com.android.systemui.screenrecord.DELETE"; |
| |
| private final RecordingController mController; |
| private Notification.Builder mRecordingNotificationBuilder; |
| |
| private ScreenRecordingAudioSource mAudioSource; |
| private boolean mShowTaps; |
| private boolean mOriginalShowTaps; |
| private ScreenMediaRecorder mRecorder; |
| private final Executor mLongExecutor; |
| |
| @Inject |
| public RecordingService(RecordingController controller, @LongRunning Executor executor) { |
| mController = controller; |
| mLongExecutor = executor; |
| } |
| |
| /** |
| * Get an intent to start the recording service. |
| * |
| * @param context Context from the requesting activity |
| * @param resultCode The result code from {@link android.app.Activity#onActivityResult(int, int, |
| * android.content.Intent)} |
| * @param data The data from {@link android.app.Activity#onActivityResult(int, int, |
| * android.content.Intent)} |
| * @param audioSource The ordinal value of the audio source |
| * {@link com.android.systemui.screenrecord.ScreenRecordingAudioSource} |
| * @param showTaps True to make touches visible while recording |
| */ |
| public static Intent getStartIntent(Context context, int resultCode, |
| int audioSource, boolean showTaps) { |
| return new Intent(context, RecordingService.class) |
| .setAction(ACTION_START) |
| .putExtra(EXTRA_RESULT_CODE, resultCode) |
| .putExtra(EXTRA_AUDIO_SOURCE, audioSource) |
| .putExtra(EXTRA_SHOW_TAPS, showTaps); |
| } |
| |
| @Override |
| public int onStartCommand(Intent intent, int flags, int startId) { |
| if (intent == null) { |
| return Service.START_NOT_STICKY; |
| } |
| String action = intent.getAction(); |
| Log.d(TAG, "onStartCommand " + action); |
| |
| NotificationManager notificationManager = |
| (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE); |
| |
| switch (action) { |
| case ACTION_START: |
| mAudioSource = ScreenRecordingAudioSource |
| .values()[intent.getIntExtra(EXTRA_AUDIO_SOURCE, 0)]; |
| Log.d(TAG, "recording with audio source" + mAudioSource); |
| mShowTaps = intent.getBooleanExtra(EXTRA_SHOW_TAPS, false); |
| |
| mOriginalShowTaps = Settings.System.getInt( |
| getApplicationContext().getContentResolver(), |
| Settings.System.SHOW_TOUCHES, 0) != 0; |
| |
| setTapsVisible(mShowTaps); |
| |
| mRecorder = new ScreenMediaRecorder( |
| getApplicationContext(), |
| getUserId(), |
| mAudioSource, |
| this |
| ); |
| startRecording(); |
| break; |
| |
| case ACTION_STOP: |
| stopRecording(); |
| notificationManager.cancel(NOTIFICATION_RECORDING_ID); |
| saveRecording(notificationManager); |
| stopSelf(); |
| break; |
| |
| case ACTION_SHARE: |
| Uri shareUri = Uri.parse(intent.getStringExtra(EXTRA_PATH)); |
| |
| Intent shareIntent = new Intent(Intent.ACTION_SEND) |
| .setType("video/mp4") |
| .putExtra(Intent.EXTRA_STREAM, shareUri); |
| String shareLabel = getResources().getString(R.string.screenrecord_share_label); |
| |
| // Close quick shade |
| sendBroadcast(new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS)); |
| |
| // Remove notification |
| notificationManager.cancel(NOTIFICATION_VIEW_ID); |
| |
| startActivity(Intent.createChooser(shareIntent, shareLabel) |
| .setFlags(Intent.FLAG_ACTIVITY_NEW_TASK)); |
| break; |
| case ACTION_DELETE: |
| // Close quick shade |
| sendBroadcast(new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS)); |
| |
| ContentResolver resolver = getContentResolver(); |
| Uri uri = Uri.parse(intent.getStringExtra(EXTRA_PATH)); |
| resolver.delete(uri, null, null); |
| |
| Toast.makeText( |
| this, |
| R.string.screenrecord_delete_description, |
| Toast.LENGTH_LONG).show(); |
| |
| // Remove notification |
| notificationManager.cancel(NOTIFICATION_VIEW_ID); |
| Log.d(TAG, "Deleted recording " + uri); |
| break; |
| } |
| return Service.START_STICKY; |
| } |
| |
| @Override |
| public IBinder onBind(Intent intent) { |
| return null; |
| } |
| |
| @Override |
| public void onCreate() { |
| super.onCreate(); |
| } |
| |
| /** |
| * Begin the recording session |
| */ |
| private void startRecording() { |
| try { |
| mRecorder.start(); |
| mController.updateState(true); |
| createRecordingNotification(); |
| } catch (IOException | RemoteException e) { |
| Toast.makeText(this, |
| R.string.screenrecord_start_error, Toast.LENGTH_LONG) |
| .show(); |
| e.printStackTrace(); |
| } |
| } |
| |
| private void createRecordingNotification() { |
| Resources res = getResources(); |
| NotificationChannel channel = new NotificationChannel( |
| CHANNEL_ID, |
| getString(R.string.screenrecord_name), |
| NotificationManager.IMPORTANCE_DEFAULT); |
| channel.setDescription(getString(R.string.screenrecord_channel_description)); |
| channel.enableVibration(true); |
| NotificationManager notificationManager = |
| (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE); |
| notificationManager.createNotificationChannel(channel); |
| |
| Bundle extras = new Bundle(); |
| extras.putString(Notification.EXTRA_SUBSTITUTE_APP_NAME, |
| res.getString(R.string.screenrecord_name)); |
| |
| String notificationTitle = mAudioSource == ScreenRecordingAudioSource.NONE |
| ? res.getString(R.string.screenrecord_ongoing_screen_only) |
| : res.getString(R.string.screenrecord_ongoing_screen_and_audio); |
| |
| mRecordingNotificationBuilder = new Notification.Builder(this, CHANNEL_ID) |
| .setSmallIcon(R.drawable.ic_screenrecord) |
| .setContentTitle(notificationTitle) |
| .setContentText(getResources().getString(R.string.screenrecord_stop_text)) |
| .setUsesChronometer(true) |
| .setColorized(true) |
| .setColor(getResources().getColor(R.color.GM2_red_700)) |
| .setOngoing(true) |
| .setContentIntent( |
| PendingIntent.getService( |
| this, REQUEST_CODE, getStopIntent(this), |
| PendingIntent.FLAG_UPDATE_CURRENT)) |
| .addExtras(extras); |
| notificationManager.notify(NOTIFICATION_RECORDING_ID, |
| mRecordingNotificationBuilder.build()); |
| Notification notification = mRecordingNotificationBuilder.build(); |
| startForeground(NOTIFICATION_RECORDING_ID, notification); |
| } |
| |
| private Notification createSaveNotification(ScreenMediaRecorder.SavedRecording recording) { |
| Uri uri = recording.getUri(); |
| Intent viewIntent = new Intent(Intent.ACTION_VIEW) |
| .setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_GRANT_READ_URI_PERMISSION) |
| .setDataAndType(uri, "video/mp4"); |
| |
| Notification.Action shareAction = new Notification.Action.Builder( |
| Icon.createWithResource(this, R.drawable.ic_screenrecord), |
| getResources().getString(R.string.screenrecord_share_label), |
| PendingIntent.getService( |
| this, |
| REQUEST_CODE, |
| getShareIntent(this, uri.toString()), |
| PendingIntent.FLAG_UPDATE_CURRENT)) |
| .build(); |
| |
| Notification.Action deleteAction = new Notification.Action.Builder( |
| Icon.createWithResource(this, R.drawable.ic_screenrecord), |
| getResources().getString(R.string.screenrecord_delete_label), |
| PendingIntent.getService( |
| this, |
| REQUEST_CODE, |
| getDeleteIntent(this, uri.toString()), |
| PendingIntent.FLAG_UPDATE_CURRENT)) |
| .build(); |
| |
| Bundle extras = new Bundle(); |
| extras.putString(Notification.EXTRA_SUBSTITUTE_APP_NAME, |
| getResources().getString(R.string.screenrecord_name)); |
| |
| Notification.Builder builder = new Notification.Builder(this, CHANNEL_ID) |
| .setSmallIcon(R.drawable.ic_screenrecord) |
| .setContentTitle(getResources().getString(R.string.screenrecord_save_message)) |
| .setContentIntent(PendingIntent.getActivity( |
| this, |
| REQUEST_CODE, |
| viewIntent, |
| Intent.FLAG_GRANT_READ_URI_PERMISSION)) |
| .addAction(shareAction) |
| .addAction(deleteAction) |
| .setAutoCancel(true) |
| .addExtras(extras); |
| |
| // Add thumbnail if available |
| Bitmap thumbnailBitmap = recording.getThumbnail(); |
| if (thumbnailBitmap != null) { |
| Notification.BigPictureStyle pictureStyle = new Notification.BigPictureStyle() |
| .bigPicture(thumbnailBitmap) |
| .bigLargeIcon((Bitmap) null); |
| builder.setLargeIcon(thumbnailBitmap).setStyle(pictureStyle); |
| } |
| return builder.build(); |
| } |
| |
| private void stopRecording() { |
| setTapsVisible(mOriginalShowTaps); |
| mRecorder.end(); |
| mController.updateState(false); |
| } |
| |
| private void saveRecording(NotificationManager notificationManager) { |
| Resources res = getApplicationContext().getResources(); |
| String notificationTitle = mAudioSource == ScreenRecordingAudioSource.NONE |
| ? res.getString(R.string.screenrecord_ongoing_screen_only) |
| : res.getString(R.string.screenrecord_ongoing_screen_and_audio); |
| Notification.Builder builder = new Notification.Builder(getApplicationContext(), CHANNEL_ID) |
| .setContentTitle(notificationTitle) |
| .setContentText( |
| getResources().getString(R.string.screenrecord_background_processing_label)) |
| .setSmallIcon(R.drawable.ic_screenrecord); |
| notificationManager.notify(NOTIFICATION_PROCESSING_ID, builder.build()); |
| |
| mLongExecutor.execute(() -> { |
| try { |
| Log.d(TAG, "saving recording"); |
| Notification notification = createSaveNotification(mRecorder.save()); |
| if (!mController.isRecording()) { |
| Log.d(TAG, "showing saved notification"); |
| notificationManager.notify(NOTIFICATION_VIEW_ID, notification); |
| } |
| } catch (IOException e) { |
| Log.e(TAG, "Error saving screen recording: " + e.getMessage()); |
| Toast.makeText(this, R.string.screenrecord_delete_error, Toast.LENGTH_LONG) |
| .show(); |
| } finally { |
| notificationManager.cancel(NOTIFICATION_PROCESSING_ID); |
| } |
| }); |
| } |
| |
| private void setTapsVisible(boolean turnOn) { |
| int value = turnOn ? 1 : 0; |
| Settings.System.putInt(getApplicationContext().getContentResolver(), |
| Settings.System.SHOW_TOUCHES, value); |
| } |
| |
| /** |
| * Get an intent to stop the recording service. |
| * @param context Context from the requesting activity |
| * @return |
| */ |
| public static Intent getStopIntent(Context context) { |
| return new Intent(context, RecordingService.class).setAction(ACTION_STOP); |
| } |
| |
| private static Intent getShareIntent(Context context, String path) { |
| return new Intent(context, RecordingService.class).setAction(ACTION_SHARE) |
| .putExtra(EXTRA_PATH, path); |
| } |
| |
| private static Intent getDeleteIntent(Context context, String path) { |
| return new Intent(context, RecordingService.class).setAction(ACTION_DELETE) |
| .putExtra(EXTRA_PATH, path); |
| } |
| |
| @Override |
| public void onInfo(MediaRecorder mr, int what, int extra) { |
| Log.d(TAG, "Media recorder info: " + what); |
| onStartCommand(getStopIntent(this), 0, 0); |
| } |
| } |