绘制有角度/旋转的多线文本 - 核心文本+核心图形

时间:2014-01-27 20:48:15

标签: ios iphone objective-c core-graphics core-text

经过几个博客&论坛我找不到使用视图上下文中的核心文本绘制倾斜/倾斜文本的合适解决方案

所以这是怎么回事。

我有一个视图,其- (void)drawRect:(CGRect)rect被调用以在屏幕上绘制字符串(多行或单行文本)。

CODE:

- (void)drawRect:(CGRect)rect
{
    NSString *text = @"This is some text being drawn by CoreText!\nAnd some more text on another line!";

    //Core Text (Create Attributed String)

    UIColor *textColor = [UIColor blackColor];
    CGColorRef color = textColor.CGColor;

    CTFontRef font = CTFontCreateWithName((CFStringRef) @"HelveticaNeue", 20.0, NULL);

    CTTextAlignment theAlignment = kCTTextAlignmentLeft;

    CFIndex theNumberOfSettings = 1;
    CTParagraphStyleSetting theSettings[1] =
    {
        { kCTParagraphStyleSpecifierAlignment, sizeof(CTTextAlignment),
            &theAlignment }
    };

    CTParagraphStyleRef paragraphStyle = CTParagraphStyleCreate(theSettings, theNumberOfSettings);

    NSDictionary *attributesDict = [NSDictionary dictionaryWithObjectsAndKeys:
                                    CFBridgingRelease(font), (NSString *)kCTFontAttributeName,
                                    color, (NSString *)kCTForegroundColorAttributeName,
                                    paragraphStyle, (NSString *) kCTParagraphStyleAttributeName,
                                    nil];


    NSAttributedString *stringToDraw = [[NSAttributedString alloc] initWithString:text attributes:attributesDict];

    CTFramesetterRef framesetter = CTFramesetterCreateWithAttributedString((CFAttributedStringRef)stringToDraw);

    //Create Frame
    CGMutablePathRef path = CGPathCreateMutable();

    CGAffineTransform transform = CGAffineTransformMakeScale(1, -1);
    //First translate your image View according to transform
    transform = CGAffineTransformTranslate(transform, 0, - self.bounds.size.height);
    // Then whenever you want any point according to UIKit related coordinates apply this transformation on the point or rect.
    CGRect frameText = CGRectMake(60, 100, 200, 200);
    CGRect newRectForUIKit = CGRectApplyAffineTransform(frameText, transform);
    CGPathAddRect(path, NULL, newRectForUIKit);

    CTFrameRef frame = CTFramesetterCreateFrame(framesetter, CFRangeMake(0, 0), path, NULL);

    CGContextRef ctx = UIGraphicsGetCurrentContext();
    CGContextSetTextMatrix(ctx, CGAffineTransformIdentity);
    CGContextTranslateCTM(ctx, 0, ([self bounds]).size.height );
    CGContextScaleCTM(ctx, 1.0, -1.0);

    //Draw Frame

    CTFrameDraw(frame, ctx);
    //Release all retained objects
    CFRelease(path);
}

输出:

enter image description here

除了绘制文本之外,我想为整个绘制文本添加一个角度。像这样的东西(必需的输出)

In this particular image the drawn text is rotated by 90 degreesA

那么如何在核心文本中为绘制的文本添加旋转角度?

注意:1)单个上下文可以有多个绘制的文本对象及其各自的角度,如下所示

enter image description here

我希望我的问题很明确。

1 个答案:

答案 0 :(得分:9)

在将CTFrameRef绘制到上下文之前,将旋转应用于上下文。

修改: 如果您想要多个角度,则每次都需要保存/恢复graphics states。 类似的东西:

- (void)drawRect:(CGRect)rect
{
    NSString *text = @"This is some text being drawn by CoreText!\nAnd some more text on another line!";

    //Core Text (Create Attributed String)

    UIColor *textColor = [UIColor blackColor];
    CGColorRef color = textColor.CGColor;

    CTFontRef font = CTFontCreateWithName((CFStringRef) @"HelveticaNeue", 20.0, NULL);

    CTTextAlignment theAlignment = kCTTextAlignmentLeft;

    CFIndex theNumberOfSettings = 1;
    CTParagraphStyleSetting theSettings[1] =
    {
        { kCTParagraphStyleSpecifierAlignment, sizeof(CTTextAlignment),
            &theAlignment }
    };

    CTParagraphStyleRef paragraphStyle = CTParagraphStyleCreate(theSettings, theNumberOfSettings);

    NSDictionary *attributesDict = [NSDictionary dictionaryWithObjectsAndKeys:
                                    CFBridgingRelease(font), (NSString *)kCTFontAttributeName,
                                    color, (NSString *)kCTForegroundColorAttributeName,
                                    paragraphStyle, (NSString *) kCTParagraphStyleAttributeName,
                                    nil];


    NSAttributedString *stringToDraw = [[NSAttributedString alloc] initWithString:text attributes:attributesDict];

    CTFramesetterRef framesetter = CTFramesetterCreateWithAttributedString((CFAttributedStringRef)stringToDraw);

    //Create Frame
    CGMutablePathRef path = CGPathCreateMutable();
    CGRect frameText = CGRectMake(60, 100, 200, 200);
    CGPathAddRect(path, NULL, frameText);

    CTFrameRef frame = CTFramesetterCreateFrame(framesetter, CFRangeMake(0, 0), path, NULL);
    CGContextRef ctx = UIGraphicsGetCurrentContext();

    CGContextSaveGState(ctx); /* save Graphic State for context rotation */

    // transform (rotate) context
    CGAffineTransform transform = CGAffineTransformMakeTranslation(self.bounds.size.width / 2.f, self.bounds.size.height / 2.f);
    CGFloat rotation = -M_PI / 2.f;
    transform = CGAffineTransformRotate(transform,rotation);
    transform = CGAffineTransformTranslate(transform,-self.bounds.size.width / 2.f, -self.bounds.size.height / 2.f);
    CGContextConcatCTM(ctx, transform);

    CGContextSaveGState(ctx);

    CGContextSetTextMatrix(ctx, CGAffineTransformIdentity);
    CGContextTranslateCTM(ctx, 0, ([self bounds]).size.height );
    CGContextScaleCTM(ctx, 1.0, -1.0);

    //Draw Frame

    CTFrameDraw(frame, ctx);
    //Release all retained objects
    CFRelease(path);

    CGContextRestoreGState(ctx);

    CGContextRestoreGState(ctx); /* restore Graphic State for context rotation */

     CGContextSaveGState(ctx); /* save Graphic States for another drawing */

     /* lets draw another string with different angle */
     attributesDict = [NSDictionary dictionaryWithObjectsAndKeys:
                                CFBridgingRelease(CTFontCreateWithName((CFStringRef) @"HelveticaNeue", 14.0, NULL)), (NSString *)kCTFontAttributeName,
                                [UIColor yellowColor].CGColor, (NSString *)kCTForegroundColorAttributeName,
                                nil];


     stringToDraw = [[NSAttributedString alloc] initWithString:@"another piece of text to drawn on same context with no angle" attributes:attributesDict];

     framesetter = CTFramesetterCreateWithAttributedString((CFAttributedStringRef)stringToDraw);

     path = CGPathCreateMutable();
     CGPathAddRect(path, NULL, CGRectMake(60, -100, 200, 200));

     frame = CTFramesetterCreateFrame(framesetter, CFRangeMake(0, 0), path, NULL);
     CFRelease(path);
     CGContextSetTextMatrix(ctx, CGAffineTransformIdentity);
     CGContextTranslateCTM(ctx, 0, ([self bounds]).size.height );
     CGContextScaleCTM(ctx, 1.0, -1.0);

     CTFrameDraw(frame, ctx);

     /**
      * @note don't forget to restore a previously saved GState, or this will be source of problems
      */
     CGContextRestoreGState(ctx);

     CFRelease(frame);
     CFRelease(framesetter);
}

enter image description here