Недавно мой заказчик заказал мне написать плугин (bundle) для Apple Mail и Safari. Данный плугин должен был отслеживать прибытие новой почты в Apple Mail, открытие письма, получение события о том, что письмо успешно отрисовалось в форме и снятие snapshot (картинки) полного контента письма.

Я не буду описывать разработку самого плугина в этой статье, опишу это в другой. Вопрос написания плугина довольно сложен. Данная область является недокументированной со стороны Apple!!! Приходилось учиться на примерах :)

Итак, для отображения письма Apple Mail, как и Safari для контента страницы, использует многоуровневый компонент WebView. Задача, сделать изображение ПОЛНОГО контента письма. Я испробовал много методов, но все они приводили к тому, что картинка снималась только с видимой части контрола + скролбары. Это не входило в мои планы. Ссылки команды Apple и многих девелоперов не давали решения. Долгие поиски (две недели!!!) были вознаграждены. Ошибкой оказался выбор дочернего контрола для WebView. Итак, мое решение:

Итак, первое что мы делаем, подписываемся на нотификацию, которая говорит о том, что рендеринг страницы прошел успешно и полностью (это же касается и Safari)

[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(webViewLoadFinished:)
name:WebViewProgressFinishedNotification
object:nil];

WebViewProgressFinishedNotification – это глобальная константа фреймворка, содержащая имя нотификации о том, что страница загружена.

Далее, пишем обработчик данной нотификации:

- (void)webViewLoadFinished:(NSNotification *)notification
{
// получаем ссылку на WebView. Она приходит в нотификации
WebView * webView = (WebView *)[notification object];
// а вот самое главное. картинку нужно снимать не с контрола, а с главного фрейма контрола, именно в нем происходит рендеринг страницы в полном маштабе.
WebFrame * frame = [webView mainFrame];
// получаем ссылку на контрол, содержащий этот фрейм.
WebFrameView * view = [frame frameView];
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
// подготавливаем репозитарий для хранения картинки, он будет получать картинку путем кеширования содержимого как изображение.
NSBitmapImageRep *imageRep = [[view documentView]
bitmapImageRepForCachingDisplayInRect:[[view documentView] frame]];
// собственно кеширование, теперь картинка в хранилище, в виде bitmap.
[[view documentView] cacheDisplayInRect:[[view documentView] frame] toBitmapImageRep:imageRep];
// перебрасываем нашу картинку в экземпляр класса NSImage
NSImage *image = [[NSImage alloc] initWithSize:NSMakeSize(1280, 1024)];
[image addRepresentation:imageRep];
// преобразуем картинку в набор байтов, используя JPEG компрессию, типов компрессий есть много, но я выбрал именно эту.
NSData * imageData = [image TIFFRepresentationUsingCompression:NSTIFFCompressionJPEG factor:0.0];
[pool release];
}

Вот и все, теперь у нас есть картинка JPEG формата полного контента страницы и мы можем делать с ней что хотим, например сохранить на диск:

NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDesktopDirectory, NSUserDomainMask, YES);
NSString *downloadDirectory = [paths objectAtIndex:0];
srand([[NSDate date] timeIntervalSince1970]);
long ext = rand();
NSString * path = [NSString stringWithFormat:@"%@/images/%@%i%@", downloadDirectory, @"web", ext, @".jpg"];
[imageData writeToFile:path atomically:YES];