Programmatically accept call in Nougat

前端 未结 4 2114
孤街浪徒
孤街浪徒 2020-12-24 09:04

From one year, I have been working over IOT product and the application attached was working fine. Now I am not able to accept call programmatically in higher versions of an

相关标签:
4条回答
  • 2020-12-24 09:11

    For answering a call using a button, set a flag whenever an incoming call is detected:

     if(state==TelephonyManager.CALL_STATE_RINGING){
               shouldAnswerCallViaNotification = true;
            } else {
                shouldAnswerCallViaNotification = false;
            }
    

    Now, create a list in your NSLogService class,

    static ArrayList<StatusBarNotification> statusBarNotifications;
    

    and in your onNotificationPosted() add StatusBarNotification to a list,

       @Override
    public void onNotificationPosted(StatusBarNotification sbn) {
    
        if (HomeScreen.shouldAnswerCallViaNotification) {
            if (statusBarNotifications == null) {
                updateNotificationList();
            }
           statusBarNotifications.add(sbn);
    
        } else {
            updateNotificationList();
        }
    
    }
     public static ArrayList<StatusBarNotification> getAllNotifications() {
        return statusBarNotifications;
    }
    
    public static void updateNotificationList() {
    
        if (statusBarNotifications != null)
            statusBarNotifications = null;
        statusBarNotifications = new ArrayList<StatusBarNotification>();
    }
    

    In your HomeScreen, on Button click, call performNotificationOperation(NLService.getAllNotifications());

    here is a definition for this method:

     private void performNotificationOperation(ArrayList<StatusBarNotification> activeNotifications) {
    
        if (activeNotifications.size()> 0) {
            main_Loop:
            for (StatusBarNotification notification : activeNotifications) {
                try {
                    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
                        if (notification.getNotification().actions != null) {
                            for (Notification.Action action : notification.getNotification().actions) {
                                //            Log.e(TAG, "" + action);
    
                                Log.e(TAG, "" + action.title);
                                if (action.title.toString().equalsIgnoreCase("Answer")) {
                                    Log.e(TAG, "" + true);
                                    PendingIntent intent = action.actionIntent;
    
                                    try {
                                        intent.send();
                                    } catch (PendingIntent.CanceledException e) {
                                        e.printStackTrace();
                                    }
                                    break main_Loop;
                                }
    
    
                            }
                        }
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
    
            }
        }
        try {
            NLService.updateNotificationList();
        } catch (Exception e) {
            e.printStackTrace();
        }
    
    }
    
    0 讨论(0)
  • 2020-12-24 09:18

    According to "Vishal Sharma" answer we can get action button title dynamically ,for supporting other languages :

      @Override
      public void onNotificationPosted(StatusBarNotification sbn) {
        try {
          if (sbn.getNotification().actions != null) {
            Notification.Action[] notificationAction=sbn.getNotification().actions;
            //Log.e(G.TAG, "" +notificationAction  + " =>"+notificationAction.length);
               String rejectTitle=""+notificationAction[0].title;
               String acceptTitle=""+notificationAction[1].title;
    
            for (Notification.Action action : notificationAction){
              if (action.title.toString().equalsIgnoreCase(acceptTitle)) {
                PendingIntent intent = action.actionIntent;
                try {
                  intent.send();
                } catch (PendingIntent.CanceledException e) {
                  e.printStackTrace();
                }
              }
            }
    
    
          }
        } catch (Exception e) {
          e.printStackTrace();
        }
      }
    
    0 讨论(0)
  • 2020-12-24 09:20

    It is sort of hack, you can use accessibility service to receive call. To enable accessibility service, you must enable your service on Setting - Accessibility - Your service.

    First, add typeWindowContentChanged to accessibilityEventTypes.

    <accessibility-service
         android:accessibilityEventTypes="typeViewClicked|typeViewFocused|typeViewScrolled|typeWindowContentChanged|typeWindowStateChanged"
         android:packageNames="com.example.android.myFirstApp, com.example.android.mySecondApp"
         android:accessibilityFeedbackType="feedbackSpoken"
         android:notificationTimeout="100"
         android:settingsActivity="com.example.android.apis.accessibility.TestBackActivity"
         android:canRetrieveWindowContent="true"
    />
    

    And do something with event or "displayed text" or "contents description".

    @Override
    public void onAccessibilityEvent(AccessibilityEvent event) {
    
        // Do something with Click or Focused event
        final int eventType = event.getEventType();
        String eventText = null;
        switch(eventType) {
            case AccessibilityEvent.TYPE_VIEW_CLICKED:
                eventText = "Focused: ";
                break;
            case AccessibilityEvent.TYPE_VIEW_FOCUSED:
                eventText = "Focused: ";
                break;
        }
        eventText = eventText + event.getContentDescription();
    
    
        // Traverse all items in screen. 
        // Do something with text.
    
        AccessibilityNodeInfo info = getRootInActiveWindow();
    
        int index;
        int count = info.getChildCount();
        AccessibilityNodeInfo child;
    
        for (index = 0; index < count; index++) {
            child = info.getChild(index);
            if (child.getText() != null)
                Log.d(TAG, "text: " + child.getText().toString() + " " + child.getContentDescription());
    
            // perform Click
            //if (child.isClickable());
                //child.performAction(AccessibilityNodeInfo.ACTION_CLICK);
        }
    }
    

    Yes, I know this is not a graceful way to solve your problem. It is a kind of hack.

    0 讨论(0)
  • 2020-12-24 09:34

    As I am also working on IOT product, this was one of the biggest issue I faced but after some Research, I think I have found some solution for this problem, or you can say a simple hack. I have tested this hack in several devices with several versions, and found that most of the devices are responding. Only Samsung devices are not responding, some Huawei devices and some Oppo devices are also not responding.(I am still looking something for these devices too).

    I noticed that Android provides one feature of Accessing Notifications. You can use NotificationListenerService to read notifications and perform some actions over them. It provides some override methods:

     onNotificationPosted()
        onNotificationRemoved()
        getActiveNotifications()
    

    ... etc

    Here is a code: Create a Service that extends NotificationListenerService

     class NLService extends NotificationListenerService {
    
         @Override
         public void onNotificationPosted(StatusBarNotification sbn) {
           ....
         }
    
         @Override
         public void onNotificationRemoved(StatusBarNotification sbn) {
           ....
         }
    

    In AndroidMenifest, add this service as:

     <service
            android:name=".NLService"
            android:label="@string/app_name"
    android:permission="android.permission.BIND_NOTIFICATION_LISTENER_SERVICE">
            <intent-filter>
                <action 
    android:name="android.service.notification.NotificationListenerService" />
            </intent-filter>
        </service>
    

    This will allow your application to read any notification received.

    Now, here is the main code:

    In onNotificationPosted(StatusBarNotification sbn) add this code:

     @Override
        public void onNotificationPosted(StatusBarNotification sbn) {
            try {
                if (sbn.getNotification().actions != null) {
                    for (Notification.Action action : sbn.getNotification().actions) 
                      {
                        Log.e(TAG, "" + action.title);
                        if (action.title.toString().equalsIgnoreCase("Answer")) {
                            Log.e(TAG, "" + true);
                            PendingIntent intent = action.actionIntent;
    
                            try {
                                intent.send();
                            } catch (PendingIntent.CanceledException e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }
              } catch (Exception e) {
                  e.printStackTrace();
              }
         }
    

    Thats it!

    All is set, Run the application and the devices except Samsung, whichever shows a notification for Incoming call, with Answer and Reject/Decline Action buttons, will allow you to answer a call.

    To open Notification Access Settings and allowing your application to read notification, use:

     Intent intent = new 
        Intent("android.settings.ACTION_NOTIFICATION_LISTENER_SETTINGS");
            startActivity(intent); 
    

    Just create a POC for this and let me know about how it works.

    Please mark my answer if this helps.

    Also, if you could provide some solution for the same regarding Samsung Devices, please update.

    Thanks

    0 讨论(0)
提交回复
热议问题