在某些情况下你可能需要对想显示在屏幕上的文字做更多的控制,而不是仅仅使用NSTextField或者NSTextView,有时候你需要更高效地在屏幕上画文字,而不想使用NSStringDrawing方法。还可能你需要在CGContext上画文字,再有可能你想在一个Carbon程序中画文字。
这个时候,你就可能会考虑CoreText了,不过你还可能想兼容Mac OS X 10.4。这里又一个简单的方法可以满足你全部的需要。
原文作者:Uli Kusterer
原文地址:http://www.zathras.de/angelweb/blog-cocoa-text-system-everywhere.htm
这个办法就是使用Cocoa文字系统。Cocoa文字系统是一组类,NSTextView、NSTextField和NSStringDrawing都会用它们在屏幕上显示字符。如果你阅读过苹果的文档,可能你会被吓到,因为苹果的文档写的非常复杂,包括了非常多的类,诸如:NSLayoutManagers, NSTextStorages, NSTextContainers, NSGlyphGenerators, NSTypesetters…不过别怕,实际使用起来还是挺简单的。
苹果其实已经在Drawing Text with NSLayoutManager这篇文档中提供了一个很好的关于如何显示文字的介绍。
如果你读过上面的文档,你可能会发现,你实际上仅仅需要使用三个类就可以在屏幕上显示任何带有效果的Unicode字符串,这些字符串包含在一个由NSLayoutManager控制的方框内。你希望显示的文字和文字的属性保存在NSTextStorage中,需要显示的位置是由NSTextContainer指定的。另外,如果你创建了这些对象,你还可以将它们缓存,这样可以明显加快显示速度。
你可以像通常一样使用+alloc/-init方法建立这些对象,然后告诉layout manager保管这些文字的外观,告诉text storage保管layout manager。苹果的例子大概是这样的:
NSTextStorage *textStorage = [[NSTextStorage alloc] initWithString:@"This is the text string."];
NSLayoutManager *layoutManager = [[NSLayoutManager alloc] init];
NSTextContainer *textContainer = [[NSTextContainer alloc] init];
[layoutManager addTextContainer:textContainer];
[textContainer release];
[textStorage addLayoutManager:layoutManager];
[layoutManager release];
// Use the objects.
[textStorage release];
想在屏幕上显示这个字符串的话,你可以指定显示的位置,然后告诉layout manager使用drawGlyphsForGlyphRange在屏幕的特定位置显示。
[textContainer setContainerSize: rect.size];
NSRange glyphRange = [layoutManager glyphRangeForTextContainer: textContainer];
[layoutManager drawGlyphsForGlyphRange: glyphRange atPoint: rect.origin];
还是很简单的,对吧?有两件事情需要注意:当你改变文字容器的尺寸时,文字需要重新排列和折行,这些操作可以通过缓存来加速。另外,你可以直接调用glyphRangeForTextContainer:方法来手动控制文字重排。
如果想在文字中添加特殊风格或者图片,也很简单,因为NSTextStorage是NSMutableAttibutedString的子类,所以指定风格、字体都很容易。
比如你希望测量一段文字要占屏幕多大位置,可以先将文字容器的一个属性:宽或者高设置成一个定值,然后将另外一个属性设置成一个很大的数值,比如FLT_MAX,然后调用-glyphRangeForTextContainer:方法重排文字,然后就可以通过-usedRectForTextContainer:方法获取到这段文字实际需要占多大的屏幕空间了:
[textContainer setContainerSize: NSMakeSize([self bounds].size.width, FLT_MAX)];
(NSRange) [layoutManager glyphRangeForTextContainer: textContainer]; // Cause re-layout.
NSRect neededBox = [layoutManager usedRectForTextContainer: textContainer];
Carbon直接使用Quartz的API。另外,在OpenGL材质中、PDF中和其他特殊地方显示文字也都直接使用Quartz。那么,如何才能在Quartz中使用上面所说的方法呢?你无法获取到NSGraphicsContext,你只有CGContext,怎么办呢?其实不难,NSGraphicsContext中都带有一个CGContext,而且可以互相转换。这样,比如你获取到一个叫做inContext的CGContextRef变量,你可以用下面的方法实现文字显示:
[NSGraphicsContext saveGraphicsState];
NSGraphicsContext* context = [NSGraphicsContext graphicsContextWithGraphicsPort: inContext flipped: true];
[NSGraphicsContext setCurrentContext: context];
// Do Cocoa drawing here.
[NSGraphicsContext restoreGraphicsState];
这样做的好处是Cocoa文字系统不仅仅比ATSUI简单,而且它的设计非常类似CoreText。因此如果你希望在你的软件未来的某个版本中使用CoreText,你可以现在就用这种方式显示文字,那么将来的转换会非常方便。
4 Responses for "随处使用Cocoa文字系统"
很詳細的介紹哦! 我一直在 Windows 上編程的, 因為iPod Touch 才回歸 Apple 開始學習 iPhone 的編程, 這麼多和詳細的資料, 在國外網站都找不到.
这个东西如果在自定义报表的时候应该很有用啊:)
对于文字处理和Layout处理,jjgod是权威,可以去他的blog看看,blog.jjgod.org
另外在使用layout manager的drawGlyphsForGlyphRange:atPoint方法时,最好加入[self lockFocus]对自身进行锁定,结束以后再使用unlcokFocus,否则在某些多线程情况下,会出现问题的。
[self lockFocus];
[layoutManager drawGlyphsForGlyphRange: glyphRange atPoint: rect.origin];
[self unlockFocus];
你好,我是mac开发的初学者。
想请教一下,如何显示旋转后的字符串?
我原来的实现方法是:
(1)先旋转NSView子类的坐标系
(2)用NSString的drawAtPoint方法显示字符串
(3)将NSView子类的坐标系旋转回来
以上过程是在NSView子类的drawRect方法中做的,带来的问题是,这样的View在print时无法正常显示;
但如果把上述过程放到drawRect外面的话,字符串的描画将找不到GraphContext
发表评论