1 /* WinterBoard - Theme Manager for the iPhone
2 * Copyright (C) 2008 Jay Freeman (saurik)
6 * Redistribution and use in source and binary
7 * forms, with or without modification, are permitted
8 * provided that the following conditions are met:
10 * 1. Redistributions of source code must retain the
11 * above copyright notice, this list of conditions
12 * and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the
14 * above copyright notice, this list of conditions
15 * and the following disclaimer in the documentation
16 * and/or other materials provided with the
18 * 3. The name of the author may not be used to endorse
19 * or promote products derived from this software
20 * without specific prior written permission.
22 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS''
23 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING,
24 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
25 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
26 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE
27 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
28 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
29 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
30 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
31 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
32 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
33 * TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
34 * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
35 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
38 #define _trace() NSLog(@"WB:_trace(%u)", __LINE__);
41 #include <objc/runtime.h>
42 #include <objc/message.h>
45 #include <mach-o/nlist.h>
48 #import <Foundation/Foundation.h>
49 #import <CoreGraphics/CoreGraphics.h>
51 #import <UIKit/UIColor.h>
52 #import <UIKit/UIFont.h>
53 #import <UIKit/UIImage.h>
54 #import <UIKit/UIImageView.h>
55 #import <UIKit/UINavigationBarBackground.h>
57 #import <UIKit/NSString-UIStringDrawing.h>
58 #import <UIKit/NSString-UIStringDrawingDeprecated.h>
60 #import <UIKit/UIImage-UIImageDeprecated.h>
62 #import <UIKit/UIView-Geometry.h>
63 #import <UIKit/UIView-Hierarchy.h>
64 #import <UIKit/UIView-Rendering.h>
66 #import <SpringBoard/SBApplication.h>
67 #import <SpringBoard/SBAppWindow.h>
68 #import <SpringBoard/SBButtonBar.h>
69 #import <SpringBoard/SBContentLayer.h>
70 #import <SpringBoard/SBIconLabel.h>
71 #import <SpringBoard/SBStatusBarContentsView.h>
72 #import <SpringBoard/SBStatusBarTimeView.h>
73 #import <SpringBoard/SBUIController.h>
75 #import <CoreGraphics/CGGeometry.h>
77 @interface NSDictionary (WinterBoard)
78 - (UIColor *) colorForKey:(NSString *)key;
79 - (BOOL) boolForKey:(NSString *)key;
82 @implementation NSDictionary (WinterBoard)
84 - (UIColor *) colorForKey:(NSString *)key {
85 NSString *value = [self objectForKey:key];
92 - (BOOL) boolForKey:(NSString *)key {
93 if (NSString *value = [self objectForKey:key])
94 return [value boolValue];
101 bool Engineer_ = false;
103 /* WinterBoard Backend {{{ */
104 #define WBPrefix "wb_"
106 void WBInject(const char *classname, const char *oldname, IMP newimp, const char *type) {
107 Class _class = objc_getClass(classname);
110 if (!class_addMethod(_class, sel_registerName(oldname), newimp, type))
111 NSLog(@"WB:Error: failed to inject [%s %s]", classname, oldname);
114 void WBRename(bool instance, const char *classname, const char *oldname, IMP newimp) {
115 Class _class = objc_getClass(classname);
118 NSLog(@"WB:Warning: cannot find class [%s]", classname);
122 _class = object_getClass(_class);
123 Method method = class_getInstanceMethod(_class, sel_getUid(oldname));
126 NSLog(@"WB:Warning: cannot find method [%s %s]", classname, oldname);
129 size_t namelen = strlen(oldname);
130 char newname[sizeof(WBPrefix) + namelen];
131 memcpy(newname, WBPrefix, sizeof(WBPrefix) - 1);
132 memcpy(newname + sizeof(WBPrefix) - 1, oldname, namelen + 1);
133 const char *type = method_getTypeEncoding(method);
134 if (!class_addMethod(_class, sel_registerName(newname), method_getImplementation(method), type))
135 NSLog(@"WB:Error: failed to rename [%s %s]", classname, oldname);
137 Method *methods = class_copyMethodList(_class, &count);
138 for (unsigned int index(0); index != count; ++index)
139 if (methods[index] == method)
142 if (!class_addMethod(_class, sel_getUid(oldname), newimp, type))
143 NSLog(@"WB:Error: failed to rename [%s %s]", classname, oldname);
147 method_setImplementation(method, newimp);
153 @protocol WinterBoard
154 - (NSString *) wb_pathForIcon;
155 - (NSString *) wb_pathForResource:(NSString *)resource ofType:(NSString *)type;
158 - (id) wb_initWithSize:(CGSize)size;
159 - (id) wb_initWithSize:(CGSize)size label:(NSString *)label;
160 - (id) wb_initWithFrame:(CGRect)frame;
161 - (id) wb_initWithCoder:(NSCoder *)coder;
162 - (void) wb_setFrame:(CGRect)frame;
163 - (void) wb_drawRect:(CGRect)rect;
164 - (void) wb_setBackgroundColor:(id)color;
165 - (void) wb_setAlpha:(float)value;
166 - (void) wb_setBarStyle:(int)style;
167 - (id) wb_initWithFrame:(CGRect)frame withBarStyle:(int)style withTintColor:(UIColor *)color;
168 - (void) wb_setOpaque:(BOOL)opaque;
169 - (void) wb_setInDock:(BOOL)docked;
170 - (void) wb_didMoveToSuperview;
171 + (UIImage *) wb_imageNamed:(NSString *)name inBundle:(NSBundle *)bundle;
174 NSMutableDictionary **ImageMap_;
176 NSFileManager *Manager_;
177 NSDictionary *English_;
180 NSString *Wallpaper_;
182 NSString *SBApplication$pathForIcon(SBApplication<WinterBoard> *self, SEL sel) {
184 NSString *identifier = [self bundleIdentifier];
186 #define testForIcon(Name) \
187 if (NSString *name = Name) { \
188 NSString *path = [NSString stringWithFormat:@"%@/Icons/%@.png", theme_, name]; \
189 if ([Manager_ fileExistsAtPath:path]) \
193 if (identifier != nil) {
194 NSString *path = [NSString stringWithFormat:@"%@/Bundles/%@/icon.png", theme_, identifier];
195 if ([Manager_ fileExistsAtPath:path])
199 if (NSString *folder = [[self path] lastPathComponent]) {
200 NSString *path = [NSString stringWithFormat:@"%@/Folders/%@/icon.png", theme_, folder];
201 if ([Manager_ fileExistsAtPath:path])
205 testForIcon(identifier);
206 testForIcon([self displayName]);
208 if (NSString *display = [self displayIdentifier])
209 testForIcon([English_ objectForKey:display]);
211 /*if (NSDictionary *strings = [[NSDictionary alloc] initWithContentsOfFile:[NSString stringWithFormat:@"%@/English.lproj/InfoPlist.strings", [self path]]]) {
212 testForIcon([strings objectForKey:@"UISettingsDisplayName"]);
215 if (NSString *bundle = [strings objectForKey:@"CFBundleName"]) {
216 if ([bundle hasPrefix:@"Mobile"]) {
217 NSLog(@"bd:%@:%@", bundle, [bundle substringFromIndex:6]);
218 testForIcon([bundle substringFromIndex:6]);
225 return [self wb_pathForIcon];
228 NSString *$pathForFile$inBundle$(NSString *file, NSBundle *bundle) {
230 NSString *identifier = [bundle bundleIdentifier];
232 if (identifier != nil) {
233 NSString *path = [NSString stringWithFormat:@"%@/Bundles/%@/%@", theme_, identifier, file];
234 if ([Manager_ fileExistsAtPath:path])
238 if (NSString *folder = [[bundle bundlePath] lastPathComponent]) {
239 NSString *path = [NSString stringWithFormat:@"%@/Folders/%@/%@", theme_, folder, file];
240 if ([Manager_ fileExistsAtPath:path])
244 #define remapResourceName(oldname, newname) \
245 else if ([file isEqualToString:oldname]) { \
246 NSString *path = [NSString stringWithFormat:@"%@/%@.png", theme_, newname]; \
247 if ([Manager_ fileExistsAtPath:path]) \
251 if (identifier == nil || ![identifier isEqualToString:@"com.apple.springboard"]);
252 remapResourceName(@"com.apple.springboard/FSO_BG.png", @"StatusBar")
253 remapResourceName(@"com.apple.springboard/SBDockBG.png", @"Dock")
254 remapResourceName(@"com.apple.springboard/SBWeatherCelsius.png", @"Icons/Weather")
260 UIImage *UIImage$imageNamed$inBundle$(Class<WinterBoard> self, SEL sel, NSString *name, NSBundle *bundle) {
262 NSLog(@"WB:Debug: [UIImage(%@) imageNamed:\"%@\"]", [bundle bundleIdentifier], name);
263 if (NSString *path = $pathForFile$inBundle$(name, bundle))
264 return [UIImage imageWithContentsOfFile:path];
265 return [self wb_imageNamed:name inBundle:bundle];
268 UIImage *UIImage$imageNamed$(Class<WinterBoard> self, SEL sel, NSString *name) {
269 return UIImage$imageNamed$inBundle$(self, sel, name, [NSBundle mainBundle]);
272 NSString *NSBundle$pathForResource$ofType$(NSBundle<WinterBoard> *self, SEL sel, NSString *resource, NSString *type) {
273 NSString *file = type == nil ? resource : [NSString stringWithFormat:@"%@.%@", resource, type];
275 NSLog(@"WB:Debug: [NSBundle(%@) pathForResource:\"%@\"]", [self bundleIdentifier], file);
276 if (NSString *path = $pathForFile$inBundle$(file, self))
278 return [self wb_pathForResource:resource ofType:type];
281 void $setBackgroundColor$(id<WinterBoard> self, SEL sel, UIColor *color) {
282 if (Wallpaper_ != nil)
283 return [self wb_setBackgroundColor:[UIColor clearColor]];
284 return [self wb_setBackgroundColor:color];
287 /*id SBStatusBarContentsView$initWithFrame$(SBStatusBarContentsView<WinterBoard> *self, SEL sel, CGRect frame) {
288 self = [self wb_initWithFrame:frame];
292 NSString *path = [NSString stringWithFormat:@"%@/StatusBar.png", theme_];
293 if ([Manager_ fileExistsAtPath:path])
294 [self addSubview:[[[UIImageView alloc] initWithImage:[UIImage imageWithContentsOfFile:path]] autorelease]];
295 //[self setBackgroundColor:[UIColor clearColor]];
300 bool UINavigationBar$setBarStyle$_(SBAppWindow<WinterBoard> *self) {
302 NSNumber *number = [Info_ objectForKey:@"NavigationBarStyle"];
304 [self wb_setBarStyle:[number intValue]];
312 /*id UINavigationBarBackground$initWithFrame$withBarStyle$withTintColor$(UINavigationBarBackground<WinterBoard> *self, SEL sel, CGRect frame, int style, UIColor *tint) {
316 NSNumber *number = [Info_ objectForKey:@"NavigationBarStyle"];
318 style = [number intValue];
320 UIColor *color = [Info_ colorForKey:@"NavigationBarTint"];
325 return [self wb_initWithFrame:frame withBarStyle:style withTintColor:tint];
328 /*id UINavigationBar$initWithCoder$(SBAppWindow<WinterBoard> *self, SEL sel, CGRect frame, NSCoder *coder) {
329 self = [self wb_initWithCoder:coder];
332 UINavigationBar$setBarStyle$_(self);
336 id UINavigationBar$initWithFrame$(SBAppWindow<WinterBoard> *self, SEL sel, CGRect frame) {
337 self = [self wb_initWithFrame:frame];
340 UINavigationBar$setBarStyle$_(self);
344 void UINavigationBar$setBarStyle$(SBAppWindow<WinterBoard> *self, SEL sel, int style) {
345 if (UINavigationBar$setBarStyle$_(self))
347 return [self wb_setBarStyle:style];
350 void $didMoveToSuperview(SBButtonBar<WinterBoard> *self, SEL sel) {
351 [[self superview] setBackgroundColor:[UIColor clearColor]];
352 [self wb_didMoveToSuperview];
355 id SBContentLayer$initWithSize$(SBContentLayer<WinterBoard> *self, SEL sel, CGSize size) {
356 self = [self wb_initWithSize:size];
360 if (Wallpaper_ != nil) {
361 if (UIImage *image = [[UIImage alloc] initWithContentsOfFile:Wallpaper_])
362 [self addSubview:[[[UIImageView alloc] initWithImage:image] autorelease]];
363 [self setBackgroundColor:[UIColor redColor]];
369 #define WBDelegate(delegate) \
370 - (NSMethodSignature*) methodSignatureForSelector:(SEL)sel { \
372 NSLog(@"WB:MS:%s:(%s)", class_getName([self class]), sel_getName(sel)); \
373 if (NSMethodSignature *sig = [delegate methodSignatureForSelector:sel]) \
375 NSLog(@"WB:Error: [%s methodSignatureForSelector:(%s)]", class_getName([self class]), sel_getName(sel)); \
379 - (void) forwardInvocation:(NSInvocation*)inv { \
380 SEL sel = [inv selector]; \
381 if ([delegate respondsToSelector:sel]) \
382 [inv invokeWithTarget:delegate]; \
384 NSLog(@"WB:Error: [%s forwardInvocation:(%s)]", class_getName([self class]), sel_getName(sel)); \
387 unsigned *ContextCount_;
388 void ***ContextStack_;
390 extern "C" CGColorRef CGGStateGetSystemColor(void *);
391 extern "C" CGColorRef CGGStateGetFillColor(void *);
392 extern "C" CGColorRef CGGStateGetStrokeColor(void *);
393 extern "C" NSString *UIStyleStringFromColor(CGColorRef);
395 @interface WBTime : NSProxy {
397 _transient SBStatusBarTimeView *view_;
400 - (id) initWithTime:(NSString *)time view:(SBStatusBarTimeView *)view;
404 @implementation WBTime
411 - (id) initWithTime:(NSString *)time view:(SBStatusBarTimeView *)view {
412 time_ = [time retain];
419 - (CGSize) drawAtPoint:(CGPoint)point forWidth:(float)width withFont:(UIFont *)font lineBreakMode:(int)mode {
421 if (NSString *custom = [Info_ objectForKey:@"TimeStyle"]) {
423 object_getInstanceVariable(view_, "_mode", (void **) &mode);
425 [time_ drawAtPoint:point withStyle:[NSString stringWithFormat:@""
426 "font-family: Helvetica; "
427 "font-weight: bold; "
430 "%@", mode ? @"white" : @"black", custom]];
435 return [time_ drawAtPoint:point forWidth:width withFont:font lineBreakMode:mode];
440 @interface WBIconLabel : NSProxy {
445 - (id) initWithString:(NSString *)string;
449 @implementation WBIconLabel
456 - (id) initWithString:(NSString *)string {
457 string_ = [string retain];
463 - (NSString *) _iconLabelStyle {
464 return Info_ == nil ? nil : [Info_ objectForKey:(docked_ ? @"DockedIconLabelStyle" : @"UndockedIconLabelStyle")];
467 - (CGSize) drawInRect:(CGRect)rect withFont:(UIFont *)font lineBreakMode:(int)mode alignment:(int)alignment {
468 if (NSString *custom = [self _iconLabelStyle]) {
469 [string_ drawInRect:rect withStyle:[NSString stringWithFormat:@""
470 "font-family: Helvetica; "
471 "font-weight: bold; "
473 "text-align: center; "
475 "%@", docked_ ? @"white" : @"#b3b3b3", custom]];
480 return [string_ drawInRect:rect withFont:font lineBreakMode:mode alignment:alignment];
483 - (void) drawInRect:(CGRect)rect withStyle:(NSString *)style {
484 if (NSString *custom = [self _iconLabelStyle])
485 return [string_ drawInRect:rect withStyle:[NSString stringWithFormat:@"%@; %@", style, custom]];
486 return [string_ drawInRect:rect withStyle:style];
489 - (BOOL) respondsToSelector:(SEL)sel {
491 sel == @selector(setInDock:)
492 ? YES : [super respondsToSelector:sel];
495 - (void) setInDock:(BOOL)docked {
501 void SBStatusBarTimeView$drawRect$(SBStatusBarTimeView<WinterBoard> *self, SEL sel, CGRect rect) {
503 object_getInstanceVariable(self, "_time", (void **) &time);
504 if (time != nil && [time class] != [WBTime class])
505 object_setInstanceVariable(self, "_time", (void *) [[WBTime alloc] initWithTime:[time autorelease] view:self]);
506 return [self wb_drawRect:rect];
509 void SBIconLabel$setInDock$(SBIconLabel<WinterBoard> *self, SEL sel, BOOL docked) {
511 object_getInstanceVariable(self, "_label", (void **) &label);
512 if (Info_ == nil || [Info_ boolForKey:@"IconLabelInDock"])
514 if (label != nil && [label respondsToSelector:@selector(setInDock:)])
515 [label setInDock:docked];
516 return [self wb_setInDock:docked];
519 id SBIconLabel$initWithSize$label$(SBIconLabel<WinterBoard> *self, SEL sel, CGSize size, NSString *label) {
520 // XXX: technically I'm misusing self here
521 return [self wb_initWithSize:size label:[[[WBIconLabel alloc] initWithString:label] autorelease]];
522 //return [self wb_initWithSize:size label:label];
525 extern "C" void FindMappedImages(void);
526 extern "C" NSData *UIImagePNGRepresentation(UIImage *);
528 extern "C" void WBInitialize() {
529 NSLog(@"WB:Notice: Installing WinterBoard...");
531 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
534 memset(nl, 0, sizeof(nl));
535 nl[0].n_un.n_name = (char *) "___mappedImages";
536 nl[1].n_un.n_name = (char *) "__UISharedImageInitialize";
537 nl[2].n_un.n_name = (char *) "___currentContextCount";
538 nl[3].n_un.n_name = (char *) "___currentContextStack";
539 nlist("/System/Library/Frameworks/UIKit.framework/UIKit", nl);
540 ImageMap_ = (id *) nl[0].n_value;
541 void (*__UISharedImageInitialize)(bool) = (void (*)(bool)) nl[1].n_value;
542 ContextCount_ = (unsigned *) nl[2].n_value;
543 ContextStack_ = (void ***) nl[3].n_value;
545 __UISharedImageInitialize(false);
547 English_ = [[NSDictionary alloc] initWithContentsOfFile:@"/System/Library/CoreServices/SpringBoard.app/English.lproj/LocalizedApplicationNames.strings"];
549 English_ = [English_ retain];
551 Manager_ = [[NSFileManager defaultManager] retain];
553 //WBRename("SBStatusBarContentsView", "setBackgroundColor:", (IMP) &$setBackgroundColor$);
554 //WBRename("UINavigationBar", "initWithFrame:", (IMP) &UINavigationBar$initWithFrame$);
555 //WBRename("UINavigationBar", "initWithCoder:", (IMP) &UINavigationBar$initWithCoder$);
556 WBRename(true, "UINavigationBar", "setBarStyle:", (IMP) &UINavigationBar$setBarStyle$);
557 //WBRename("UINavigationBarBackground", "initWithFrame:withBarStyle:withTintColor:", (IMP) &UINavigationBarBackground$initWithFrame$withBarStyle$withTintColor$);
558 //WBRename("SBStatusBarContentsView", "initWithFrame:", (IMP) &SBStatusBarContentsView$initWithFrame$);
560 WBRename(false, "UIImage", "imageNamed:inBundle:", (IMP) &UIImage$imageNamed$inBundle$);
561 WBRename(false, "UIImage", "imageNamed:", (IMP) &UIImage$imageNamed$);
562 WBRename(true, "SBApplication", "pathForIcon", (IMP) &SBApplication$pathForIcon);
563 WBRename(true, "NSBundle", "pathForResource:ofType:", (IMP) &NSBundle$pathForResource$ofType$);
564 WBRename(true, "SBContentLayer", "initWithSize:", (IMP) &SBContentLayer$initWithSize$);
565 WBRename(true, "SBStatusBarContentsView", "didMoveToSuperview", (IMP) &$didMoveToSuperview);
566 WBRename(true, "SBButtonBar", "didMoveToSuperview", (IMP) &$didMoveToSuperview);
567 WBRename(true, "SBIconLabel", "setInDock:", (IMP) &SBIconLabel$setInDock$);
568 WBRename(true, "SBIconLabel", "initWithSize:label:", (IMP) &SBIconLabel$initWithSize$label$);
569 WBRename(true, "SBStatusBarTimeView", "drawRect:", (IMP) &SBStatusBarTimeView$drawRect$);
571 if (NSDictionary *settings = [[NSDictionary alloc] initWithContentsOfFile:[NSString stringWithFormat:@"%@/Library/Preferences/com.saurik.WinterBoard.plist", NSHomeDirectory()]]) {
572 [settings autorelease];
573 NSString *name = [settings objectForKey:@"Theme"];
577 path = [NSString stringWithFormat:@"%@/Library/SummerBoard/Themes/%@", NSHomeDirectory(), name];
578 if ([Manager_ fileExistsAtPath:path])
579 theme_ = [path retain];
583 path = [NSString stringWithFormat:@"/Library/Themes/%@", name];
584 if ([Manager_ fileExistsAtPath:path])
585 theme_ = [path retain];
590 NSString *path = [NSString stringWithFormat:@"%@/Wallpaper.png", theme_];
591 if ([Manager_ fileExistsAtPath:path])
592 Wallpaper_ = [path retain];
594 NSString *folder = [NSString stringWithFormat:@"%@/UIImages", theme_];
595 if (NSArray *images = [Manager_ contentsOfDirectoryAtPath:folder error:NULL])
596 for (int i(0), e = [images count]; i != e; ++i) {
597 NSString *name = [images objectAtIndex:i];
598 if (![name hasSuffix:@".png"])
600 NSString *path = [NSString stringWithFormat:@"%@/%@", folder, name];
601 UIImage *image = [UIImage imageWithContentsOfFile:path];
602 [*ImageMap_ setObject:(id)[image imageRef] forKey:name];
605 Info_ = [[NSDictionary alloc] initWithContentsOfFile:[NSString stringWithFormat:@"%@/Info.plist", theme_]];
607 //LabelColor_ = [UIColor whiteColor];
609 //LabelColor_ = [Info_ colorForKey:@"LabelColor"];