How do I find out the current keyboard used on iOS

2019-03-11 06:16发布

问题:

You can get a list of the keyboards installed on the iOS device using:

NSUserDefaults *userDeafaults = [NSUserDefaults standardUserDefaults];
NSDictionary * userDefaultsDict = [userDeafaults dictionaryRepresentation];
NSLog(@"%@", userDefaultsDict);

This yields something in the console like:

{
    ...
    AppleKeyboards =     (
        "en_US@hw=US;sw=QWERTY",
        "es_ES@hw=Spanish - ISO;sw=QWERTY-Spanish",
        "emoji@sw=Emoji",
        "com.swiftkey.SwiftKeyApp.Keyboard"
    );
    AppleKeyboardsExpanded = 1;
    ...
}

This tells me that the device has the Spanish, Emoji and SwiftKey keyboards installed, but it tells me nothing about which will be used when the keyboard comes up.

Is there a way to tell?

回答1:

There is no public API for this, but I found a solution for you, which requires very little "gray area API" (I define API as "gray area" if an API is not normally exposed, but can be hidden with little to no work).

iOS has the following class: UITextInputMode

This class gives you all the input methods the user can use. Using the following query will give you the currently used, only when the keyboard is open:

UITextInputMode* inputMode = [[[UITextInputMode activeInputModes] filteredArrayUsingPredicate:[NSPredicate predicateWithFormat:@"isDisplayed = YES"]] lastObject];

To get the display name of the extension (or regular Apple keyboard), use:

[inputMode valueForKey:@"displayName"]

or

[inputMode valueForKey:@"extendedDisplayName"]

This only works when the keyboard is visible. So you will have to monitor input mode change yourself using

[[NSNotificationCenter defaultCenter] addObserverForName:UITextInputCurrentInputModeDidChangeNotification object:nil queue:[NSOperationQueue mainQueue] usingBlock:^(NSNotification *note)
 {
     dispatch_async(dispatch_get_main_queue(), ^{
         NSLog(@"%@", [[[[UITextInputMode activeInputModes] filteredArrayUsingPredicate:[NSPredicate predicateWithFormat:@"isDisplayed = YES"]] lastObject] valueForKey:@"extendedDisplayName"]);
     });
 }];

We actually need to delay obtaining the current input mode, as the notification is sent before the keyboard internal implementation has updated the system with new value. Obtaining it on the next runloop works well.



回答2:

Leo Natan's answer is great, but I would like to add something to it. You can actually get the current input mode at any time, not just when the keyboard is open, like this:

UITextView *textView = [[UITextView alloc] init];
UITextInputMode *inputMode = textView.textInputMode;

Please note that textView.textInputMode is nil for the Emoji keyboard for some reason.

Also, in addition to displayName and extendedDisplayName, there are other keys you can retrieve, such as identifier, normalizedIdentifier (iOS 8+), hardwareLayout, ... See the full API here:

https://github.com/nst/iOS-Runtime-Headers/blob/master/Frameworks/UIKit.framework/UIKeyboardInputMode.h

Now I'm not sure if using any of those is more risky than displayName for App Store approval...



回答3:

@Leo Natan's answers is cool but it's may return nil when the keyboard have not display.

So here I use the string to find the UIKeyboardInputMode's property.

I can tell you that this can find out the current keyboard because it's comes from Apple's Private API.

Code here:

+ (BOOL)isTheCustomKeyboard
{
    UITextInputMode* inputMode = [UITextInputMode currentInputMode];
    if ([inputMode respondsToSelector:NSSelectorFromString(@"identifier")])
    {
        NSString* indentifier = [inputMode performSelector:NSSelectorFromString(@"identifier")];
        if ([indentifier isEqualToString: YOUR_APP_ID])
        {
            return YES;
        }
    }
   return NO;
}

And more:

+ (BOOL)isContaintCustomKeyboard
{
    NSArray * inputModes = [UITextInputMode activeInputModes];
    for (id inputModel in inputModes)
    {
        if ([inputModel respondsToSelector:NSSelectorFromString(@"identifier")])
        {
            NSString* indentifier = [inputModel performSelector:NSSelectorFromString(@"identifier")];
            if ([indentifier isEqualToString: YOUR_APP_ID])
            {
                return YES;
            }
        }
    }
    return NO;
}

Actually we can also use the displayName or the identifier and more.