+// %hook -[NSBundle pathForResource:ofType:] {{{
+MSInstanceMessageHook2(NSString *, NSBundle, pathForResource,ofType, NSString *, resource, NSString *, type) {
+ NSString *file = type == nil ? resource : [NSString stringWithFormat:@"%@.%@", resource, type];
+ if (Debug_)
+ NSLog(@"WB:Debug: [NSBundle(%@) pathForResource:\"%@\"]", [self bundleIdentifier], file);
+ if (NSString *path = $pathForFile$inBundle$(file, self, false))
+ return path;
+ return MSOldCall(resource, type);
+}
+// }}}
+
+static struct WBStringDrawingState {
+ WBStringDrawingState *next_;
+ unsigned count_;
+ NSString *base_;
+ NSString *info_;
+} *stringDrawingState_;
+
+MSInstanceMessageHook4(CGSize, NSString, drawAtPoint,forWidth,withFont,lineBreakMode, CGPoint, point, float, width, UIFont *, font, int, mode) {
+ WBStringDrawingState *state(stringDrawingState_);
+ if (state == NULL)
+ return MSOldCall(point, width, font, mode);
+
+ if (--state->count_ == 0)
+ stringDrawingState_ = state->next_;
+ if (state->info_ == nil)
+ return MSOldCall(point, width, font, mode);
+
+ NSString *info([Info_ objectForKey:state->info_]);
+ if (info == nil)
+ return MSOldCall(point, width, font, mode);
+
+ NSString *base(state->base_ ?: @"");
+ [self drawAtPoint:point withStyle:[NSString stringWithFormat:@"%@;%@;%@", [font markupDescription], base, info]];
+ return CGSizeZero;
+}
+
+MSInstanceMessageHook2(CGSize, NSString, drawAtPoint,withFont, CGPoint, point, UIFont *, font) {
+ WBStringDrawingState *state(stringDrawingState_);
+ if (state == NULL)
+ return MSOldCall(point, font);
+
+ if (--state->count_ == 0)
+ stringDrawingState_ = state->next_;
+ if (state->info_ == nil)
+ return MSOldCall(point, font);
+
+ NSString *info([Info_ objectForKey:state->info_]);
+ if (info == nil)
+ return MSOldCall(point, font);
+
+ NSString *base(state->base_ ?: @"");
+ [self drawAtPoint:point withStyle:[NSString stringWithFormat:@"%@;%@;%@", [font markupDescription], base, info]];
+ return CGSizeZero;
+}
+
+MSInstanceMessageHook1(CGSize, NSString, sizeWithFont, UIFont *, font) {
+ WBStringDrawingState *state(stringDrawingState_);
+ if (state == NULL)
+ return MSOldCall(font);
+
+ if (--state->count_ == 0)
+ stringDrawingState_ = state->next_;
+ if (state->info_ == nil)
+ return MSOldCall(font);
+
+ NSString *info([Info_ objectForKey:state->info_]);
+ if (info == nil)
+ return MSOldCall(font);
+
+ NSString *base(state->base_ ?: @"");
+ return [self sizeWithStyle:[NSString stringWithFormat:@"%@;%@;%@", [font markupDescription], base, info] forWidth:65535];
+}
+
+MSInstanceMessageHook1(UIImage *, SBIconBadgeFactory, checkoutBadgeImageForText, NSString *, text) {
+ WBStringDrawingState badgeState = {NULL, 1, @""
+ "color: white;"
+ , @"BadgeStyle"};
+
+ stringDrawingState_ = &badgeState;
+
+ UIImage *image(MSOldCall(text));
+
+ stringDrawingState_ = NULL;
+ return image;
+}
+
+MSInstanceMessageHook1(UIImage *, SBCalendarApplicationIcon, generateIconImage, int, type) {
+ WBStringDrawingState dayState = {NULL, 2, @""
+ "color: white;"
+ // XXX: this is only correct on an iPod dock
+ "text-shadow: rgba(0, 0, 0, 0.2) -1px -1px 2px;"
+ , @"CalendarIconDayStyle"};
+
+ WBStringDrawingState sizeState = {&dayState, 7, nil, nil};
+
+ WBStringDrawingState dateState = {&sizeState, 2, @""
+ "color: #333333;"
+ , @"CalendarIconDateStyle"};
+
+ stringDrawingState_ = &dateState;
+
+ UIImage *image(MSOldCall(type));
+
+ stringDrawingState_ = NULL;
+ return image;
+}
+
+MSHook(void, SBCalendarIconContentsView$drawRect$, SBCalendarIconContentsView *self, SEL sel, CGRect rect) {
+ NSBundle *bundle([NSBundle mainBundle]);
+
+ CFLocaleRef locale(CFLocaleCopyCurrent());
+ CFDateFormatterRef formatter(CFDateFormatterCreate(NULL, locale, kCFDateFormatterNoStyle, kCFDateFormatterNoStyle));
+ CFRelease(locale);
+
+ CFDateRef now(CFDateCreate(NULL, CFAbsoluteTimeGetCurrent()));
+
+ CFDateFormatterSetFormat(formatter, (CFStringRef) [bundle localizedStringForKey:@"CALENDAR_ICON_DAY_NUMBER_FORMAT" value:@"d" table:@"SpringBoard"]);
+ CFStringRef date(CFDateFormatterCreateStringWithDate(NULL, formatter, now));
+ CFDateFormatterSetFormat(formatter, (CFStringRef) [bundle localizedStringForKey:@"CALENDAR_ICON_DAY_NAME_FORMAT" value:@"cccc" table:@"SpringBoard"]);
+ CFStringRef day(CFDateFormatterCreateStringWithDate(NULL, formatter, now));
+
+ CFRelease(now);
+
+ CFRelease(formatter);
+
+ NSString *datestyle([@""
+ "font-family: Helvetica; "
+ "font-weight: bold; "
+ "color: #333333; "
+ "alpha: 1.0; "
+ "" stringByAppendingString:(IsWild_
+ ? @"font-size: 54px; "
+ : @"font-size: 39px; "
+ )]);
+
+ NSString *daystyle([@""
+ "font-family: Helvetica; "
+ "font-weight: bold; "
+ "color: white; "
+ "text-shadow: rgba(0, 0, 0, 0.2) -1px -1px 2px; "
+ "" stringByAppendingString:(IsWild_
+ ? @"font-size: 11px; "
+ : @"font-size: 9px; "
+ )]);
+
+ if (NSString *style = [Info_ objectForKey:@"CalendarIconDateStyle"])
+ datestyle = [datestyle stringByAppendingString:style];
+ if (NSString *style = [Info_ objectForKey:@"CalendarIconDayStyle"])
+ daystyle = [daystyle stringByAppendingString:style];
+
+ float width([self bounds].size.width);
+ float leeway(10);
+ CGSize datesize = [(NSString *)date sizeWithStyle:datestyle forWidth:(width + leeway)];
+ CGSize daysize = [(NSString *)day sizeWithStyle:daystyle forWidth:(width + leeway)];
+
+ unsigned base0(IsWild_ ? 89 : 70);
+ if ($GSFontGetUseLegacyFontMetrics())
+ base0 = base0 + 1;
+ unsigned base1(IsWild_ ? 18 : 16);
+
+ if (Four_) {
+ ++base0;
+ ++base1;
+ }
+
+ [(NSString *)date drawAtPoint:CGPointMake(
+ (width + 1 - datesize.width) / 2, (base0 - datesize.height) / 2
+ ) withStyle:datestyle];
+
+ [(NSString *)day drawAtPoint:CGPointMake(
+ (width + 1 - daysize.width) / 2, (base1 - daysize.height) / 2
+ ) withStyle:daystyle];
+
+ CFRelease(date);
+ CFRelease(day);
+}
+
+// %hook -[{NavigationBar,Toolbar} setBarStyle:] {{{
+void $setBarStyle$_(NSString *name, int &style) {
+ if (Debug_)
+ NSLog(@"WB:Debug:%@Style:%d", name, style);
+ NSNumber *number = nil;
+ if (number == nil)
+ number = [Info_ objectForKey:[NSString stringWithFormat:@"%@Style-%d", name, style]];
+ if (number == nil)
+ number = [Info_ objectForKey:[NSString stringWithFormat:@"%@Style", name]];
+ if (number != nil) {
+ style = [number intValue];
+ if (Debug_)
+ NSLog(@"WB:Debug:%@Style=%d", name, style);
+ }
+}
+
+MSInstanceMessageHook1(void, UIToolbar, setBarStyle, int, style) {
+ $setBarStyle$_(@"Toolbar", style);
+ return MSOldCall(style);
+}
+
+MSInstanceMessageHook1(void, UINavigationBar, setBarStyle, int, style) {
+ $setBarStyle$_(@"NavigationBar", style);
+ return MSOldCall(style);
+}
+// }}}
+
+MSHook(void, SBButtonBar$didMoveToSuperview, UIView *self, SEL sel) {
+ [[self superview] setBackgroundColor:[UIColor clearColor]];
+ _SBButtonBar$didMoveToSuperview(self, sel);
+}
+
+MSHook(void, SBStatusBarContentsView$didMoveToSuperview, UIView *self, SEL sel) {
+ [[self superview] setBackgroundColor:[UIColor clearColor]];
+ _SBStatusBarContentsView$didMoveToSuperview(self, sel);
+}
+
+static NSArray *Wallpapers_;
+static bool Papered_;
+static bool Docked_;
+static NSString *WallpaperFile_;
+static UIImageView *WallpaperImage_;
+static UIWebDocumentView *WallpaperPage_;
+static NSURL *WallpaperURL_;
+
+#define _release(object) \
+ do if (object != nil) { \
+ [object release]; \
+ object = nil; \
+ } while (false)
+
+static UIImage *$getImage$(NSString *path) {
+ UIImage *image([UIImage imageWithContentsOfFile:path]);
+
+ unsigned scale($getScale$(path));
+ if (scale != 1 && [image respondsToSelector:@selector(setScale)])
+ [image setScale:scale];
+
+ return image;
+}
+
+static UIImage *$getDefaultDesktopImage$() {
+ if (NSString *path = $getTheme$($useScale$([NSArray arrayWithObjects:@"LockBackground.png", @"LockBackground.jpg", nil])))
+ return $getImage$(path);
+ return nil;
+}
+
+MSClassMessageHook0(UIImage *, UIImage, defaultDesktopImage) {
+ return $getDefaultDesktopImage$() ?: MSOldCall();
+}
+
+MSInstanceMessageHook0(UIImage *, SBSlidingAlertDisplay, _defaultDesktopImage) {
+ return $getDefaultDesktopImage$() ?: MSOldCall();
+}
+
+MSInstanceMessageHook0(void, SBWallpaperView, resetCurrentImageToWallpaper) {
+ for (UIView *parent([self superview]); parent != nil; parent = [parent superview])
+ if ([parent isKindOfClass:$SBSlidingAlertDisplay]) {
+ if (UIImage *image = $getDefaultDesktopImage$()) {
+ [self setImage:image];
+ return;
+ }
+
+ break;
+ }
+
+ MSOldCall();
+}
+
+// %hook -[SBUIController init] {{{
+MSInstanceMessageHook0(id, SBUIController, init) {
+ self = MSOldCall();
+ if (self == nil)
+ return nil;
+
+ NSString *paper($getTheme$(Wallpapers_));
+ if (paper != nil)
+ paper = [paper stringByDeletingLastPathComponent];
+
+ {
+ size_t size;
+ sysctlbyname("hw.machine", NULL, &size, NULL, 0);
+ char *machine = new char[size];
+
+ if (sysctlbyname("hw.machine", machine, &size, NULL, 0) == -1) {
+ perror("sysctlbyname(\"hw.machine\", ?)");
+ delete [] machine;
+ machine = NULL;
+ }
+
+ IsWild_ = machine != NULL && strncmp(machine, "iPad", 4) == 0;
+ }
+
+ if (Debug_)
+ NSLog(@"WB:Debug:Info = %@", [Info_ description]);
+
+ if (paper != nil) {
+ UIImageView *&_wallpaperView(MSHookIvar<UIImageView *>(self, "_wallpaperView"));
+ if (&_wallpaperView != NULL) {
+ [_wallpaperView removeFromSuperview];
+ [_wallpaperView release];
+ _wallpaperView = nil;
+ }
+ }
+
+ UIView *&_contentLayer(MSHookIvar<UIView *>(self, "_contentLayer"));
+ UIView *&_contentView(MSHookIvar<UIView *>(self, "_contentView"));
+
+ UIView **player;
+ if (&_contentLayer != NULL)
+ player = &_contentLayer;
+ else if (&_contentView != NULL)
+ player = &_contentView;
+ else
+ player = NULL;
+ UIView *layer(player == NULL ? nil : *player);
+
+ UIWindow *window([[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]]);
+ UIView *content([[[UIView alloc] initWithFrame:[window frame]] autorelease]);
+ [window setContentView:content];
+
+ UIWindow *&_window(MSHookIvar<UIWindow *>(self, "_window"));
+ [window setBackgroundColor:[_window backgroundColor]];
+ [_window setBackgroundColor:[UIColor clearColor]];
+
+ [window setLevel:-1000];
+ [window setHidden:NO];
+
+ /*if (player != NULL)
+ *player = content;*/
+
+ [content setBackgroundColor:[layer backgroundColor]];
+ [layer setBackgroundColor:[UIColor clearColor]];
+
+ UIView *indirect;
+ if (!SummerBoard_ || !IsWild_)
+ indirect = content;
+ else {
+ CGRect bounds([content bounds]);
+ bounds.origin.y = -30;
+ indirect = [[[UIView alloc] initWithFrame:bounds] autorelease];
+ [content addSubview:indirect];
+ [indirect zoomToScale:2.4];
+ }
+
+ _release(WallpaperFile_);
+ _release(WallpaperImage_);
+ _release(WallpaperPage_);
+ _release(WallpaperURL_);
+
+ if (paper != nil) {
+ NSArray *themes([NSArray arrayWithObject:paper]);
+
+ if (NSString *path = $getTheme$([NSArray arrayWithObject:@"Wallpaper.mp4"], themes)) {
+#if UseAVController
+ NSError *error;
+
+ static AVController *controller_(nil);
+ if (controller_ == nil) {
+ AVQueue *queue([AVQueue avQueue]);
+ controller_ = [[AVController avControllerWithQueue:queue error:&error] retain];
+ }
+
+ AVQueue *queue([controller_ queue]);
+
+ UIView *video([[[UIView alloc] initWithFrame:[indirect bounds]] autorelease]);
+ [controller_ setLayer:[video _layer]];
+
+ AVItem *item([[[AVItem alloc] initWithPath:path error:&error] autorelease]);
+ [queue appendItem:item error:&error];
+
+ [controller_ play:&error];
+#elif UseMPMoviePlayerController
+ NSURL *url([NSURL fileURLWithPath:path]);
+ MPMoviePlayerController *controller = [[$MPMoviePlayerController alloc] initWithContentURL:url];
+ controller.movieControlMode = MPMovieControlModeHidden;
+ [controller play];
+#else
+ MPVideoView *video = [[[$MPVideoView alloc] initWithFrame:[indirect bounds]] autorelease];
+ [video setMovieWithPath:path];
+ [video setRepeatMode:1];
+ [video setRepeatGap:-1];
+ [video playFromBeginning];;
+#endif
+
+ [indirect addSubview:video];
+ }
+
+ if (NSString *path = $getTheme$($useScale$([NSArray arrayWithObjects:@"Wallpaper.png", @"Wallpaper.jpg", nil]), themes)) {
+ if (UIImage *image = $getImage$(path)) {
+ WallpaperFile_ = [path retain];
+ WallpaperImage_ = [[UIImageView alloc] initWithImage:image];
+ if (NSNumber *number = [Info_ objectForKey:@"WallpaperAlpha"])
+ [WallpaperImage_ setAlpha:[number floatValue]];
+ [indirect addSubview:WallpaperImage_];
+ }
+ }
+
+ if (NSString *path = $getTheme$([NSArray arrayWithObject:@"Wallpaper.html"], themes)) {
+ CGRect bounds = [indirect bounds];
+
+ UIWebDocumentView *view([[[UIWebDocumentView alloc] initWithFrame:bounds] autorelease]);
+ [view setAutoresizes:true];
+
+ WallpaperPage_ = [view retain];
+ WallpaperURL_ = [[NSURL fileURLWithPath:path] retain];
+
+ [WallpaperPage_ loadRequest:[NSURLRequest requestWithURL:WallpaperURL_]];
+
+ [view setBackgroundColor:[UIColor clearColor]];
+ if ([view respondsToSelector:@selector(setDrawsBackground:)])
+ [view setDrawsBackground:NO];
+ [[view webView] setDrawsBackground:NO];
+
+ [indirect addSubview:view];
+ }
+ }
+
+ for (size_t i(0), e([Themes_ count]); i != e; ++i) {
+ NSString *theme = [Themes_ objectAtIndex:(e - i - 1)];
+ NSString *html = [theme stringByAppendingPathComponent:@"Widget.html"];
+ if ([Manager_ fileExistsAtPath:html]) {
+ CGRect bounds = [indirect bounds];
+
+ UIWebDocumentView *view([[[UIWebDocumentView alloc] initWithFrame:bounds] autorelease]);
+ [view setAutoresizes:true];
+
+ NSURL *url = [NSURL fileURLWithPath:html];
+ [view loadRequest:[NSURLRequest requestWithURL:url]];
+
+ [view setBackgroundColor:[UIColor clearColor]];
+ if ([view respondsToSelector:@selector(setDrawsBackground:)])
+ [view setDrawsBackground:NO];
+ [[view webView] setDrawsBackground:NO];
+
+ [indirect addSubview:view];
+ }
+ }
+
+ return self;
+}
+// }}}
+
+MSHook(void, SBAwayView$updateDesktopImage$, SBAwayView *self, SEL sel, UIImage *image) {
+ NSString *path = $getTheme$([NSArray arrayWithObject:@"LockBackground.html"]);
+ UIView *&_backgroundView(MSHookIvar<UIView *>(self, "_backgroundView"));