summaryrefslogtreecommitdiffstats
path: root/WebCore/platform/gtk/ScrollbarGtk.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'WebCore/platform/gtk/ScrollbarGtk.cpp')
-rw-r--r--WebCore/platform/gtk/ScrollbarGtk.cpp150
1 files changed, 150 insertions, 0 deletions
diff --git a/WebCore/platform/gtk/ScrollbarGtk.cpp b/WebCore/platform/gtk/ScrollbarGtk.cpp
new file mode 100644
index 0000000..099895d
--- /dev/null
+++ b/WebCore/platform/gtk/ScrollbarGtk.cpp
@@ -0,0 +1,150 @@
+/*
+ * Copyright (C) 2007 Holger Hans Peter Freyther zecke@selfish.org
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ */
+
+#include "config.h"
+#include "ScrollbarGtk.h"
+
+#include "IntRect.h"
+#include "GraphicsContext.h"
+#include "FrameView.h"
+#include "NotImplemented.h"
+#include "ScrollbarTheme.h"
+#include "gtkdrawing.h"
+
+#include <gtk/gtk.h>
+
+using namespace WebCore;
+
+PassRefPtr<Scrollbar> Scrollbar::createNativeScrollbar(ScrollbarClient* client, ScrollbarOrientation orientation, ScrollbarControlSize size)
+{
+ return adoptRef(new ScrollbarGtk(client, orientation, size));
+}
+
+static gboolean gtkScrollEventCallback(GtkWidget* widget, GdkEventScroll* event, ScrollbarGtk*)
+{
+ /* Scroll only if our parent rejects the scroll event. The rationale for
+ * this is that we want the main frame to scroll when we move the mouse
+ * wheel over a child scrollbar in most cases. */
+ return gtk_widget_event(gtk_widget_get_parent(widget), reinterpret_cast<GdkEvent*>(event));
+}
+
+ScrollbarGtk::ScrollbarGtk(ScrollbarClient* client, ScrollbarOrientation orientation,
+ ScrollbarControlSize controlSize)
+ : Scrollbar(client, orientation, controlSize)
+ , m_adjustment(GTK_ADJUSTMENT(gtk_adjustment_new(0.0, 0.0, 0.0, 0.0, 0.0, 0.0)))
+{
+ GtkScrollbar* scrollBar = orientation == HorizontalScrollbar ?
+ GTK_SCROLLBAR(::gtk_hscrollbar_new(m_adjustment)) :
+ GTK_SCROLLBAR(::gtk_vscrollbar_new(m_adjustment));
+ gtk_widget_show(GTK_WIDGET(scrollBar));
+ g_object_ref(G_OBJECT(scrollBar));
+ g_signal_connect(G_OBJECT(scrollBar), "value-changed", G_CALLBACK(ScrollbarGtk::gtkValueChanged), this);
+ g_signal_connect(G_OBJECT(scrollBar), "scroll-event", G_CALLBACK(gtkScrollEventCallback), this);
+
+ setPlatformWidget(GTK_WIDGET(scrollBar));
+
+ /*
+ * assign a sane default width and height to the Scrollbar, otherwise
+ * we will end up with a 0 width scrollbar.
+ */
+ resize(ScrollbarTheme::nativeTheme()->scrollbarThickness(),
+ ScrollbarTheme::nativeTheme()->scrollbarThickness());
+}
+
+ScrollbarGtk::~ScrollbarGtk()
+{
+ /*
+ * the Widget does not take over ownership.
+ */
+ g_signal_handlers_disconnect_by_func(G_OBJECT(platformWidget()), (gpointer)ScrollbarGtk::gtkValueChanged, this);
+ g_signal_handlers_disconnect_by_func(G_OBJECT(platformWidget()), (gpointer)gtkScrollEventCallback, this);
+ g_object_unref(G_OBJECT(platformWidget()));
+}
+
+void ScrollbarGtk::frameRectsChanged() const
+{
+ if (!parent() || !parent()->isScrollViewScrollbar(this))
+ return;
+
+ IntPoint loc = parent()->convertToContainingWindow(frameRect().location());
+
+ // Don't allow the allocation size to be negative
+ IntSize sz = frameRect().size();
+ sz.clampNegativeToZero();
+
+ GtkAllocation allocation = { loc.x(), loc.y(), sz.width(), sz.height() };
+ gtk_widget_size_allocate(platformWidget(), &allocation);
+}
+
+void ScrollbarGtk::updateThumbPosition()
+{
+ if (m_adjustment->value != m_currentPos) {
+ m_adjustment->value = m_currentPos;
+ gtk_adjustment_value_changed(m_adjustment);
+ }
+}
+
+void ScrollbarGtk::updateThumbProportion()
+{
+ m_adjustment->step_increment = m_lineStep;
+ m_adjustment->page_increment = m_pageStep;
+ m_adjustment->page_size = m_visibleSize;
+ m_adjustment->upper = m_totalSize;
+ gtk_adjustment_changed(m_adjustment);
+}
+
+void ScrollbarGtk::setFrameRect(const IntRect& rect)
+{
+ Widget::setFrameRect(rect);
+ frameRectsChanged();
+}
+
+void ScrollbarGtk::frameRectsChanged()
+{
+ if (!parent())
+ return;
+
+ ASSERT(parent()->isFrameView());
+
+ FrameView* frameView = static_cast<FrameView*>(parent());
+ IntRect windowRect = IntRect(frameView->contentsToWindow(frameRect().location()), frameRect().size());
+
+ IntSize sz = frameRect().size();
+ sz.clampNegativeToZero();
+
+ GtkAllocation allocation = { windowRect.x(), windowRect.y(), sz.width(), sz.height() };
+ gtk_widget_size_allocate(platformWidget(), &allocation);
+}
+
+void ScrollbarGtk::gtkValueChanged(GtkAdjustment*, ScrollbarGtk* that)
+{
+ that->setValue(static_cast<int>(gtk_adjustment_get_value(that->m_adjustment)));
+}
+
+void ScrollbarGtk::setEnabled(bool shouldEnable)
+{
+ if (enabled() == shouldEnable)
+ return;
+
+ Scrollbar::setEnabled(shouldEnable);
+ if (platformWidget())
+ gtk_widget_set_sensitive(platformWidget(), shouldEnable);
+}
+
+
+