diff options
author | oshima@chromium.org <oshima@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-05-04 18:43:30 +0000 |
---|---|---|
committer | oshima@chromium.org <oshima@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-05-04 18:43:30 +0000 |
commit | a53d1bedda4f266feb841fb3a61e704df5ea49e3 (patch) | |
tree | 3c455257445452873379909a40975ddec70326f7 /views | |
parent | f5ed21559387149aa986b5857942b4fa1e7df1b8 (diff) | |
download | chromium_src-a53d1bedda4f266feb841fb3a61e704df5ea49e3.zip chromium_src-a53d1bedda4f266feb841fb3a61e704df5ea49e3.tar.gz chromium_src-a53d1bedda4f266feb841fb3a61e704df5ea49e3.tar.bz2 |
Add transparency to TYPE_CHILD WidgetGtk.
* This does not work with renderer as parent yet.(I need to investigate renderer what it's doing)
* gtk_widget_set_back_pixmap is not necessary and removed.
* changed to use clear operation to fill a background with transparent color.
BUG=none
TEST=manual: added transparent widget example to view_examples. I also tested with chrome and confirmed that info bubble, app launcher and tab dragging works.
Review URL: http://codereview.chromium.org/1629024
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@46374 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'views')
-rw-r--r-- | views/examples/widget_example.h | 38 | ||||
-rw-r--r-- | views/widget/widget_gtk.cc | 206 | ||||
-rw-r--r-- | views/widget/widget_gtk.h | 7 |
3 files changed, 193 insertions, 58 deletions
diff --git a/views/examples/widget_example.h b/views/examples/widget_example.h index 609e6ed..b84e8f8 100644 --- a/views/examples/widget_example.h +++ b/views/examples/widget_example.h @@ -43,7 +43,17 @@ class BoxLayout : public views::LayoutManager { } virtual gfx::Size GetPreferredSize(views::View* host) { - return host->GetPreferredSize(); + int count = host->GetChildViewCount(); + gfx::Rect bounds; + int y = 0; + for (int i = 0; i < count; i++) { + views::View* child = host->GetChildViewAt(i); + gfx::Size size = child->GetPreferredSize(); + gfx::Rect child_bounds(0, y, size.width(), size.height()); + bounds = bounds.Union(child_bounds); + y += size.height(); + } + return bounds.size(); } private: @@ -123,19 +133,29 @@ class WidgetExample : public ExampleBase, public views::ButtonListener { void InitWidget(Widget* widget, const Widget::TransparencyParam transparency) { - // Add a button to close the popup widget. + // Add view/native buttons to close the popup widget. views::TextButton* close_button = new views::TextButton(this, L"Close"); close_button->set_tag(CLOSE_WIDGET); + // TODO(oshima): support transparent native view. + views::NativeButton* native_button + = new views::NativeButton(this, L"Native Close"); + native_button->set_tag(CLOSE_WIDGET); + + views::View* button_container = new views::View(); + button_container->SetLayoutManager(new BoxLayout); + button_container->AddChildView(close_button); + button_container->AddChildView(native_button); + views::View* widget_container = new views::View(); widget_container->SetLayoutManager(new CenterLayout); - if (transparency == Widget::Transparent) - close_button->set_background( - views::Background::CreateStandardPanelBackground()); - else + widget_container->AddChildView(button_container); + + widget->SetContentsView(widget_container); + + if (transparency != Widget::Transparent) { widget_container->set_background( views::Background::CreateStandardPanelBackground()); - widget_container->AddChildView(close_button); - widget->GetRootView()->SetContentsView(widget_container); + } // Show the widget. widget->Show(); @@ -151,7 +171,7 @@ class WidgetExample : public ExampleBase, public views::ButtonListener { // Compute where to place the child widget. // We'll place it at the center of the root widget. views::WidgetGtk* parent_widget = - static_cast<views::WidgetGtk*>(parent->GetWidget()->GetRootWidget()); + static_cast<views::WidgetGtk*>(parent->GetWidget()); gfx::Rect bounds; parent_widget->GetBounds(&bounds, false); // Child widget is 200x200 square. diff --git a/views/widget/widget_gtk.cc b/views/widget/widget_gtk.cc index 4bde019..f3eaa49 100644 --- a/views/widget/widget_gtk.cc +++ b/views/widget/widget_gtk.cc @@ -16,6 +16,8 @@ #include "app/os_exchange_data_provider_gtk.h" #include "base/auto_reset.h" #include "base/compiler_specific.h" +#include "base/message_loop.h" +#include "base/ref_counted.h" #include "base/utf_string_conversions.h" #include "gfx/path.h" #include "views/widget/default_theme_provider.h" @@ -25,6 +27,104 @@ #include "views/widget/tooltip_manager_gtk.h" #include "views/window/window_gtk.h" +namespace { + +// g_object data keys to associate a WidgetGtk object to a GtkWidget. +const char* kWidgetKey = "__VIEWS_WIDGET__"; +// A g_object data key to associate a CompositePainter object to a GtkWidget. +const char* kCompositePainterKey = "__VIEWS_COMPOSITE_PAINTER__"; +// A g_object data key to associate the flag whether or not the widget +// is composited to a GtkWidget. gtk_widget_is_composited simply tells +// if x11 supports composition and cannot be used to tell if given widget +// is composited. +const char* kCompositeEnabledKey = "__VIEWS_COMPOSITE_ENABLED__"; + +// CompositePainter draws a composited child widgets image into its +// drawing area. This object is ref counted so that it can disconnect +// expose event handler when all transparent widgets are deleted. +class CompositePainter : public base::RefCounted<CompositePainter> { + public: + explicit CompositePainter(GtkWidget* parent) + : parent_object_(G_OBJECT(parent)) { + handler_id_ = g_signal_connect_after( + parent_object_, "expose_event", G_CALLBACK(OnCompositePaint), NULL); + } + + static void AddCompositePainter(GtkWidget* widget) { + CompositePainter* painter = static_cast<CompositePainter*>( + g_object_get_data(G_OBJECT(widget), kCompositePainterKey)); + if (painter) { + painter->AddRef(); + } else { + g_object_set_data(G_OBJECT(widget), kCompositePainterKey, + new CompositePainter(widget)); + } + } + + static void RemoveCompositePainter(GtkWidget* widget) { + CompositePainter* painter = reinterpret_cast<CompositePainter*>( + g_object_get_data(G_OBJECT(widget), kCompositePainterKey)); + DCHECK(painter); + painter->Release(); + } + + // Enable the composition. + static void SetComposited(GtkWidget* widget) { + DCHECK(GTK_WIDGET_REALIZED(widget)); + gdk_window_set_composited(widget->window, true); + g_object_set_data(G_OBJECT(widget), kCompositeEnabledKey, + const_cast<char*>("")); + } + + // Returns true if the |widget| is composited. + static bool IsComposited(GtkWidget* widget) { + return g_object_get_data(G_OBJECT(widget), kCompositeEnabledKey) != NULL; + } + + private: + friend class base::RefCounted<CompositePainter>; + virtual ~CompositePainter() { + g_signal_handler_disconnect(parent_object_, handler_id_); + g_object_set_data(parent_object_, kCompositePainterKey, NULL); + } + + // Composes a image from one child. + static void CompositeChildWidget(GtkWidget* child, gpointer data) { + GdkEventExpose* event = static_cast<GdkEventExpose*>(data); + GtkWidget* parent = gtk_widget_get_parent(child); + DCHECK(parent); + if (IsComposited(child)) { + cairo_t* cr = gdk_cairo_create(parent->window); + gdk_cairo_set_source_pixmap(cr, child->window, + child->allocation.x, + child->allocation.y); + GdkRegion* region = gdk_region_rectangle(&child->allocation); + gdk_region_intersect(region, event->region); + gdk_cairo_region(cr, region); + cairo_clip(cr); + cairo_set_operator(cr, CAIRO_OPERATOR_OVER); + cairo_paint(cr); + cairo_destroy(cr); + } + } + + // Expose-event handler that compose & draws children's image into + // the |parent|'s drawing area. + static gboolean OnCompositePaint(GtkWidget* parent, GdkEventExpose* event) { + gtk_container_foreach(GTK_CONTAINER(parent), + CompositeChildWidget, + event); + return false; + } + + GObject* parent_object_; + gulong handler_id_; + + DISALLOW_COPY_AND_ASSIGN(CompositePainter); +}; + +} // namespace + namespace views { // During drag and drop GTK sends a drag-leave during a drop. This means we @@ -62,9 +162,6 @@ class WidgetGtk::DropObserver : public MessageLoopForUI::Observer { DISALLOW_COPY_AND_ASSIGN(DropObserver); }; -static const char* kWidgetKey = "__VIEWS_WIDGET__"; -static const wchar_t* kWidgetWideKey = L"__VIEWS_WIDGET__"; - // Returns the position of a widget on screen. static void GetWidgetPositionOnScreen(GtkWidget* widget, int* x, int *y) { // First get the root window. @@ -188,9 +285,8 @@ GtkWindow* WidgetGtk::GetTransientParent() const { } bool WidgetGtk::MakeTransparent() { - // Transparency can only be enabled for windows/popups and only if we haven't - // realized the widget. - DCHECK(!widget_ && type_ != TYPE_CHILD); + // Transparency can only be enabled only if we haven't realized the widget. + DCHECK(!widget_); if (!gdk_screen_is_composited(gdk_screen_get_default())) { // Transparency is only supported for compositing window managers. @@ -397,6 +493,7 @@ void WidgetGtk::Init(GtkWidget* parent, g_signal_connect_after(G_OBJECT(window_contents_), "size_allocate", G_CALLBACK(&OnSizeAllocateThunk), this); + gtk_widget_set_app_paintable(window_contents_, true); g_signal_connect(window_contents_, "expose_event", G_CALLBACK(&OnPaintThunk), this); g_signal_connect(window_contents_, "enter_notify_event", @@ -440,10 +537,6 @@ void WidgetGtk::Init(GtkWidget* parent, G_CALLBACK(&OnKeyPressThunk), this); g_signal_connect(widget_, "key_release_event", G_CALLBACK(&OnKeyReleaseThunk), this); - if (transparent_) { - g_signal_connect(widget_, "expose_event", - G_CALLBACK(&OnWindowPaintThunk), this); - } // Drag and drop. gtk_drag_dest_set(window_contents_, static_cast<GtkDestDefaults>(0), @@ -580,6 +673,13 @@ void WidgetGtk::Close() { void WidgetGtk::CloseNow() { if (widget_) { + if (transparent_ && type_ == TYPE_CHILD) { + GtkWidget* parent = gtk_widget_get_parent(widget_); + if (parent) { + DCHECK(parent != null_parent_); + CompositePainter::RemoveCompositePainter(parent); + } + } gtk_widget_destroy(widget_); widget_ = NULL; } @@ -790,6 +890,10 @@ void WidgetGtk::OnSizeAllocate(GtkWidget* widget, GtkAllocation* allocation) { } gboolean WidgetGtk::OnPaint(GtkWidget* widget, GdkEventExpose* event) { + if (transparent_) { + // Clear the background before drawing any view and native components. + DrawTransparentBackground(widget, event); + } root_view_->OnPaint(event); return false; // False indicates other widgets should get the event as well. } @@ -1054,22 +1158,10 @@ RootView* WidgetGtk::CreateRootView() { } gboolean WidgetGtk::OnWindowPaint(GtkWidget* widget, GdkEventExpose* event) { - // NOTE: for reasons I don't understand this code is never hit. It should - // be hit when transparent_, but we never get the expose-event for the - // window in this case, even though a stand alone test case triggers it. I'm - // leaving it in just in case. - - // Fill the background totally transparent. We don't need to paint the root - // view here as that is done by OnPaint. + // Clear the background to be totally transparent. We don't need to + // paint the root view here as that is done by OnPaint. DCHECK(transparent_); - int width, height; - gtk_window_get_size(GTK_WINDOW(widget), &width, &height); - cairo_t* cr = gdk_cairo_create(widget->window); - cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE); - cairo_set_source_rgba(cr, 0, 0, 0, 0); - cairo_rectangle(cr, 0, 0, width, height); - cairo_fill(cr); - cairo_destroy(cr); + DrawTransparentBackground(widget, event); return false; } @@ -1193,7 +1285,19 @@ void WidgetGtk::CreateGtkWidget(GtkWidget* parent, const gfx::Rect& bounds) { gtk_container_add(GTK_CONTAINER(popup), null_parent_); gtk_widget_realize(null_parent_); } + if (transparent_) { + // transparency has to be configured before widget is realized. + DCHECK(parent) << "Transparent widget must have parent when initialized"; + ConfigureWidgetForTransparentBackground(parent); + } gtk_container_add(GTK_CONTAINER(parent ? parent : null_parent_), widget_); + gtk_widget_realize(widget_); + if (transparent_) { + // The widget has to be realized to set composited flag. + // I tried "realize" signal to set this flag, but it did not work + // when the top level is popup. + CompositePainter::SetComposited(widget_); + } } else { widget_ = gtk_window_new( (type_ == TYPE_WINDOW || type_ == TYPE_DECORATED_WINDOW) ? @@ -1228,27 +1332,23 @@ void WidgetGtk::CreateGtkWidget(GtkWidget* parent, const gfx::Rect& bounds) { static_cast<Widget*>(this)); if (transparent_) - ConfigureWidgetForTransparentBackground(); + ConfigureWidgetForTransparentBackground(NULL); if (ignore_events_) ConfigureWidgetForIgnoreEvents(); SetAlwaysOnTop(always_on_top_); + // The widget needs to be realized before handlers like size-allocate can + // function properly. + gtk_widget_realize(widget_); } - - // The widget needs to be realized before handlers like size-allocate can - // function properly. - gtk_widget_realize(widget_); - - // Associate this object with the widget. - SetNativeWindowProperty(kWidgetWideKey, this); } -void WidgetGtk::ConfigureWidgetForTransparentBackground() { - DCHECK(widget_ && window_contents_ && widget_ != window_contents_); +void WidgetGtk::ConfigureWidgetForTransparentBackground(GtkWidget* parent) { + DCHECK(widget_ && window_contents_); GdkColormap* rgba_colormap = - gdk_screen_get_rgba_colormap(gdk_screen_get_default()); + gdk_screen_get_rgba_colormap(gtk_widget_get_screen(widget_)); if (!rgba_colormap) { transparent_ = false; return; @@ -1257,19 +1357,21 @@ void WidgetGtk::ConfigureWidgetForTransparentBackground() { // on both the window and fixed. In addition we need to make sure no // decorations are drawn. The last bit is to make sure the widget doesn't // attempt to draw a pixmap in it's background. - gtk_widget_set_colormap(widget_, rgba_colormap); - gtk_widget_set_app_paintable(widget_, true); - gtk_widget_realize(widget_); - gdk_window_set_decorations(widget_->window, - static_cast<GdkWMDecoration>(0)); - // Widget must be realized before setting pixmap. - gdk_window_set_back_pixmap(widget_->window, NULL, FALSE); - + if (type_ != TYPE_CHILD) { + DCHECK(parent == NULL); + gtk_widget_set_colormap(widget_, rgba_colormap); + gtk_widget_set_app_paintable(widget_, true); + g_signal_connect(widget_, "expose_event", + G_CALLBACK(&OnWindowPaintThunk), this); + gtk_widget_realize(widget_); + gdk_window_set_decorations(widget_->window, + static_cast<GdkWMDecoration>(0)); + } else { + DCHECK(parent); + CompositePainter::AddCompositePainter(parent); + } + DCHECK(!GTK_WIDGET_REALIZED(window_contents_)); gtk_widget_set_colormap(window_contents_, rgba_colormap); - gtk_widget_set_app_paintable(window_contents_, true); - gtk_widget_realize(window_contents_); - // Widget must be realized before setting pixmap. - gdk_window_set_back_pixmap(window_contents_->window, NULL, FALSE); } void WidgetGtk::ConfigureWidgetForIgnoreEvents() { @@ -1301,6 +1403,14 @@ void WidgetGtk::HandleGrabBroke() { } } +void WidgetGtk::DrawTransparentBackground(GtkWidget* widget, + GdkEventExpose* event) { + cairo_t* cr = gdk_cairo_create(widget->window); + cairo_set_operator(cr, CAIRO_OPERATOR_CLEAR); + gdk_cairo_region(cr, event->region); + cairo_fill(cr); + cairo_destroy(cr); +} //////////////////////////////////////////////////////////////////////////////// // Widget, public: diff --git a/views/widget/widget_gtk.h b/views/widget/widget_gtk.h index cb5f103..824c11d 100644 --- a/views/widget/widget_gtk.h +++ b/views/widget/widget_gtk.h @@ -75,6 +75,7 @@ class WidgetGtk // invoked before Init. This does a couple of checks and returns true if // the window can be made transparent. The actual work of making the window // transparent is done by ConfigureWidgetForTransparentBackground. + // This works for both child and window types. bool MakeTransparent(); bool is_transparent() const { return transparent_; } @@ -314,7 +315,7 @@ class WidgetGtk // Invoked from create widget to enable the various bits needed for a // transparent background. This is only invoked if MakeTransparent has been // invoked. - void ConfigureWidgetForTransparentBackground(); + void ConfigureWidgetForTransparentBackground(GtkWidget* parent); // Invoked from create widget to enable the various bits needed for a // window which doesn't receive events. This is only invoked if @@ -324,6 +325,10 @@ class WidgetGtk // TODO(sky): documentation void HandleGrabBroke(); + // A utility function to draw a transparent background onto the |widget|. + static void DrawTransparentBackground(GtkWidget* widget, + GdkEventExpose* event); + const Type type_; // Our native views. If we're a window/popup, then widget_ is the window and |