// Copyright (c) 2009 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include #include "base/gfx/rect.h" #include "webkit/glue/plugins/fake_plugin_window_tracker_mac.h" namespace webkit_glue { void NotifyBrowserOfPluginSelectWindow(uint32 window_id, CGRect bounds, bool modal); void NotifyBrowserOfPluginShowWindow(uint32 window_id, CGRect bounds, bool modal); void NotifyBrowserOfPluginHideWindow(uint32 window_id, CGRect bounds); void NotifyBrowserOfPluginDisposeWindow(uint32 window_id, CGRect bounds); } // namespace webkit_glue // The process that was frontmost when a plugin created a new window; generally // we expect this to be the browser UI process. static ProcessSerialNumber g_saved_front_process = { 0, 0 }; // Bring the plugin process to the front so that the user can see it. // TODO: Make this an IPC to order the plugin process above the browser // process but not necessarily the frontmost. static void SwitchToPluginProcess() { ProcessSerialNumber this_process, front_process; if (GetCurrentProcess(&this_process) != noErr) return; if (GetFrontProcess(&front_process) != noErr) return; Boolean matched = false; if (SameProcess(&this_process, &front_process, &matched) != noErr) return; if (!matched) { g_saved_front_process = front_process; SetFrontProcess(&this_process); } } // If the plugin process is still the front process, bring the prior // front process (normally this will be the browser process) back to // the front. // TODO: Make this an IPC message so that the browser can properly // reactivate the window. static void SwitchToSavedProcess() { ProcessSerialNumber this_process, front_process; if (GetCurrentProcess(&this_process) != noErr) return; if (GetFrontProcess(&front_process) != noErr) return; Boolean matched = false; if (SameProcess(&this_process, &front_process, &matched) != noErr) return; if (matched) { SetFrontProcess(&g_saved_front_process); } } // Checks to see if there are any plugin-opened windows still showing, and if // not reactivates the saved process. // Should be called after any window has been closed. static void MaybeReactivateSavedProcess() { bool window_is_visible = false; WindowRef window = GetWindowList(); while (window != NULL) { if (IsWindowVisible(window)) { window_is_visible = true; break; } window = GetNextWindow(window); } if (!window_is_visible) SwitchToSavedProcess(); } // Returns true if the given window is modal. static bool IsModalWindow(WindowRef window) { WindowModality modality = kWindowModalityNone; WindowRef modal_target = NULL; OSStatus status = GetWindowModality(window, &modality, &modal_target); return (status == noErr) && (modality != kWindowModalityNone); } static bool IsContainingWindowActive(const WebPluginDelegateImpl* delegate) { // TODO(stuartmorgan): We need a way to find out if the delegate is in the // active window; for now we just lie and always say yes so plugins don't // throw events away. return true; } #pragma mark - static Boolean ChromePluginIsWindowActive(WindowRef window) { const WebPluginDelegateImpl* delegate = FakePluginWindowTracker::SharedInstance()->GetDelegateForFakeWindow( window); return delegate ? IsContainingWindowActive(delegate) : IsWindowActive(window); } static Boolean ChromePluginIsWindowHilited(WindowRef window) { const WebPluginDelegateImpl* delegate = FakePluginWindowTracker::SharedInstance()->GetDelegateForFakeWindow( window); return delegate ? IsContainingWindowActive(delegate) : IsWindowHilited(window); } static CGRect CGRectForWindow(WindowRef window) { CGRect bounds = { { 0, 0 }, { 0, 0 } }; HIWindowGetBounds(window, kWindowContentRgn, kHICoordSpace72DPIGlobal, &bounds); return bounds; } static void ChromePluginSelectWindow(WindowRef window) { SwitchToPluginProcess(); SelectWindow(window); webkit_glue::NotifyBrowserOfPluginSelectWindow(HIWindowGetCGWindowID(window), CGRectForWindow(window), IsModalWindow(window)); } static void ChromePluginShowWindow(WindowRef window) { SwitchToPluginProcess(); ShowWindow(window); webkit_glue::NotifyBrowserOfPluginShowWindow(HIWindowGetCGWindowID(window), CGRectForWindow(window), IsModalWindow(window)); } static void ChromePluginDisposeWindow(WindowRef window) { CGWindowID window_id = HIWindowGetCGWindowID(window); CGRect window_rect = CGRectForWindow(window); DisposeWindow(window); webkit_glue::NotifyBrowserOfPluginDisposeWindow(window_id, window_rect); MaybeReactivateSavedProcess(); } static void ChromePluginHideWindow(WindowRef window) { CGWindowID window_id = HIWindowGetCGWindowID(window); CGRect window_rect = CGRectForWindow(window); HideWindow(window); webkit_glue::NotifyBrowserOfPluginHideWindow(window_id, window_rect); MaybeReactivateSavedProcess(); } static void ChromePluginDisposeDialog(DialogRef dialog) { WindowRef window = GetDialogWindow(dialog); CGWindowID window_id = HIWindowGetCGWindowID(window); CGRect window_rect = CGRectForWindow(window); DisposeDialog(dialog); webkit_glue::NotifyBrowserOfPluginDisposeWindow(window_id, window_rect); MaybeReactivateSavedProcess(); } #pragma mark - struct interpose_substitution { const void* replacement; const void* original; }; #define INTERPOSE_FUNCTION(function) \ { reinterpret_cast(ChromePlugin##function), \ reinterpret_cast(function) } __attribute__((used)) static const interpose_substitution substitutions[] __attribute__((section("__DATA, __interpose"))) = { INTERPOSE_FUNCTION(IsWindowActive), INTERPOSE_FUNCTION(IsWindowHilited), INTERPOSE_FUNCTION(SelectWindow), INTERPOSE_FUNCTION(ShowWindow), INTERPOSE_FUNCTION(DisposeWindow), INTERPOSE_FUNCTION(HideWindow), INTERPOSE_FUNCTION(DisposeDialog), };