]> git.saurik.com Git - winterboard.git/blame - Settings.mm
Added some really complex path scanning code.
[winterboard.git] / Settings.mm
CommitLineData
224d48e3 1/* WinterBoard - Theme Manager for the iPhone
43fd3a0b 2 * Copyright (C) 2009-2010 Jay Freeman (saurik)
224d48e3
JF
3*/
4
5/*
6 * Redistribution and use in source and binary
7 * forms, with or without modification, are permitted
8 * provided that the following conditions are met:
9 *
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
17 * distribution.
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.
21 *
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.
36*/
37
38#import <Foundation/Foundation.h>
39#import <UIKit/UIKit.h>
b39b993d
DH
40#import <Preferences/PSRootController.h>
41#import <Preferences/PSViewController.h>
224d48e3
JF
42#import <Preferences/PSListController.h>
43#import <Preferences/PSSpecifier.h>
44#import <Preferences/PSTableCell.h>
45#import <UIKit/UINavigationButton.h>
46
7c88a3f1 47#include <dlfcn.h>
b39b993d 48#include <objc/runtime.h>
7c88a3f1
JF
49
50static BOOL (*IsIconHiddenDisplayId)(NSString *);
51static BOOL (*HideIconViaDisplayId)(NSString *);
52static BOOL (*UnHideIconViaDisplayId)(NSString *);
53
54static const NSString *WinterBoardDisplayID = @"com.saurik.WinterBoard";
55
224d48e3
JF
56extern NSString *PSTableCellKey;
57extern "C" UIImage *_UIImageWithName(NSString *);
58
59static UIImage *checkImage;
60static UIImage *uncheckedImage;
61
62static BOOL settingsChanged;
63static NSMutableDictionary *_settings;
64static NSString *_plist;
65
265e19b2
JF
66/* [NSObject yieldToSelector:(withObject:)] {{{*/
67@interface NSObject (wb$yieldToSelector)
68- (id) wb$yieldToSelector:(SEL)selector withObject:(id)object;
69- (id) wb$yieldToSelector:(SEL)selector;
70@end
71
72@implementation NSObject (Cydia)
73
74- (void) wb$doNothing {
75}
76
77- (void) wb$_yieldToContext:(NSMutableArray *)context {
78 NSAutoreleasePool *pool([[NSAutoreleasePool alloc] init]);
79
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]));
83
84 /* XXX: deal with exceptions */
85 id value([self performSelector:selector withObject:object]);
86
87 NSMethodSignature *signature([self methodSignatureForSelector:selector]);
88 [context removeAllObjects];
89 if ([signature methodReturnLength] != 0 && value != nil)
90 [context addObject:value];
91
92 stopped = true;
93
94 [self
95 performSelectorOnMainThread:@selector(wb$doNothing)
96 withObject:nil
97 waitUntilDone:NO
98 ];
99
100 [pool release];
101}
102
103- (id) wb$yieldToSelector:(SEL)selector withObject:(id)object {
104 /*return [self performSelector:selector withObject:object];*/
105
106 volatile bool stopped(false);
107
108 NSMutableArray *context([NSMutableArray arrayWithObjects:
109 [NSValue valueWithPointer:selector],
110 [NSValue valueWithNonretainedObject:object],
111 [NSValue valueWithPointer:const_cast<bool *>(&stopped)],
112 nil]);
113
114 NSThread *thread([[[NSThread alloc]
115 initWithTarget:self
116 selector:@selector(wb$_yieldToContext:)
117 object:context
118 ] autorelease]);
119
120 [thread start];
121
122 NSRunLoop *loop([NSRunLoop currentRunLoop]);
123 NSDate *future([NSDate distantFuture]);
124
125 while (!stopped && [loop runMode:NSDefaultRunLoopMode beforeDate:future]);
126
127 return [context count] == 0 ? nil : [context objectAtIndex:0];
128}
129
130- (id) wb$yieldToSelector:(SEL)selector {
131 return [self wb$yieldToSelector:selector withObject:nil];
132}
133
134@end
135/* }}} */
136
224d48e3
JF
137/* Theme Settings Controller {{{ */
138@interface WBSThemesController: PSViewController <UITableViewDelegate, UITableViewDataSource> {
139 UITableView *_tableView;
140 NSMutableArray *_themes;
141}
142
143@property (nonatomic, retain) NSMutableArray *themes;
144
145+ (void) load;
146
147- (id) initForContentSize:(CGSize)size;
148- (id) view;
149- (id) navigationTitle;
150- (void) themesChanged;
151
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;
7c88a3f1 161
224d48e3
JF
162@end
163
164@implementation WBSThemesController
165
166@synthesize themes = _themes;
167
168+ (void) load {
169 NSAutoreleasePool *pool([[NSAutoreleasePool alloc] init]);
170 checkImage = [_UIImageWithName(@"UIPreferencesBlueCheck.png") retain];
171 uncheckedImage = [[UIImage imageWithContentsOfFile:@"/System/Library/PreferenceBundles/WinterBoardSettings.bundle/SearchResultsCheckmarkClear.png"] retain];
172 [pool release];
173}
174
175- (id) initForContentSize:(CGSize)size {
176 if ((self = [super initForContentSize:size]) != nil) {
177 self.themes = [_settings objectForKey:@"Themes"];
178 if (!_themes) {
179 if (NSString *theme = [_settings objectForKey:@"Theme"]) {
180 self.themes = [NSMutableArray arrayWithObject:
181 [NSMutableDictionary dictionaryWithObjectsAndKeys:
182 theme, @"Name",
183 [NSNumber numberWithBool:YES], @"Active", nil]];
184 [_settings removeObjectForKey:@"Theme"];
185 }
186 if (!_themes)
187 self.themes = [NSMutableArray array];
188 [_settings setObject:_themes forKey:@"Themes"];
189 }
190
191 NSMutableArray *themesOnDisk([NSMutableArray array]);
192
193 [themesOnDisk
194 addObjectsFromArray:[[NSFileManager defaultManager]
195 contentsOfDirectoryAtPath:@"/Library/Themes" error:NULL]
196 ];
197
198 [themesOnDisk addObjectsFromArray:[[NSFileManager defaultManager]
199 contentsOfDirectoryAtPath:[NSString stringWithFormat:@"%@/Library/SummerBoard/Themes", NSHomeDirectory()]
200 error:NULL
201 ]];
202
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]];
207 }
208
209 NSMutableSet *themesSet([NSMutableSet set]);
210
211 for (int i = 0, count = [_themes count]; i < count; i++) {
212 NSDictionary *theme([_themes objectAtIndex:i]);
213 NSString *name([theme objectForKey:@"Name"]);
214
215 if (!name || ![themesOnDisk containsObject:name]) {
216 [_themes removeObjectAtIndex:i];
217 i--;
218 count--;
219 } else {
220 [themesSet addObject:name];
221 }
222 }
223
224 for (NSString *theme in themesOnDisk) {
225 if ([themesSet containsObject:theme])
226 continue;
227 [themesSet addObject:theme];
228
229 [_themes insertObject:[NSMutableDictionary dictionaryWithObjectsAndKeys:
230 theme, @"Name",
231 [NSNumber numberWithBool:NO], @"Active",
232 nil] atIndex:0];
233 }
234
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];
b39b993d
DH
240 if ([self respondsToSelector:@selector(setView:)])
241 [self setView:_tableView];
224d48e3
JF
242 }
243 return self;
244}
245
246- (void) dealloc {
247 [_tableView release];
248 [_themes release];
249 [super dealloc];
250}
251
252- (id) navigationTitle {
253 return @"Themes";
254}
255
256- (id) view {
257 return _tableView;
258}
259
260- (void) themesChanged {
261 settingsChanged = YES;
262}
263
264/* UITableViewDelegate / UITableViewDataSource Methods {{{ */
265- (int) numberOfSectionsInTableView:(UITableView *)tableView {
266 return 1;
267}
268
269- (id) tableView:(UITableView *)tableView titleForHeaderInSection:(int)section {
270 return nil;
271}
272
273- (int) tableView:(UITableView *)tableView numberOfRowsInSection:(int)section {
274 return _themes.count;
275}
276
277- (id) tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
278 UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"ThemeCell"];
279 if (!cell) {
280 cell = [[[UITableViewCell alloc] initWithFrame:CGRectMake(0, 0, 100, 100) reuseIdentifier:@"ThemeCell"] autorelease];
281 //[cell setTableViewStyle:UITableViewCellStyleDefault];
282 }
283
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)];
290 return cell;
291}
292
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];
302}
303
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)
308 return;
309 NSMutableDictionary *theme = [[[_themes objectAtIndex:fromIndex] retain] autorelease];
310 [_themes removeObjectAtIndex:fromIndex];
311 [_themes insertObject:theme atIndex:toIndex];
312 [self themesChanged];
313}
314
315- (UITableViewCellEditingStyle) tableView:(UITableView *)tableView editingStyleForRowAtIndexPath:(NSIndexPath *)indexPath {
316 return UITableViewCellEditingStyleNone;
317}
318
319- (BOOL) tableView:(UITableView *)tableView shouldIndentWhileEditingRowAtIndexPath:(NSIndexPath *)indexPath {
320 return NO;
321}
322
323- (BOOL) tableView:(UITableView *)tableView canMoveRowAtIndexPath:(NSIndexPath *)indexPath {
324 return YES;
325}
326/* }}} */
327@end
328/* }}} */
329
330@interface WBSettingsController: PSListController {
331}
332
333- (id) initForContentSize:(CGSize)size;
334- (void) dealloc;
335- (void) suspend;
336- (void) navigationBarButtonClicked:(int)buttonIndex;
337- (void) viewWillRedisplay;
338- (void) pushController:(id)controller;
339- (id) specifiers;
340- (void) settingsChanged;
341- (NSString *) title;
342- (void) setPreferenceValue:(id)value specifier:(PSSpecifier *)spec;
343- (id) readPreferenceValue:(PSSpecifier *)spec;
344
345@end
346
347@implementation WBSettingsController
348
7c88a3f1 349+ (void) load {
7c88a3f1 350 void *libhide(dlopen("/usr/lib/hide.dylib", RTLD_LAZY));
7c88a3f1 351 IsIconHiddenDisplayId = reinterpret_cast<BOOL (*)(NSString *)>(dlsym(libhide, "IsIconHiddenDisplayId"));
7c88a3f1 352 HideIconViaDisplayId = reinterpret_cast<BOOL (*)(NSString *)>(dlsym(libhide, "HideIconViaDisplayId"));
7c88a3f1 353 UnHideIconViaDisplayId = reinterpret_cast<BOOL (*)(NSString *)>(dlsym(libhide, "UnHideIconViaDisplayId"));
7c88a3f1
JF
354}
355
224d48e3
JF
356- (id) initForContentSize:(CGSize)size {
357 if ((self = [super initForContentSize:size]) != nil) {
358 _plist = [[NSString stringWithFormat:@"%@/Library/Preferences/com.saurik.WinterBoard.plist", NSHomeDirectory()] retain];
359 _settings = [([NSMutableDictionary dictionaryWithContentsOfFile:_plist] ?: [NSMutableDictionary dictionary]) retain];
7c88a3f1
JF
360
361 [_settings setObject:[NSNumber numberWithBool:IsIconHiddenDisplayId(WinterBoardDisplayID)] forKey:@"IconHidden"];
224d48e3
JF
362 } return self;
363}
364
365- (void) dealloc {
366 [_settings release];
367 [_plist release];
368 [super dealloc];
369}
370
43fd3a0b 371- (void) __optimizeThemes {
265e19b2
JF
372 system("/usr/libexec/winterboard/Optimize");
373}
374
375- (void) optimizeThemes {
43fd3a0b
JF
376 UIActionSheet *sheet([[[UIActionSheet alloc]
377 initWithTitle:@"Optimize Themes"
378 buttons:[NSArray arrayWithObjects:@"Optimize", @"Cancel", nil]
379 defaultButtonIndex:1
380 delegate:self
381 context:@"optimize"
382 ] autorelease]);
383
384 [sheet setBodyText:@"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."];
385 [sheet setNumberOfRows:1];
386 [sheet setDestructiveButtonIndex:0];
387 [sheet popupAlertAnimated:YES];
388}
389
390- (void) _optimizeThemes {
265e19b2
JF
391 UIView *view([self view]);
392 UIWindow *window([view window]);
393
394 UIProgressHUD *hud([[[UIProgressHUD alloc] initWithWindow:window] autorelease]);
395 [hud setText:@"Reticulating Splines\nPlease Wait (Minutes)"];
396
397 [window setUserInteractionEnabled:NO];
398
399 [window addSubview:hud];
400 [hud show:YES];
43fd3a0b 401 [self wb$yieldToSelector:@selector(__optimizeThemes)];
265e19b2
JF
402 [hud removeFromSuperview];
403
404 [window setUserInteractionEnabled:YES];
405
406 [self settingsChanged];
407}
408
43fd3a0b
JF
409- (void) alertSheet:(UIActionSheet *)sheet buttonClicked:(int)button {
410 NSString *context([sheet context]);
411
412 if ([context isEqualToString:@"optimize"]) {
413 switch (button) {
414 case 1:
415 [self performSelector:@selector(_optimizeThemes) withObject:nil afterDelay:0];
416 break;
417 }
418
419 [sheet dismiss];
420 } else
421 [super alertSheet:sheet buttonClicked:button];
422
423}
424
224d48e3
JF
425- (void) suspend {
426 if (!settingsChanged)
427 return;
428
429 NSData *data([NSPropertyListSerialization dataFromPropertyList:_settings format:NSPropertyListBinaryFormat_v1_0 errorDescription:NULL]);
430 if (!data)
431 return;
432 if (![data writeToFile:_plist options:NSAtomicWrite error:NULL])
433 return;
434
7c88a3f1
JF
435 ([[_settings objectForKey:@"IconHidden"] boolValue] ? HideIconViaDisplayId : UnHideIconViaDisplayId)(WinterBoardDisplayID);
436
224d48e3
JF
437 unlink("/User/Library/Caches/com.apple.springboard-imagecache-icons");
438 unlink("/User/Library/Caches/com.apple.springboard-imagecache-icons.plist");
439 unlink("/User/Library/Caches/com.apple.springboard-imagecache-smallicons");
440 unlink("/User/Library/Caches/com.apple.springboard-imagecache-smallicons.plist");
bbea033b 441
546a205b
JF
442 unlink("/User/Library/Caches/com.apple.SpringBoard.folderSwitcherLinen");
443 unlink("/User/Library/Caches/com.apple.SpringBoard.notificationCenterLinen");
bbea033b 444
546a205b
JF
445 unlink("/User/Library/Caches/com.apple.SpringBoard.folderSwitcherLinen.0");
446 unlink("/User/Library/Caches/com.apple.SpringBoard.folderSwitcherLinen.1");
447 unlink("/User/Library/Caches/com.apple.SpringBoard.folderSwitcherLinen.2");
448 unlink("/User/Library/Caches/com.apple.SpringBoard.folderSwitcherLinen.3");
7c88a3f1
JF
449
450 system("rm -rf /User/Library/Caches/SpringBoardIconCache");
451 system("rm -rf /User/Library/Caches/SpringBoardIconCache-small");
f64efe8d 452 system("rm -rf /User/Library/Caches/com.apple.IconsCache");
546a205b 453 system("rm -rf /User/Library/Caches/com.apple.newsstand");
bbea033b 454 system("rm -rf /User/Library/Caches/com.apple.springboard.sharedimagecache");
7c88a3f1 455
f64efe8d 456 system("killall lsd SpringBoard");
224d48e3
JF
457}
458
b39b993d
DH
459- (void) cancelChanges {
460 [_settings release];
461 [_plist release];
462 _plist = [[NSString stringWithFormat:@"%@/Library/Preferences/com.saurik.WinterBoard.plist", NSHomeDirectory()] retain];
463 _settings = [([NSMutableDictionary dictionaryWithContentsOfFile:_plist] ?: [NSMutableDictionary dictionary]) retain];
464
465 [_settings setObject:[NSNumber numberWithBool:IsIconHiddenDisplayId(WinterBoardDisplayID)] forKey:@"IconHidden"];
466 [self reloadSpecifiers];
467 if (![[PSViewController class] instancesRespondToSelector:@selector(showLeftButton:withStyle:rightButton:withStyle:)]) {
1d3b613f
JF
468 [[self navigationItem] setLeftBarButtonItem:nil];
469 [[self navigationItem] setRightBarButtonItem:nil];
838dbf4c
DH
470 } else {
471 [self showLeftButton:nil withStyle:0 rightButton:nil withStyle:0];
b39b993d
DH
472 }
473 settingsChanged = NO;
474}
475
224d48e3
JF
476- (void) navigationBarButtonClicked:(int)buttonIndex {
477 if (!settingsChanged) {
478 [super navigationBarButtonClicked:buttonIndex];
479 return;
480 }
481
838dbf4c 482 if (buttonIndex == 0) {
b39b993d 483 [self cancelChanges];
838dbf4c
DH
484 return;
485 }
224d48e3
JF
486
487 [self suspend];
488 [self.rootController popController];
489}
490
b39b993d
DH
491- (void) settingsConfirmButtonClicked:(UIBarButtonItem *)button {
492 [self navigationBarButtonClicked:button.tag];
493}
494
224d48e3
JF
495- (void) viewWillRedisplay {
496 if (settingsChanged)
497 [self settingsChanged];
498 [super viewWillRedisplay];
499}
500
b39b993d
DH
501- (void) viewWillAppear:(BOOL)animated {
502 if (settingsChanged)
503 [self settingsChanged];
1d3b613f
JF
504 if ([super respondsToSelector:@selector(viewWillAppear:)])
505 [super viewWillAppear:animated];
b39b993d
DH
506}
507
224d48e3
JF
508- (void) pushController:(id)controller {
509 [self hideNavigationBarButtons];
510 [super pushController:controller];
511}
512
513- (id) specifiers {
514 if (!_specifiers)
515 _specifiers = [[self loadSpecifiersFromPlistName:@"WinterBoard" target:self] retain];
516 return _specifiers;
517}
518
519- (void) settingsChanged {
b39b993d
DH
520 if (![[PSViewController class] instancesRespondToSelector:@selector(showLeftButton:withStyle:rightButton:withStyle:)]) {
521 UIBarButtonItem *respringButton([[UIBarButtonItem alloc] initWithTitle:@"Respring" style:UIBarButtonItemStyleDone target:self action:@selector(settingsConfirmButtonClicked:)]);
522 UIBarButtonItem *cancelButton([[UIBarButtonItem alloc] initWithTitle:@"Cancel" style:UIBarButtonItemStylePlain target:self action:@selector(settingsConfirmButtonClicked:)]);
523 cancelButton.tag = 0;
524 respringButton.tag = 1;
1d3b613f
JF
525 [[self navigationItem] setLeftBarButtonItem:respringButton];
526 [[self navigationItem] setRightBarButtonItem:cancelButton];
b39b993d
DH
527 [respringButton release];
528 [cancelButton release];
529 } else {
530 [self showLeftButton:@"Respring" withStyle:2 rightButton:@"Cancel" withStyle:0];
531 }
224d48e3
JF
532 settingsChanged = YES;
533}
534
535- (NSString *) title {
536 return @"WinterBoard";
537}
538
539- (void) setPreferenceValue:(id)value specifier:(PSSpecifier *)spec {
7c88a3f1 540 NSString *key([spec propertyForKey:@"key"]);
224d48e3
JF
541 if ([[spec propertyForKey:@"negate"] boolValue])
542 value = [NSNumber numberWithBool:(![value boolValue])];
7c88a3f1 543 [_settings setValue:value forKey:key];
224d48e3
JF
544 [self settingsChanged];
545}
546
547- (id) readPreferenceValue:(PSSpecifier *)spec {
548 NSString *key([spec propertyForKey:@"key"]);
549 id defaultValue([spec propertyForKey:@"default"]);
550 id plistValue([_settings objectForKey:key]);
551 if (!plistValue)
552 return defaultValue;
553 if ([[spec propertyForKey:@"negate"] boolValue])
554 plistValue = [NSNumber numberWithBool:(![plistValue boolValue])];
555 return plistValue;
556}
557
558@end
b39b993d
DH
559
560#define WBSAddMethod(_class, _sel, _imp, _type) \
561 if (![[_class class] instancesRespondToSelector:@selector(_sel)]) \
562 class_addMethod([_class class], @selector(_sel), (IMP)_imp, _type)
563void $PSRootController$popController(PSRootController *self, SEL _cmd) {
564 [self popViewControllerAnimated:YES];
565}
566
567void $PSViewController$hideNavigationBarButtons(PSRootController *self, SEL _cmd) {
568}
569
570id $PSViewController$initForContentSize$(PSRootController *self, SEL _cmd, CGRect contentSize) {
571 return [self init];
572}
573
574static __attribute__((constructor)) void __wbsInit() {
575 WBSAddMethod(PSRootController, popController, $PSRootController$popController, "v@:");
576 WBSAddMethod(PSViewController, hideNavigationBarButtons, $PSViewController$hideNavigationBarButtons, "v@:");
577 WBSAddMethod(PSViewController, initForContentSize:, $PSViewController$initForContentSize$, "@@:{ff}");
578}