虽然很早就知道Three20这个强大的库,可是一直没有学习怎么用。个人觉得学习最好的方式之一就是记笔记,一来方便以后的温故,二来也可以让人“挑刺”达到交流的目的。最后么,希望通过blog这个平台,可以监督自己的学习进程。
20110517
学习的时候首先关注了TTCore(看到了core,所以就先从这个下手了)中对cocoa类的category。比如说NSString,就有判断空格换行,解析url请求,生成MD5等等。还有NSDate,有很多格式化时间输出字符串的方法,都很实用。这些category都很容易看懂,使用时从名字能方便的看出效果。在这些方法中,有个函数比较陌生,我把注释一并写在下面。
- ( id )performSelector:(SEL)selector withObject:( id )p1 withObject:( id )p2 withObject:( id )p3 {
NSMethodSignature * sig = [self methodSignatureForSelector:selector];
// 记录方法的参数和返回值类型,通常用于对象间传递消息;通常随后会创建NSInvocation对象,来调用方法
if (sig) {
NSInvocation * invo = [NSInvocation invocationWithMethodSignature:sig]; // 对象间信息存储和传递
[invo setTarget:self];
[invo setSelector:selector];
[invo setArgument: & p1 atIndex: 2 ]; // 注意索引号
[invo setArgument: & p2 atIndex: 3 ];
[invo setArgument: & p3 atIndex: 4 ];
[invo invoke]; // 调用函数,判断是否有返回值
if (sig.methodReturnLength) {
id anObject;
[invo getReturnValue: & anObject];
return anObject;
} else {
return nil;
}
} else {
return nil;
}
}
做ios的大多要和navigationController,tabBarController还有tableviewController打交道,那就得看看Three20是如何操作的。在此之前我们需要了解基于url的navigator。TTNavigator就是将url形式的字符串通过map隐射到响应的类,当需要某个类的时候,只需要生成负荷映射规则的url即可。
TTNavigator * navigator = [TTNavigator navigator];
navigator.window = window;
TTURLMap * map = navigator.URLMap;
[map from: @" tt://restaurant/(initWithName:) "
toViewController:[RestaurantController class ]];
要调用RestaurantController的时候,只需要生成URL,如:@"tt://restaurant/Chinese",就可以调用。
[navigator openURLAction:[TTURLAction actionWithURLPath: @" http://github.com/jverkoey " ]];
[[TTNavigator navigator] openURLAction:
[[TTURLAction actionWithURLPath: @" tt://restaurant/Chotchkie's " ] applyAnimated:YES]]
TTNavigator的另一个巨大的优点,就是持久化。举个使用tabBarController的例子,如果你关闭程序前在tab3页面上,那么你再次打开的时候还是在tab3页面上。当你使用URL的时候,TTNavigator会记住URL的调用栈,下次再使用的时候,只有最近的一个viewController会被初始化。由于所有的url都记录下来了,所以你也不需要担心,如果我需要回到上一级页面时怎么back。TTNavigator会根据URL按需实例化,这样就大大节省了内存。
接下来,我们看一下,URL是如何映射到类的。先看一个映射:
[map from: @" tt://menu/(initWithMenu:)/(page:) "
toSharedViewController:[MenuController class ]];
那么使用@"tt://menu/1/5"就相当于调用:
[[MenuController alloc] initWithMenu: 1 page: 5 ]
而320中的tableViewController是TTTableViewController,那有很多优异特性。一个可能的优点可能是省去了很多datasource和delegate的实现代码。它生成dataSource的方式很简单,比如:
self.dataSource = [TTSectionedDataSource dataSourceWithObjects: @" A " ,
[TTTableTextItem itemWithText: @" Alen " URL: @" tt://myDetail/a " ],
[TTTableTextItem itemWithText: @" Alva " URL: @" tt://myDetail/b " ],
nil];
// 还有TTListDataSource,按需使用
知道了以上这些规则,我们基本上可以写一个navigationController或者tabBarController的例子了(先拿这个练练手,主要是熟悉URL的写法规则)。
- ( void )applicationDidFinishLaunching:(UIApplication * )application
{
TTNavigator * navigator = [TTNavigator navigator];
// 选择persistence模式
navigator.persistenceMode = TTNavigatorPersistenceModeAll;
navigator.window = [[[UIWindow alloc] initWithFrame:TTScreenBounds()] autorelease];
TTURLMap * map = navigator.URLMap;
// Any URL that doesn't match will fall back on this one, and open in the web browser
[map from: @" * " toViewController:[TTWebController class ]];
// The tab bar controller is shared, meaning there will only ever be one created. Loading
// This URL will make the existing tab bar controller appear if it was not visible.
[map from: @" tt://tabBar " toSharedViewController:[MyTabController class ]];
[map from: @" tt://menu/(initWithMenu:) " toSharedViewController:[MenuController class ]];
// [map from:@"tt: // myMenu/(initWithNumber:)" toSharedViewController:[MenuController class]];
[map from: @" tt://myDetail/(loadFromNib:) " toViewController:[MyDetailViewController class ]];
if ( ! [navigator restoreViewControllers]) {
// This is the first launch, so we just start with the tab bar
[navigator openURLAction:[TTURLAction actionWithURLPath: @" tt://tabBar " ]];
}
}
最后需要注意的是,一定要将原来的
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
注释掉,因为似乎applicationDidFinishLaunching:的优先级没有它高。
更多信息请参考Three20Info。