/* * Copyright (C) 2015 Google Inc. All Rights Reserved. * * 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.example.android.wearable.runtimepermissions; import android.Manifest; import android.app.Activity; import android.content.Intent; import android.content.pm.PackageManager; import android.os.Environment; import android.os.Looper; import android.support.v4.app.ActivityCompat; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.util.Log; import android.view.View; import android.widget.Button; import android.widget.TextView; import com.example.android.wearable.runtimepermissions.common.Constants; import com.google.android.gms.common.ConnectionResult; import com.google.android.gms.common.api.GoogleApiClient; import com.google.android.gms.common.api.PendingResult; import com.google.android.gms.common.api.ResultCallback; import com.google.android.gms.wearable.CapabilityApi; import com.google.android.gms.wearable.CapabilityInfo; import com.google.android.gms.wearable.DataMap; import com.google.android.gms.wearable.MessageApi; import com.google.android.gms.wearable.MessageEvent; import com.google.android.gms.wearable.Node; import com.google.android.gms.wearable.Wearable; import java.io.File; import java.util.Set; import java.util.concurrent.TimeUnit; /** * Displays data that requires runtime permissions both locally (READ_EXTERNAL_STORAGE) and * remotely on wear (BODY_SENSORS). * * The class also handles sending back the results of a permission request from a remote wear device * when the permission has not been approved yet on the phone (uses EXTRA as trigger). In that case, * the IncomingRequestPhoneService launches the splash Activity (PhonePermissionRequestActivity) to * inform user of permission request. After the user decides what to do, it falls back to this * Activity (which has all the GoogleApiClient code) to handle sending data across and keeps user * in app experience. */ public class MainPhoneActivity extends AppCompatActivity implements GoogleApiClient.ConnectionCallbacks, GoogleApiClient.OnConnectionFailedListener, CapabilityApi.CapabilityListener, MessageApi.MessageListener, ResultCallback { private static final String TAG = "MainPhoneActivity"; /* * Alerts Activity that the initial request for permissions came from wear, and the Activity * needs to send back the results (data or permission rejection). */ public static final String EXTRA_PROMPT_PERMISSION_FROM_WEAR = "com.example.android.wearable.runtimepermissions.extra.PROMPT_PERMISSION_FROM_WEAR"; private static final int REQUEST_WEAR_PERMISSION_RATIONALE = 1; private boolean mWearBodySensorsPermissionApproved; private boolean mPhoneStoragePermissionApproved; private boolean mWearRequestingPhoneStoragePermission; private Button mWearBodySensorsPermissionButton; private Button mPhoneStoragePermissionButton; private TextView mOutputTextView; private Set mWearNodeIds; private GoogleApiClient mGoogleApiClient; @Override protected void onCreate(Bundle savedInstanceState) { Log.d(TAG, "onCreate()"); super.onCreate(savedInstanceState); /* * Since this is a remote permission, we initialize it to false and then check the remote * permission once the GoogleApiClient is connected. */ mWearBodySensorsPermissionApproved = false; setContentView(R.layout.activity_main); // Checks if wear app requested phone permission (permission request opens later if true). mWearRequestingPhoneStoragePermission = getIntent().getBooleanExtra(EXTRA_PROMPT_PERMISSION_FROM_WEAR, false); mPhoneStoragePermissionButton = (Button) findViewById(R.id.phoneStoragePermissionButton); mWearBodySensorsPermissionButton = (Button) findViewById(R.id.wearBodySensorsPermissionButton); mOutputTextView = (TextView) findViewById(R.id.output); mGoogleApiClient = new GoogleApiClient.Builder(this) .addApi(Wearable.API) .addConnectionCallbacks(this) .addOnConnectionFailedListener(this) .build(); } public void onClickWearBodySensors(View view) { logToUi("Requested info from wear device(s). New approval may be required."); DataMap dataMap = new DataMap(); dataMap.putInt(Constants.KEY_COMM_TYPE, Constants.COMM_TYPE_REQUEST_DATA); sendMessage(dataMap); } public void onClickPhoneStorage(View view) { if (mPhoneStoragePermissionApproved) { logToUi(getPhoneStorageInformation()); } else { // On 23+ (M+) devices, Storage permission not granted. Request permission. Intent startIntent = new Intent(this, PhonePermissionRequestActivity.class); startActivity(startIntent); } } @Override protected void onPause() { Log.d(TAG, "onPause()"); super.onPause(); if ((mGoogleApiClient != null) && (mGoogleApiClient.isConnected())) { Wearable.CapabilityApi.removeCapabilityListener( mGoogleApiClient, this, Constants.CAPABILITY_WEAR_APP); Wearable.MessageApi.removeListener(mGoogleApiClient, this); mGoogleApiClient.disconnect(); } } @Override protected void onResume() { Log.d(TAG, "onResume()"); super.onResume(); /* Enables app to handle 23+ (M+) style permissions. It also covers user changing * permission in settings and coming back to the app. */ mPhoneStoragePermissionApproved = ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED; if (mPhoneStoragePermissionApproved) { mPhoneStoragePermissionButton.setCompoundDrawablesWithIntrinsicBounds( R.drawable.ic_permission_approved, 0, 0, 0); } if (mGoogleApiClient != null) { mGoogleApiClient.connect(); } } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { Log.d(TAG, "onActivityResult()"); if (requestCode == REQUEST_WEAR_PERMISSION_RATIONALE) { if (resultCode == Activity.RESULT_OK) { logToUi("Requested permission on wear device(s)."); DataMap dataMap = new DataMap(); dataMap.putInt(Constants.KEY_COMM_TYPE, Constants.COMM_TYPE_REQUEST_PROMPT_PERMISSION); sendMessage(dataMap); } } } @Override public void onConnected(Bundle bundle) { Log.d(TAG, "onConnected()"); // Set up listeners for capability and message changes. Wearable.CapabilityApi.addCapabilityListener( mGoogleApiClient, this, Constants.CAPABILITY_WEAR_APP); Wearable.MessageApi.addListener(mGoogleApiClient, this); // Initial check of capabilities to find the wear nodes. PendingResult pendingResult = Wearable.CapabilityApi.getCapability( mGoogleApiClient, Constants.CAPABILITY_WEAR_APP, CapabilityApi.FILTER_REACHABLE); pendingResult.setResultCallback(new ResultCallback() { @Override public void onResult(CapabilityApi.GetCapabilityResult getCapabilityResult) { CapabilityInfo capabilityInfo = getCapabilityResult.getCapability(); String capabilityName = capabilityInfo.getName(); boolean wearSupportsSampleApp = capabilityName.equals(Constants.CAPABILITY_WEAR_APP); if (wearSupportsSampleApp) { mWearNodeIds = capabilityInfo.getNodes(); /* * Upon getting all wear nodes, we now need to check if the original request to * launch this activity (and PhonePermissionRequestActivity) was initiated by * a wear device. If it was, we need to send back the permission results (data * or rejection of permission) to the wear device. * * Also, note we set variable to false, this enables the user to continue * changing permissions without sending updates to the wear every time. */ if (mWearRequestingPhoneStoragePermission) { mWearRequestingPhoneStoragePermission = false; sendWearPermissionResults(); } } } }); } @Override public void onConnectionSuspended(int i) { Log.d(TAG, "onConnectionSuspended(): connection to location client suspended"); } @Override public void onConnectionFailed(ConnectionResult connectionResult) { Log.e(TAG, "onConnectionFailed(): connection to location client failed"); } public void onCapabilityChanged(CapabilityInfo capabilityInfo) { Log.d(TAG, "onCapabilityChanged(): " + capabilityInfo); mWearNodeIds = capabilityInfo.getNodes(); } public void onMessageReceived(MessageEvent messageEvent) { Log.d(TAG, "onMessageReceived(): " + messageEvent); String messagePath = messageEvent.getPath(); if (messagePath.equals(Constants.MESSAGE_PATH_PHONE)) { DataMap dataMap = DataMap.fromByteArray(messageEvent.getData()); int commType = dataMap.getInt(Constants.KEY_COMM_TYPE, 0); if (commType == Constants.COMM_TYPE_RESPONSE_PERMISSION_REQUIRED) { mWearBodySensorsPermissionApproved = false; updateWearButtonOnUiThread(); /* Because our request for remote data requires a remote permission, we now launch * a splash activity informing the user we need those permissions (along with * other helpful information to approve). */ Intent wearPermissionRationale = new Intent(this, WearPermissionRequestActivity.class); startActivityForResult(wearPermissionRationale, REQUEST_WEAR_PERMISSION_RATIONALE); } else if (commType == Constants.COMM_TYPE_RESPONSE_USER_APPROVED_PERMISSION) { mWearBodySensorsPermissionApproved = true; updateWearButtonOnUiThread(); logToUi("User approved permission on remote device, requesting data again."); DataMap outgoingDataRequestDataMap = new DataMap(); outgoingDataRequestDataMap.putInt(Constants.KEY_COMM_TYPE, Constants.COMM_TYPE_REQUEST_DATA); sendMessage(outgoingDataRequestDataMap); } else if (commType == Constants.COMM_TYPE_RESPONSE_USER_DENIED_PERMISSION) { mWearBodySensorsPermissionApproved = false; updateWearButtonOnUiThread(); logToUi("User denied permission on remote device."); } else if (commType == Constants.COMM_TYPE_RESPONSE_DATA) { mWearBodySensorsPermissionApproved = true; String storageDetails = dataMap.getString(Constants.KEY_PAYLOAD); updateWearButtonOnUiThread(); logToUi(storageDetails); } else { Log.d(TAG, "Unrecognized communication type received."); } } } @Override public void onResult(MessageApi.SendMessageResult sendMessageResult) { if (!sendMessageResult.getStatus().isSuccess()) { Log.d(TAG, "Sending message failed, onResult: " + sendMessageResult); updateWearButtonOnUiThread(); logToUi("Sending message failed."); } else { Log.d(TAG, "Message sent."); } } private void sendMessage(DataMap dataMap) { Log.d(TAG, "sendMessage(): " + mWearNodeIds); if ((mWearNodeIds != null) && (!mWearNodeIds.isEmpty())) { PendingResult pendingResult; for (Node node : mWearNodeIds) { pendingResult = Wearable.MessageApi.sendMessage( mGoogleApiClient, node.getId(), Constants.MESSAGE_PATH_WEAR, dataMap.toByteArray()); pendingResult.setResultCallback(this, Constants.CONNECTION_TIME_OUT_MS, TimeUnit.SECONDS); } } else { // Unable to retrieve node with proper capability mWearBodySensorsPermissionApproved = false; updateWearButtonOnUiThread(); logToUi("Wear devices not available to send message."); } } private void updateWearButtonOnUiThread() { runOnUiThread(new Runnable() { @Override public void run() { if (mWearBodySensorsPermissionApproved) { mWearBodySensorsPermissionButton.setCompoundDrawablesWithIntrinsicBounds( R.drawable.ic_permission_approved, 0, 0, 0); } else { mWearBodySensorsPermissionButton.setCompoundDrawablesWithIntrinsicBounds( R.drawable.ic_permission_denied, 0, 0, 0); } } }); } /* * Handles all messages for the UI coming on and off the main thread. Not all callbacks happen * on the main thread. */ private void logToUi(final String message) { boolean mainUiThread = (Looper.myLooper() == Looper.getMainLooper()); if (mainUiThread) { if (!message.isEmpty()) { Log.d(TAG, message); mOutputTextView.setText(message); } } else { if (!message.isEmpty()) { runOnUiThread(new Runnable() { @Override public void run() { Log.d(TAG, message); mOutputTextView.setText(message); } }); } } } private String getPhoneStorageInformation() { StringBuilder stringBuilder = new StringBuilder(); String state = Environment.getExternalStorageState(); boolean isExternalStorageReadable = Environment.MEDIA_MOUNTED.equals(state) || Environment.MEDIA_MOUNTED_READ_ONLY.equals(state); if (isExternalStorageReadable) { File externalStorageDirectory = Environment.getExternalStorageDirectory(); String[] fileList = externalStorageDirectory.list(); if (fileList.length > 0) { stringBuilder.append("List of files\n"); for (String file : fileList) { stringBuilder.append(" - " + file + "\n"); } } else { stringBuilder.append("No files in external storage."); } } else { stringBuilder.append("No external media is available."); } return stringBuilder.toString(); } private void sendWearPermissionResults() { Log.d(TAG, "sendWearPermissionResults()"); DataMap dataMap = new DataMap(); if (mPhoneStoragePermissionApproved) { dataMap.putInt(Constants.KEY_COMM_TYPE, Constants.COMM_TYPE_RESPONSE_USER_APPROVED_PERMISSION); } else { dataMap.putInt(Constants.KEY_COMM_TYPE, Constants.COMM_TYPE_RESPONSE_USER_DENIED_PERMISSION); } sendMessage(dataMap); } }