11

ドキュメントによると、NSTextStorageをサブクラス化して、テキストビューで使用しようとしています。

/*
 NSTextStorage is a semi-abstract subclass of NSMutableAttributedString. It
 implements change management (beginEditing/endEditing), verification of
 attributes, delegate handling, and layout management notification. The one
 aspect it does not implement is the actual attributed string storage --- this is
 left up to the subclassers, which need to override the two
 NSMutableAttributedString primitives:

 - (void)replaceCharactersInRange:(NSRange)range withString:(NSString *)str;
 - (void)setAttributes:(NSDictionary *)attrs range:(NSRange)range;

*/

そこで、デリゲートを使用して、同じ機能で必要なすべてのイベントを処理しようとします。

@implementation MyTextStorage

- (id) init {
    self = [super init];

    if (self != nil) {
        storage = [[NSMutableAttributedString alloc] init];
    }

    return self;
}

- (void) dealloc {
    [storage release];
    [super dealloc];
}

- (NSString *) string {
    return [storage string];
}

- (void) replaceCharactersInRange:(NSRange)range withString:(NSString *)str {
    [storage replaceCharactersInRange:range withString:str];
}

- (void)setAttributes:(NSDictionary *)attrs range:(NSRange)range {
    [storage setAttributes:attrs range:range];
}


@end

デリゲートとして何を使用しているかは関係ありません:NSTextStorageまたはNSMutableAttributedString、結果は同じです:

キャッチされない例外が発生 しました*NSRunStorage、_NSBlockNumberForIndex():配列の境界を超えるインデックス(18446744073709551615)*キャッチされない例外によるアプリの終了'NSRangeException'、理由:'*** NSRunStorage、_NSBlockNumberForIndex():インデックス(18446744073709551615)を超える配列の境界(0) '

スタックトレース:

0   CoreFoundation                      0x00007fff840cd7b4 __exceptionPreprocess + 180
    1   libobjc.A.dylib                     0x00007fff885390f3 objc_exception_throw + 45
    2   CoreFoundation                      0x00007fff840cd5d7 +[NSException raise:format:arguments:] + 103
    3   CoreFoundation                      0x00007fff840cd564 +[NSException raise:format:] + 148
    4   AppKit                              0x00007fff86cc6288 _NSBlockNumberForIndex + 86
    5   AppKit                              0x00007fff86cc71d5 -[NSLayoutManager textContainerForGlyphAtIndex:effectiveRange:] + 364
    6   AppKit                              0x00007fff86d1f121 -[NSTextView(NSSharing) didChangeText] + 340
    7   AppKit                              0x00007fff86d44b68 -[NSTextView insertText:replacementRange:] + 2763
    8   CocoaCalculator                     0x0000000100002312 -[CalcTextView insertText:] + 65
    9   CocoaCalculator                     0x00000001000027ac -[CalcTextContainer initWithFrame:] + 1176
    10  AppKit                              0x00007fff86c23d44 -[NSCustomView nibInstantiate] + 646
    11  AppKit                              0x00007fff86b7be17 -[NSIBObjectData instantiateObject:] + 259
    12  AppKit                              0x00007fff86b7b202 -[NSIBObjectData nibInstantiateWithOwner:topLevelObjects:] + 336
    13  AppKit                              0x00007fff86b7988d loadNib + 226
    14  AppKit                              0x00007fff86b78d9a +[NSBundle(NSNibLoading) _loadNibFile:nameTable:withZone:ownerBundle:]

電話をかけようとすると終了が始まります

[textView insertText:@"123"];

ただし、標準のNSTextStorageバージョンでは、すべて正しく機能します。

このクラスを拡張するには何をする必要がありますか?

4

2 に答える 2

11

実装が必要なNSAttributedStringプリミティブメソッドに加えて、とedited:range:changeInLength:のオーバーライド内replaceCharactersInRange:withString:setAttributes:range:それぞれ呼び出す必要があると思います。

何かのようなもの:

- (void) replaceCharactersInRange:(NSRange)range withString:(NSString *)str {
    [storage replaceCharactersInRange:range withString:str];
    [self edited:NSTextStorageEditedCharacters range:range changeInLength:[str length] - range.length];
}

- (void)setAttributes:(NSDictionary *)attrs range:(NSRange)range {
    [storage setAttributes:attrs range:range];
    [self edited:NSTextStorageEditedAttributes range:range changeInLength:0];
}
于 2011-05-12T03:38:08.283 に答える
5

このNSTextStorage領域のドキュメントは少しあいまいです。プリミティブの2つをオーバーライドするだけでよいという質問で引用したコメントはNSMutableAttributedString、他のプリミティブをオーバーライドする必要がないことを意味しますNSMutableAttributedString

それでも、スーパークラスのプリミティブをオーバーライドする必要がありますNSAttributedString-attributesAtIndex:effectiveRange:つまり、、、を実装する必要があり-lengthます-string。の実装が含まれていることに気付いた-stringので、残りの2つはそれを実行する必要があります。

追加する場合:

- (NSUInteger)length
{
     return [storage length];
}

- (NSDictionary *)attributesAtIndex:(NSUInteger)location effectiveRange:(NSRangePointer)range
{
      return [storage attributesAtIndex:location effectiveRange:range];
}

...そしてうまくいけばそれはこの問題を修正するでしょう。

于 2011-05-11T23:37:16.350 に答える