1 /* WinterBoard - Theme Manager for the iPhone
2 * Copyright (C) 2009-2010 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 #import <Foundation/Foundation.h>
39 #import <UIKit/UIKit.h>
40 #import <Preferences/PSRootController.h>
41 #import <Preferences/PSViewController.h>
42 #import <Preferences/PSListController.h>
43 #import <Preferences/PSSpecifier.h>
44 #import <Preferences/PSTableCell.h>
45 #import <UIKit/UINavigationButton.h>
48 #include <objc/runtime.h>
50 static BOOL (*IsIconHiddenDisplayId)(NSString *);
51 static BOOL (*HideIconViaDisplayId)(NSString *);
52 static BOOL (*UnHideIconViaDisplayId)(NSString *);
54 static const NSString *WinterBoardDisplayID = @"com.saurik.WinterBoard";
56 extern NSString *PSTableCellKey;
57 extern "C" UIImage *_UIImageWithName(NSString *);
59 static UIImage *checkImage;
60 static UIImage *uncheckedImage;
62 static BOOL settingsChanged;
63 static NSMutableDictionary *_settings;
64 static NSString *_plist;
66 /* [NSObject yieldToSelector:(withObject:)] {{{*/
67 @interface NSObject (wb$yieldToSelector)
68 - (id) wb$yieldToSelector:(SEL)selector withObject:(id)object;
69 - (id) wb$yieldToSelector:(SEL)selector;
72 @implementation NSObject (Cydia)
74 - (void) wb$doNothing {
77 - (void) wb$_yieldToContext:(NSMutableArray *)context {
78 NSAutoreleasePool *pool([[NSAutoreleasePool alloc] init]);
80 SEL selector(reinterpret_cast<SEL>([[context objectAtIndex:0] pointerValue]));
81 id object([[context objectAtIndex:1] nonretainedObjectValue]);
82 volatile bool &stopped(*reinterpret_cast<bool *>([[context objectAtIndex:2] pointerValue]));
84 /* XXX: deal with exceptions */
85 id value([self performSelector:selector withObject:object]);
87 NSMethodSignature *signature([self methodSignatureForSelector:selector]);
88 [context removeAllObjects];
89 if ([signature methodReturnLength] != 0 && value != nil)
90 [context addObject:value];
95 performSelectorOnMainThread:@selector(wb$doNothing)
103 - (id) wb$yieldToSelector:(SEL)selector withObject:(id)object {
104 /*return [self performSelector:selector withObject:object];*/
106 volatile bool stopped(false);
108 NSMutableArray *context([NSMutableArray arrayWithObjects:
109 [NSValue valueWithPointer:selector],
110 [NSValue valueWithNonretainedObject:object],
111 [NSValue valueWithPointer:const_cast<bool *>(&stopped)],
114 NSThread *thread([[[NSThread alloc]
116 selector:@selector(wb$_yieldToContext:)
122 NSRunLoop *loop([NSRunLoop currentRunLoop]);
123 NSDate *future([NSDate distantFuture]);
125 while (!stopped && [loop runMode:NSDefaultRunLoopMode beforeDate:future]);
127 return [context count] == 0 ? nil : [context objectAtIndex:0];
130 - (id) wb$yieldToSelector:(SEL)selector {
131 return [self wb$yieldToSelector:selector withObject:nil];
137 /* Theme Settings Controller {{{ */
138 @interface WBSThemesController: PSViewController <UITableViewDelegate, UITableViewDataSource> {
139 UITableView *_tableView;
140 NSMutableArray *_themes;
143 @property (nonatomic, retain) NSMutableArray *themes;
147 - (id) initForContentSize:(CGSize)size;
149 - (id) navigationTitle;
150 - (void) themesChanged;
152 - (int) numberOfSectionsInTableView:(UITableView *)tableView;
153 - (id) tableView:(UITableView *)tableView titleForHeaderInSection:(int)section;
154 - (int) tableView:(UITableView *)tableView numberOfRowsInSection:(int)section;
155 - (id) tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath;
156 - (void) tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath;
157 - (void) tableView:(UITableView *)tableView moveRowAtIndexPath:(NSIndexPath *)fromIndexPath toIndexPath:(NSIndexPath *)toIndexPath;
158 - (UITableViewCellEditingStyle) tableView:(UITableView *)tableView editingStyleForRowAtIndexPath:(NSIndexPath *)indexPath;
159 - (BOOL) tableView:(UITableView *)tableView shouldIndentWhileEditingRowAtIndexPath:(NSIndexPath *)indexPath;
160 - (BOOL) tableView:(UITableView *)tableView canMoveRowAtIndexPath:(NSIndexPath *)indexPath;
164 @implementation WBSThemesController
166 @synthesize themes = _themes;
169 NSAutoreleasePool *pool([[NSAutoreleasePool alloc] init]);
170 checkImage = [_UIImageWithName(@"UIPreferencesBlueCheck.png") retain];
171 uncheckedImage = [[UIImage imageWithContentsOfFile:@"/System/Library/PreferenceBundles/WinterBoardSettings.bundle/SearchResultsCheckmarkClear.png"] retain];
175 - (id) initForContentSize:(CGSize)size {
176 if ((self = [super initForContentSize:size]) != nil) {
177 self.themes = [_settings objectForKey:@"Themes"];
179 if (NSString *theme = [_settings objectForKey:@"Theme"]) {
180 self.themes = [NSMutableArray arrayWithObject:
181 [NSMutableDictionary dictionaryWithObjectsAndKeys:
183 [NSNumber numberWithBool:YES], @"Active", nil]];
184 [_settings removeObjectForKey:@"Theme"];
187 self.themes = [NSMutableArray array];
188 [_settings setObject:_themes forKey:@"Themes"];
191 NSMutableArray *themesOnDisk([NSMutableArray array]);
194 addObjectsFromArray:[[NSFileManager defaultManager]
195 contentsOfDirectoryAtPath:@"/Library/Themes" error:NULL]
198 [themesOnDisk addObjectsFromArray:[[NSFileManager defaultManager]
199 contentsOfDirectoryAtPath:[NSString stringWithFormat:@"%@/Library/SummerBoard/Themes", NSHomeDirectory()]
203 for (int i = 0, count = [themesOnDisk count]; i < count; i++) {
204 NSString *theme = [themesOnDisk objectAtIndex:i];
205 if ([theme hasSuffix:@".theme"])
206 [themesOnDisk replaceObjectAtIndex:i withObject:[theme stringByDeletingPathExtension]];
209 NSMutableSet *themesSet([NSMutableSet set]);
211 for (int i = 0, count = [_themes count]; i < count; i++) {
212 NSDictionary *theme([_themes objectAtIndex:i]);
213 NSString *name([theme objectForKey:@"Name"]);
215 if (!name || ![themesOnDisk containsObject:name]) {
216 [_themes removeObjectAtIndex:i];
220 [themesSet addObject:name];
224 for (NSString *theme in themesOnDisk) {
225 if ([themesSet containsObject:theme])
227 [themesSet addObject:theme];
229 [_themes insertObject:[NSMutableDictionary dictionaryWithObjectsAndKeys:
231 [NSNumber numberWithBool:NO], @"Active",
235 _tableView = [[UITableView alloc] initWithFrame:CGRectMake(0, 0, 320, 480-64) style:UITableViewStyleGrouped];
236 [_tableView setDataSource:self];
237 [_tableView setDelegate:self];
238 [_tableView setEditing:YES];
239 [_tableView setAllowsSelectionDuringEditing:YES];
240 if ([self respondsToSelector:@selector(setView:)])
241 [self setView:_tableView];
247 [_tableView release];
252 - (id) navigationTitle {
260 - (void) themesChanged {
261 settingsChanged = YES;
264 /* UITableViewDelegate / UITableViewDataSource Methods {{{ */
265 - (int) numberOfSectionsInTableView:(UITableView *)tableView {
269 - (id) tableView:(UITableView *)tableView titleForHeaderInSection:(int)section {
273 - (int) tableView:(UITableView *)tableView numberOfRowsInSection:(int)section {
274 return _themes.count;
277 - (id) tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
278 UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"ThemeCell"];
280 cell = [[[UITableViewCell alloc] initWithFrame:CGRectMake(0, 0, 100, 100) reuseIdentifier:@"ThemeCell"] autorelease];
281 //[cell setTableViewStyle:UITableViewCellStyleDefault];
284 NSDictionary *theme([_themes objectAtIndex:indexPath.row]);
285 cell.text = [theme objectForKey:@"Name"];
286 cell.hidesAccessoryWhenEditing = NO;
287 NSNumber *active([theme objectForKey:@"Active"]);
288 BOOL inactive(active == nil || ![active boolValue]);
289 [cell setImage:(inactive ? uncheckedImage : checkImage)];
293 - (void) tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
294 UITableViewCell *cell = [tableView cellForRowAtIndexPath:indexPath];
295 NSMutableDictionary *theme = [_themes objectAtIndex:indexPath.row];
296 NSNumber *active = [theme objectForKey:@"Active"];
297 BOOL inactive = active == nil || ![active boolValue];
298 [theme setObject:[NSNumber numberWithBool:inactive] forKey:@"Active"];
299 [cell setImage:(!inactive ? uncheckedImage : checkImage)];
300 [tableView deselectRowAtIndexPath:(NSIndexPath *)indexPath animated:YES];
301 [self themesChanged];
304 - (void) tableView:(UITableView *)tableView moveRowAtIndexPath:(NSIndexPath *)fromIndexPath toIndexPath:(NSIndexPath *)toIndexPath {
305 NSUInteger fromIndex = [fromIndexPath row];
306 NSUInteger toIndex = [toIndexPath row];
307 if (fromIndex == toIndex)
309 NSMutableDictionary *theme = [[[_themes objectAtIndex:fromIndex] retain] autorelease];
310 [_themes removeObjectAtIndex:fromIndex];
311 [_themes insertObject:theme atIndex:toIndex];
312 [self themesChanged];
315 - (UITableViewCellEditingStyle) tableView:(UITableView *)tableView editingStyleForRowAtIndexPath:(NSIndexPath *)indexPath {
316 return UITableViewCellEditingStyleNone;
319 - (BOOL) tableView:(UITableView *)tableView shouldIndentWhileEditingRowAtIndexPath:(NSIndexPath *)indexPath {
323 - (BOOL) tableView:(UITableView *)tableView canMoveRowAtIndexPath:(NSIndexPath *)indexPath {
330 @interface WBAdvancedController: PSListController {
334 - (void) settingsChanged;
338 @implementation WBAdvancedController
342 _specifiers = [[self loadSpecifiersFromPlistName:@"Advanced" target:self] retain];
346 - (void) settingsChanged {
347 settingsChanged = YES;
350 - (void) setPreferenceValue:(id)value specifier:(PSSpecifier *)spec {
351 NSString *key([spec propertyForKey:@"key"]);
352 if ([[spec propertyForKey:@"negate"] boolValue])
353 value = [NSNumber numberWithBool:(![value boolValue])];
354 [_settings setValue:value forKey:key];
355 [self settingsChanged];
358 - (id) readPreferenceValue:(PSSpecifier *)spec {
359 NSString *key([spec propertyForKey:@"key"]);
360 id defaultValue([spec propertyForKey:@"default"]);
361 id plistValue([_settings objectForKey:key]);
364 if ([[spec propertyForKey:@"negate"] boolValue])
365 plistValue = [NSNumber numberWithBool:(![plistValue boolValue])];
369 - (void) __optimizeThemes {
370 system("/usr/libexec/winterboard/Optimize");
373 - (void) optimizeThemes {
374 UIAlertView *alert([[[UIAlertView alloc]
375 initWithTitle:@"Optimize Themes"
376 message:@"Please note that this setting /replaces/ the PNG files that came with the theme. PNG files that have been iPhone-optimized cannot be viewed on a normal computer unless they are first deoptimized. You can use Cydia to reinstall themes that have been optimized in order to revert to the original PNG files."
378 cancelButtonTitle:@"Cancel"
379 otherButtonTitles:@"Optimize", nil
382 [alert setContext:@"optimize"];
383 [alert setNumberOfRows:1];
387 - (void) _optimizeThemes {
388 UIView *view([self view]);
389 UIWindow *window([view window]);
391 UIProgressHUD *hud([[[UIProgressHUD alloc] initWithWindow:window] autorelease]);
392 [hud setText:@"Reticulating Splines\nPlease Wait (Minutes)"];
394 [window setUserInteractionEnabled:NO];
396 [window addSubview:hud];
398 [self wb$yieldToSelector:@selector(__optimizeThemes)];
399 [hud removeFromSuperview];
401 [window setUserInteractionEnabled:YES];
403 [self settingsChanged];
406 - (void) alertView:(UIAlertView *)alert clickedButtonAtIndex:(NSInteger)button {
407 NSString *context([alert context]);
409 if ([context isEqualToString:@"optimize"]) {
410 if (button == [alert firstOtherButtonIndex]) {
411 [self performSelector:@selector(_optimizeThemes) withObject:nil afterDelay:0];
414 [alert dismissWithClickedButtonIndex:-1 animated:YES];
416 /*else if ([super respondsToSelector:@selector(alertView:clickedButtonAtIndex:)])
417 [super alertView:alert clickedButtonAtIndex:button];*/
422 @interface WBSettingsController: PSListController {
425 - (id) initForContentSize:(CGSize)size;
428 - (void) navigationBarButtonClicked:(int)buttonIndex;
429 - (void) viewWillRedisplay;
430 - (void) pushController:(id)controller;
432 - (void) settingsChanged;
433 - (NSString *) title;
434 - (void) setPreferenceValue:(id)value specifier:(PSSpecifier *)spec;
435 - (id) readPreferenceValue:(PSSpecifier *)spec;
439 @implementation WBSettingsController
442 void *libhide(dlopen("/usr/lib/hide.dylib", RTLD_LAZY));
443 IsIconHiddenDisplayId = reinterpret_cast<BOOL (*)(NSString *)>(dlsym(libhide, "IsIconHiddenDisplayId"));
444 HideIconViaDisplayId = reinterpret_cast<BOOL (*)(NSString *)>(dlsym(libhide, "HideIconViaDisplayId"));
445 UnHideIconViaDisplayId = reinterpret_cast<BOOL (*)(NSString *)>(dlsym(libhide, "UnHideIconViaDisplayId"));
448 - (void) _wb$loadSettings {
449 _plist = [[NSString stringWithFormat:@"%@/Library/Preferences/com.saurik.WinterBoard.plist", NSHomeDirectory()] retain];
450 _settings = [NSMutableDictionary dictionaryWithContentsOfFile:_plist];
453 if (_settings != nil)
457 _settings = [NSMutableDictionary dictionary];
460 _settings = [_settings retain];
462 if ([_settings objectForKey:@"SummerBoard"] == nil)
463 [_settings setObject:[NSNumber numberWithBool:set] forKey:@"SummerBoard"];
465 [_settings setObject:[NSNumber numberWithBool:IsIconHiddenDisplayId(WinterBoardDisplayID)] forKey:@"IconHidden"];
468 - (id) initForContentSize:(CGSize)size {
469 if ((self = [super initForContentSize:size]) != nil) {
470 [self _wb$loadSettings];
481 if (!settingsChanged)
484 NSData *data([NSPropertyListSerialization dataFromPropertyList:_settings format:NSPropertyListBinaryFormat_v1_0 errorDescription:NULL]);
487 if (![data writeToFile:_plist options:NSAtomicWrite error:NULL])
490 ([[_settings objectForKey:@"IconHidden"] boolValue] ? HideIconViaDisplayId : UnHideIconViaDisplayId)(WinterBoardDisplayID);
492 unlink("/User/Library/Caches/com.apple.springboard-imagecache-icons");
493 unlink("/User/Library/Caches/com.apple.springboard-imagecache-icons.plist");
494 unlink("/User/Library/Caches/com.apple.springboard-imagecache-smallicons");
495 unlink("/User/Library/Caches/com.apple.springboard-imagecache-smallicons.plist");
497 unlink("/User/Library/Caches/com.apple.SpringBoard.folderSwitcherLinen");
498 unlink("/User/Library/Caches/com.apple.SpringBoard.notificationCenterLinen");
500 unlink("/User/Library/Caches/com.apple.SpringBoard.folderSwitcherLinen.0");
501 unlink("/User/Library/Caches/com.apple.SpringBoard.folderSwitcherLinen.1");
502 unlink("/User/Library/Caches/com.apple.SpringBoard.folderSwitcherLinen.2");
503 unlink("/User/Library/Caches/com.apple.SpringBoard.folderSwitcherLinen.3");
505 system("rm -rf /User/Library/Caches/SpringBoardIconCache");
506 system("rm -rf /User/Library/Caches/SpringBoardIconCache-small");
507 system("rm -rf /User/Library/Caches/com.apple.IconsCache");
508 system("rm -rf /User/Library/Caches/com.apple.newsstand");
509 system("rm -rf /User/Library/Caches/com.apple.springboard.sharedimagecache");
511 system("killall lsd SpringBoard");
514 - (void) cancelChanges {
518 [self _wb$loadSettings];
520 [self reloadSpecifiers];
521 if (![[PSViewController class] instancesRespondToSelector:@selector(showLeftButton:withStyle:rightButton:withStyle:)]) {
522 [[self navigationItem] setLeftBarButtonItem:nil];
523 [[self navigationItem] setRightBarButtonItem:nil];
525 [self showLeftButton:nil withStyle:0 rightButton:nil withStyle:0];
527 settingsChanged = NO;
530 - (void) navigationBarButtonClicked:(int)buttonIndex {
531 if (!settingsChanged) {
532 [super navigationBarButtonClicked:buttonIndex];
536 if (buttonIndex == 0) {
537 [self cancelChanges];
542 [self.rootController popController];
545 - (void) settingsConfirmButtonClicked:(UIBarButtonItem *)button {
546 [self navigationBarButtonClicked:button.tag];
549 - (void) viewWillRedisplay {
551 [self settingsChanged];
552 [super viewWillRedisplay];
555 - (void) viewWillAppear:(BOOL)animated {
557 [self settingsChanged];
558 if ([super respondsToSelector:@selector(viewWillAppear:)])
559 [super viewWillAppear:animated];
562 - (void) pushController:(id)controller {
563 [self hideNavigationBarButtons];
564 [super pushController:controller];
569 _specifiers = [[self loadSpecifiersFromPlistName:@"WinterBoard" target:self] retain];
573 - (void) settingsChanged {
574 if (![[PSViewController class] instancesRespondToSelector:@selector(showLeftButton:withStyle:rightButton:withStyle:)]) {
575 UIBarButtonItem *respringButton([[UIBarButtonItem alloc] initWithTitle:@"Respring" style:UIBarButtonItemStyleDone target:self action:@selector(settingsConfirmButtonClicked:)]);
576 UIBarButtonItem *cancelButton([[UIBarButtonItem alloc] initWithTitle:@"Cancel" style:UIBarButtonItemStylePlain target:self action:@selector(settingsConfirmButtonClicked:)]);
577 cancelButton.tag = 0;
578 respringButton.tag = 1;
579 [[self navigationItem] setLeftBarButtonItem:respringButton];
580 [[self navigationItem] setRightBarButtonItem:cancelButton];
581 [respringButton release];
582 [cancelButton release];
584 [self showLeftButton:@"Respring" withStyle:2 rightButton:@"Cancel" withStyle:0];
586 settingsChanged = YES;
589 - (NSString *) title {
590 return @"WinterBoard";
593 - (void) setPreferenceValue:(id)value specifier:(PSSpecifier *)spec {
594 NSString *key([spec propertyForKey:@"key"]);
595 if ([[spec propertyForKey:@"negate"] boolValue])
596 value = [NSNumber numberWithBool:(![value boolValue])];
597 [_settings setValue:value forKey:key];
598 [self settingsChanged];
601 - (id) readPreferenceValue:(PSSpecifier *)spec {
602 NSString *key([spec propertyForKey:@"key"]);
603 id defaultValue([spec propertyForKey:@"default"]);
604 id plistValue([_settings objectForKey:key]);
607 if ([[spec propertyForKey:@"negate"] boolValue])
608 plistValue = [NSNumber numberWithBool:(![plistValue boolValue])];
614 #define WBSAddMethod(_class, _sel, _imp, _type) \
615 if (![[_class class] instancesRespondToSelector:@selector(_sel)]) \
616 class_addMethod([_class class], @selector(_sel), (IMP)_imp, _type)
617 void $PSRootController$popController(PSRootController *self, SEL _cmd) {
618 [self popViewControllerAnimated:YES];
621 void $PSViewController$hideNavigationBarButtons(PSRootController *self, SEL _cmd) {
624 id $PSViewController$initForContentSize$(PSRootController *self, SEL _cmd, CGRect contentSize) {
628 static __attribute__((constructor)) void __wbsInit() {
629 WBSAddMethod(PSRootController, popController, $PSRootController$popController, "v@:");
630 WBSAddMethod(PSViewController, hideNavigationBarButtons, $PSViewController$hideNavigationBarButtons, "v@:");
631 WBSAddMethod(PSViewController, initForContentSize:, $PSViewController$initForContentSize$, "@@:{ff}");