2015-03-23 13:28:42 -07:00
|
|
|
/**
|
|
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This source code is licensed under the BSD-style license found in the
|
|
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
|
|
*/
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
#import "RCTKeyCommands.h"
|
|
|
|
|
|
|
|
#import <UIKit/UIKit.h>
|
|
|
|
|
2015-06-22 08:08:35 -07:00
|
|
|
#import "RCTDefines.h"
|
2015-02-19 20:10:52 -08:00
|
|
|
#import "RCTUtils.h"
|
|
|
|
|
2015-06-22 08:08:35 -07:00
|
|
|
#if RCT_DEV
|
|
|
|
|
|
|
|
static BOOL RCTIsIOS8OrEarlier()
|
|
|
|
{
|
|
|
|
return [UIDevice currentDevice].systemVersion.floatValue < 9;
|
|
|
|
}
|
|
|
|
|
2015-06-19 08:08:14 -07:00
|
|
|
@interface RCTKeyCommand : NSObject <NSCopying>
|
|
|
|
|
|
|
|
@property (nonatomic, strong) UIKeyCommand *keyCommand;
|
|
|
|
@property (nonatomic, copy) void (^block)(UIKeyCommand *);
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation RCTKeyCommand
|
|
|
|
|
|
|
|
- (instancetype)initWithKeyCommand:(UIKeyCommand *)keyCommand
|
|
|
|
block:(void (^)(UIKeyCommand *))block
|
|
|
|
{
|
|
|
|
if ((self = [super init])) {
|
|
|
|
_keyCommand = keyCommand;
|
2015-06-22 08:08:35 -07:00
|
|
|
_block = block;
|
2015-06-19 08:08:14 -07:00
|
|
|
}
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
2015-08-24 09:14:33 -01:00
|
|
|
RCT_NOT_IMPLEMENTED(- (instancetype)init)
|
2015-06-19 08:08:14 -07:00
|
|
|
|
|
|
|
- (id)copyWithZone:(__unused NSZone *)zone
|
|
|
|
{
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSUInteger)hash
|
|
|
|
{
|
|
|
|
return _keyCommand.input.hash ^ _keyCommand.modifierFlags;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isEqual:(RCTKeyCommand *)object
|
|
|
|
{
|
|
|
|
if (![object isKindOfClass:[RCTKeyCommand class]]) {
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
return [self matchesInput:object.keyCommand.input
|
|
|
|
flags:object.keyCommand.modifierFlags];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)matchesInput:(NSString *)input flags:(UIKeyModifierFlags)flags
|
|
|
|
{
|
|
|
|
return [_keyCommand.input isEqual:input] && _keyCommand.modifierFlags == flags;
|
|
|
|
}
|
|
|
|
|
2015-06-22 08:08:35 -07:00
|
|
|
- (NSString *)description
|
|
|
|
{
|
|
|
|
return [NSString stringWithFormat:@"<%@:%p input=\"%@\" flags=%zd hasBlock=%@>",
|
|
|
|
[self class], self, _keyCommand.input, _keyCommand.modifierFlags,
|
|
|
|
_block ? @"YES" : @"NO"];
|
|
|
|
}
|
|
|
|
|
2015-06-19 08:08:14 -07:00
|
|
|
@end
|
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
@interface RCTKeyCommands ()
|
|
|
|
|
2015-06-19 08:08:14 -07:00
|
|
|
@property (nonatomic, strong) NSMutableSet *commands;
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
@end
|
|
|
|
|
2015-06-22 08:08:35 -07:00
|
|
|
@implementation UIResponder (RCTKeyCommands)
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
- (NSArray *)RCT_keyCommands
|
|
|
|
{
|
2015-06-19 08:08:14 -07:00
|
|
|
NSSet *commands = [RCTKeyCommands sharedInstance].commands;
|
2015-06-22 08:08:35 -07:00
|
|
|
return [[commands valueForKeyPath:@"keyCommand"] allObjects];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)RCT_handleKeyCommand:(UIKeyCommand *)key
|
|
|
|
{
|
|
|
|
// NOTE: throttle the key handler because on iOS 9 the handleKeyCommand:
|
|
|
|
// method gets called repeatedly if the command key is held down.
|
|
|
|
|
|
|
|
static NSTimeInterval lastCommand = 0;
|
|
|
|
if (RCTIsIOS8OrEarlier() || CACurrentMediaTime() - lastCommand > 0.5) {
|
|
|
|
for (RCTKeyCommand *command in [RCTKeyCommands sharedInstance].commands) {
|
|
|
|
if ([command.keyCommand.input isEqualToString:key.input] &&
|
|
|
|
command.keyCommand.modifierFlags == key.modifierFlags) {
|
|
|
|
if (command.block) {
|
|
|
|
command.block(key);
|
|
|
|
lastCommand = CACurrentMediaTime();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-02-19 20:10:52 -08:00
|
|
|
}
|
|
|
|
|
2015-06-22 08:08:35 -07:00
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation UIApplication (RCTKeyCommands)
|
|
|
|
|
|
|
|
// Required for iOS 8.x
|
2015-02-19 20:10:52 -08:00
|
|
|
- (BOOL)RCT_sendAction:(SEL)action to:(id)target from:(id)sender forEvent:(UIEvent *)event
|
|
|
|
{
|
|
|
|
if (action == @selector(RCT_handleKeyCommand:)) {
|
2015-06-22 08:08:35 -07:00
|
|
|
[self RCT_handleKeyCommand:sender];
|
|
|
|
return YES;
|
2015-02-19 20:10:52 -08:00
|
|
|
}
|
|
|
|
return [self RCT_sendAction:action to:target from:sender forEvent:event];
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation RCTKeyCommands
|
|
|
|
|
|
|
|
+ (void)initialize
|
2015-03-23 13:28:42 -07:00
|
|
|
{
|
2015-06-22 08:08:35 -07:00
|
|
|
if (RCTIsIOS8OrEarlier()) {
|
|
|
|
|
|
|
|
//swizzle UIApplication
|
|
|
|
RCTSwapInstanceMethods([UIApplication class],
|
|
|
|
@selector(keyCommands),
|
|
|
|
@selector(RCT_keyCommands));
|
|
|
|
|
|
|
|
RCTSwapInstanceMethods([UIApplication class],
|
|
|
|
@selector(sendAction:to:from:forEvent:),
|
|
|
|
@selector(RCT_sendAction:to:from:forEvent:));
|
|
|
|
} else {
|
|
|
|
|
|
|
|
//swizzle UIResponder
|
|
|
|
RCTSwapInstanceMethods([UIResponder class],
|
|
|
|
@selector(keyCommands),
|
|
|
|
@selector(RCT_keyCommands));
|
|
|
|
}
|
2015-02-19 20:10:52 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (instancetype)sharedInstance
|
|
|
|
{
|
|
|
|
static RCTKeyCommands *sharedInstance;
|
|
|
|
static dispatch_once_t onceToken;
|
|
|
|
dispatch_once(&onceToken, ^{
|
2015-08-17 07:35:34 -07:00
|
|
|
sharedInstance = [self new];
|
2015-02-19 20:10:52 -08:00
|
|
|
});
|
2015-03-23 13:28:42 -07:00
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
return sharedInstance;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (instancetype)init
|
|
|
|
{
|
|
|
|
if ((self = [super init])) {
|
2015-08-17 07:35:34 -07:00
|
|
|
_commands = [NSMutableSet new];
|
2015-02-19 20:10:52 -08:00
|
|
|
}
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)registerKeyCommandWithInput:(NSString *)input
|
|
|
|
modifierFlags:(UIKeyModifierFlags)flags
|
|
|
|
action:(void (^)(UIKeyCommand *))block
|
|
|
|
{
|
|
|
|
RCTAssertMainThread();
|
2015-03-23 13:28:42 -07:00
|
|
|
|
2015-06-22 08:08:35 -07:00
|
|
|
if (input.length && flags && RCTIsIOS8OrEarlier()) {
|
2015-05-01 06:21:03 -07:00
|
|
|
|
|
|
|
// Workaround around the first cmd not working: http://openradar.appspot.com/19613391
|
|
|
|
// You can register just the cmd key and do nothing. This ensures that
|
2015-06-22 08:08:35 -07:00
|
|
|
// command-key modified commands will work first time. Fixed in iOS 9.
|
2015-05-01 06:21:03 -07:00
|
|
|
|
|
|
|
[self registerKeyCommandWithInput:@""
|
|
|
|
modifierFlags:flags
|
|
|
|
action:nil];
|
|
|
|
}
|
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
UIKeyCommand *command = [UIKeyCommand keyCommandWithInput:input
|
|
|
|
modifierFlags:flags
|
|
|
|
action:@selector(RCT_handleKeyCommand:)];
|
2015-04-11 15:08:00 -07:00
|
|
|
|
2015-06-22 08:08:35 -07:00
|
|
|
RCTKeyCommand *keyCommand = [[RCTKeyCommand alloc] initWithKeyCommand:command block:block];
|
|
|
|
[_commands removeObject:keyCommand];
|
|
|
|
[_commands addObject:keyCommand];
|
2015-02-19 20:10:52 -08:00
|
|
|
}
|
|
|
|
|
2015-04-11 15:08:00 -07:00
|
|
|
- (void)unregisterKeyCommandWithInput:(NSString *)input
|
|
|
|
modifierFlags:(UIKeyModifierFlags)flags
|
2015-02-19 20:10:52 -08:00
|
|
|
{
|
|
|
|
RCTAssertMainThread();
|
2015-03-23 13:28:42 -07:00
|
|
|
|
2015-06-19 08:08:14 -07:00
|
|
|
for (RCTKeyCommand *command in _commands.allObjects) {
|
|
|
|
if ([command matchesInput:input flags:flags]) {
|
|
|
|
[_commands removeObject:command];
|
2015-02-19 20:10:52 -08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isKeyCommandRegisteredForInput:(NSString *)input
|
|
|
|
modifierFlags:(UIKeyModifierFlags)flags
|
|
|
|
{
|
|
|
|
RCTAssertMainThread();
|
2015-03-23 13:28:42 -07:00
|
|
|
|
2015-06-19 08:08:14 -07:00
|
|
|
for (RCTKeyCommand *command in _commands) {
|
|
|
|
if ([command matchesInput:input flags:flags]) {
|
2015-02-19 20:10:52 -08:00
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
2015-06-22 08:08:35 -07:00
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
@implementation RCTKeyCommands
|
|
|
|
|
|
|
|
+ (instancetype)sharedInstance
|
|
|
|
{
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)registerKeyCommandWithInput:(NSString *)input
|
|
|
|
modifierFlags:(UIKeyModifierFlags)flags
|
|
|
|
action:(void (^)(UIKeyCommand *))block {}
|
|
|
|
|
|
|
|
- (void)unregisterKeyCommandWithInput:(NSString *)input
|
|
|
|
modifierFlags:(UIKeyModifierFlags)flags {}
|
|
|
|
|
|
|
|
- (BOOL)isKeyCommandRegisteredForInput:(NSString *)input
|
|
|
|
modifierFlags:(UIKeyModifierFlags)flags
|
|
|
|
{
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
#endif
|