/* * Copyright (C) 2014 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.example.android.wearable.delayedconfirmation; import android.app.Activity; import android.app.Notification; import android.app.NotificationManager; import android.os.Bundle; import android.support.wearable.view.DelayedConfirmationView; import android.util.Log; import android.view.View; import android.widget.Toast; import com.google.android.gms.common.ConnectionResult; import com.google.android.gms.common.api.GoogleApiClient; 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.MessageApi; import com.google.android.gms.wearable.Node; import com.google.android.gms.wearable.Wearable; import java.util.Set; public class MainActivity extends Activity implements DelayedConfirmationView.DelayedConfirmationListener, GoogleApiClient.OnConnectionFailedListener, GoogleApiClient.ConnectionCallbacks, CapabilityApi.CapabilityListener { private static final String TAG = "DelayedConfirmation"; private static final int NUM_SECONDS = 5; private static final String TIMER_SELECTED_PATH = "/timer_selected"; private static final String TIMER_FINISHED_PATH = "/timer_finished"; /* name of the capability that the phone side provides */ private static final String CONFIRMATION_HANDLER_CAPABILITY_NAME = "confirmation_handler"; private DelayedConfirmationView delayedConfirmationView; private GoogleApiClient mGoogleApiClient; /* the preferred note that can handle the confirmation capability */ private Node mConfirmationHandlerNode; @Override public void onCreate(Bundle b) { super.onCreate(b); setContentView(R.layout.main_activity); delayedConfirmationView = (DelayedConfirmationView) findViewById(R.id.delayed_confirmation); delayedConfirmationView.setTotalTimeMs(NUM_SECONDS * 1000); mGoogleApiClient = new GoogleApiClient.Builder(this) .addApi(Wearable.API) .addOnConnectionFailedListener(this) .addConnectionCallbacks(this) .build(); } @Override protected void onResume() { super.onResume(); if (!mGoogleApiClient.isConnected()) { mGoogleApiClient.connect(); } } @Override protected void onPause() { if (mGoogleApiClient.isConnected()) { Wearable.CapabilityApi.removeCapabilityListener(mGoogleApiClient, this, CONFIRMATION_HANDLER_CAPABILITY_NAME); mGoogleApiClient.disconnect(); } super.onPause(); } /** * Starts the DelayedConfirmationView when user presses "Start Timer" button. */ public void onStartTimer(View view) { delayedConfirmationView.start(); delayedConfirmationView.setListener(this); } @Override public void onTimerSelected(View v) { v.setPressed(true); Notification notification = new Notification.Builder(this) .setSmallIcon(R.drawable.ic_launcher) .setContentTitle(getString(R.string.notification_title)) .setContentText(getString(R.string.notification_timer_selected)) .build(); ((NotificationManager) getSystemService(NOTIFICATION_SERVICE)).notify(0, notification); sendMessageToCompanion(TIMER_SELECTED_PATH); // Prevent onTimerFinished from being heard. ((DelayedConfirmationView) v).setListener(null); finish(); } @Override public void onTimerFinished(View v) { Notification notification = new Notification.Builder(this) .setSmallIcon(R.drawable.ic_launcher) .setContentTitle(getString(R.string.notification_title)) .setContentText(getString(R.string.notification_timer_finished)) .build(); ((NotificationManager) getSystemService(NOTIFICATION_SERVICE)).notify(0, notification); sendMessageToCompanion(TIMER_FINISHED_PATH); finish(); } @Override public void onConnectionFailed(ConnectionResult connectionResult) { Log.e(TAG, "Failed to connect to Google Api Client"); mConfirmationHandlerNode = null; } private void sendMessageToCompanion(final String path) { if (mConfirmationHandlerNode != null) { Wearable.MessageApi.sendMessage(mGoogleApiClient, mConfirmationHandlerNode.getId(), path, new byte[0]) .setResultCallback(getSendMessageResultCallback(mConfirmationHandlerNode)); } else { Toast.makeText(this, R.string.no_device_found, Toast.LENGTH_SHORT).show(); } } private ResultCallback getSendMessageResultCallback( final Node node) { return new ResultCallback() { @Override public void onResult(MessageApi.SendMessageResult sendMessageResult) { if (!sendMessageResult.getStatus().isSuccess()) { Log.e(TAG, "Failed to send message with status " + sendMessageResult.getStatus()); } else { Log.d(TAG, "Sent confirmation message to node " + node.getDisplayName()); } } }; } private void setupConfirmationHandlerNode() { Wearable.CapabilityApi.addCapabilityListener( mGoogleApiClient, this, CONFIRMATION_HANDLER_CAPABILITY_NAME); Wearable.CapabilityApi.getCapability( mGoogleApiClient, CONFIRMATION_HANDLER_CAPABILITY_NAME, CapabilityApi.FILTER_REACHABLE).setResultCallback( new ResultCallback() { @Override public void onResult(CapabilityApi.GetCapabilityResult result) { if (!result.getStatus().isSuccess()) { Log.e(TAG, "setupConfirmationHandlerNode() Failed to get capabilities, " + "status: " + result.getStatus().getStatusMessage()); return; } updateConfirmationCapability(result.getCapability()); } }); } private void updateConfirmationCapability(CapabilityInfo capabilityInfo) { Set connectedNodes = capabilityInfo.getNodes(); if (connectedNodes.isEmpty()) { mConfirmationHandlerNode = null; } else { mConfirmationHandlerNode = pickBestNode(connectedNodes); } } /** * We pick a node that is capabale of handling the confirmation. If there is more than one, * then we would prefer the one that is directly connected to this device. In general, * depending on the situation and requirements, the "best" node might be picked based on other * criteria. */ private Node pickBestNode(Set connectedNodes) { Node best = null; if (connectedNodes != null) { for (Node node : connectedNodes) { if (node.isNearby()) { return node; } best = node; } } return best; } @Override public void onConnected(Bundle bundle) { setupConfirmationHandlerNode(); } @Override public void onConnectionSuspended(int cause) { mConfirmationHandlerNode = null; } @Override public void onCapabilityChanged(CapabilityInfo capabilityInfo) { updateConfirmationCapability(capabilityInfo); } }