| OLD | NEW |
| 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #import <Cocoa/Cocoa.h> | 5 #import <Cocoa/Cocoa.h> |
| 6 #import <QuartzCore/QuartzCore.h> | 6 #import <QuartzCore/QuartzCore.h> |
| 7 | 7 |
| 8 #include "webkit/plugins/npapi/webplugin_delegate_impl.h" | 8 #include "webkit/plugins/npapi/webplugin_delegate_impl.h" |
| 9 | 9 |
| 10 #include <string> | 10 #include <string> |
| 11 #include <unistd.h> | 11 #include <unistd.h> |
| 12 #include <set> | 12 #include <set> |
| 13 | 13 |
| 14 #include "base/file_util.h" | 14 #include "base/file_util.h" |
| 15 #include "base/message_loop.h" | 15 #include "base/message_loop.h" |
| 16 #include "base/metrics/stats_counters.h" | 16 #include "base/metrics/stats_counters.h" |
| 17 #include "base/scoped_ptr.h" | 17 #include "base/scoped_ptr.h" |
| 18 #include "base/string_util.h" | 18 #include "base/string_util.h" |
| 19 #include "base/utf_string_conversions.h" | 19 #include "base/utf_string_conversions.h" |
| 20 #include "base/sys_info.h" |
| 20 #include "base/sys_string_conversions.h" | 21 #include "base/sys_string_conversions.h" |
| 21 #include "third_party/WebKit/Source/WebKit/chromium/public/WebInputEvent.h" | 22 #include "third_party/WebKit/Source/WebKit/chromium/public/WebInputEvent.h" |
| 22 #include "webkit/glue/webkit_glue.h" | 23 #include "webkit/glue/webkit_glue.h" |
| 23 #include "webkit/plugins/npapi/plugin_instance.h" | 24 #include "webkit/plugins/npapi/plugin_instance.h" |
| 24 #include "webkit/plugins/npapi/plugin_lib.h" | 25 #include "webkit/plugins/npapi/plugin_lib.h" |
| 25 #include "webkit/plugins/npapi/plugin_list.h" | 26 #include "webkit/plugins/npapi/plugin_list.h" |
| 26 #include "webkit/plugins/npapi/plugin_stream_url.h" | 27 #include "webkit/plugins/npapi/plugin_stream_url.h" |
| 27 #include "webkit/plugins/npapi/plugin_web_event_converter_mac.h" | 28 #include "webkit/plugins/npapi/plugin_web_event_converter_mac.h" |
| 28 #include "webkit/plugins/npapi/webplugin.h" | 29 #include "webkit/plugins/npapi/webplugin.h" |
| 29 #include "webkit/plugins/npapi/webplugin_accelerated_surface_mac.h" | 30 #include "webkit/plugins/npapi/webplugin_accelerated_surface_mac.h" |
| (...skipping 229 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 259 quirks_(0), | 260 quirks_(0), |
| 260 buffer_context_(NULL), | 261 buffer_context_(NULL), |
| 261 layer_(nil), | 262 layer_(nil), |
| 262 surface_(NULL), | 263 surface_(NULL), |
| 263 renderer_(nil), | 264 renderer_(nil), |
| 264 containing_window_has_focus_(false), | 265 containing_window_has_focus_(false), |
| 265 initial_window_focus_(false), | 266 initial_window_focus_(false), |
| 266 container_is_visible_(false), | 267 container_is_visible_(false), |
| 267 have_called_set_window_(false), | 268 have_called_set_window_(false), |
| 268 ime_enabled_(false), | 269 ime_enabled_(false), |
| 270 keyup_ignore_count_(0), |
| 269 external_drag_tracker_(new ExternalDragTracker()), | 271 external_drag_tracker_(new ExternalDragTracker()), |
| 270 handle_event_depth_(0), | 272 handle_event_depth_(0), |
| 271 first_set_window_call_(true), | 273 first_set_window_call_(true), |
| 272 plugin_has_focus_(false), | 274 plugin_has_focus_(false), |
| 273 has_webkit_focus_(false), | 275 has_webkit_focus_(false), |
| 274 containing_view_has_focus_(true), | 276 containing_view_has_focus_(true), |
| 275 creation_succeeded_(false) { | 277 creation_succeeded_(false) { |
| 276 memset(&window_, 0, sizeof(window_)); | 278 memset(&window_, 0, sizeof(window_)); |
| 277 #ifndef NP_NO_CARBON | 279 #ifndef NP_NO_CARBON |
| 278 memset(&np_cg_context_, 0, sizeof(np_cg_context_)); | 280 memset(&np_cg_context_, 0, sizeof(np_cg_context_)); |
| (...skipping 219 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 498 content_origin.y() != content_area_origin_.y()) { | 500 content_origin.y() != content_area_origin_.y()) { |
| 499 DLOG(WARNING) << "Stale plugin content area location: " | 501 DLOG(WARNING) << "Stale plugin content area location: " |
| 500 << content_area_origin_ << " instead of " | 502 << content_area_origin_ << " instead of " |
| 501 << content_origin; | 503 << content_origin; |
| 502 SetContentAreaOrigin(content_origin); | 504 SetContentAreaOrigin(content_origin); |
| 503 } | 505 } |
| 504 | 506 |
| 505 current_windowless_cursor_.GetCursorInfo(cursor_info); | 507 current_windowless_cursor_.GetCursorInfo(cursor_info); |
| 506 } | 508 } |
| 507 | 509 |
| 510 // Per the Cocoa Plugin IME spec, plugins shoudn't receive keydown or keyup |
| 511 // events while composition is in progress. Treat them as handled, however, |
| 512 // since IME is consuming them on behalf of the plugin. |
| 513 if ((event.type == WebInputEvent::KeyDown && ime_enabled_) || |
| 514 (event.type == WebInputEvent::KeyUp && keyup_ignore_count_)) { |
| 515 // Composition ends on a keydown, so ime_enabled_ will be false at keyup; |
| 516 // because the keydown wasn't sent to the plugin, the keyup shouldn't be |
| 517 // either (per the spec). |
| 518 if (event.type == WebInputEvent::KeyDown) |
| 519 ++keyup_ignore_count_; |
| 520 else |
| 521 --keyup_ignore_count_; |
| 522 return true; |
| 523 } |
| 524 |
| 508 #ifndef NP_NO_CARBON | 525 #ifndef NP_NO_CARBON |
| 509 if (instance()->event_model() == NPEventModelCarbon) { | 526 if (instance()->event_model() == NPEventModelCarbon) { |
| 510 #ifndef NP_NO_QUICKDRAW | 527 #ifndef NP_NO_QUICKDRAW |
| 511 if (instance()->drawing_model() == NPDrawingModelQuickDraw) { | 528 if (instance()->drawing_model() == NPDrawingModelQuickDraw) { |
| 512 if (quirks_ & PLUGIN_QUIRK_ALLOW_FASTER_QUICKDRAW_PATH) { | 529 if (quirks_ & PLUGIN_QUIRK_ALLOW_FASTER_QUICKDRAW_PATH) { |
| 513 // Mouse event handling doesn't work correctly in the fast path mode, | 530 // Mouse event handling doesn't work correctly in the fast path mode, |
| 514 // so any time we get a mouse event turn the fast path off, but set a | 531 // so any time we get a mouse event turn the fast path off, but set a |
| 515 // time to switch it on again (we don't rely just on MouseLeave because | 532 // time to switch it on again (we don't rely just on MouseLeave because |
| 516 // we don't want poor performance in the case of clicking the play | 533 // we don't want poor performance in the case of clicking the play |
| 517 // button and then leaving the mouse there). | 534 // button and then leaving the mouse there). |
| (...skipping 60 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 578 | 595 |
| 579 // Send the plugin the event. | 596 // Send the plugin the event. |
| 580 scoped_ptr<ScopedCurrentPluginEvent> event_scope(NULL); | 597 scoped_ptr<ScopedCurrentPluginEvent> event_scope(NULL); |
| 581 if (instance()->event_model() == NPEventModelCocoa) { | 598 if (instance()->event_model() == NPEventModelCocoa) { |
| 582 event_scope.reset(new ScopedCurrentPluginEvent( | 599 event_scope.reset(new ScopedCurrentPluginEvent( |
| 583 instance(), static_cast<NPCocoaEvent*>(plugin_event))); | 600 instance(), static_cast<NPCocoaEvent*>(plugin_event))); |
| 584 } | 601 } |
| 585 int16_t handle_response = instance()->NPP_HandleEvent(plugin_event); | 602 int16_t handle_response = instance()->NPP_HandleEvent(plugin_event); |
| 586 bool handled = handle_response != kNPEventNotHandled; | 603 bool handled = handle_response != kNPEventNotHandled; |
| 587 | 604 |
| 588 if (handled && event.type == WebInputEvent::KeyDown) { | 605 // Start IME if requested by the plugin. |
| 589 // Update IME state as requested by the plugin. | 606 if (handled && handle_response == kNPEventStartIME && |
| 590 SetImeEnabled(handle_response == kNPEventStartIME); | 607 event.type == WebInputEvent::KeyDown) { |
| 608 StartIme(); |
| 609 ++keyup_ignore_count_; |
| 591 } | 610 } |
| 592 | 611 |
| 593 // Plugins don't give accurate information about whether or not they handled | 612 // Plugins don't give accurate information about whether or not they handled |
| 594 // events, so browsers on the Mac ignore the return value. | 613 // events, so browsers on the Mac ignore the return value. |
| 595 // Scroll events are the exception, since the Cocoa spec defines a meaning | 614 // Scroll events are the exception, since the Cocoa spec defines a meaning |
| 596 // for the return value. | 615 // for the return value. |
| 597 if (WebInputEvent::isMouseEventType(event.type)) { | 616 if (WebInputEvent::isMouseEventType(event.type)) { |
| 598 handled = true; | 617 handled = true; |
| 599 } else if (WebInputEvent::isKeyboardEventType(event.type)) { | 618 } else if (WebInputEvent::isKeyboardEventType(event.type)) { |
| 600 // For Command-key events, trust the return value since eating all menu | 619 // For Command-key events, trust the return value since eating all menu |
| (...skipping 185 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 786 // states (WindowlessSetWindow will then send it on the first call). | 805 // states (WindowlessSetWindow will then send it on the first call). |
| 787 if (!have_called_set_window_) { | 806 if (!have_called_set_window_) { |
| 788 initial_window_focus_ = has_focus; | 807 initial_window_focus_ = has_focus; |
| 789 return; | 808 return; |
| 790 } | 809 } |
| 791 | 810 |
| 792 if (has_focus == containing_window_has_focus_) | 811 if (has_focus == containing_window_has_focus_) |
| 793 return; | 812 return; |
| 794 containing_window_has_focus_ = has_focus; | 813 containing_window_has_focus_ = has_focus; |
| 795 | 814 |
| 796 if (!has_focus) | |
| 797 SetImeEnabled(false); | |
| 798 | |
| 799 #ifndef NP_NO_QUICKDRAW | 815 #ifndef NP_NO_QUICKDRAW |
| 800 // Make sure controls repaint with the correct look. | 816 // Make sure controls repaint with the correct look. |
| 801 if (quirks_ & PLUGIN_QUIRK_ALLOW_FASTER_QUICKDRAW_PATH) | 817 if (quirks_ & PLUGIN_QUIRK_ALLOW_FASTER_QUICKDRAW_PATH) |
| 802 SetQuickDrawFastPathEnabled(false); | 818 SetQuickDrawFastPathEnabled(false); |
| 803 #endif | 819 #endif |
| 804 | 820 |
| 805 ScopedActiveDelegate active_delegate(this); | 821 ScopedActiveDelegate active_delegate(this); |
| 806 switch (instance()->event_model()) { | 822 switch (instance()->event_model()) { |
| 807 #ifndef NP_NO_CARBON | 823 #ifndef NP_NO_CARBON |
| 808 case NPEventModelCarbon: { | 824 case NPEventModelCarbon: { |
| (...skipping 16 matching lines...) Expand all Loading... |
| 825 instance()->NPP_HandleEvent(&focus_event); | 841 instance()->NPP_HandleEvent(&focus_event); |
| 826 break; | 842 break; |
| 827 } | 843 } |
| 828 } | 844 } |
| 829 } | 845 } |
| 830 | 846 |
| 831 bool WebPluginDelegateImpl::PlatformSetPluginHasFocus(bool focused) { | 847 bool WebPluginDelegateImpl::PlatformSetPluginHasFocus(bool focused) { |
| 832 if (!have_called_set_window_) | 848 if (!have_called_set_window_) |
| 833 return false; | 849 return false; |
| 834 | 850 |
| 835 if (!focused) | 851 plugin_->FocusChanged(focused); |
| 836 SetImeEnabled(false); | |
| 837 | 852 |
| 838 ScopedActiveDelegate active_delegate(this); | 853 ScopedActiveDelegate active_delegate(this); |
| 839 | 854 |
| 840 switch (instance()->event_model()) { | 855 switch (instance()->event_model()) { |
| 841 #ifndef NP_NO_CARBON | 856 #ifndef NP_NO_CARBON |
| 842 case NPEventModelCarbon: { | 857 case NPEventModelCarbon: { |
| 843 NPEvent focus_event = { 0 }; | 858 NPEvent focus_event = { 0 }; |
| 844 if (focused) | 859 if (focused) |
| 845 focus_event.what = NPEventType_GetFocusEvent; | 860 focus_event.what = NPEventType_GetFocusEvent; |
| 846 else | 861 else |
| (...skipping 45 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 892 if (!clip_rect_.IsEmpty()) | 907 if (!clip_rect_.IsEmpty()) |
| 893 instance()->webplugin()->InvalidateRect(gfx::Rect()); | 908 instance()->webplugin()->InvalidateRect(gfx::Rect()); |
| 894 } | 909 } |
| 895 | 910 |
| 896 void WebPluginDelegateImpl::WindowFrameChanged(const gfx::Rect& window_frame, | 911 void WebPluginDelegateImpl::WindowFrameChanged(const gfx::Rect& window_frame, |
| 897 const gfx::Rect& view_frame) { | 912 const gfx::Rect& view_frame) { |
| 898 instance()->set_window_frame(window_frame); | 913 instance()->set_window_frame(window_frame); |
| 899 SetContentAreaOrigin(gfx::Point(view_frame.x(), view_frame.y())); | 914 SetContentAreaOrigin(gfx::Point(view_frame.x(), view_frame.y())); |
| 900 } | 915 } |
| 901 | 916 |
| 902 void WebPluginDelegateImpl::ImeCompositionConfirmed(const string16& text) { | 917 void WebPluginDelegateImpl::ImeCompositionCompleted(const string16& text) { |
| 903 if (instance()->event_model() != NPEventModelCocoa) { | 918 if (instance()->event_model() != NPEventModelCocoa) { |
| 904 DLOG(ERROR) << "IME text receieved in Carbon event model"; | 919 DLOG(ERROR) << "IME notification receieved in Carbon event model"; |
| 905 return; | 920 return; |
| 906 } | 921 } |
| 907 | 922 |
| 908 NPCocoaEvent text_event; | 923 ime_enabled_ = false; |
| 909 memset(&text_event, 0, sizeof(NPCocoaEvent)); | 924 |
| 910 text_event.type = NPCocoaEventTextInput; | 925 // If |text| is empty this was just called to tell us composition was |
| 911 text_event.data.text.text = | 926 // cancelled externally (e.g., the user pressed esc). |
| 912 reinterpret_cast<NPNSString*>(base::SysUTF16ToNSString(text)); | 927 if (!text.empty()) { |
| 913 instance()->NPP_HandleEvent(&text_event); | 928 NPCocoaEvent text_event; |
| 929 memset(&text_event, 0, sizeof(NPCocoaEvent)); |
| 930 text_event.type = NPCocoaEventTextInput; |
| 931 text_event.data.text.text = |
| 932 reinterpret_cast<NPNSString*>(base::SysUTF16ToNSString(text)); |
| 933 instance()->NPP_HandleEvent(&text_event); |
| 934 } |
| 914 } | 935 } |
| 915 | 936 |
| 916 void WebPluginDelegateImpl::SetThemeCursor(ThemeCursor cursor) { | 937 void WebPluginDelegateImpl::SetThemeCursor(ThemeCursor cursor) { |
| 917 current_windowless_cursor_.InitFromThemeCursor(cursor); | 938 current_windowless_cursor_.InitFromThemeCursor(cursor); |
| 918 } | 939 } |
| 919 | 940 |
| 920 void WebPluginDelegateImpl::SetCursor(const Cursor* cursor) { | 941 void WebPluginDelegateImpl::SetCursor(const Cursor* cursor) { |
| 921 current_windowless_cursor_.InitFromCursor(cursor); | 942 current_windowless_cursor_.InitFromCursor(cursor); |
| 922 } | 943 } |
| 923 | 944 |
| (...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 964 if (plugin_visible && !redraw_timer_->IsRunning() && windowed_handle()) { | 985 if (plugin_visible && !redraw_timer_->IsRunning() && windowed_handle()) { |
| 965 redraw_timer_->Start( | 986 redraw_timer_->Start( |
| 966 base::TimeDelta::FromMilliseconds(kCoreAnimationRedrawPeriodMs), | 987 base::TimeDelta::FromMilliseconds(kCoreAnimationRedrawPeriodMs), |
| 967 this, &WebPluginDelegateImpl::DrawLayerInSurface); | 988 this, &WebPluginDelegateImpl::DrawLayerInSurface); |
| 968 } else if (!plugin_visible) { | 989 } else if (!plugin_visible) { |
| 969 redraw_timer_->Stop(); | 990 redraw_timer_->Stop(); |
| 970 } | 991 } |
| 971 } | 992 } |
| 972 } | 993 } |
| 973 | 994 |
| 974 void WebPluginDelegateImpl::SetImeEnabled(bool enabled) { | 995 void WebPluginDelegateImpl::StartIme() { |
| 975 if (instance()->event_model() != NPEventModelCocoa) | 996 if (instance()->event_model() != NPEventModelCocoa || |
| 997 !IsImeSupported()) { |
| 976 return; | 998 return; |
| 977 if (enabled == ime_enabled_) | 999 } |
| 1000 if (ime_enabled_) |
| 978 return; | 1001 return; |
| 979 ime_enabled_ = enabled; | 1002 ime_enabled_ = true; |
| 980 plugin_->SetImeEnabled(enabled); | 1003 plugin_->StartIme(); |
| 1004 } |
| 1005 |
| 1006 bool WebPluginDelegateImpl::IsImeSupported() { |
| 1007 // Currently the plugin IME implementation only works on 10.6. |
| 1008 static BOOL sImeSupported = NO; |
| 1009 static BOOL sHaveCheckedSupport = NO; |
| 1010 if (!sHaveCheckedSupport) { |
| 1011 int32 major, minor, bugfix; |
| 1012 base::SysInfo::OperatingSystemVersionNumbers(&major, &minor, &bugfix); |
| 1013 sImeSupported = major > 10 || (major == 10 && minor > 5); |
| 1014 sHaveCheckedSupport = YES; |
| 1015 } |
| 1016 return sImeSupported; |
| 981 } | 1017 } |
| 982 | 1018 |
| 983 #pragma mark - | 1019 #pragma mark - |
| 984 #pragma mark Core Animation Support | 1020 #pragma mark Core Animation Support |
| 985 | 1021 |
| 986 void WebPluginDelegateImpl::DrawLayerInSurface() { | 1022 void WebPluginDelegateImpl::DrawLayerInSurface() { |
| 987 // If we haven't plumbed up the surface yet, don't try to draw. | 1023 // If we haven't plumbed up the surface yet, don't try to draw. |
| 988 if (!windowed_handle() || !renderer_) | 1024 if (!windowed_handle() || !renderer_) |
| 989 return; | 1025 return; |
| 990 | 1026 |
| (...skipping 149 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1140 qd_manager_->SetFastPathEnabled(enabled); | 1176 qd_manager_->SetFastPathEnabled(enabled); |
| 1141 qd_port_.port = qd_manager_->port(); | 1177 qd_port_.port = qd_manager_->port(); |
| 1142 WindowlessSetWindow(); | 1178 WindowlessSetWindow(); |
| 1143 // Send a paint event so that the new buffer gets updated immediately. | 1179 // Send a paint event so that the new buffer gets updated immediately. |
| 1144 WindowlessPaint(buffer_context_, clip_rect_); | 1180 WindowlessPaint(buffer_context_, clip_rect_); |
| 1145 } | 1181 } |
| 1146 #endif // !NP_NO_QUICKDRAW | 1182 #endif // !NP_NO_QUICKDRAW |
| 1147 | 1183 |
| 1148 } // namespace npapi | 1184 } // namespace npapi |
| 1149 } // namespace webkit | 1185 } // namespace webkit |
| OLD | NEW |