How does Google's custom iOS keyboard, Gboard, programmatically dismiss the frontmost app?

后端 未结 1 1268
死守一世寂寞
死守一世寂寞 2020-12-04 15:38

Google\'s custom iOS app, Gboard, has an interesting feature that can\'t be accomplished using public APIs for in the iOS SDK (as of iOS 10). I\'d like to know exac

相关标签:
1条回答
  • 2020-12-04 16:05

    Your guess is correct — Gboard is using private API to do it.

    … though not through exploring view hierarchy or event injection.

    When the voice-to-text action is done, we can check the syslog from Xcode or Console that it calls the -[AVAudioSession setActive:withOptions:error:] method. So I've reverse-engineered the Gboard app and look for the stack trace related to this.

    Climbing up the call stack we can find the -[GKBVoiceRecognitionViewController navigateBackToPreviousApp] method, and…

    …_systemNavigationAction? Yep, definitely private API.

    Since class_getInstanceVariable is a public API and "_systemNavigationAction" is a string literal, the automatic checker is not able to note the private API usage, and the human reviewers probably don't see anything wrong with the "jump back to the previous app" behavior. Or probably because they are Google and you are not…


    The actual code that performs the "jump back to previous app" action is like this:

    @import UIKit;
    @import ObjectiveC.runtime;
    
    @interface UISystemNavigationAction : NSObject
    @property(nonatomic, readonly, nonnull) NSArray<NSNumber*>* destinations;
    -(BOOL)sendResponseForDestination:(NSUInteger)destination;
    @end
    
    inline BOOL jumpBackToPreviousApp() {
        Ivar sysNavIvar = class_getInstanceVariable(UIApplication.class, "_systemNavigationAction");
        UIApplication* app = UIApplication.sharedApplication;
        UISystemNavigationAction* action = object_getIvar(app, sysNavIvar);
        if (!action) {
            return NO;
        }
        NSUInteger destination = action.destinations.firstObject.unsignedIntegerValue;
        return [action sendResponseForDestination:destination];
    }
    

    In particular, the -sendResponseForDestination: method performs the actual "go back" action.

    (Since the API is undocumented, Gboard is actually using the API incorrectly. They used the wrong signature -(void)sendResponseForDestination:(id)destination. But it happens that all numbers other than 1 will work the same, so the Google developers are lucky this time)

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