この同じ問題の解決策を見つけようとして、これに出くわしました。オンラインで何も見つかりませんでしたが、今のところうまくいくように見えるものを思いつきました。これが私がしていることです:
NSTextView(または使用しているもの)をサブクラス化し、インスタンス変数を作成して、キーダウンイベントを一時的に保存します。。。
@interface MyTextView : NSTextView {
NSEvent* _keyDownEvent;
}
@end
次に、ビューのメソッドを次のように定義します(自動参照カウントを使用している場合は、保持/解放のジャンクを削除します)。
@implementation MyTextView
- (id)initWithFrame:(NSRect)frame {
if (self = [super initWithFrame:frame]) {
_keyDownEvent = nil;
}
return self;
}
- (void)keyDown:(NSEvent*)event {
[_keyDownEvent release];
_keyDownEvent = [event retain];
[super keyDown:event];
}
- (void)doCommandBySelector:(SEL)selector {
if (_keyDownEvent && selector == @selector(noop:)) {
if ([self nextResponder]) {
[[self nextResponder] keyDown:[_keyDownEvent autorelease]];
} else {
[_keyDownEvent release];
}
_keyDownEvent = nil;
} else {
[super doCommandBySelector:selector];
}
}
- (void)dealloc {
[_keyDownEvent release];
[super dealloc];
}
@end
これが私がこれにたどり着いた方法です。キーを押さないと、ビープ音が鳴ります。そこで、NSBeep()にブレークポイントを設定し、プログラムが壊れたときに、GDBでスタックトレースを吐き出しました。
#0 0x00007fff96eb1c2d in NSBeep ()
#1 0x00007fff96e6d739 in -[NSResponder doCommandBySelector:] ()
#2 0x00007fff96e6d72b in -[NSResponder doCommandBySelector:] ()
#3 0x00007fff96fda826 in -[NSWindow doCommandBySelector:] ()
#4 0x00007fff96e6d72b in -[NSResponder doCommandBySelector:] ()
#5 0x00007fff96e6d72b in -[NSResponder doCommandBySelector:] ()
#6 0x00007fff96e6d72b in -[NSResponder doCommandBySelector:] ()
#7 0x00007fff96e6d72b in -[NSResponder doCommandBySelector:] ()
#8 0x00007fff96e6d72b in -[NSResponder doCommandBySelector:] ()
#9 0x00007fff96e6d72b in -[NSResponder doCommandBySelector:] ()
#10 0x00007fff96e6d72b in -[NSResponder doCommandBySelector:] ()
#11 0x00007fff96f486ce in -[NSTextView doCommandBySelector:] ()
#12 0x00007fff96da1c93 in -[NSKeyBindingManager(NSKeyBindingManager_MultiClients) interpretEventAsCommand:forClient:] ()
#13 0x00007fff970f5382 in -[NSTextInputContext handleEvent:] ()
#14 0x00007fff96fbfd2a in -[NSView interpretKeyEvents:] ()
#15 0x00007fff96f38a25 in -[NSTextView keyDown:] ()
#16 0x0000000100012889 in -[MyTextView keyDown:] (self=0x1004763a0, _cmd=0x7fff972b0234, event=0x100197320) at /path/MyTextView.m:24
#17 0x00007fff96a16b44 in -[NSWindow sendEvent:] ()
#18 0x00007fff969af16d in -[NSApplication sendEvent:] ()
#19 0x00007fff969451f2 in -[NSApplication run] ()
#20 0x00007fff96bc3b88 in NSApplicationMain ()
#21 0x00000001000015e2 in main (argc=3, argv=0x7fff5fbff8f0) at /path/main.m:12
何が起こっているのかというと、キーダウンイベントがテキスト入力に使用されていない場合、「noop」コマンドが応答チェーンの上位に送信されます。デフォルトでは、応答チェーンから外れるとビープ音が鳴ります。私のソリューションでは、NSTextViewサブクラスがnoopコマンドをキャッチし、代わりに元のkeyDownイベントを応答チェーンにスローします。次に、NSWindowまたは他のビューは、通常どおり未使用のkeyDownイベントを取得します。