1 | // ===========================================================================
|
---|
2 | //
|
---|
3 | // File: APEMain.m
|
---|
4 | //
|
---|
5 | // Contains: ICeCoffEE APE Module code
|
---|
6 | //
|
---|
7 | // Copyright: Copyright (c) 2003, Nicholas Riley
|
---|
8 | // All Rights Reserved.
|
---|
9 | //
|
---|
10 | // Author(s): Nicholas Riley (Sun Jan 19 2003)
|
---|
11 | //
|
---|
12 | // ===========================================================================
|
---|
13 |
|
---|
14 | #import <Carbon/Carbon.h>
|
---|
15 | #import <ApplicationEnhancer/ApplicationEnhancer.h>
|
---|
16 | #import <ApplicationEnhancer/APETools.h>
|
---|
17 | #import <objc/objc-runtime.h>
|
---|
18 | #import "ICeCoffEE.h"
|
---|
19 | #import "CPS.h"
|
---|
20 |
|
---|
21 | //¥¥¥ Our settings
|
---|
22 |
|
---|
23 | //¥¥¥ Function prototypes
|
---|
24 | static void ICCF_ReloadPrefs(); // reloads our preferences
|
---|
25 | static void ICCF_MigratePrefs(); // migrates prefs from 1.0Ð1.2
|
---|
26 |
|
---|
27 | //¥¥¥ Enter sandman, the code begins --
|
---|
28 |
|
---|
29 | #define ICCF_GET_PATCHCLASS(patchclass) \
|
---|
30 | struct objc_class *patchclass = objc_getClass(patchclass ## Name); \
|
---|
31 | if (patchclass == NULL) { \
|
---|
32 | ICapeprintf("can't get %s\n", patchclass ## Name); \
|
---|
33 | return NO; \
|
---|
34 | }
|
---|
35 |
|
---|
36 | #define ICCF_GET_METHOD(name, patchclass, sel) \
|
---|
37 | Method name = class_getInstanceMethod(patchclass, sel); \
|
---|
38 | if (name == NULL) { \
|
---|
39 | ICapeprintf("can't get %s\n", patchclass ## Name); \
|
---|
40 | return NO; \
|
---|
41 | }
|
---|
42 |
|
---|
43 | BOOL ICCF_PatchMethod(char *patcheeClassName,
|
---|
44 | char *patchClassName,
|
---|
45 | char *patchSuperclassName,
|
---|
46 | char *selectorString) {
|
---|
47 |
|
---|
48 | ICCF_GET_PATCHCLASS(patcheeClass);
|
---|
49 | ICCF_GET_PATCHCLASS(patchClass);
|
---|
50 | ICCF_GET_PATCHCLASS(patchSuperclass);
|
---|
51 |
|
---|
52 | SEL selector = sel_getUid(selectorString);
|
---|
53 |
|
---|
54 | ICCF_GET_METHOD(patcheeMethod, patcheeClass, selector);
|
---|
55 | ICCF_GET_METHOD(patchMethod, patchClass, selector);
|
---|
56 | ICCF_GET_METHOD(patchSuperMethod, patchSuperclass, selector);
|
---|
57 |
|
---|
58 | if (APEPatchCreate(patchSuperMethod->method_imp,
|
---|
59 | APEPatchCreate(patcheeMethod->method_imp, patchMethod->method_imp)) == NULL) {
|
---|
60 | ICapeprintf("can't patch class %s with [%s %s] super %s", patcheeClassName, patchClassName, selectorString, patchSuperclassName);
|
---|
61 | return NO;
|
---|
62 | }
|
---|
63 | return YES;
|
---|
64 | }
|
---|
65 |
|
---|
66 | CFBundleRef ICCF_bundle;
|
---|
67 |
|
---|
68 | // With APE 1.3, if we're in the exclude list, APEBundleMainEarlyLoad doesn't get invoked; don't need to use APETools. But we need to do our own management to avoid loading in non-GUI applications.
|
---|
69 | Boolean ICCF_shouldLoad;
|
---|
70 |
|
---|
71 | void APEBundleMainEarlyLoad(CFBundleRef inBundle, CFStringRef inAPEToolsApplicationID)
|
---|
72 | {
|
---|
73 | ICCF_MigratePrefs();
|
---|
74 |
|
---|
75 | UInt32 icVersion = CFBundleGetVersionNumber(inBundle);
|
---|
76 | ICapeprintf("ICeCoffEE APE: bundle version is %ld (0x%x)\n", icVersion, icVersion);
|
---|
77 | CFNumberRef icVersionRef = CFNumberCreate(NULL, kCFNumberLongType, &icVersion);
|
---|
78 | CFPreferencesSetAppValue(kICLastLoadedVersion, icVersionRef, kICBundleIdentifier);
|
---|
79 | SAFE_RELEASE(icVersionRef);
|
---|
80 |
|
---|
81 | ICCF_shouldLoad = false;
|
---|
82 |
|
---|
83 | CPSProcessSerNum psn;
|
---|
84 | OSStatus err = CPSGetCurrentProcess(&psn);
|
---|
85 | if (err != noErr) {
|
---|
86 | #if ICCF_DEBUG
|
---|
87 | apeprintf("ICeCoffEE APE: Can't get process serial number for current process (error %ld); still loading because of ICCF_DEBUG\n", err);
|
---|
88 | #else
|
---|
89 | apeprintf("ICeCoffEE APE: Can't get process serial number for current process (error %ld); not loading in this application\n", err);
|
---|
90 | return;
|
---|
91 | #endif
|
---|
92 | }
|
---|
93 |
|
---|
94 | CPSProcessInfoRec info;
|
---|
95 | err = CPSGetProcessInfo(&psn, &info, NULL, 0, NULL, NULL, 0);
|
---|
96 | if (err != noErr) {
|
---|
97 | #if ICCF_DEBUG
|
---|
98 | apeprintf("ICeCoffEE APE: Can't get process information (error %ld); still loading because of ICCF_DEBUG\n", err);
|
---|
99 | #else
|
---|
100 | apeprintf("ICeCoffEE APE: Can't get process information (error %ld); not loading in this application\n", err);
|
---|
101 | return;
|
---|
102 | #endif
|
---|
103 | } else {
|
---|
104 | ICapeprintf("ICeCoffEE APE: got process attributes = 0x%lx\n", info.Attributes);
|
---|
105 | if (info.Attributes & (kCPSBGOnlyAttr | kCPSUIElementAttr | kCPSFullScreenAttr)) {
|
---|
106 | ICapeprintf("ICeCoffEE APE: not loading as this application is background-only\n");
|
---|
107 | return;
|
---|
108 | }
|
---|
109 | }
|
---|
110 | ICCF_shouldLoad = true;
|
---|
111 | }
|
---|
112 |
|
---|
113 | Boolean ICCF_CFBundleIDMatches(CFStringRef bundleID, CFStringRef test) {
|
---|
114 | return CFStringCompare(bundleID, test, kCFCompareCaseInsensitive) == kCFCompareEqualTo;
|
---|
115 | }
|
---|
116 |
|
---|
117 | void APEBundleMainLateLoad(CFBundleRef inBundle, CFStringRef inAPEToolsApplicationID)
|
---|
118 | {
|
---|
119 | if (!ICCF_shouldLoad) return;
|
---|
120 |
|
---|
121 | ICCF_bundle = CFBundleGetBundleWithIdentifier(kICBundleIdentifier);
|
---|
122 | CFStringRef bundleID = CFBundleGetIdentifier(CFBundleGetMainBundle());
|
---|
123 | BOOL shouldLoadInNSTextView = YES;
|
---|
124 |
|
---|
125 | // XXX handle patching error return from ICCF_PatchMethod
|
---|
126 | if (bundleID != NULL) {
|
---|
127 | if (ICCF_CFBundleIDMatches(bundleID, CFSTR("com.apple.xcode"))) {
|
---|
128 | ICCF_PatchMethod("PBXTextView", "ICeCoffEEMenuOnly", "ICeCoffEEMenuSuper", "menuForEvent:");
|
---|
129 | ICapeprintf("ICeCoffEE APE: loaded in PBXTextView for Xcode\n");
|
---|
130 | shouldLoadInNSTextView = NO;
|
---|
131 | } else if (ICCF_CFBundleIDMatches(bundleID, CFSTR("com.apple.terminal"))) {
|
---|
132 | ICCF_PatchMethod("TermSubview", "ICeCoffEEMenuOnly", "ICeCoffEEMenuSuper", "menuForEvent:") &&
|
---|
133 | ICCF_PatchMethod("TermSubview", "ICeCoffEETerminal", "ICECoffEETermSubviewSuper", "selectedRange") &&
|
---|
134 | ICCF_PatchMethod("TermSubview", "ICeCoffEETerminal", "ICECoffEETermSubviewSuper", "attributedSubstringFromRange:") &&
|
---|
135 | ICCF_PatchMethod("TermSubview", "ICeCoffEETerminal", "ICECoffEETermSubviewSuper", "mouseDown:") &&
|
---|
136 | ICCF_PatchMethod("TermSubview", "ICeCoffEETerminal", "ICECoffEETermSubviewSuper", "mouseUp:") &&
|
---|
137 | ICCF_PatchMethod("TermSubview", "ICeCoffEETerminal", "ICECoffEETermSubviewSuper", "draggingEntered:") &&
|
---|
138 | ICCF_PatchMethod("TermSubview", "ICeCoffEETerminal", "ICECoffEETermSubviewSuper", "_optionClickEvent:::");
|
---|
139 | ICapeprintf("ICeCoffEE APE: loaded in TermSubview for Terminal\n");
|
---|
140 | } else if (ICCF_CFBundleIDMatches(bundleID, CFSTR("org.mozilla.camino"))) {
|
---|
141 | ICCF_PatchMethod("ChildView", "ICeCoffEEMenuOnly", "ICeCoffEEMenuSuper", "menuForEvent:");
|
---|
142 | ICapeprintf("ICeCoffEE APE: loaded in ChildView for Camino\n");
|
---|
143 | }
|
---|
144 | }
|
---|
145 |
|
---|
146 | ICCF_PatchMethod("WebHTMLView", "ICeCoffEEWebKit", "ICeCoffEEWebKitSuper", "mouseUp:") &&
|
---|
147 | ICCF_PatchMethod("WebHTMLView", "ICeCoffEEWebKit", "ICeCoffEEWebKitSuper", "mouseDown:") &&
|
---|
148 | ICCF_PatchMethod("WebHTMLView", "ICeCoffEEWebKit", "ICeCoffEEWebKitSuper", "menuForEvent:");
|
---|
149 | ICapeprintf("ICeCoffEE APE: loaded in WebHTMLView for WebKit/Safari\n");
|
---|
150 |
|
---|
151 | if (shouldLoadInNSTextView) {
|
---|
152 | ICCF_PatchMethod("NSTextView", "ICeCoffEE", "ICeCoffEESuper", "mouseDown:") &&
|
---|
153 | ICCF_PatchMethod("NSTextView", "ICeCoffEE", "ICeCoffEESuper", "menuForEvent:");
|
---|
154 | ICapeprintf("ICeCoffEE APE: loaded generic NSTextView support\n");
|
---|
155 | }
|
---|
156 |
|
---|
157 | ICCF_ReloadPrefs();
|
---|
158 |
|
---|
159 | return;
|
---|
160 | }
|
---|
161 |
|
---|
162 | // We define APEBundleMessage so we can receive messages from our preference pane. We actually only know the Refresh message that instructs our APE module to reload the preferences. Why do we reload preferences here and not do it in our patch? Well, although you can poll CFPreferences in every patch call, that'll result in some performance loss - we better cache the settings in a static variable and refresh them from prefs only when needed.
|
---|
163 | OSStatus APEBundleMessage(CFStringRef message,CFDataRef inData,CFDataRef *outData)
|
---|
164 | {
|
---|
165 | ICapeprintf("ICeCoffEE APE: message '%@' (inData = %@)\n", message, inData);
|
---|
166 |
|
---|
167 | if (CFStringCompare(message, kICPreferencesChanged, 0) == kCFCompareEqualTo)
|
---|
168 | { // request to reload prefs from our preference pane
|
---|
169 | ICCF_ReloadPrefs();
|
---|
170 | }
|
---|
171 |
|
---|
172 | return noErr;
|
---|
173 | }
|
---|
174 |
|
---|
175 | Boolean ICCF_GetBooleanPref(CFStringRef prefKey, Boolean defaultValue) {
|
---|
176 | Boolean keyExists;
|
---|
177 | Boolean value = CFPreferencesGetAppBooleanValue(prefKey, kICBundleIdentifier, &keyExists);
|
---|
178 | if (keyExists) return value;
|
---|
179 | CFPreferencesSetAppValue(prefKey, defaultValue ? kCFBooleanTrue : kCFBooleanFalse, kICBundleIdentifier);
|
---|
180 | return defaultValue;
|
---|
181 | }
|
---|
182 |
|
---|
183 | CFIndex ICCF_GetCFIndexPref(CFStringRef prefKey, CFIndex defaultValue) {
|
---|
184 | Boolean keyExists;
|
---|
185 | CFIndex value = CFPreferencesGetAppIntegerValue(prefKey, kICBundleIdentifier, &keyExists);
|
---|
186 | if (keyExists) return value;
|
---|
187 | CFNumberRef defaultValueNumber = CFNumberCreate(NULL, kCFNumberCFIndexType, &defaultValue);
|
---|
188 | CFPreferencesSetAppValue(prefKey, defaultValueNumber, kICBundleIdentifier);
|
---|
189 | CFRelease(defaultValueNumber);
|
---|
190 | return defaultValue;
|
---|
191 | }
|
---|
192 |
|
---|
193 | void ICCF_GetCFTypePref(CFStringRef prefKey, CFTypeRef *val, CFTypeID type) {
|
---|
194 | if (*val != NULL) {
|
---|
195 | CFRelease(*val);
|
---|
196 | *val = NULL;
|
---|
197 | }
|
---|
198 |
|
---|
199 | *val = CFPreferencesCopyAppValue(prefKey, kICBundleIdentifier);
|
---|
200 |
|
---|
201 | if (*val == NULL) return;
|
---|
202 |
|
---|
203 | if (CFGetTypeID(*val) != type) {
|
---|
204 | CFRelease(*val);
|
---|
205 | *val = NULL;
|
---|
206 | return;
|
---|
207 | }
|
---|
208 | }
|
---|
209 |
|
---|
210 | // to test: defaults write net.sabi.ICeCoffEE "Excluded Applications" -array '{CFBundleID = "com.apple.projectbuilder"; }' '{CFBundleID = "net.sabi.Pester"; }' '{CFBundleID = "com.apple.foobar"; }'
|
---|
211 | static void ICCF_MigratePrefs() {
|
---|
212 |
|
---|
213 | CFArrayRef prefExcludedApps = CFPreferencesCopyAppValue(kIC12PrefExcluded, kICBundleIdentifier);
|
---|
214 |
|
---|
215 | if (prefExcludedApps == NULL) return;
|
---|
216 |
|
---|
217 | if (CFGetTypeID(prefExcludedApps) != CFArrayGetTypeID()) {
|
---|
218 | CFRelease(prefExcludedApps);
|
---|
219 | return;
|
---|
220 | }
|
---|
221 |
|
---|
222 | CFMutableArrayRef excludedApps = CFArrayCreateMutableCopy(NULL, 0, prefExcludedApps);
|
---|
223 | CFRelease(prefExcludedApps); prefExcludedApps = NULL;
|
---|
224 |
|
---|
225 | ICapeprintf("Excluded apps: %@\n", excludedApps);
|
---|
226 |
|
---|
227 | CFIndex excludedAppCount = CFArrayGetCount(excludedApps);
|
---|
228 | CFIndex excludedAppIndex;
|
---|
229 |
|
---|
230 | CFDictionaryRef excludedAppSpecifiers = NULL;
|
---|
231 | CFStringRef excludedAppBundleID = NULL;
|
---|
232 | CFURLRef excludedAppURL = NULL;
|
---|
233 | CFBundleRef excludedAppBundle = NULL;
|
---|
234 |
|
---|
235 | BOOL postponeMigrationForApp;
|
---|
236 |
|
---|
237 | for (excludedAppIndex = excludedAppCount - 1 ; excludedAppIndex >= 0 ; excludedAppIndex--) {
|
---|
238 | if ( (excludedAppSpecifiers = CFArrayGetValueAtIndex(excludedApps, excludedAppIndex)) == NULL || CFGetTypeID(excludedAppSpecifiers) != CFDictionaryGetTypeID()) {
|
---|
239 | CFArrayRemoveValueAtIndex(excludedApps, excludedAppIndex);
|
---|
240 | continue;
|
---|
241 | }
|
---|
242 |
|
---|
243 | postponeMigrationForApp = NO;
|
---|
244 | if ( (excludedAppBundleID = CFDictionaryGetValue(excludedAppSpecifiers, kIC12PrefExcludedAppSpecifierBundleID)) != NULL
|
---|
245 | && CFGetTypeID(excludedAppBundleID) == CFStringGetTypeID()
|
---|
246 | && CFStringCompare(excludedAppBundleID, CFSTR("com.apple.projectbuilder"), kCFCompareCaseInsensitive) != kCFCompareEqualTo) {
|
---|
247 | if (LSFindApplicationForInfo(kLSUnknownCreator, excludedAppBundleID, NULL, NULL, &excludedAppURL) == noErr) {
|
---|
248 | excludedAppBundle = CFBundleCreate(NULL, excludedAppURL);
|
---|
249 | if (excludedAppBundle != NULL) {
|
---|
250 | APEToolsAddToExcludeList(kICBundleIdentifier, excludedAppBundle, CFSTR("Migrated from ICeCoffEE 1.0-1.2"), NO);
|
---|
251 | } else postponeMigrationForApp = YES; // can't create bundle
|
---|
252 | } else postponeMigrationForApp = YES; // can't find app
|
---|
253 | }
|
---|
254 | // don't release excludedAppSpecifiers, used Get
|
---|
255 | // don't release excludedAppBundleID, used Get
|
---|
256 | if (excludedAppURL != NULL) { CFRelease(excludedAppURL); excludedAppURL = NULL; }
|
---|
257 | if (excludedAppBundle != NULL) { CFRelease(excludedAppBundle); excludedAppBundle = NULL; }
|
---|
258 | if (!postponeMigrationForApp) CFArrayRemoveValueAtIndex(excludedApps, excludedAppIndex);
|
---|
259 | }
|
---|
260 |
|
---|
261 | ICapeprintf("Excluded apps remaining: %@\n", excludedApps);
|
---|
262 | CFPreferencesSetAppValue(kIC12PrefExcluded, (CFArrayGetCount(excludedApps) == 0 ? NULL : excludedApps), kICBundleIdentifier);
|
---|
263 | CFRelease(excludedApps);
|
---|
264 |
|
---|
265 | CFPreferencesAppSynchronize(kICBundleIdentifier);
|
---|
266 | }
|
---|
267 |
|
---|
268 | static void ICCF_ReloadPrefs() {
|
---|
269 | CFPreferencesAppSynchronize(kICBundleIdentifier);
|
---|
270 |
|
---|
271 | ICCF_prefs.commandClickEnabled = ICCF_GetBooleanPref(kICCommandClickEnabled, YES);
|
---|
272 | ICCF_prefs.textBlinkEnabled = ICCF_GetBooleanPref(kICTextBlinkEnabled, YES);
|
---|
273 | ICCF_prefs.textBlinkCount = ICCF_GetCFIndexPref(kICTextBlinkCount, 3);
|
---|
274 | ICCF_prefs.servicesInContextualMenu = ICCF_GetBooleanPref(kICServicesInContextualMenu, YES);
|
---|
275 | ICCF_prefs.servicesInMenuBar = ICCF_GetBooleanPref(kICServicesInMenuBar, NO);
|
---|
276 | ICCF_GetCFTypePref(kICServiceOptions, (CFTypeRef *)&ICCF_prefs.serviceOptions, CFDictionaryGetTypeID());
|
---|
277 | ICCF_prefs.terminalRequireOptionForSelfDrag = ICCF_GetBooleanPref(kICTerminalRequireOptionForSelfDrag, NO);
|
---|
278 | ICCF_prefs.errorSoundEnabled = ICCF_GetBooleanPref(kICErrorSoundEnabled, NO);
|
---|
279 | ICCF_prefs.errorDialogEnabled = ICCF_GetBooleanPref(kICErrorDialogEnabled, YES);
|
---|
280 |
|
---|
281 | CFPreferencesAppSynchronize(kICBundleIdentifier);
|
---|
282 |
|
---|
283 | ICCF_AddRemoveServicesMenu();
|
---|
284 | }
|
---|