Home | History | Annotate | Download | only in com.example.android.wearable.watchface
      1 /*
      2  * Copyright (C) 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.wearable.watchface;
     18 
     19 import android.app.Activity;
     20 import android.app.AlertDialog;
     21 import android.content.ComponentName;
     22 import android.content.DialogInterface;
     23 import android.graphics.Color;
     24 import android.net.Uri;
     25 import android.os.Bundle;
     26 import android.support.wearable.companion.WatchFaceCompanion;
     27 import android.util.Log;
     28 import android.view.View;
     29 import android.widget.AdapterView;
     30 import android.widget.Spinner;
     31 import android.widget.TextView;
     32 
     33 import com.google.android.gms.common.ConnectionResult;
     34 import com.google.android.gms.common.api.GoogleApiClient;
     35 import com.google.android.gms.common.api.ResultCallback;
     36 import com.google.android.gms.wearable.DataApi;
     37 import com.google.android.gms.wearable.DataItem;
     38 import com.google.android.gms.wearable.DataMap;
     39 import com.google.android.gms.wearable.DataMapItem;
     40 import com.google.android.gms.wearable.Wearable;
     41 
     42 /**
     43  * The phone-side config activity for {@code DigitalWatchFaceService}. Like the watch-side config
     44  * activity ({@code DigitalWatchFaceWearableConfigActivity}), allows for setting the background
     45  * color. Additionally, enables setting the color for hour, minute and second digits.
     46  */
     47 public class DigitalWatchFaceCompanionConfigActivity extends Activity
     48         implements GoogleApiClient.ConnectionCallbacks, GoogleApiClient.OnConnectionFailedListener,
     49                 ResultCallback<DataApi.DataItemResult> {
     50     private static final String TAG = "DigitalWatchFaceConfig";
     51 
     52     // TODO: use the shared constants (needs covering all the samples with Gradle build model)
     53     private static final String KEY_BACKGROUND_COLOR = "BACKGROUND_COLOR";
     54     private static final String KEY_HOURS_COLOR = "HOURS_COLOR";
     55     private static final String KEY_MINUTES_COLOR = "MINUTES_COLOR";
     56     private static final String KEY_SECONDS_COLOR = "SECONDS_COLOR";
     57     private static final String PATH_WITH_FEATURE = "/watch_face_config/Digital";
     58 
     59     private GoogleApiClient mGoogleApiClient;
     60     private String mPeerId;
     61 
     62     @Override
     63     protected void onCreate(Bundle savedInstanceState) {
     64         super.onCreate(savedInstanceState);
     65         setContentView(R.layout.activity_digital_watch_face_config);
     66 
     67         mPeerId = getIntent().getStringExtra(WatchFaceCompanion.EXTRA_PEER_ID);
     68         mGoogleApiClient = new GoogleApiClient.Builder(this)
     69                 .addConnectionCallbacks(this)
     70                 .addOnConnectionFailedListener(this)
     71                 .addApi(Wearable.API)
     72                 .build();
     73 
     74         ComponentName name = getIntent().getParcelableExtra(
     75                 WatchFaceCompanion.EXTRA_WATCH_FACE_COMPONENT);
     76         TextView label = (TextView)findViewById(R.id.label);
     77         label.setText(label.getText() + " (" + name.getClassName() + ")");
     78     }
     79 
     80     @Override
     81     protected void onStart() {
     82         super.onStart();
     83         mGoogleApiClient.connect();
     84     }
     85 
     86     @Override
     87     protected void onStop() {
     88         if (mGoogleApiClient != null && mGoogleApiClient.isConnected()) {
     89             mGoogleApiClient.disconnect();
     90         }
     91         super.onStop();
     92     }
     93 
     94     @Override // GoogleApiClient.ConnectionCallbacks
     95     public void onConnected(Bundle connectionHint) {
     96         if (Log.isLoggable(TAG, Log.DEBUG)) {
     97             Log.d(TAG, "onConnected: " + connectionHint);
     98         }
     99 
    100         if (mPeerId != null) {
    101             Uri.Builder builder = new Uri.Builder();
    102             Uri uri = builder.scheme("wear").path(PATH_WITH_FEATURE).authority(mPeerId).build();
    103             Wearable.DataApi.getDataItem(mGoogleApiClient, uri).setResultCallback(this);
    104         } else {
    105             displayNoConnectedDeviceDialog();
    106         }
    107     }
    108 
    109     @Override // ResultCallback<DataApi.DataItemResult>
    110     public void onResult(DataApi.DataItemResult dataItemResult) {
    111         if (dataItemResult.getStatus().isSuccess() && dataItemResult.getDataItem() != null) {
    112             DataItem configDataItem = dataItemResult.getDataItem();
    113             DataMapItem dataMapItem = DataMapItem.fromDataItem(configDataItem);
    114             DataMap config = dataMapItem.getDataMap();
    115             setUpAllPickers(config);
    116         } else {
    117             // If DataItem with the current config can't be retrieved, select the default items on
    118             // each picker.
    119             setUpAllPickers(null);
    120         }
    121     }
    122 
    123     @Override // GoogleApiClient.ConnectionCallbacks
    124     public void onConnectionSuspended(int cause) {
    125         if (Log.isLoggable(TAG, Log.DEBUG)) {
    126             Log.d(TAG, "onConnectionSuspended: " + cause);
    127         }
    128     }
    129 
    130     @Override // GoogleApiClient.OnConnectionFailedListener
    131     public void onConnectionFailed(ConnectionResult result) {
    132         if (Log.isLoggable(TAG, Log.DEBUG)) {
    133             Log.d(TAG, "onConnectionFailed: " + result);
    134         }
    135     }
    136 
    137     private void displayNoConnectedDeviceDialog() {
    138         AlertDialog.Builder builder = new AlertDialog.Builder(this);
    139         String messageText = getResources().getString(R.string.title_no_device_connected);
    140         String okText = getResources().getString(R.string.ok_no_device_connected);
    141         builder.setMessage(messageText)
    142                 .setCancelable(false)
    143                 .setPositiveButton(okText, new DialogInterface.OnClickListener() {
    144                     public void onClick(DialogInterface dialog, int id) { }
    145                 });
    146         AlertDialog alert = builder.create();
    147         alert.show();
    148     }
    149 
    150     /**
    151      * Sets up selected items for all pickers according to given {@code config} and sets up their
    152      * item selection listeners.
    153      *
    154      * @param config the {@code DigitalWatchFaceService} config {@link DataMap}. If null, the
    155      *         default items are selected.
    156      */
    157     private void setUpAllPickers(DataMap config) {
    158         setUpColorPickerSelection(R.id.background, KEY_BACKGROUND_COLOR, config,
    159                 R.string.color_black);
    160         setUpColorPickerSelection(R.id.hours, KEY_HOURS_COLOR, config, R.string.color_white);
    161         setUpColorPickerSelection(R.id.minutes, KEY_MINUTES_COLOR, config, R.string.color_white);
    162         setUpColorPickerSelection(R.id.seconds, KEY_SECONDS_COLOR, config, R.string.color_gray);
    163 
    164         setUpColorPickerListener(R.id.background, KEY_BACKGROUND_COLOR);
    165         setUpColorPickerListener(R.id.hours, KEY_HOURS_COLOR);
    166         setUpColorPickerListener(R.id.minutes, KEY_MINUTES_COLOR);
    167         setUpColorPickerListener(R.id.seconds, KEY_SECONDS_COLOR);
    168     }
    169 
    170     private void setUpColorPickerSelection(int spinnerId, final String configKey, DataMap config,
    171             int defaultColorNameResId) {
    172         String defaultColorName = getString(defaultColorNameResId);
    173         int defaultColor = Color.parseColor(defaultColorName);
    174         int color;
    175         if (config != null) {
    176             color = config.getInt(configKey, defaultColor);
    177         } else {
    178             color = defaultColor;
    179         }
    180         Spinner spinner = (Spinner) findViewById(spinnerId);
    181         String[] colorNames = getResources().getStringArray(R.array.color_array);
    182         for (int i = 0; i < colorNames.length; i++) {
    183             if (Color.parseColor(colorNames[i]) == color) {
    184                 spinner.setSelection(i);
    185                 break;
    186             }
    187         }
    188     }
    189 
    190     private void setUpColorPickerListener(int spinnerId, final String configKey) {
    191         Spinner spinner = (Spinner) findViewById(spinnerId);
    192         spinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
    193             @Override
    194             public void onItemSelected(AdapterView<?> adapterView, View view, int pos, long id) {
    195                 final String colorName = (String) adapterView.getItemAtPosition(pos);
    196                 sendConfigUpdateMessage(configKey, Color.parseColor(colorName));
    197             }
    198 
    199             @Override
    200             public void onNothingSelected(AdapterView<?> adapterView) { }
    201         });
    202     }
    203 
    204     private void sendConfigUpdateMessage(String configKey, int color) {
    205         if (mPeerId != null) {
    206             DataMap config = new DataMap();
    207             config.putInt(configKey, color);
    208             byte[] rawData = config.toByteArray();
    209             Wearable.MessageApi.sendMessage(mGoogleApiClient, mPeerId, PATH_WITH_FEATURE, rawData);
    210 
    211             if (Log.isLoggable(TAG, Log.DEBUG)) {
    212                 Log.d(TAG, "Sent watch face config message: " + configKey + " -> "
    213                         + Integer.toHexString(color));
    214             }
    215         }
    216     }
    217 }
    218