]> git.saurik.com Git - safemode-ios.git/blame - Tweak.xm
Compile for ARM64 (using Xcode 5 for armv6 hacks).
[safemode-ios.git] / Tweak.xm
CommitLineData
df972f42 1/* Cydia Substrate - Powerful Code Insertion Platform
bef4d375 2 * Copyright (C) 2008-2013 Jay Freeman (saurik)
25f84761
JF
3*/
4
df972f42 5/* GNU Lesser General Public License, Version 3 {{{ */
25f84761 6/*
5bb6a76c 7 * Substrate is free software: you can redistribute it and/or modify it under
df972f42
JF
8 * the terms of the GNU Lesser General Public License as published by the
9 * Free Software Foundation, either version 3 of the License, or (at your
10 * option) any later version.
25f84761 11 *
5bb6a76c 12 * Substrate is distributed in the hope that it will be useful, but WITHOUT
df972f42
JF
13 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
14 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
15 * License for more details.
25f84761 16 *
df972f42 17 * You should have received a copy of the GNU Lesser General Public License
5bb6a76c 18 * along with Substrate. If not, see <http://www.gnu.org/licenses/>.
df972f42
JF
19**/
20/* }}} */
25f84761
JF
21
22#import <CoreFoundation/CoreFoundation.h>
23#import <Foundation/Foundation.h>
24#import <CoreGraphics/CGGeometry.h>
39f17851
JF
25#import <UIKit/UIKit.h>
26
88e0e8ac 27#include <substrate.h>
2c75d26c 28
39f17851 29Class $SafeModeAlertItem;
39f17851 30
2c75d26c
JF
31@interface SBAlertItem : NSObject {
32}
33- (UIAlertView *) alertSheet;
34- (void) dismiss;
35@end
36
37@interface SBAlertItemsController : NSObject {
38}
39+ (SBAlertItemsController *) sharedInstance;
40- (void) activateAlertItem:(SBAlertItem *)item;
41@end
42
43@interface SBStatusBarTimeView : UIView {
44}
45- (id) textFont;
46@end
47
88d77501
JF
48@interface UIApplication (CydiaSubstrate)
49- (void) applicationOpenURL:(id)url;
50@end
51
52@interface UIAlertView (CydiaSubstrate)
53- (void) setForceHorizontalButtonsLayout:(BOOL)force;
54- (void) setBodyText:(NSString *)body;
55- (void) setNumberOfRows:(NSInteger)rows;
56@end
57
39f17851
JF
58void SafeModeAlertItem$alertSheet$buttonClicked$(id self, SEL sel, id sheet, int button) {
59 switch (button) {
60 case 1:
61 break;
62
63 case 2:
8d6b6bf4 64 if (kCFCoreFoundationVersionNumber >= 700)
d10105f2
JF
65 system("killall backboardd");
66 else
67 // XXX: there are better ways of restarting SpringBoard that would actually save state
68 exit(0);
39f17851
JF
69 break;
70
71 case 3:
88d77501 72 [[UIApplication sharedApplication] applicationOpenURL:[NSURL URLWithString:@"http://cydia.saurik.com/safemode/"]];
39f17851
JF
73 break;
74 }
75
76 [self dismiss];
77}
78
79void SafeModeAlertItem$configure$requirePasscodeForActions$(id self, SEL sel, BOOL configure, BOOL require) {
2c75d26c 80 UIAlertView *sheet([self alertSheet]);
0625b5c3 81
39f17851 82 [sheet setDelegate:self];
a6712361 83 [sheet setBodyText:@"We apologize for the inconvenience, but SpringBoard has just crashed.\n\nMobileSubstrate /did not/ cause this problem: it has protected you from it.\n\nYour device is now running in Safe Mode. All extensions that support this safety system are disabled.\n\nReboot (or restart SpringBoard) to return to the normal mode. To return to this dialog touch the status bar.\n\nTap \"Help\" below for more tips."];
39f17851
JF
84 [sheet addButtonWithTitle:@"OK"];
85 [sheet addButtonWithTitle:@"Restart"];
86 [sheet addButtonWithTitle:@"Help"];
87 [sheet setNumberOfRows:1];
0625b5c3 88
e5a54d69
JF
89 if ([sheet respondsToSelector:@selector(setForceHorizontalButtonsLayout:)])
90 [sheet setForceHorizontalButtonsLayout:YES];
39f17851
JF
91}
92
93void SafeModeAlertItem$performUnlockAction(id self, SEL sel) {
88e0e8ac 94 [[%c(SBAlertItemsController) sharedInstance] activateAlertItem:self];
39f17851
JF
95}
96
97static void MSAlert() {
98 if ($SafeModeAlertItem == nil)
99 $SafeModeAlertItem = objc_lookUpClass("SafeModeAlertItem");
100 if ($SafeModeAlertItem == nil) {
101 $SafeModeAlertItem = objc_allocateClassPair(objc_getClass("SBAlertItem"), "SafeModeAlertItem", 0);
102 if ($SafeModeAlertItem == nil)
103 return;
104
105 class_addMethod($SafeModeAlertItem, @selector(alertSheet:buttonClicked:), (IMP) &SafeModeAlertItem$alertSheet$buttonClicked$, "v@:@i");
106 class_addMethod($SafeModeAlertItem, @selector(configure:requirePasscodeForActions:), (IMP) &SafeModeAlertItem$configure$requirePasscodeForActions$, "v@:cc");
107 class_addMethod($SafeModeAlertItem, @selector(performUnlockAction), (IMP) SafeModeAlertItem$performUnlockAction, "v@:");
108 objc_registerClassPair($SafeModeAlertItem);
109 }
110
88e0e8ac
JF
111 if (%c(SBAlertItemsController) != nil)
112 [[%c(SBAlertItemsController) sharedInstance] activateAlertItem:[[[$SafeModeAlertItem alloc] init] autorelease]];
39f17851
JF
113}
114
0625b5c3
JF
115
116// XXX: on iOS 5.0, we really would prefer avoiding
117
88e0e8ac
JF
118%hook SBStatusBar
119- (void) touchesEnded:(id)touches withEvent:(id)event {
b4f9ed46 120 MSAlert();
88e0e8ac
JF
121 %orig(touches, event);
122} %end
b4f9ed46 123
88e0e8ac
JF
124%hook SBStatusBar
125- (void) mouseDown:(void *)event {
39f17851 126 MSAlert();
88e0e8ac
JF
127 %orig(event);
128} %end
39f17851 129
88e0e8ac
JF
130%hook UIStatusBar
131- (void) touchesBegan:(void *)touches withEvent:(void *)event {
e5a54d69 132 MSAlert();
88e0e8ac
JF
133 %orig(touches, event);
134} %end
e5a54d69 135
0625b5c3
JF
136
137// this fairly complex code came from Grant, to solve the "it Safe Mode"-in-bar bug
138
88e0e8ac
JF
139%hook SBStatusBarDataManager
140- (void) _updateTimeString {
22a7ee91
JF
141 char *_data(&MSHookIvar<char>(self, "_data"));
142 if (_data == NULL)
143 return;
144
145 Ivar _itemIsEnabled(object_getInstanceVariable(self, "_itemIsEnabled", NULL));
146 if (_itemIsEnabled == NULL)
147 return;
148
149 Ivar _itemIsCloaked(object_getInstanceVariable(self, "_itemIsCloaked", NULL));
150 if (_itemIsCloaked == NULL)
151 return;
152
153 size_t enabledOffset(ivar_getOffset(_itemIsEnabled));
154 size_t cloakedOffset(ivar_getOffset(_itemIsCloaked));
155 if (enabledOffset >= cloakedOffset)
156 return;
157
158 size_t offset(cloakedOffset - enabledOffset);
159 char *timeString(_data + offset);
160 strcpy(timeString, "Exit Safe Mode");
88e0e8ac 161} %end
e5a54d69 162
0625b5c3 163
f7673201
JF
164static bool alerted_;
165
166static void AlertIfNeeded() {
167 if (alerted_)
39f17851 168 return;
f7673201 169 alerted_ = true;
39f17851 170 MSAlert();
25f84761
JF
171}
172
0625b5c3
JF
173
174// on iOS 4.3 and above we can use this advertisement, which seems to check every time the user unlocks
175// XXX: verify that this still works on iOS 5.0
176
88e0e8ac
JF
177%hook AAAccountManager
178+ (void) showMobileMeOfferIfNecessary {
f7673201 179 AlertIfNeeded();
88e0e8ac 180} %end
f7673201 181
0625b5c3
JF
182
183// -[SBIconController showInfoAlertIfNeeded] explains how to drag icons around the iPhone home screen
184// it used to be shown to users when they unlocked their screen for the first time, and happened every unlock
185// however, as of iOS 4.3, it got relegated to only appearing once the user installed an app or web clip
186
88e0e8ac
JF
187%hook SBIconController
188- (void) showInfoAlertIfNeeded {
f7673201 189 AlertIfNeeded();
88e0e8ac 190} %end
f7673201 191
0625b5c3
JF
192
193// the icon state, including crazy configurations like Five Icon Dock, is stored in SpringBoard's defaults
194// unfortunately, SpringBoard on iOS 2.0 and 2.1 (maybe 2.2 as well) buffer overrun with more than 4 icons
195// there is a third party package called IconSupport that remedies this, but not everyone is using it yet
196
88e0e8ac
JF
197%hook SBButtonBar
198- (int) maxIconColumns {
25f84761
JF
199 static int max;
200 if (max == 0) {
88e0e8ac 201 max = %orig();
25f84761
JF
202 if (NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults])
203 if (NSDictionary *iconState = [defaults objectForKey:@"iconState"])
204 if (NSDictionary *buttonBar = [iconState objectForKey:@"buttonBar"])
205 if (NSArray *iconMatrix = [buttonBar objectForKey:@"iconMatrix"])
206 if ([iconMatrix count] != 0)
207 if (NSArray *row = [iconMatrix objectAtIndex:0]) {
208 int count([row count]);
209 if (max < count)
210 max = count;
211 }
212 } return max;
88e0e8ac 213} %end
25f84761 214
0625b5c3 215
88e0e8ac
JF
216%hook SBUIController
217- (id) init {
218 if ((self = %orig()) != nil) {
b4f9ed46
JF
219 UIView *&_contentLayer(MSHookIvar<UIView *>(self, "_contentLayer"));
220 UIView *&_contentView(MSHookIvar<UIView *>(self, "_contentView"));
221
222 UIView *layer;
223 if (&_contentLayer != NULL)
224 layer = _contentLayer;
225 else if (&_contentView != NULL)
226 layer = _contentView;
227 else
228 layer = nil;
229
230 if (layer != nil)
231 [layer setBackgroundColor:[UIColor darkGrayColor]];
232 } return self;
88e0e8ac 233} %end
25f84761 234
1c74ea29 235#define Paper_ "/Library/MobileSubstrate/MobileSafety.png"
dbbe0f32 236
88e0e8ac
JF
237%hook UIImage
238+ (UIImage *) defaultDesktopImage {
dbbe0f32 239 return [UIImage imageWithContentsOfFile:@Paper_];
88e0e8ac 240} %end
dbbe0f32 241
88e0e8ac
JF
242%hook SBStatusBarTimeView
243- (void) tile {
39f17851
JF
244 NSString *&_time(MSHookIvar<NSString *>(self, "_time"));
245 CGRect &_textRect(MSHookIvar<CGRect>(self, "_textRect"));
25f84761 246 if (_time != nil)
39f17851 247 [_time release];
dbbe0f32 248 _time = [@"Exit Safe Mode" retain];
2c75d26c 249 id font([self textFont]);
88d77501 250 CGSize size([_time sizeWithFont:font]);
39f17851 251 CGRect frame([self frame]);
39f17851
JF
252 _textRect.size = size;
253 _textRect.origin.x = (frame.size.width - size.width) / 2;
254 _textRect.origin.y = (frame.size.height - size.height) / 2;
88e0e8ac 255} %end
e9502538 256
0625b5c3
JF
257
258// notification widgets ("wee apps" or "bulletin board sections") are capable of crashing SpringBoard
259// unfortunately, which ones are in use are stored in SpringBoard's defaults, so we need to turn them off
260
88e0e8ac
JF
261%hook BBSectionInfo
262- (BOOL) showsInNotificationCenter {
e9502538 263 return NO;
88e0e8ac 264} %end
e06a3b1a
JF
265
266
267// on iOS 6.0, Apple split parts of SpringBoard into a daemon called backboardd, including app launches
268// in order to allow safe mode to propogate into applications, we need to then tell backboardd here
269// XXX: (all of this should be replaced, however, with per-process launchd-mediated exception handling)
270
88e0e8ac
JF
271%hook BKSApplicationLaunchSettings
272- (void) setEnvironment:(NSDictionary *)original {
e06a3b1a 273 if (original == nil)
88e0e8ac 274 return %orig(nil);
e06a3b1a
JF
275
276 NSMutableDictionary *modified([original mutableCopy]);
277 [modified setObject:@"1" forKey:@"_MSSafeMode"];
88e0e8ac
JF
278 return %orig(modified);
279} %end
43b2fcfa 280
88e0e8ac 281%ctor {
43b2fcfa
JF
282 NSAutoreleasePool *pool([[NSAutoreleasePool alloc] init]);
283
284 // on iOS 6, backboardd is in charge of brightness, and freaks out when SpringBoard restarts :(
285 // the result is that the device is super dark until we attempt to update the brightness here.
286
287 if (kCFCoreFoundationVersionNumber >= 700) {
288 if (void (*GSEventSetBacklightLevel)(float) = reinterpret_cast<void (*)(float)>(dlsym(RTLD_DEFAULT, "GSEventSetBacklightLevel")))
289 if (NSMutableDictionary *defaults = [NSMutableDictionary dictionaryWithContentsOfFile:[NSString stringWithFormat:@"%@/Library/Preferences/com.apple.springboard.plist", NSHomeDirectory()]])
290 if (NSNumber *level = [defaults objectForKey:@"SBBacklightLevel2"])
291 GSEventSetBacklightLevel([level floatValue]);
292 }
293
294 [pool release];
295}