• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2010 Google Inc. All Rights Reserved
2 
3 
4 #include "talk/base/macwindowpicker.h"
5 
6 #include <ApplicationServices/ApplicationServices.h>
7 #include <CoreFoundation/CoreFoundation.h>
8 #include <dlfcn.h>
9 
10 #include "talk/base/logging.h"
11 #include "talk/base/macutils.h"
12 
13 namespace talk_base {
14 
15 static const char* kCoreGraphicsName =
16     "/System/Library/Frameworks/ApplicationServices.framework/Frameworks/"
17     "CoreGraphics.framework/CoreGraphics";
18 
19 static const char* kWindowListCopyWindowInfo = "CGWindowListCopyWindowInfo";
20 static const char* kWindowListCreateDescriptionFromArray =
21     "CGWindowListCreateDescriptionFromArray";
22 
23 // Function pointer for holding the CGWindowListCopyWindowInfo function.
24 typedef CFArrayRef(*CGWindowListCopyWindowInfoProc)(CGWindowListOption,
25                                                     CGWindowID);
26 
27 // Function pointer for holding the CGWindowListCreateDescriptionFromArray
28 // function.
29 typedef CFArrayRef(*CGWindowListCreateDescriptionFromArrayProc)(CFArrayRef);
30 
MacWindowPicker()31 MacWindowPicker::MacWindowPicker() : lib_handle_(NULL), get_window_list_(NULL),
32                                      get_window_list_desc_(NULL) {
33 }
34 
~MacWindowPicker()35 MacWindowPicker::~MacWindowPicker() {
36   if (lib_handle_ != NULL) {
37     dlclose(lib_handle_);
38   }
39 }
40 
Init()41 bool MacWindowPicker::Init() {
42   // TODO: If this class grows to use more dynamically functions
43   // from the CoreGraphics framework, consider using
44   // talk/base/latebindingsymboltable.h.
45   lib_handle_ = dlopen(kCoreGraphicsName, RTLD_NOW);
46   if (lib_handle_ == NULL) {
47     LOG(LS_ERROR) << "Could not load CoreGraphics";
48     return false;
49   }
50 
51   get_window_list_ = dlsym(lib_handle_, kWindowListCopyWindowInfo);
52   get_window_list_desc_ =
53       dlsym(lib_handle_, kWindowListCreateDescriptionFromArray);
54   if (get_window_list_ == NULL || get_window_list_desc_ == NULL) {
55     // The CGWindowListCopyWindowInfo and the
56     // CGWindowListCreateDescriptionFromArray functions was introduced
57     // in Leopard(10.5) so this is a normal failure on Tiger.
58     LOG(LS_INFO) << "Failed to load Core Graphics symbols";
59     dlclose(lib_handle_);
60     lib_handle_ = NULL;
61     return false;
62   }
63 
64   return true;
65 }
66 
IsVisible(const WindowId & id)67 bool MacWindowPicker::IsVisible(const WindowId& id) {
68   // Init if we're not already inited.
69   if (get_window_list_desc_ == NULL && !Init()) {
70     return false;
71   }
72   CGWindowID ids[1];
73   ids[0] = id.id();
74   CFArrayRef window_id_array =
75       CFArrayCreate(NULL, reinterpret_cast<const void **>(&ids), 1, NULL);
76 
77   CFArrayRef window_array =
78       reinterpret_cast<CGWindowListCreateDescriptionFromArrayProc>(
79           get_window_list_desc_)(window_id_array);
80   if (window_array == NULL || 0 == CFArrayGetCount(window_array)) {
81     // Could not find the window. It might have been closed.
82     LOG(LS_INFO) << "Window not found";
83     CFRelease(window_id_array);
84     return false;
85   }
86 
87   CFDictionaryRef window = reinterpret_cast<CFDictionaryRef>(
88       CFArrayGetValueAtIndex(window_array, 0));
89   CFBooleanRef is_visible = reinterpret_cast<CFBooleanRef>(
90       CFDictionaryGetValue(window, kCGWindowIsOnscreen));
91 
92   // Check that the window is visible. If not we might crash.
93   bool visible = false;
94   if (is_visible != NULL) {
95     visible = CFBooleanGetValue(is_visible);
96   }
97   CFRelease(window_id_array);
98   CFRelease(window_array);
99   return visible;
100 }
101 
MoveToFront(const WindowId & id)102 bool MacWindowPicker::MoveToFront(const WindowId& id) {
103   // Init if we're not already initialized.
104   if (get_window_list_desc_ == NULL && !Init()) {
105     return false;
106   }
107   CGWindowID ids[1];
108   ids[0] = id.id();
109   CFArrayRef window_id_array =
110       CFArrayCreate(NULL, reinterpret_cast<const void **>(&ids), 1, NULL);
111 
112   CFArrayRef window_array =
113       reinterpret_cast<CGWindowListCreateDescriptionFromArrayProc>(
114           get_window_list_desc_)(window_id_array);
115   if (window_array == NULL || 0 == CFArrayGetCount(window_array)) {
116     // Could not find the window. It might have been closed.
117     LOG(LS_INFO) << "Window not found";
118     CFRelease(window_id_array);
119     return false;
120   }
121 
122   CFDictionaryRef window = reinterpret_cast<CFDictionaryRef>(
123       CFArrayGetValueAtIndex(window_array, 0));
124   CFStringRef window_name_ref = reinterpret_cast<CFStringRef>(
125       CFDictionaryGetValue(window, kCGWindowName));
126   CFNumberRef application_pid = reinterpret_cast<CFNumberRef>(
127       CFDictionaryGetValue(window, kCGWindowOwnerPID));
128 
129   int pid_val;
130   CFNumberGetValue(application_pid, kCFNumberIntType, &pid_val);
131   std::string window_name;
132   ToUtf8(window_name_ref, &window_name);
133 
134   // Build an applescript that sets the selected window to front
135   // within the application. Then set the application to front.
136   bool result = true;
137   std::stringstream ss;
138   ss << "tell application \"System Events\"\n"
139      << "set proc to the first item of (every process whose unix id is "
140      << pid_val
141      << ")\n"
142      << "tell proc to perform action \"AXRaise\" of window \""
143      << window_name
144      << "\"\n"
145      << "set the frontmost of proc to true\n"
146      << "end tell";
147   if (!RunAppleScript(ss.str())) {
148     // This might happen to for example X applications where the X
149     // server spawns of processes with their own PID but the X server
150     // is still registered as owner to the application windows. As a
151     // workaround, we put the X server process to front, meaning that
152     // all X applications will show up. The drawback with this
153     // workaround is that the application that we really wanted to set
154     // to front might be behind another X application.
155     ProcessSerialNumber psn;
156     pid_t pid = pid_val;
157     int res = GetProcessForPID(pid, &psn);
158     if (res != 0) {
159       LOG(LS_ERROR) << "Failed getting process for pid";
160       result = false;
161     }
162     res = SetFrontProcess(&psn);
163     if (res != 0) {
164       LOG(LS_ERROR) << "Failed setting process to front";
165       result = false;
166     }
167   }
168   CFRelease(window_id_array);
169   CFRelease(window_array);
170   return result;
171 }
172 
GetDesktopList(DesktopDescriptionList * descriptions)173 bool MacWindowPicker::GetDesktopList(DesktopDescriptionList* descriptions) {
174   const uint32_t kMaxDisplays = 128;
175   CGDirectDisplayID active_displays[kMaxDisplays];
176   uint32_t display_count = 0;
177 
178   CGError err = CGGetActiveDisplayList(kMaxDisplays,
179                                        active_displays,
180                                        &display_count);
181   if (err != kCGErrorSuccess) {
182     LOG_E(LS_ERROR, OS, err) << "Failed to enumerate the active displays.";
183     return false;
184   }
185   for (uint32_t i = 0; i < display_count; ++i) {
186     DesktopId id(active_displays[i], static_cast<int>(i));
187     // TODO: Figure out an appropriate desktop title.
188     DesktopDescription desc(id, "");
189     desc.set_primary(CGDisplayIsMain(id.id()));
190     descriptions->push_back(desc);
191   }
192   return display_count > 0;
193 }
194 
GetDesktopDimensions(const DesktopId & id,int * width,int * height)195 bool MacWindowPicker::GetDesktopDimensions(const DesktopId& id,
196                                            int* width,
197                                            int* height) {
198   *width = CGDisplayPixelsWide(id.id());
199   *height = CGDisplayPixelsHigh(id.id());
200   return true;
201 }
202 
GetWindowList(WindowDescriptionList * descriptions)203 bool MacWindowPicker::GetWindowList(WindowDescriptionList* descriptions) {
204   // Init if we're not already inited.
205   if (get_window_list_ == NULL && !Init()) {
206     return false;
207   }
208 
209   // Only get onscreen, non-desktop windows.
210   CFArrayRef window_array =
211       reinterpret_cast<CGWindowListCopyWindowInfoProc>(get_window_list_)(
212           kCGWindowListOptionOnScreenOnly | kCGWindowListExcludeDesktopElements,
213           kCGNullWindowID);
214   if (window_array == NULL) {
215     return false;
216   }
217 
218   // Check windows to make sure they have an id, title, and use window layer 0.
219   CFIndex i;
220   CFIndex count = CFArrayGetCount(window_array);
221   for (i = 0; i < count; ++i) {
222     CFDictionaryRef window = reinterpret_cast<CFDictionaryRef>(
223         CFArrayGetValueAtIndex(window_array, i));
224     CFStringRef window_title = reinterpret_cast<CFStringRef>(
225         CFDictionaryGetValue(window, kCGWindowName));
226     CFNumberRef window_id = reinterpret_cast<CFNumberRef>(
227         CFDictionaryGetValue(window, kCGWindowNumber));
228     CFNumberRef window_layer = reinterpret_cast<CFNumberRef>(
229         CFDictionaryGetValue(window, kCGWindowLayer));
230     if (window_title != NULL && window_id != NULL && window_layer != NULL) {
231       std::string title_str;
232       int id_val, layer_val;
233       ToUtf8(window_title, &title_str);
234       CFNumberGetValue(window_id, kCFNumberIntType, &id_val);
235       CFNumberGetValue(window_layer, kCFNumberIntType, &layer_val);
236 
237       // Discard windows without a title.
238       if (layer_val == 0 && title_str.length() > 0) {
239         WindowId id(static_cast<CGWindowID>(id_val));
240         WindowDescription desc(id, title_str);
241         descriptions->push_back(desc);
242       }
243     }
244   }
245 
246   CFRelease(window_array);
247   return true;
248 }
249 
250 }  // namespace talk_base
251