Рисование иерархии представлений в конкретном контексте в Какао

Для части моего приложения мне необходимо создать изображение определенного представления и всех его подвидов.

Для этого я создаю контекст, который обертывает растровое изображение того же размера, что и представление, но я не уверен, как нарисовать в нем иерархию представления. Я могу нарисовать одно представление, просто задав контекст и явно вызывая drawRect, но это не касается всех подвидов.

Я не вижу в интерфейсе NSView ничего, что могло бы помочь в этом, поэтому я подозреваю, что решение может лежать на более высоком уровне.

Ответов (5)

Решение

Я обнаружил, что написать код рисования самому - лучший способ:

  • устранение потенциальных проблем с прозрачностью (некоторые другие параметры добавляют белый фон ко всему изображению)
  • производительность была намного лучше

Приведенный ниже код не идеален, потому что он не решает проблемы масштабирования при переходе от границ к кадрам, но он учитывает состояние isFlipped и очень хорошо работает для того, для чего я его использовал. Обратите внимание, что он рисует только подпредставления (и подподпредставления, ... рекурсивно), но заставить его также рисовать себя очень легко, просто добавьте [self drawRect:[self bounds]] в реализацию imageWithSubviews .

- (void)drawSubviews
{
    BOOL flipped = [self isFlipped];

    for ( NSView *subview in [self subviews] ) {

        // changes the coordinate system so that the local coordinates of the subview (bounds) become the coordinates of the superview (frame)
        // the transform assumes bounds and frame have the same size, and bounds origin is (0,0)
        // handling of 'isFlipped' also probably unreliable
        NSAffineTransform *transform = [NSAffineTransform transform];
        if ( flipped ) {
            [transform translateXBy:subview.frame.origin.x yBy:NSMaxY(subview.frame)];
            [transform scaleXBy:+1.0 yBy:-1.0];
        } else
            [transform translateXBy:subview.frame.origin.x yBy:subview.frame.origin.y];
        [transform concat];

        // recursively draw the subview and sub-subviews
        [subview drawRect:[subview bounds]];
        [subview drawSubviews];

        // reset the transform to get back a clean graphic contexts for the rest of the drawing
        [transform invert];
        [transform concat];
    }
}

- (NSImage *)imageWithSubviews
{
    NSImage *image = [[[NSImage alloc] initWithSize:[self bounds].size] autorelease];
    [image lockFocus];
    // it seems NSImage cannot use flipped coordinates the way NSView does (the method 'setFlipped:' does not seem to help)
    // Use instead an NSAffineTransform
    if ( [self isFlipped] ) {
        NSAffineTransform *transform = [NSAffineTransform transform];
        [transform translateXBy:0 yBy:NSMaxY(self.bounds)];
        [transform scaleXBy:+1.0 yBy:-1.0];
        [transform concat];
    }
    [self drawSubviews];
    [image unlockFocus];
    return image;
}

Да, для этого конкретного использования немасштабированные данные изображения - это именно то, что мне нужно.

Вы можете использовать -[NSView dataWithPDFInsideRect:]для рендеринга всей иерархии представления, в которое вы его отправляете, в PDF-файл, возвращенный как NSData объект. Затем вы можете делать с этим все, что хотите, включая рендеринг в растровое изображение.

Вы уверены, что хотите получить растровое изображение? В конце концов, этот PDF может быть (по крайней мере теоретически) независимым от разрешения.

You can use -[NSBitmapImageRep initWithFocusedViewRect:] after locking focus on a view to have the view render itself (and its subviews) into the given rectangle.

То, что вы хотите сделать, уже явно доступно. См. Раздел «API перенаправления чертежей NSView» в примечаниях к выпуску 10.4 AppKit.

Сделайте NSBitmapImageRep для кеширования и очистите его:

NSGraphicsContext *bitmapGraphicsContext = [NSGraphicsContext graphicsContextWithBitmapImageRep:cacheBitmapImageRep];
[NSGraphicsContext saveGraphicsState];
[NSGraphicsContext setCurrentContext:bitmapGraphicsContext];
[[NSColor clearColor] set];
NSRectFill(NSMakeRect(0, 0, [cacheBitmapImageRep size].width, [cacheBitmapImageRep size].height));
[NSGraphicsContext restoreGraphicsState];

Кеш к нему:

-[NSView cacheDisplayInRect:toBitmapImageRep:]

Если вы хотите в более общем плане правильно рисовать в заданном контексте, обрабатывая рекурсию и прозрачность представления,

-[NSView displayRectIgnoringOpacity:inContext:]