Home | History | Annotate | Download | only in suggestions
      1 /*
      2  * Copyright (C) 2017 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.android.settingslib.suggestions;
     18 
     19 import android.content.ComponentName;
     20 import android.content.Context;
     21 import android.content.Intent;
     22 import android.content.ServiceConnection;
     23 import android.os.IBinder;
     24 import android.os.RemoteException;
     25 import android.service.settings.suggestions.ISuggestionService;
     26 import android.service.settings.suggestions.Suggestion;
     27 import android.util.Log;
     28 
     29 import androidx.annotation.Nullable;
     30 import androidx.annotation.WorkerThread;
     31 
     32 import java.util.List;
     33 
     34 /**
     35  * A controller class to access suggestion data.
     36  */
     37 public class SuggestionController {
     38 
     39     /**
     40      * Callback interface when service is connected/disconnected.
     41      */
     42     public interface ServiceConnectionListener {
     43         /**
     44          * Called when service is connected.
     45          */
     46         void onServiceConnected();
     47 
     48         /**
     49          * Called when service is disconnected.
     50          */
     51         void onServiceDisconnected();
     52     }
     53 
     54     private static final String TAG = "SuggestionController";
     55     private static final boolean DEBUG = false;
     56 
     57     private final Context mContext;
     58     private final Intent mServiceIntent;
     59 
     60     private ServiceConnection mServiceConnection;
     61     private ISuggestionService mRemoteService;
     62     private ServiceConnectionListener mConnectionListener;
     63 
     64     /**
     65      * Create a new controller instance.
     66      *
     67      * @param context  caller context
     68      * @param service  The component name for service.
     69      * @param listener listener to receive service connected/disconnected event.
     70      */
     71     public SuggestionController(Context context, ComponentName service,
     72             ServiceConnectionListener listener) {
     73         mContext = context.getApplicationContext();
     74         mConnectionListener = listener;
     75         mServiceIntent = new Intent().setComponent(service);
     76         mServiceConnection = createServiceConnection();
     77     }
     78 
     79     /**
     80      * Start the controller.
     81      */
     82     public void start() {
     83         mContext.bindServiceAsUser(mServiceIntent, mServiceConnection, Context.BIND_AUTO_CREATE,
     84                 android.os.Process.myUserHandle());
     85     }
     86 
     87     /**
     88      * Stop the controller.
     89      */
     90     public void stop() {
     91         if (mRemoteService != null) {
     92             mRemoteService = null;
     93             mContext.unbindService(mServiceConnection);
     94         }
     95     }
     96 
     97     /**
     98      * Get setting suggestions.
     99      */
    100     @Nullable
    101     @WorkerThread
    102     public List<Suggestion> getSuggestions() {
    103         if (!isReady()) {
    104             return null;
    105         }
    106         try {
    107             return mRemoteService.getSuggestions();
    108         } catch (NullPointerException e) {
    109             Log.w(TAG, "mRemote service detached before able to query", e);
    110             return null;
    111         } catch (RemoteException | RuntimeException e) {
    112             Log.w(TAG, "Error when calling getSuggestion()", e);
    113             return null;
    114         }
    115     }
    116 
    117     public void dismissSuggestions(Suggestion suggestion) {
    118         if (!isReady()) {
    119             Log.w(TAG, "SuggestionController not ready, cannot dismiss " + suggestion.getId());
    120             return;
    121         }
    122         try {
    123             mRemoteService.dismissSuggestion(suggestion);
    124         } catch (RemoteException | RuntimeException e) {
    125             Log.w(TAG, "Error when calling dismissSuggestion()", e);
    126         }
    127     }
    128 
    129     public void launchSuggestion(Suggestion suggestion) {
    130         if (!isReady()) {
    131             Log.w(TAG, "SuggestionController not ready, cannot launch " + suggestion.getId());
    132             return;
    133         }
    134 
    135         try {
    136             mRemoteService.launchSuggestion(suggestion);
    137         } catch (RemoteException | RuntimeException e) {
    138             Log.w(TAG, "Error when calling launchSuggestion()", e);
    139         }
    140     }
    141 
    142     /**
    143      * Whether or not the manager is ready
    144      */
    145     private boolean isReady() {
    146         return mRemoteService != null;
    147     }
    148 
    149     /**
    150      * Create a new {@link ServiceConnection} object to handle service connect/disconnect event.
    151      */
    152     private ServiceConnection createServiceConnection() {
    153         return new ServiceConnection() {
    154 
    155             @Override
    156             public void onServiceConnected(ComponentName name, IBinder service) {
    157                 if (DEBUG) {
    158                     Log.d(TAG, "Service is connected");
    159                 }
    160                 mRemoteService = ISuggestionService.Stub.asInterface(service);
    161                 if (mConnectionListener != null) {
    162                     mConnectionListener.onServiceConnected();
    163                 }
    164             }
    165 
    166             @Override
    167             public void onServiceDisconnected(ComponentName name) {
    168                 if (mConnectionListener != null) {
    169                     mRemoteService = null;
    170                     mConnectionListener.onServiceDisconnected();
    171                 }
    172             }
    173         };
    174     }
    175 }
    176