8

我正在构建一个包含其他几个子视图(、、...)的自定义NSTextField视图WebView。当其中一个子视图是第一响应者时,我想让我的自定义视图绘制不同的边框,并作为可以使用菜单项和键盘快捷键操作的单个项目。它看起来像这样:

+-------------+
| NSTextField |
+-------------+
| WebView     |
+-------------+

到目前为止,我已经成功地进行了子类化,并且其他人可以在何时和被调用NSTextField时通知委托。但是这种方法不起作用,因为它本身包含许多子视图——我不能将它们全部子类化!- (BOOL)becomeFirstResponder- (BOOL)resignFirstResponderWebView

有没有更好的方法来检测子视图何时更改其第一响应者状态?还是创建自定义视图的更好方法?

4

3 回答 3

3

另一种方法是覆盖该-makeFirstResponder:方法NSWindow以发送通知。

- (BOOL)makeFirstResponder:(NSResponder *)responder {
  id previous = self.firstResponder ?: [NSNull null];
  id next = responder ?: [NSNull null];

  NSDictionary *userInfo = @{
    BrFirstResponderPreviousKey: previous,
    BrFirstResponderNextKey: next,
  };

  [[NSNotificationCenter defaultCenter] postNotificationName:BrFirstResponderWillChangeNotification object:self userInfo:userInfo];

  return [super makeFirstResponder:responder];
}

然后,您可以在自定义视图或视图控制器中收听通知,并检查上一个或下一个响应者是否是子视图,并根据需要使用-isDescendantOf:和设置needsDisplay

虽然这不是一个理想的解决方案,因为自定义视图不再是独立的。它现在有效,但希望能分享更好的方法。

于 2013-08-08T11:04:56.863 回答
1

我在 iOS / 中遇到了这个问题UIWebView,它没有makeFirstResponderUIWindow,webViewDidEndEditing或中实现shouldBeginEditingInDOMRange。然而,通过使用 Swizzling,我能够创建一个帮助类别,允许检索当前的第一响应者,并在每次第一响应者更改时发布通知。真的令人沮丧,这一切应该是公共 API,但不是,因为 swizzle 通常不是第一个 goto,但这已经足够好了。

首先,设置您的类别标题:

@interface UIResponder (Swizzle)
+ (UIResponder *)currentFirstResponder;
- (BOOL)customBecomeFirstResponder;
@end

然后分类实现

@implementation UIResponder (Swizzle)
// It's insanity that there is no better way to get a notification when the first responder changes, but there it is.
static UIResponder *sCurrentFirstResponder;
+ (UIResponder *)currentFirstResponder {
    return sCurrentFirstResponder;
}

- (BOOL)customBecomeFirstResponder {
    NSMutableDictionary *userInfo = [NSMutableDictionary dictionaryWithCapacity:2];
    if(sCurrentFirstResponder) {
        [userInfo setObject:sCurrentFirstResponder forKey:NSKeyValueChangeOldKey];
    }
    sCurrentFirstResponder = self;
    if(sCurrentFirstResponder) {
        [userInfo setObject:sCurrentFirstResponder forKey:NSKeyValueChangeNewKey];
    }
    [[NSNotificationCenter defaultCenter] postNotificationName:kFirstResponderDidChangeNotification
                                                        object:nil
                                                      userInfo:userInfo];
    return [self customBecomeFirstResponder];
}
@end

最后,使用JR Swizzle之类的助手,交换类。

#import "JRSwizzle.h"

- (void)applicationLoaded {  
    if(![UIResponder jr_swizzleMethod:@selector(becomeFirstResponder) withMethod:@selector(customBecomeFirstResponder) error:&error]) { 
        NSLog(@"Error swizzling - %@",error);
    }
}

以为我会分享。在 App Store 中有效,因为它不使用私有 API,虽然 Apple 警告不要混合基类,但没有禁止这样做的法令。

于 2015-02-17T19:38:30.303 回答
1

两个WebViewEditingDelegate方法都会被调用,

辞职第一响应者:

-(void)webViewDidEndEditing:(NSNotification *)notification
{

}

当成为第一响应者时:

-(BOOL)webView:(WebView *)webView shouldBeginEditingInDOMRange:(DOMRange *)range
{
    return YES;
}
于 2013-11-07T18:02:59.190 回答