微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

objective-c – 如何以编程方式将项目符号列表添加到NSTextView

这个问题可能听起来很奇怪,但我已经挣扎了好几天了.

我有一个NSTextView,可以显示一些带有一些格式选项的文本.其中之一是能够为选择或当前行打开/关闭子弹列表(最简单的一个).

我知道在NSTextView上有一个orderFrontListPanel:方法打开窗口,其中包含可供选择和编辑的可用列表参数(例如,当您按Menu-> Format-> List …时在TextView中).
我已经想出并实现了手动添加项目符号,NSTextView似乎几乎正确地使用它们.通过说几乎我的意思是它保留标签位置,继续“输入”列表,等等.但是有一些小故障不适合我,不同于标准实施.

我试图找到以编程方式设置列表的认方式,就像通过“列表…”菜单完成一样没有运气.

我请求帮助,每一点点的信息都会受到赞赏:).

P.S.:我已经查看了TextView源代码,发现了很多有趣但没有迹象或线索如何以编程方式启用列表.

更新

还在调查.我发现当你发送orderFrontListPanel:到你的NSTextView然后选择项目符号并按回车键时,没有特殊的消息被发送到NSTextView.这意味着可以在弹出式面板中的某处构建项目符号列表,并直接设置为TextView的文本容器…

解决方法

以编程方式将项目符号列表添加到NSTextView的两种方法

方法1:

以下链接引导我使用第一种方法,但除非您想为子弹使用一些特殊的非Unicode字形,否则它将不必要地转向:

> Display hidden characters in NSTextView
> How to draw one NSGlyph,that hasn’t unicode representation?
> appendBezierPathWithGlyph fails in [NSBezierPath currentPoint]

这需要:(1)子类化布局管理器,用子弹字形替换某些任意字符; (2)带有firstLineHeadindent的段落样式,比该缩进略大的制表位,以及用于组合两者的包裹线的headindent.

布局管理器如下所示:

#import <Foundation/Foundation.h>

@interface TickerLayoutManager : NSLayoutManager {

// Might as well let this class hold all the fonts used by the progress ticker.
// That way they're all defined in one place,the init method.
NSFont *fontnormal;
NSFont *fontIndent; // smaller,for indented lines
NSFont *fontBold;

NSGlyph glyphBullet;
CGFloat fWidthGlyPHPlusspace;

}

@property (nonatomic,retain) NSFont *fontnormal;
@property (nonatomic,retain) NSFont *fontIndent; 
@property (nonatomic,retain) NSFont *fontBold;
@property NSGlyph glyphBullet;
@property CGFloat fWidthGlyPHPlusspace;

@end

#import "TickerLayoutManager.h"

@implementation TickerLayoutManager

@synthesize fontnormal;
@synthesize fontIndent; 
@synthesize fontBold;
@synthesize glyphBullet;
@synthesize fWidthGlyPHPlusspace;

- (id)init {
    self = [super init];
    if (self) {
        self.fontnormal = [NSFont fontWithName:@"Baskerville" size:14.0f];
        self.fontIndent = [NSFont fontWithName:@"Baskerville" size:12.0f];
        self.fontBold = [NSFont fontWithName:@"Baskerville Bold" size:14.0f];
        // Get the bullet glyph.
        self.glyphBullet = [self.fontIndent glyphWithName:@"bullet"];
        // To determine its point size,put it in a Bezier path and take its bounds.
        NSBezierPath *bezierPath = [NSBezierPath bezierPath];
        [bezierPath movetoPoint:NSMakePoint(0.0f,0.0f)]; // prevents "No current point for line" exception
        [bezierPath appendBezierPathWithGlyph:self.glyphBullet inFont:self.fontIndent];
        NSRect rectGlyphOutline = [bezierPath bounds];
        // The bullet should be followed with a space,so get the combined size...
        NSSize sizeSpace = [@" " sizeWithAttributes:[NSDictionary dictionaryWithObject:self.fontIndent forKey:NSFontAttributeName]];
        self.fWidthGlyPHPlusspace = rectGlyphOutline.size.width + sizeSpace.width;
        // ...which is for some reason inexact. If this number is too low,your bulleted text will be thrown to the line below,so add some boost.
        self.fWidthGlyPHPlusspace *= 1.5; // 
    }

    return self;
}

- (void)drawGlyphsForGlyphRange:(NSRange)range 
                        atPoint:(NSPoint)origin {

    // The following prints only once,even though the textview's string is set 4 times,so this implementation is not too expensive.
    printf("\nCalling TickerLayoutManager's drawGlyphs method.");

    Nsstring *string = [[self textStorage] string];
    for (int i = range.location; i < range.length; i++) {
        // Replace all occurrences of the ">" char with the bullet glyph.
        if ([string characteratIndex:i] == '>')
            [self replaceGlyphAtIndex:i withGlyph:self.glyphBullet];
    }

    [super drawGlyphsForGlyphRange:range atPoint:origin];
}

@end

将布局管理器分配给窗口/视图控制器的awakeFromNib中的textview,如下所示:

- (void) awakeFromNib {

    // regular setup...

    // Give the ticker display NSTextView its subclassed layout manager.
    TickerLayoutManager *newLayoutMgr = [[TickerLayoutManager alloc] init];
    NSTextContainer *textContainer = [self.txvProgressticker textContainer];
    // Use "replaceLM" rather than "setLM," in order to keep shared relnshps intact. 
    [textContainer replaceLayoutManager:newLayoutMgr];
    [newLayoutMgr release];
    // (Note: It is possible that all text-displaying controls in this class’s window will share this text container,as they would a field editor (a textview),although the fact that the ticker display is itself a textview might isolate it. Apple's "Text System Overview" is not clear on this point.)

}

然后添加一个类似这样的方法

- (void) addProgresstickerLine:(Nsstring *)string 
                   inStyle:(uint8_t)uiStyle {

    // Null check.
    if (!string)
        return;

    // Prepare the font.
    // (As noted above,TickerLayoutManager holds all 3 ticker display fonts.)
    NSFont *font = nil;
    TickerLayoutManager *tickerLayoutMgr = (TickerLayoutManager *)[self.txvProgressticker layoutManager];
    switch (uiStyle) {
        case kTickerStylenormal:
            font = tickerLayoutMgr.fontnormal;
            break;
        case kTickerStyleIndent:
            font = tickerLayoutMgr.fontIndent;
            break;
        case kTickerStyleBold:
            font = tickerLayoutMgr.fontBold;
            break;
        default:
            font = tickerLayoutMgr.fontnormal;
            break;
    }


    // Prepare the paragraph style,to govern indentation.    
    // CAUTION: If you propertize it for re-use,make sure you don't mutate it once it has been assigned to an attributed string. (See warning in class ref.)
    // At the same time,add the initial line break and,if indented,the tab.
    NSMutableParagraphStyle *paragStyle = [[NSParagraphStyle defaultParagraphStyle] mutablecopy]; // ALLOC
    [paragStyle setAlignment:NSLeftTextAlignment]; // default,but just in case
    if (uiStyle == kTickerStyleIndent) {
        // (The custom layout mgr will replace ‘>’ char with a bullet,so it should be followed with an extra space.)
        string = [@"\n>\t" stringByAppendingString:string];
        // Indent the first line up to where the bullet should appear.
        [paragStyle setFirstLineHeadindent:15.0f];
        // Define a tab stop to the right of the bullet glyph.
        NSTextTab *textTabFllwgBullet = [[NSTextTab alloc] initWithType:NSLeftTabStopType location:15.0f + tickerLayoutMgr.fWidthGlyPHPlusspace];
        [paragStyle setTabStops:[NSArray arrayWithObject:textTabFllwgBullet]];  
        [textTabFllwgBullet release];
        // Set the indentation for the wrapped lines to the same place as the tab stop.
        [paragStyle setHeadindent:15.0f + tickerLayoutMgr.fWidthGlyPHPlusspace];
    }
    else {
        string = [@"\n" stringByAppendingString:string];
    }


    // PUT IT ALL TOGETHER.
    // Combine the above into a dictionary of attributes.
    NSDictionary *dict = [NSDictionary dictionaryWithObjectsAndKeys:
                            font,NSFontAttributeName,paragStyle,NSParagraphStyleAttributeName,nil];
    // Use the attributes dictionary to make an attributed string out of the plain string.
    NSAttributedString *attrs = [[NSAttributedString alloc] initWithString:string attributes:dict]; // ALLOC
    // Append the attributed string to the ticker display.
    [[self.txvProgressticker textStorage] appendAttributedString:attrs];

    // RELEASE
    [attrs release];
    [paragStyle release];

}

测试出来:

Nsstring *sTicker = NSLocalizedString(@"First normal line of ticker should wrap to left margin",@"First normal line of ticker should wrap to left margin");
[self addProgresstickerLine:sTicker inStyle:kTickerStylenormal];
sTicker = NSLocalizedString(@"Indented ticker line should have bullet point and should wrap farther to right.",@"Indented ticker line should have bullet point and should wrap farther to right.");
[self addProgresstickerLine:sTicker inStyle:kTickerStyleIndent];
sTicker = NSLocalizedString(@"Try a second indented line,to make sure both line up.",@"Try a second indented line,to make sure both line up.");
[self addProgresstickerLine:sTicker inStyle:kTickerStyleIndent];
sTicker = NSLocalizedString(@"Final bold line",@"Final bold line");
[self addProgresstickerLine:sTicker inStyle:kTickerStyleBold];

你得到这个:

方法2:

但子弹是一个常规的Unicode字符,在十六进制2022.所以你可以直接把它放在字符串中,并得到一个精确的测量,如下所示:

Nsstring *stringWithGlyph = [Nsstring stringWithUTF8String:"\u2022"];
    Nsstring *stringWithGlyPHPlusspace = [stringWithGlyph stringByAppendingString:@" "];
    NSSize sizeGlyPHPlusspace = [stringWithGlyPHPlusspace sizeWithAttributes:[NSDictionary dictionaryWithObject:self.fontIndent forKey:NSFontAttributeName]];
    self.fWidthGlyPHPlusspace = sizeGlyPHPlusspace.width;

因此不需要自定义布局管理器.只需像上面一样设置paragStyle缩进,并将文本字符串附加到一个字符串,该字符串包含行返回项目符号空格(或选项卡,在这种情况下,您仍然希望该制表符停止).

使用空格,这产生了更严格的结果:

想要使用子弹以外的角色吗?这是一个很好的Unicode图表:http://www.danshort.com/unicode/

原文地址:https://www.jb51.cc/c/112481.html

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐