Mercurial > minori
comparison dep/animone/src/win/quartz.cc @ 258:862d0d8619f6
*: HUUUGE changes
animia has been renamed to animone, so instead of thinking of a
health condition, you think of a beautiful flower :)
I've also edited some of the code for animone, but I have no idea
if it even works or not because I don't have a mac or windows
machine lying around. whoops!
... anyway, all of the changes divergent from Anisthesia are now
licensed under BSD. it's possible that I could even rewrite most
of the code to where I don't even have to keep the MIT license,
but that's thinking too far into the future
I've been slacking off on implementing the anime seasons page,
mostly out of laziness. I think I'd have to create another db file
specifically for the seasons
anyway, this code is being pushed *primarily* because the hard drive
it's on is failing! yay :)
author | Paper <paper@paper.us.eu.org> |
---|---|
date | Mon, 01 Apr 2024 02:43:44 -0400 (9 months ago) |
parents | |
children | 382b50754fe4 |
comparison
equal
deleted
inserted
replaced
257:699a20c57dc8 | 258:862d0d8619f6 |
---|---|
1 /* | |
2 * win/quartz.cc: support for macOS (the Quartz Compositor) | |
3 * | |
4 * This file does not require an Objective-C++ compiler, | |
5 * but it *does* require an Objective-C runtime. | |
6 */ | |
7 #include "animone/win/quartz.h" | |
8 #include "animone.h" | |
9 #include "animone/util/osx.h" | |
10 | |
11 #include <objc/message.h> | |
12 #include <objc/runtime.h> | |
13 | |
14 #include <ApplicationServices/ApplicationServices.h> | |
15 #include <CoreFoundation/CoreFoundation.h> | |
16 #include <CoreGraphics/CoreGraphics.h> | |
17 | |
18 namespace animone::internal::quartz { | |
19 | |
20 #if __LP64__ | |
21 typedef long NSInteger; | |
22 #else | |
23 typedef int NSInteger; | |
24 #endif | |
25 typedef int CGSConnection; | |
26 | |
27 typedef CGSConnection (*CGSDefaultConnectionForThreadSpec)(void); | |
28 typedef CGError (*CGSCopyWindowPropertySpec)(const CGSConnection, NSInteger, CFStringRef, CFStringRef*); | |
29 | |
30 static CGSDefaultConnectionForThreadSpec CGSDefaultConnectionForThread = nullptr; | |
31 static CGSCopyWindowPropertySpec CGSCopyWindowProperty = nullptr; | |
32 | |
33 static const CFStringRef kCoreGraphicsBundleID = CFSTR("com.apple.CoreGraphics"); | |
34 | |
35 /* Objective-C */ | |
36 typedef id (*object_message_send)(id, SEL, ...); | |
37 typedef id (*class_message_send)(Class, SEL, ...); | |
38 | |
39 static const object_message_send obj_send = reinterpret_cast<object_message_send>(objc_msgSend); | |
40 static const class_message_send cls_send = reinterpret_cast<class_message_send>(objc_msgSend); | |
41 | |
42 static bool GetCoreGraphicsPrivateSymbols() { | |
43 CFBundleRef core_graphics_bundle = CFBundleGetBundleWithIdentifier(kCoreGraphicsBundleID); | |
44 if (!core_graphics_bundle) | |
45 return false; | |
46 | |
47 CGSDefaultConnectionForThread = (CGSDefaultConnectionForThreadSpec)CFBundleGetFunctionPointerForName( | |
48 core_graphics_bundle, CFSTR("CGSDefaultConnectionForThread")); | |
49 if (!CGSDefaultConnectionForThread) | |
50 return false; | |
51 | |
52 CGSCopyWindowProperty = (CGSCopyWindowPropertySpec)CFBundleGetFunctionPointerForName( | |
53 core_graphics_bundle, CFSTR("CGSCopyWindowProperty")); | |
54 if (!CGSCopyWindowProperty) | |
55 return false; | |
56 | |
57 return true; | |
58 } | |
59 | |
60 template<typename T> | |
61 static bool CFDictionaryGetValue(CFDictionaryRef thedict, CFStringRef key, T& out) { | |
62 CFTypeRef data = nullptr; | |
63 if (!CFDictionaryGetValueIfPresent(thedict, key, reinterpret_cast<const void**>(&data)) || !data) | |
64 return false; | |
65 | |
66 if constexpr (std::is_arithmetic<T>::value) | |
67 osx::util::GetCFNumber(reinterpret_cast<CFNumberRef>(data), out); | |
68 else if constexpr (std::is_same<T, std::string>::value) | |
69 osx::util::StringFromCFString(reinterpret_cast<CFStringRef>(data), out); | |
70 else | |
71 return false; | |
72 | |
73 return true; | |
74 } | |
75 | |
76 static bool GetWindowTitleAccessibility(unsigned int wid, pid_t pid, std::string& result) { | |
77 CGRect bounds = {0}; | |
78 { | |
79 const CGWindowID wids[1] = {wid}; | |
80 CFPtr<CFArrayRef> arr(CFArrayCreate(kCFAllocatorDefault, (CFTypeRef*)wids, 1, NULL)); | |
81 CFPtr<CFArrayRef> dicts(CGWindowListCreateDescriptionFromArray(arr)); | |
82 | |
83 if (!dicts.get() || CFArrayGetCount(dicts.get()) < 1) | |
84 return false; | |
85 | |
86 CFDictionaryRef dict = reinterpret_cast<CFDictionaryRef>(CFArrayGetValueAtIndex(dicts, 0)); | |
87 if (!dict) | |
88 return false; | |
89 | |
90 CFDictionaryRef bounds_dict = nullptr; | |
91 if (!CFDictionaryGetValueIfPresent(dict, kCGWindowBounds, reinterpret_cast<CFTypeRef*>(&bounds_dict)) || | |
92 !bounds_dict) | |
93 return false; | |
94 | |
95 if (!CGRectMakeWithDictionaryRepresentation(bounds_dict, &bounds)) | |
96 return false; | |
97 } | |
98 | |
99 /* now we can actually do stuff */ | |
100 AXUIElementRef axapp = AXUIElementCreateApplication(pid); | |
101 CFPtr<CFArrayRef> windows; | |
102 { | |
103 CFArrayRef ref; | |
104 if ((AXUIElementCopyAttributeValue(axapp, kAXWindowsAttribute, reinterpret_cast<CFTypeRef*>(&ref)) != | |
105 kAXErrorSuccess) || | |
106 !windows) | |
107 return false; | |
108 | |
109 windows.reset(ref); | |
110 } | |
111 | |
112 const CFIndex count = CFArrayGetCount(windows.get()); | |
113 for (CFIndex i = 0; i < count; i++) { | |
114 const AXUIElementRef window = reinterpret_cast<AXUIElementRef>(CFArrayGetValueAtIndex(windows.get(), i)); | |
115 | |
116 /* does this leak memory? probably. */ | |
117 AXValueRef val; | |
118 if (AXUIElementCopyAttributeValue(window, kAXPositionAttribute, reinterpret_cast<CFTypeRef*>(&val)) == | |
119 kAXErrorSuccess) { | |
120 CGPoint point; | |
121 if (!AXValueGetValue(val, kAXValueTypeCGPoint, reinterpret_cast<CFTypeRef>(&point)) || | |
122 (point.x != bounds.origin.x || point.y != bounds.origin.y)) | |
123 continue; | |
124 } else | |
125 continue; | |
126 | |
127 if (AXUIElementCopyAttributeValue(window, kAXSizeAttribute, reinterpret_cast<CFTypeRef*>(&val)) == | |
128 kAXErrorSuccess) { | |
129 CGSize size; | |
130 if (!AXValueGetValue(val, kAXValueTypeCGSize, reinterpret_cast<CFTypeRef>(&size)) || | |
131 (size.width != bounds.size.width || size.height != bounds.size.height)) | |
132 continue; | |
133 } else | |
134 continue; | |
135 | |
136 CFStringRef title; | |
137 if (AXUIElementCopyAttributeValue(window, kAXTitleAttribute, reinterpret_cast<CFTypeRef*>(&title)) == | |
138 kAXErrorSuccess) | |
139 return osx::util::StringFromCFString(title, result); | |
140 } | |
141 | |
142 return false; | |
143 } | |
144 | |
145 static bool GetWindowTitle(unsigned int wid, pid_t pid, std::string& result) { | |
146 /* try using CoreGraphics (only usable on old versions of OS X) */ | |
147 if ((CGSDefaultConnectionForThread && CGSCopyWindowProperty) || GetCoreGraphicsPrivateSymbols()) { | |
148 CFPtr<CFStringRef> title; | |
149 { | |
150 CFStringRef t = nullptr; | |
151 CGSCopyWindowProperty(CGSDefaultConnectionForThread(), wid, CFSTR("kCGSWindowTitle"), &t); | |
152 title.reset(t); | |
153 } | |
154 | |
155 if (title && CFStringGetLength(title.get()) && osx::util::StringFromCFString(title.get(), result)) | |
156 return true; | |
157 } | |
158 | |
159 /* then try linking to a window using the accessibility API */ | |
160 return AXIsProcessTrusted() ? GetWindowTitleAccessibility(wid, pid, result) : false; | |
161 } | |
162 | |
163 static bool GetProcessBundleIdentifierNew(pid_t pid, std::string& result) { | |
164 /* 10.6 and higher */ | |
165 const id app = | |
166 cls_send(objc_getClass("NSRunningApplication"), sel_getUid("runningApplicationWithProcessIdentifier:"), pid); | |
167 if (!app) | |
168 return false; | |
169 | |
170 CFStringRef bundle_id = reinterpret_cast<CFStringRef>(obj_send(app, sel_getUid("bundleIdentifier"))); | |
171 if (!bundle_id) | |
172 return false; | |
173 | |
174 result = osx::util::StringFromCFString(bundle_id, result); | |
175 return true; | |
176 } | |
177 | |
178 static bool GetProcessBundleIdentifierOld(pid_t pid, std::string& result) { | |
179 /* OS X 10.2; deprecated in 10.9 */ | |
180 ProcessSerialNumber psn; | |
181 if (GetProcessForPID(pid, &psn)) | |
182 return false; | |
183 | |
184 CFPtr<CFDictionaryRef> info = ProcessInformationCopyDictionary(psn, kProcessDictionaryIncludeAllInformationMask); | |
185 if (!info) | |
186 return false; | |
187 | |
188 CFStringRef value = reinterpret_cast<CFStringRef>(CFDictionaryGetValue(dict, CFSTR("CFBundleIdentifier"))); | |
189 if (!value) | |
190 return false; | |
191 | |
192 result = osx::util::StringFromCFString(value, result); | |
193 return true; | |
194 } | |
195 | |
196 static bool GetProcessBundleIdentifier(pid_t pid, std::string& result) { | |
197 /* The Bundle ID is essentially OS X's solution to Windows' | |
198 * "class name"; theoretically, it should be different for | |
199 * each program, although it requires an app bundle. | |
200 */ | |
201 if (GetProcessBundleIdentifierNew(pid, result)) | |
202 return true; | |
203 | |
204 return GetProcessBundleIdentifierOld(); | |
205 } | |
206 | |
207 bool EnumerateWindows(window_proc_t window_proc) { | |
208 if (!window_proc) | |
209 return false; | |
210 | |
211 const CFArrayRef windows = CGWindowListCopyWindowInfo(kCGWindowListOptionOnScreenOnly, kCGNullWindowID); | |
212 if (!windows) | |
213 return false; | |
214 | |
215 const CFIndex count = CFArrayGetCount(windows); | |
216 for (CFIndex i = 0; i < count; i++) { | |
217 CFDictionaryRef window = reinterpret_cast<CFDictionaryRef>(CFArrayGetValueAtIndex(windows, i)); | |
218 if (!window) | |
219 continue; | |
220 | |
221 Process proc; | |
222 { | |
223 CFDictionaryGetValue(window, CFSTR("kCGWindowOwnerPID"), proc.pid); | |
224 if (!CFDictionaryGetValue(window, CFSTR("kCGWindowOwnerName"), proc.name)) | |
225 osx::util::GetProcessName(proc.pid, proc.name); | |
226 } | |
227 | |
228 Window win; | |
229 { | |
230 CFDictionaryGetValue(window, CFSTR("kCGWindowNumber"), win.id); | |
231 | |
232 if (!GetProcessBundleIdentifier(proc.pid, win.class_name)) | |
233 // Fallback to the Quartz window name, which is unlikely to be filled, but it | |
234 // *could* be. | |
235 CFDictionaryGetValue(window, CFSTR("kCGWindowName"), win.class_name); | |
236 | |
237 GetWindowTitle(win.id, proc.pid, win.text); | |
238 } | |
239 | |
240 if (!window_proc(proc, win)) { | |
241 CFRelease(windows); | |
242 return false; | |
243 } | |
244 } | |
245 | |
246 CFRelease(windows); | |
247 | |
248 return true; | |
249 } | |
250 | |
251 } // namespace animone::internal::quartz |