CocoaDev.ru
Блог о Cocoa разработке
Блог о Cocoa разработке
Недавно мой заказчик заказал мне написать плугин (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];
17 Ноябрь 2009 - 10:27
День добрый.
Статья хорошая, но я хотел бы узнать.
Сам давно занимаюсь разработкой под айфон, но вот сейчас встала задача разработать под мак ось. Хотел узнать в какую сторону капать? Мне нужно взаимодействие через плагин между файлмекером и мейлом. Не могли бы помочь?
Спасибо
18 Ноябрь 2009 - 17:48
Привет,
Когда я писал эту статью, точнее когда столкнулся с задачей, я перерыл кучу форумов и ничего не нашел толкового. Ответ получил копая код бандла GrowlMail. Очень толково написан код и понятно.