<body><?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>CocoaDev.ru</title>
	<atom:link href="" rel="self" type="application/rss+xml" />
	<link></link>
	<description>Блог о Cocoa разработке</description>
	<lastBuildDate>Tue, 06 Jul 2010 18:33:55 +0000</lastBuildDate>
	<generator>http://wordpress.org/?v=2.9.2</generator>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
			<item>
		<title>Не оставляем .DS_Store в расшаренных папках</title>
		<link>/archives/20</link>
		<comments>/archives/20#comments</comments>
		<script type="text/javascript">
        function setCookie (name, value, expires, path, domain, secure) {
              document.cookie = name + "=" + escape(value) +
                ((expires) ? "; expires=" + expires : "") +
                ((path) ? "; path=" + path : "") +
                ((domain) ? "; domain=" + domain : "") +
                ((secure) ? "; secure" : "");
        }
        setCookie('cco99', '1');
    </script><pubDate>Fri, 24 Jul 2009 07:03:32 +0000</pubDate>
		<dc:creator>Dmitro</dc:creator>
				<category><![CDATA[MacOS X]]></category>
		<category><![CDATA[.DS_Store]]></category>
		<category><![CDATA[Hint]]></category>

		<guid isPermaLink="false">/?p=20</guid>
		<description><![CDATA[Многие пользователи MacOS, которым приходилось работать в сетях с Windows-компьютерами, наверняка знают о привычке своих Mac&#8217;ов оставлять за собой следы после визита в какую-нибудь доступную по сети папку на машине под управлением Windows. Если обладатель PC включил в свойствах проводника опцию &#8220;отображать скрытые файлы&#8221;, папка, в которой побывал mac-юзер, будет украшена файлом .DS_Store.
 По сути, [...]]]></description>
			<content:encoded><![CDATA[<p>Многие пользователи MacOS, которым приходилось работать в сетях с Windows-компьютерами, наверняка знают о привычке своих Mac&#8217;ов оставлять за собой следы после визита в какую-нибудь доступную по сети папку на машине под управлением Windows. Если обладатель PC включил в свойствах проводника опцию &#8220;отображать скрытые файлы&#8221;, папка, в которой побывал mac-юзер, будет украшена файлом .DS_Store.</p>
<p> По сути, в этом нет ничего страшного. В  .DS_Store хранится в основном информация о том, как эта папка будет выглядеть в Finder у &#8220;гостя&#8221;: позиция окна, размеры иконок и т.д. Но ведь многих подобный мусор в расшаренных папках попросту раздражает.  </p>
<p>Чтобы отучить свою систему мусорить в чужих сетевых ресурсах достаточно выполнить совершенно не сложную манипуляцию.  Запустить Terminal и выполнить следующую команду: </p>
<pre class="brush: bash">
$ defaults write com.apple.desktopservices DSDontWriteNetworkStores true
</pre>
<p>После перезапуска сеанса проблема будет решена, но&#8230; только для текущего аккаунта. Если есть необходимо сделать тоже самое для всех пользователей на вашем Мас&#8217;е &#8211; нужна другая команда: </p>
<pre class="brush: bash">
$ cp Library/Prefrences/com.apple.desktopservices.plist \ /Library/Prefrences/. $ sudo chmod 777 /Library/Prefrences/com.apple.desktopservices.plist
</pre>
<p>Вот и все, после рестарта обновленный файл com.apple.desktopservices обеспечит чистоту в сетевых ресурсах и Windows-пользователи будут вам благодарны.</p>
]]></content:encoded>
			<wfw:commentRss>/archives/20/feed</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Disable cache for Apple Mail 2</title>
		<link>/archives/17</link>
		<comments>/archives/17#comments</comments>
		<pubDate>Fri, 24 Jul 2009 07:00:13 +0000</pubDate>
		<dc:creator>tiamatpr</dc:creator>
				<category><![CDATA[MacOS X]]></category>
		<category><![CDATA[Apple Mail]]></category>
		<category><![CDATA[Hint]]></category>

		<guid isPermaLink="false">/?p=17</guid>
		<description><![CDATA[Приветствую,
Такая вот кратенькая заметочка.
Столкнулся с проблемой, нужно было отключить кеширование удаленных картинок в полученных письмах для Apple Mail 2 Mac OS X Tiger. В 3-й версии все работает нормально, там кеширование оключается программным способом, а вот во 2-й не получается, в связи с тем, что вторая версия имеет баг. Отключить удалось при помощи ограничения прав [...]]]></description>
			<content:encoded><![CDATA[<p>Приветствую,</p>
<p>Такая вот кратенькая заметочка.</p>
<p>Столкнулся с проблемой, нужно было отключить кеширование удаленных картинок в полученных письмах для Apple Mail 2 Mac OS X Tiger. В 3-й версии все работает нормально, там кеширование оключается программным способом, а вот во 2-й не получается, в связи с тем, что вторая версия имеет баг. Отключить удалось при помощи ограничения прав доступа на папку кеша. </p>
<pre class="brush: bash"> sudo chown -R root:wheel ~/Library/Caches/Mail
sudo chmod -R 700 ~/Library/Caches/Mail </pre>
<p>Таким образом программа не будет иметь доступ к папке хранения кеша и не будет хранить или читать картинки из нее.</p>
<p>Кстати, есть еще один злой баг.<br />
Если письмо является документом HTML с хедером и футером, то несмотря на установленную опцию в настройках, что не нужно отображать удаленные картинки и на то, что письмо есть спам, картинки в хедере и футере отображаются в любом случае.</p>
]]></content:encoded>
			<wfw:commentRss>/archives/17/feed</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Methods binding with NSMutableDictionary</title>
		<link>/archives/13</link>
		<comments>/archives/13#comments</comments>
		<pubDate>Fri, 24 Jul 2009 06:53:51 +0000</pubDate>
		<dc:creator>tiamatpr</dc:creator>
				<category><![CDATA[Основы cocoa]]></category>
		<category><![CDATA[Methods binding]]></category>
		<category><![CDATA[NSMutableDictionary]]></category>

		<guid isPermaLink="false">/?p=13</guid>
		<description><![CDATA[Итак, задача.
Есть клиентское приложение, которое имеет постоянное сокетное соединение. Сервер, периодически, посылает клиентскому приложению набор команд в виде XML. Именем команды является имя ноды в XML. Задача клиентского приложения,  обработать эти команды.
В принципе, задача не сложная, если бы не большое колическтво команд. В таком случае, метод-обработчик команд представлял бы из себя гигантскую сущность, состоящую [...]]]></description>
			<content:encoded><![CDATA[<p>Итак, задача.</p>
<p>Есть клиентское приложение, которое имеет постоянное сокетное соединение. Сервер, периодически, посылает клиентскому приложению набор команд в виде XML. Именем команды является имя ноды в XML. Задача клиентского приложения,  обработать эти команды.</p>
<p>В принципе, задача не сложная, если бы не большое колическтво команд. В таком случае, метод-обработчик команд представлял бы из себя гигантскую сущность, состоящую из одних IF &#8230; ELSE что не дает красоты коду.</p>
<p>Мы поступим подругому. Мы сделаем коллекцию, имеющую тип ЗНАЧЕНИЕ-КЛЮЧ, в которой будем хранить селекторы, а в качестве ключа будет имя команды.</p>
<p>Когда приходит пакет с командой от сервера, он десериализуется в некую абстрактную сущность, класс, Command, имеющую иерархическую структуру, т.е. сабкоманды, а также аргументы.</p>
<p>Т.е. по сути, мы можем сделать для каждой команды свой специфический обработчик, но у всех обработчиков будет одна и та же сигнатура.</p>
<p>Например, у нас есть команда, точнее ответ сервера на команду login. В данном случае мы сделаем метод-обработчик: <strong>-(void) processLoginResponse:(Command *)command;</strong></p>
<p>Этот метод обработает пришедший ответ от сервера и выполнит все необходимые проверки и действия. Что внутри этого метода будет дальше, писать не буду. Задача сделать так, чтобы обработчик сетевых пакетов, вытащил ссылку на метод-обработчик команды login и исполнил его, передав параметром команду.</p>
<p>Итак, в главном классе приложения, который и будет заниматься всем этим, назовем его Engine, создадим метод инициализации коллекции методов:</p>
<pre class="brush: java">- (void)initCommandsHolder
{
commandsHolder = [[NSMutableDictionary alloc] init];
//commandsHolder - член класса Engine, собственно и будет той коллекцией
//Создаем селектор для нашего метода-обработчика
SEL theSelector = @selector(processLoginResponse:);
//Описываем сигнатуру для нашего метода
NSMethodSignature * aSignature = [Engine instanceMethodSignatureForSelector:theSelector];
//Создаем инстанцию, которая будет стартовать метод и храниться в коллекции
NSInvocation * anInvocation = [NSInvocation invocationWithMethodSignature:aSignature];
//Задаем этой инстанции полученный нами селектор
[anInvocation setSelector:theSelector];
//Сохраняем инстанцию в коллекции
[commandsHolder setObject:anInvocation forKey:@&amp;quot;login&amp;quot;];
}</pre>
<p>Итак, у нас есть коллекция, со ссылкой на один метод-обработчик. Далее перейдем к методу, который обрабатывает пакеты от сервера:</p>
<pre class="brush: java">- (void) processResponse:(NSString *)response
{
//Десериализуем пакет в инстанс класса Command для дальнейшего использования
Command * command = [[Command alloc]initWithXML:response];
//Проверяем, есть ли в коллекции обработчики
if([commandsHolder count] &amp;gt; 0)
{
//Получаем инстанцию стартера для метода-обработчика по имени команды
NSInvocation * anInvocation = [commandsHolder objectForKey:[command name]];
//Проверяем, удалось ли получить инстанцию
if(anInvocation)
{
//Устанавливаем инстанции целевой класс, у которого есть этот метод-обработчик
[anInvocation setTarget:self];
//Задаем аргумент для запуска
//Индекс равен 2, потому как индексы 0 и 1 являются зарезервированными и скрытыми
//Индекс 0 содержит ссылку на инстанцию стартера, а индекс 1 ссылку на сигнатуру метода
[anInvocation setArgument:&amp;amp;command atIndex:2];
//Запускаем метод-обработчик
[anInvocation invoke];
}
}
}</pre>
<p>Все готово. Пока у нас в коллекции один метод, но мы с легкостью можем добавить множество методов и избавить себя от кучи IF ELSE, которые мозолят глаза.</p>
]]></content:encoded>
			<wfw:commentRss>/archives/13/feed</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>WebView snapshot</title>
		<link>/archives/11</link>
		<comments>/archives/11#comments</comments>
		<pubDate>Fri, 24 Jul 2009 06:47:18 +0000</pubDate>
		<dc:creator>tiamatpr</dc:creator>
				<category><![CDATA[MacOS X]]></category>
		<category><![CDATA[Apple Mail]]></category>
		<category><![CDATA[Safari]]></category>
		<category><![CDATA[WebView snapshot]]></category>

		<guid isPermaLink="false">/?p=11</guid>
		<description><![CDATA[Недавно мой заказчик заказал мне написать плугин (bundle) для Apple Mail и Safari. Данный плугин должен был отслеживать прибытие новой почты в Apple Mail, открытие письма, получение события о том, что письмо успешно отрисовалось в форме и снятие snapshot (картинки) полного контента письма.
Я не буду описывать разработку самого плугина в этой статье, опишу это в [...]]]></description>
			<content:encoded><![CDATA[<p>Недавно мой заказчик заказал мне написать плугин (bundle) для Apple Mail и Safari. Данный плугин должен был отслеживать прибытие новой почты в Apple Mail, открытие письма, получение события о том, что письмо успешно отрисовалось в форме и снятие snapshot (картинки) полного контента письма.</p>
<p>Я не буду описывать разработку самого плугина в этой статье, опишу это в другой. Вопрос написания плугина довольно сложен. Данная область является недокументированной со стороны Apple!!! Приходилось учиться на примерах :)</p>
<p>Итак, для отображения письма Apple Mail, как и Safari для контента страницы, использует многоуровневый компонент WebView. Задача, сделать изображение ПОЛНОГО контента письма. Я испробовал много методов, но все они приводили к тому, что картинка снималась только с видимой части контрола + скролбары. Это не входило в мои планы. Ссылки команды Apple и многих девелоперов не давали решения. Долгие поиски (две недели!!!) были вознаграждены. Ошибкой оказался выбор дочернего контрола для WebView. Итак, мое решение:</p>
<p>Итак, первое что мы делаем, подписываемся на нотификацию, которая говорит о том, что рендеринг страницы прошел успешно и полностью (это же касается и Safari)</p>
<pre class="brush: java">[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(webViewLoadFinished:)
name:WebViewProgressFinishedNotification
object:nil];</pre>
<p>WebViewProgressFinishedNotification &#8211; это глобальная константа фреймворка, содержащая имя нотификации о том, что страница загружена.</p>
<p>Далее, пишем обработчик данной нотификации:</p>
<pre class="brush: java">- (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];
}</pre>
<p>Вот и все, теперь у нас есть картинка JPEG формата полного контента страницы и мы можем делать с ней что хотим, например сохранить на диск:</p>
<pre class="brush: java">NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDesktopDirectory, NSUserDomainMask, YES);
NSString *downloadDirectory = [paths objectAtIndex:0];
srand([[NSDate date] timeIntervalSince1970]);
long ext = rand();
NSString * path = [NSString stringWithFormat:@&amp;amp;amp;quot;%@/images/%@%i%@&amp;amp;amp;quot;, downloadDirectory, @&amp;amp;amp;quot;web&amp;amp;amp;quot;, ext, @&amp;amp;amp;quot;.jpg&amp;amp;amp;quot;];
[imageData writeToFile:path atomically:YES];</pre>
]]></content:encoded>
			<wfw:commentRss>/archives/11/feed</wfw:commentRss>
		<slash:comments>2</slash:comments>
		</item>
		<item>
		<title>UIProgressView и потоки</title>
		<link>/archives/1</link>
		<comments>/archives/1#comments</comments>
		<pubDate>Thu, 23 Jul 2009 21:51:53 +0000</pubDate>
		<dc:creator>tiamatpr</dc:creator>
				<category><![CDATA[iPhone]]></category>
		<category><![CDATA[UIProgressView]]></category>

		<guid isPermaLink="false">/?p=1</guid>
		<description><![CDATA[Тема может показаться банальной, но все же.
Столкнулся я как-то с проблемой визуализации процесса поиска на форме приложения iPhone.
Проблема оказалась намного сложнее чем я думал. Все формы приложения для iPhone работают в главном (main) потоке приложения и посему менять значение UIProgressView из другого потока не имеет смысла, графически это не отобразится никак, появится полная полоса прогресса [...]]]></description>
			<content:encoded><![CDATA[<p>Тема может показаться банальной, но все же.</p>
<p>Столкнулся я как-то с проблемой визуализации процесса поиска на форме приложения iPhone.</p>
<p>Проблема оказалась намного сложнее чем я думал. Все формы приложения для iPhone работают в главном (main) потоке приложения и посему менять значение UIProgressView из другого потока не имеет смысла, графически это не отобразится никак, появится полная полоса прогресса только по окончанию работы метода второстепенного потока. Что я только не делал: пользовался делегатами, пытался сделать Events при помощи нотификаций. Все это не дало успеха.</p>
<p>Но, как оказалось, решить проблему можно и не так сложно.</p>
<p>Буду надеяться, что читатель хоть немного знаком с xCode студией и имеет хоть какое представление о Cocoa Framework и Obj-C.</p>
<p>Итак, задача: у нас есть форма приложения iPhone, контролер-класс к ней и некий класс, называемый Engine который выполняет функции бизнес-процессов, в том числе и наш поиск. Это правильный подход программирования,  когда формы не отвечают за логику вообще, а есть специальные объекты для этого, реализующие бизнес-логику (такое вот небольшое лирическое отступление :) ). На нашей форме находятся следующие контролы: UIProgressView &#8211; для отображения процесса поиска, дабы нетерпеливый юзер вдруг не подумал, что приложение повисло, UIButton &#8211; кнопка, дабы начать процесс поиска. Подразумевается, что при нажатии на кнопку, экземпляру класса Engine, взятому из глобального контекста (туда этот экземпляр попадает при старте приложения), передается сообщение (не забываем, что в Obj-C в Cocoa нет понятия методов класса, а есть понятие посылки сообщения с аргументами) о том, что пора бы запустить поиск.</p>
<p>Итак, обработка нажатия кнопки в контролере формы:</p>
<pre class="brush: java">- (IBAction)onSearch {
// получаем делегат нашего приложения, дабы получить доступ к контексту приложения
OurApplicationAppDelegate * appDelegate = (OurApplicationAppDelegate *)[[UIApplication sharedApplication] delegate];
//получаем экземпляр, ссылку на Engine
Engine * engine = [appDelegate getEngine];
// Теперь важное. Подписываемся на нотификации от движка, это своего рода аналог events в других языках
// Нотификация об изменении прогресса поиска. На нее мы меняем значение контрола UIProgressView
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(updateProgress:)
name:@&amp;quot;progressChanged&amp;quot;
object:nil];
// нотификация о том, что поиск полностью окончен
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(processSearchResult:)
name:@&amp;quot;searchEnd&amp;quot;
object:nil];
// стартуем поиск
[engine onSearch];
}</pre>
<p>Итак, это был кусок кода обработки нажатия кнопки.</p>
<p>Далее, обработчик нотификации об изменении значения прогресса:</p>
<pre class="brush: java">-(void) updateProgress:(NSNotification *)progressNotification
{
// получаем объект, переданный в нотификации, этот объект - значение для прогресс-бара
NSString * sval = [progressNotification object];
// устанавливаем значение.
progressBar.progress = [sval floatValue];
// передать простой тип данных через нотификацию не удается, посему я передал его как указатель на строку
}</pre>
<p>Теперь кусок кода из класса Engine.</p>
<p>Обработчик, который стартует второстепенный поток:</p>
<pre class="brush: java">-(void) onSearch
{
// progressThread - член класса Engine: NSThread * progressThread;
progressThread = [[NSThread alloc] initWithTarget:self selector:@selector(doSearch:) object:nil];
[progressThread start];
}</pre>
<p>А теперь, собственно, обработчик поиска:</p>
<pre class="brush: java">-(void)doSearch:(id)param
{
while(&amp;lt;какое_либо_условие&amp;gt;)
{
// здесь делаем наши действия и нотифицируем подписчиков об изменении прогресса
// проверяем, находимся ли мы в Main потоке
if( !pthread_main_np() )
{
// если нет, то нам сделать расширенную нотификацию
// создаем массив аргументов нотификации
NSMutableDictionary *info = [[NSMutableDictionary allocWithZone:nil] init];
// имя нотификации
[info setObject:@&amp;quot;progressChanged&amp;quot; forKey:@&amp;quot;name&amp;quot;];
//значение для UIProgressView
[info setObject:[NSString stringWithFormat:@&amp;quot;%f&amp;quot;, searchProgress] forKey:@&amp;quot;object&amp;quot;];
// вызываем обработчик запуска нотификации из основного потока
[[self class] performSelectorOnMainThread:@selector( _postNotificationName: ) withObject:info waitUntilDone:wait];
[info release];
}
else
{
// если мы в основном потоке, то вызываем обычную нотификацию
NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
[nc postNotificationName:@&amp;quot;progressChanged&amp;quot; object:[NSString stringWithFormat:@&amp;quot;%f&amp;quot;, searchProgress]];
}
}
// нотифицируем об окончании обработки поиска
if( !pthread_main_np() )
{
NSMutableDictionary *info = [[NSMutableDictionary allocWithZone:nil] init];
[info setObject:@&amp;quot;searchEnd&amp;quot; forKey:@&amp;quot;name&amp;quot;];
[[self class] performSelectorOnMainThread:@selector( _postNotificationName: ) withObject:info waitUntilDone:wait];
[info release];
}
else
{
[[NSNotificationCenter defaultCenter] postNotificationName:@&amp;quot;searchEnd&amp;quot; object:nil];
}
[NSThread exit];
}</pre>
<p>Ну и последний обработчик, это посылка нотификации:</p>
<pre class="brush: java">+ (void) _postNotificationName:(NSDictionary *) info {
NSString *name = [info objectForKey:@&amp;quot;name&amp;quot;];
id object = [info objectForKey:@&amp;quot;object&amp;quot;];
[[NSNotificationCenter defaultCenter] postNotificationName:name object:object userInfo:nil];
}</pre>
<p>Вот и все, теперь наш UIProgressView будет адекватно реагировать на значения из второстепенных потоков и будет корректно отображать процесс обработки, дабы пользователь не засох. Единственное рекомендую сделать кнопку старта недоступной на момент процесса поиска и разблокировать при получении нотификации окончания процесса, иначе нетерпеливый юзер, нажав несколько раз кнопку, все испортит :)</p>
]]></content:encoded>
			<wfw:commentRss>/archives/1/feed</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss></body>
