// // GtkPlus.cs: Wraps GTK+, exposing its Widget painting features to code // that uses System.Drawing types (such as IDeviceContext, Rectangle). // // Permission is hereby granted, free of charge, to any person obtaining // a copy of this software and associated documentation files (the // "Software"), to deal in the Software without restriction, including // without limitation the rights to use, copy, modify, merge, publish, // distribute, sublicense, and/or sell copies of the Software, and to // permit persons to whom the Software is furnished to do so, subject to // the following conditions: // // The above copyright notice and this permission notice shall be // included in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // // Copyright (c) 2008 George Giolfan // // Authors: // George Giolfan (georgegiolfan@yahoo.com) // using gpointer = System.IntPtr; using gboolean = System.Boolean; using gint = System.Int32; using guint = System.UInt32; using guint8 = System.Byte; using guint16 = System.UInt16; using guint32 = System.UInt32; using gfloat = System.Single; using gdouble = System.Double; using PangoFontDescriptionPointer = System.IntPtr; using GDataPointer = System.IntPtr; using GObjectPointer = System.IntPtr; using GSListPointer = System.IntPtr; using GType = System.IntPtr; using GTypeClassPointer = System.IntPtr; using GTypeInstancePointer = System.IntPtr; using GdkColormapPointer = System.IntPtr; using GdkDrawablePointer = System.IntPtr; using GdkGCPointer = System.IntPtr; using GdkNativeWindowPointer = System.IntPtr; using GdkPixbufPointer = System.IntPtr; using GdkPixmapPointer = System.IntPtr; using GdkWindowPointer = System.IntPtr; using GtkAdjustmentPointer = System.IntPtr; using GtkBinPointer = System.IntPtr; using GtkContainerPointer = System.IntPtr; using GtkObjectPointer = System.IntPtr; using GtkStylePointer = System.IntPtr; using GtkToolbarPointer = System.IntPtr; using GtkToolItemPointer = System.IntPtr; using GtkTreeViewPointer = System.IntPtr; using GtkTreeViewColumnPointer = System.IntPtr; using GtkWidgetPointer = System.IntPtr; using GtkWindowPointer = System.IntPtr; using GtkAllocation = ShiftUI.VisualStyles.GtkPlus.GdkRectangle; using GInitiallyUnowned = ShiftUI.VisualStyles.GtkPlus.GObject; using System.Collections; using System.Drawing; using System.Drawing.Imaging; using System.Reflection; using System.Reflection.Emit; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; using System; namespace ShiftUI.VisualStyles { /// /// /// /// Inspired by ThemeGtk and QGtkStyle (http://labs.trolltech.com/page/Projects/Styles/GtkStyle). Tries to use the exact GTK+ logic in order to be compatible with all its themes. /// class GtkPlus { #region Instance static GtkPlus instance; public static GtkPlus Instance { get { return instance; } } #endregion #region GTK+ initialization public static bool Initialize () { try { if (gtk_check_version (2, 10, 0) != IntPtr.Zero) return false; //TODO: If we need to pass the actual arguments, duplicate the required code from GTK#. int argc = 0; string [] argv = new string [1]; bool result = gtk_init_check (ref argc, ref argv); if (result) instance = new GtkPlus (); return result; } catch (DllNotFoundException) { return false; } } #endregion #region Fields readonly int WidgetTypeCount = Enum.GetNames (typeof (WidgetType)).Length; readonly GtkWidgetPointer [] widgets; readonly GtkWidgetPointer window; readonly GtkWidgetPointer @fixed; readonly GtkStylePointer [] styles; #region ComboBox readonly GtkWidgetPointer combo_box_drop_down_toggle_button; readonly GtkWidgetPointer combo_box_drop_down_arrow; GtkStylePointer combo_box_drop_down_toggle_button_style; GtkStylePointer combo_box_drop_down_arrow_style; #endregion #region ToolBar readonly GtkWidgetPointer tool_bar_button; readonly GtkWidgetPointer tool_bar_toggle_button; GtkStylePointer tool_bar_button_style; GtkStylePointer tool_bar_toggle_button_style; #endregion #region TreeView readonly GtkTreeViewColumnPointer tree_view_column; readonly GtkWidgetPointer tree_view_column_button; GtkStylePointer tree_view_column_button_style; #endregion #region Painters readonly ButtonPainter button_painter = new ButtonPainter (); readonly CheckBoxPainter check_box_painter = new CheckBoxPainter (); readonly RadioButtonPainter radio_button_painter = new RadioButtonPainter (); #region ComboBox readonly ComboBoxDropDownButtonPainter combo_box_drop_down_button_painter = new ComboBoxDropDownButtonPainter (); readonly ComboBoxBorderPainter combo_box_border_painter = new ComboBoxBorderPainter (); #endregion #region GroupBox readonly GroupBoxPainter group_box_painter = new GroupBoxPainter (); #endregion #region Header readonly HeaderPainter header_painter = new HeaderPainter (); #endregion #region ProgressBar readonly ProgressBarBarPainter progress_bar_bar_painter = new ProgressBarBarPainter (); readonly ProgressBarChunkPainter progress_bar_chunk_painter = new ProgressBarChunkPainter (); #endregion #region ScrollBar readonly ScrollBarArrowButtonPainter scroll_bar_arrow_button_painter = new ScrollBarArrowButtonPainter (); readonly ScrollBarThumbButtonPainter scroll_bar_thumb_button_painter = new ScrollBarThumbButtonPainter (); readonly ScrollBarTrackPainter scroll_bar_track_painter = new ScrollBarTrackPainter (); #endregion #region StatusBar readonly StatusBarGripperPainter status_bar_gripper_painter = new StatusBarGripperPainter (); #endregion #region TabWidget readonly TabWidgetPanePainter tab_Widget_pane_painter = new TabWidgetPanePainter (); readonly TabWidgetTabItemPainter tab_Widget_tab_item_painter = new TabWidgetTabItemPainter (); #endregion readonly TextBoxPainter text_box_painter = new TextBoxPainter (); #region ToolBar readonly ToolBarPainter tool_bar_painter = new ToolBarPainter (); readonly ToolBarButtonPainter tool_bar_button_painter = new ToolBarButtonPainter (); readonly ToolBarCheckedButtonPainter tool_bar_checked_button_painter = new ToolBarCheckedButtonPainter (); #endregion #region TrackBar readonly TrackBarTrackPainter track_bar_track_painter = new TrackBarTrackPainter (); readonly TrackBarThumbPainter track_bar_thumb_painter = new TrackBarThumbPainter (); #endregion readonly TreeViewGlyphPainter tree_view_glyph_painter = new TreeViewGlyphPainter (); readonly UpDownPainter up_down_painter = new UpDownPainter (); #endregion #endregion #region Constructor and finalizer protected GtkPlus () { widgets = new GtkWidgetPointer [WidgetTypeCount]; styles = new GtkStylePointer [WidgetTypeCount]; window = gtk_window_new (GtkWindowType.GTK_WINDOW_TOPLEVEL); @fixed = gtk_fixed_new (); gtk_container_add (window, @fixed); #region Widget types #region Button gtk_container_add (@fixed, widgets [(int)WidgetType.Button] = gtk_button_new ()); GTK_WIDGET_SET_FLAGS (widgets [(int)WidgetType.Button], GtkWidgetFlags.GTK_CAN_DEFAULT); #endregion #region CheckBox gtk_container_add (@fixed, widgets [(int)WidgetType.CheckBox] = gtk_check_button_new ()); #endregion #region ComboBox gtk_container_add (@fixed, widgets [(int)WidgetType.ComboBox] = gtk_combo_box_entry_new ()); gtk_widget_realize (widgets [(int)WidgetType.ComboBox]); combo_box_drop_down_toggle_button = GetFirstChildWidgetOfType.Get (widgets [(int)WidgetType.ComboBox], gtk_toggle_button_get_type ()); gtk_widget_realize (combo_box_drop_down_toggle_button); combo_box_drop_down_arrow = GetFirstChildWidgetOfType.Get (combo_box_drop_down_toggle_button, gtk_arrow_get_type ()); g_object_ref (combo_box_drop_down_toggle_button_style = GetWidgetStyle (combo_box_drop_down_toggle_button)); g_object_ref (combo_box_drop_down_arrow_style = GetWidgetStyle (combo_box_drop_down_arrow)); #endregion #region GroupBox gtk_container_add (@fixed, widgets [(int)WidgetType.GroupBox] = gtk_frame_new (null)); #endregion #region ProgressBar gtk_container_add (@fixed, widgets [(int)WidgetType.ProgressBar] = gtk_progress_bar_new ()); #endregion #region RadioButton gtk_container_add (@fixed, widgets [(int)WidgetType.RadioButton] = gtk_radio_button_new (IntPtr.Zero)); #endregion #region ScrollBar gtk_container_add (@fixed, widgets [(int)WidgetType.HScrollBar] = gtk_hscrollbar_new (IntPtr.Zero)); gtk_container_add (@fixed, widgets [(int)WidgetType.VScrollBar] = gtk_vscrollbar_new (IntPtr.Zero)); #endregion #region StatusBar gtk_container_add (@fixed, widgets [(int)WidgetType.StatusBar] = gtk_statusbar_new ()); #endregion #region TabWidget gtk_container_add (@fixed, widgets [(int)WidgetType.TabWidget] = gtk_notebook_new ()); #endregion #region TextBox gtk_container_add (@fixed, widgets [(int)WidgetType.TextBox] = gtk_entry_new ()); #endregion #region ToolBar gtk_container_add (@fixed, widgets [(int)WidgetType.ToolBar] = gtk_toolbar_new ()); GtkToolItemPointer tool_button = gtk_tool_button_new (IntPtr.Zero, null); gtk_toolbar_insert (widgets [(int)WidgetType.ToolBar], tool_button, -1); tool_bar_button = gtk_bin_get_child (tool_button); g_object_ref (tool_bar_button_style = GetWidgetStyle (tool_bar_button)); GtkToolItemPointer toggle_tool_button = gtk_toggle_tool_button_new (); gtk_toolbar_insert (widgets [(int)WidgetType.ToolBar], toggle_tool_button, -1); tool_bar_toggle_button = gtk_bin_get_child (toggle_tool_button); g_object_ref (tool_bar_toggle_button_style = GetWidgetStyle (tool_bar_toggle_button)); #endregion #region TrackBar gtk_container_add (@fixed, widgets [(int)WidgetType.HorizontalTrackBar] = gtk_hscale_new_with_range (0, 1, 1)); gtk_container_add (@fixed, widgets [(int)WidgetType.VerticalTrackBar] = gtk_vscale_new_with_range (0, 1, 1)); #endregion #region TreeView gtk_container_add (@fixed, widgets [(int)WidgetType.TreeView] = gtk_tree_view_new ()); tree_view_column = gtk_tree_view_column_new (); gtk_tree_view_insert_column (widgets [(int)WidgetType.TreeView], tree_view_column, -1); GtkTreeViewColumn column_structure = (GtkTreeViewColumn)Marshal.PtrToStructure (tree_view_column, typeof (GtkTreeViewColumn)); tree_view_column_button = column_structure.button; g_object_ref (tree_view_column_button_style = GetWidgetStyle (tree_view_column_button)); #endregion #region UpDown GtkAdjustmentPointer adjustment = gtk_adjustment_new (0, 0, 0, 0, 0, 0); gtk_container_add (@fixed, widgets [(int)WidgetType.UpDown] = gtk_spin_button_new (adjustment, 0, 0)); #endregion #endregion for (int widget_index = 0; widget_index < WidgetTypeCount; widget_index++) g_object_ref (styles [widget_index] = GetWidgetStyle (widgets [widget_index])); } ~GtkPlus () { gtk_object_destroy (window); for (int widget_index = 0; widget_index < WidgetTypeCount; widget_index++) g_object_unref (styles [widget_index]); #region ComboBox g_object_unref (combo_box_drop_down_toggle_button_style); g_object_unref (combo_box_drop_down_arrow_style); #endregion #region ToolBar g_object_unref (tool_bar_button_style); g_object_unref (tool_bar_toggle_button_style); #endregion #region TreeView g_object_unref (tree_view_column_button_style); #endregion } #endregion #region Widgets #region Button public void ButtonPaint (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, bool @default, GtkPlusState state) { button_painter.Configure (@default, state); Paint (WidgetType.Button, bounds, dc, clippingArea, button_painter); } #endregion #region CheckBox public void CheckBoxPaint (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, GtkPlusState state, GtkPlusToggleButtonValue value) { check_box_painter.Configure (state, value); Paint (WidgetType.CheckBox, bounds, dc, clippingArea, check_box_painter); } Size GetGtkCheckButtonIndicatorSize (WidgetType widgetType) { int indicator_size = GetWidgetStyleInteger (widgets [(int)widgetType], "indicator-size"); return new Size (indicator_size, indicator_size); } public Size CheckBoxGetSize () { return GetGtkCheckButtonIndicatorSize (WidgetType.CheckBox); } #endregion #region ComboBox public void ComboBoxPaintDropDownButton (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, GtkPlusState state) { combo_box_drop_down_button_painter.Configure (state); Paint (WidgetType.ComboBox, bounds, dc, clippingArea, combo_box_drop_down_button_painter); } public void ComboBoxPaintBorder (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea) { Paint (WidgetType.ComboBox, bounds, dc, clippingArea, combo_box_border_painter); } #endregion #region GroupBox public void GroupBoxPaint (IDeviceContext dc, Rectangle bounds, Rectangle excludedArea, GtkPlusState state) { group_box_painter.Configure (state); PaintExcludingArea (WidgetType.GroupBox, bounds, dc, excludedArea, group_box_painter); } #endregion #region Header public void HeaderPaint (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, GtkPlusState state) { header_painter.Configure (state); Paint (WidgetType.TreeView, bounds, dc, clippingArea, header_painter); } #endregion #region ProgressBar public void ProgressBarPaintBar (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea) { Paint (WidgetType.ProgressBar, bounds, dc, clippingArea, progress_bar_bar_painter); } public void ProgressBarPaintChunk (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea) { Paint (WidgetType.ProgressBar, bounds, dc, clippingArea, progress_bar_chunk_painter); } public Rectangle ProgressBarGetBackgroundContentRectagle (Rectangle bounds) { GtkStyle style = (GtkStyle)Marshal.PtrToStructure (gtk_widget_get_style (widgets [(int)WidgetType.ProgressBar]), typeof(GtkStyle)); bounds.Inflate (-style.xthickness, -style.ythickness); return bounds; } #endregion #region RadioButton public void RadioButtonPaint (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, GtkPlusState state, GtkPlusToggleButtonValue value) { radio_button_painter.Configure (state, value); Paint (WidgetType.RadioButton, bounds, dc, clippingArea, radio_button_painter); } public Size RadioButtonGetSize () { return GetGtkCheckButtonIndicatorSize (WidgetType.RadioButton); } #endregion #region ScrollBar public void ScrollBarPaintArrowButton (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, GtkPlusState state, bool horizontal, bool upOrLeft) { scroll_bar_arrow_button_painter.Configure (state, horizontal, upOrLeft); Paint (horizontal ? WidgetType.HScrollBar : WidgetType.VScrollBar, bounds, dc, clippingArea, scroll_bar_arrow_button_painter); } public void ScrollBarPaintThumbButton (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, GtkPlusState state, bool horizontal) { scroll_bar_thumb_button_painter.Configure (state, horizontal); Paint (horizontal ? WidgetType.HScrollBar : WidgetType.VScrollBar, bounds, dc, clippingArea, scroll_bar_thumb_button_painter); } public void ScrollBarPaintTrack (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, GtkPlusState state, bool horizontal, bool upOrLeft) { scroll_bar_track_painter.Configure (state, upOrLeft); Paint (horizontal ? WidgetType.HScrollBar : WidgetType.VScrollBar, bounds, dc, clippingArea, scroll_bar_track_painter); } #endregion #region StatusBar public void StatusBarPaintGripper (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea) { Paint (WidgetType.StatusBar, bounds, dc, clippingArea, status_bar_gripper_painter); } #endregion #region TabWidget public void TabWidgetPaintPane (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea) { Paint (WidgetType.TabWidget, bounds, dc, clippingArea, tab_Widget_pane_painter); } public void TabWidgetPaintTabItem (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, GtkPlusState state) { tab_Widget_tab_item_painter.Configure (state); Paint (WidgetType.TabWidget, bounds, dc, clippingArea, tab_Widget_tab_item_painter); } #endregion #region TextBox public void TextBoxPaint (IDeviceContext dc, Rectangle bounds, Rectangle excludedArea, GtkPlusState state) { text_box_painter.Configure (state); PaintExcludingArea (WidgetType.TextBox, bounds, dc, excludedArea, text_box_painter); } #endregion #region ToolBar public void ToolBarPaint (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea) { Paint (WidgetType.ToolBar, bounds, dc, clippingArea, tool_bar_painter); } public void ToolBarPaintButton (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, GtkPlusState state) { tool_bar_button_painter.Configure (state); Paint (WidgetTypeNotNeeded, bounds, dc, clippingArea, tool_bar_button_painter); } public void ToolBarPaintCheckedButton (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea) { Paint (WidgetTypeNotNeeded, bounds, dc, clippingArea, tool_bar_checked_button_painter); } #endregion #region TrackBar public void TrackBarPaintTrack (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, bool horizontal) { Paint (horizontal ? WidgetType.HorizontalTrackBar : WidgetType.VerticalTrackBar, bounds, dc, clippingArea, track_bar_track_painter); } public void TrackBarPaintThumb (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, GtkPlusState state, bool horizontal) { track_bar_thumb_painter.Configure (state, horizontal); Paint (horizontal ? WidgetType.HorizontalTrackBar : WidgetType.VerticalTrackBar, bounds, dc, clippingArea, track_bar_thumb_painter); } #endregion #region TreeView public void TreeViewPaintGlyph (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, bool closed) { tree_view_glyph_painter.Configure (closed); Paint (WidgetType.TreeView, bounds, dc, clippingArea, tree_view_glyph_painter); } #endregion #region UpDown public void UpDownPaint (IDeviceContext dc, Rectangle bounds, Rectangle clippingArea, bool up, GtkPlusState state) { up_down_painter.Configure (up, state); Paint (WidgetType.UpDown, bounds, dc, clippingArea, up_down_painter); } #endregion #endregion #region Painting void Paint (WidgetType widgetType, Rectangle bounds, IDeviceContext dc, Rectangle clippingArea, Painter painter) { Paint (widgetType, bounds, dc, TransparencyType.Alpha, Color.Black, DeviceContextType.Native, clippingArea, painter, Rectangle.Empty); } void PaintExcludingArea (WidgetType widgetType, Rectangle bounds, IDeviceContext dc, Rectangle excludedArea, Painter painter) { Paint (widgetType, bounds, dc, TransparencyType.Alpha, Color.Black, DeviceContextType.Native, bounds, painter, excludedArea); } void Paint (WidgetType widgetType, Rectangle bounds, IDeviceContext dc, TransparencyType transparencyType, Color background, DeviceContextType deviceContextType, Rectangle clippingArea, Painter painter, Rectangle excludedArea) { Rectangle painted_area = Rectangle.Intersect (bounds, clippingArea); if (painted_area.Width == 0 || painted_area.Height == 0) return; painted_area.Offset (-bounds.X, -bounds.Y); excludedArea.Offset (-bounds.X, -bounds.Y); GdkDrawablePointer drawable = gdk_pixmap_new (IntPtr.Zero, bounds.Width, bounds.Height, 24); painter.AttachStyle (widgetType, drawable, this); GdkPixbufPointer pixbuf; IntPtr pixel_data; int rowstride; GdkGCPointer gc = gdk_gc_new (drawable); GdkColor color = new GdkColor (background); gdk_gc_set_rgb_fg_color (gc, ref color); Paint (drawable, gc, bounds, widgetType, out pixbuf, out pixel_data, out rowstride, painted_area, painter, excludedArea); GdkPixbufPointer white_pixbuf = IntPtr.Zero; IntPtr white_pixel_data = IntPtr.Zero; int white_rowstride = 0; GdkColor white_color = new GdkColor(); if (transparencyType == TransparencyType.Alpha) { white_color.red = guint16.MaxValue; white_color.green = guint16.MaxValue; white_color.blue = guint16.MaxValue; gdk_gc_set_rgb_fg_color (gc, ref white_color); Paint (drawable, gc, bounds, widgetType, out white_pixbuf, out white_pixel_data, out white_rowstride, painted_area, painter, excludedArea); } g_object_unref (gc); unsafe { byte* row = (byte*)pixel_data; byte* pixel; byte* white_row = (byte*)white_pixel_data; byte* white_pixel; for (int row_index = 0; row_index < painted_area.Height; row_index++) { pixel = row; white_pixel = white_row; for (int pixel_index = 0; pixel_index < painted_area.Width; pixel_index++) { const int GdkRedOffset = 0; const int GdkGreenOffset = 1; const int GdkBlueOffset = 2; const int BitmapAlphaOffset = 3; const int BitmapRedOffset = 2; const int BitmapBlueOffset = 0; switch (transparencyType) { case TransparencyType.Alpha: pixel [BitmapAlphaOffset] = (byte)(pixel [GdkRedOffset] - white_pixel [GdkRedOffset] + byte.MaxValue); break; case TransparencyType.Color: if ( pixel [GdkRedOffset] == background.R && pixel [GdkGreenOffset] == background.G && pixel [GdkBlueOffset] == background.B) { const int AlphaFullyTransparent = 0; pixel [BitmapAlphaOffset] = AlphaFullyTransparent; } break; } byte temporary = pixel [GdkRedOffset]; pixel [BitmapBlueOffset] = pixel [GdkBlueOffset]; pixel [BitmapRedOffset] = temporary; const int PixelSize = 4; pixel += PixelSize; white_pixel += PixelSize; } row += rowstride; white_row += white_rowstride; } } if (transparencyType == TransparencyType.Alpha) g_object_unref (white_pixbuf); g_object_unref (drawable); Bitmap bitmap = new Bitmap (painted_area.Width, painted_area.Height, rowstride, PixelFormat.Format32bppPArgb, pixel_data); Graphics g; bool graphics_is_from_hdc = false; switch (deviceContextType) { case DeviceContextType.Graphics: g = (Graphics)dc; break; case DeviceContextType.Native: g = Graphics.FromHdc (dc.GetHdc ()); break; default: g = dc as Graphics; if (g == null) { graphics_is_from_hdc = true; g = Graphics.FromHdc (dc.GetHdc ()); } else graphics_is_from_hdc = false; break; } painted_area.Offset (bounds.X, bounds.Y); g.DrawImage (bitmap, painted_area.Location); switch (deviceContextType) { case DeviceContextType.Graphics: break; case DeviceContextType.Native: g.Dispose (); dc.ReleaseHdc (); break; default: if (graphics_is_from_hdc) { g.Dispose (); dc.ReleaseHdc (); } break; } bitmap.Dispose (); g_object_unref (pixbuf); } void Paint (GdkDrawablePointer drawable, GdkGCPointer gc, Rectangle rectangle, WidgetType widgetType, out GdkPixbufPointer pixbuf, out IntPtr pixelData, out int rowstride, Rectangle clippingArea, Painter painter, Rectangle excludedArea) { gdk_draw_rectangle (drawable, gc, true, clippingArea.X, clippingArea.Y, clippingArea.Width, clippingArea.Height); painter.Paint (styles [(int)widgetType], drawable, new GdkRectangle(clippingArea), widgets [(int)widgetType], 0, 0, rectangle.Width, rectangle.Height, this); if (excludedArea.Width != 0) gdk_draw_rectangle (drawable, gc, true, excludedArea.X, excludedArea.Y, excludedArea.Width, excludedArea.Height); if ( (pixbuf = gdk_pixbuf_new (GdkColorspace.GDK_COLORSPACE_RGB, true, 8, clippingArea.Width, clippingArea.Height)) == IntPtr.Zero || gdk_pixbuf_get_from_drawable (pixbuf, drawable, IntPtr.Zero, clippingArea.X, clippingArea.Y, 0, 0, clippingArea.Width, clippingArea.Height) == IntPtr.Zero) throw new OutOfMemoryException (); pixelData = gdk_pixbuf_get_pixels (pixbuf); rowstride = gdk_pixbuf_get_rowstride (pixbuf); } abstract class Painter { public virtual void AttachStyle (WidgetType widgetType, GdkDrawablePointer drawable, GtkPlus gtkPlus) { gtkPlus.styles [(int)widgetType] = gtk_style_attach (gtkPlus.styles [(int)widgetType], drawable); } public abstract void Paint (GtkStylePointer style, GdkWindowPointer window, GdkRectangle area, GtkWidgetPointer widget, gint x, gint y, gint width, gint height, GtkPlus gtkPlus); } enum TransparencyType { None, Color, Alpha } enum DeviceContextType { Unknown, Graphics, Native } #endregion #region Painters #region Button class ButtonPainter : Painter { bool @default; GtkPlusState state; public void Configure (bool @default, GtkPlusState state) { this.@default = @default; this.state = state; } public override void Paint (GtkStylePointer style, GdkWindowPointer window, GdkRectangle area, GtkWidgetPointer widget, gint x, gint y, gint width, gint height, GtkPlus gtkPlus) { if (@default) { gtk_window_set_default (gtkPlus.window, widget); gtk_paint_box ( style, window, GtkStateType.GTK_STATE_NORMAL, GtkShadowType.GTK_SHADOW_IN, ref area, widget, "buttondefault", x, y, width, height); gtk_window_set_default (gtkPlus.window, IntPtr.Zero); } else gtk_paint_box ( style, window, (GtkStateType)state, state == GtkPlusState.Pressed ? GtkShadowType.GTK_SHADOW_IN : GtkShadowType.GTK_SHADOW_OUT, ref area, widget, "button", x, y, width, height); } } #endregion #region CheckBox, RadioButton abstract class ToggleButtonPainter : Painter { GtkPlusState state; GtkPlusToggleButtonValue value; public void Configure (GtkPlusState state, GtkPlusToggleButtonValue value) { this.state = state; this.value = value; } public override void Paint (GtkStylePointer style, GdkWindowPointer window, GdkRectangle area, GtkWidgetPointer widget, gint x, gint y, gint width, gint height, GtkPlus gtkPlus) { PaintFunction ( style, window, (GtkStateType)state, (GtkShadowType)value, ref area, widget, Detail, x, y, width,height); } protected abstract string Detail { get; } protected abstract ToggleButtonPaintFunction PaintFunction { get; } } delegate void ToggleButtonPaintFunction (GtkStylePointer style, GdkWindowPointer window, GtkStateType state_type, GtkShadowType shadow_type, ref GdkRectangle area, GtkWidgetPointer widget, string detail, gint x, gint y, gint width, gint height); class CheckBoxPainter : ToggleButtonPainter { protected override string Detail { get { return "checkbutton"; } } protected override ToggleButtonPaintFunction PaintFunction { get { return gtk_paint_check; } } } class RadioButtonPainter : ToggleButtonPainter { protected override string Detail { get { return "radiobutton"; } } protected override ToggleButtonPaintFunction PaintFunction { get { return gtk_paint_option; } } } #endregion #region ComboBox class ComboBoxDropDownButtonPainter : Painter { GtkPlusState state; public void Configure (GtkPlusState state) { this.state = state; } public override void AttachStyle (WidgetType widgetType, IntPtr drawable, GtkPlus gtkPlus) { gtkPlus.combo_box_drop_down_toggle_button_style = gtk_style_attach (gtkPlus.combo_box_drop_down_toggle_button_style, drawable); gtkPlus.combo_box_drop_down_arrow_style = gtk_style_attach (gtkPlus.combo_box_drop_down_arrow_style, drawable); } public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { GtkShadowType shadow_type; switch (state) { case GtkPlusState.Disabled: shadow_type = GtkShadowType.GTK_SHADOW_ETCHED_IN; break; case GtkPlusState.Pressed: shadow_type = GtkShadowType.GTK_SHADOW_IN; break; default: shadow_type = GtkShadowType.GTK_SHADOW_OUT; break; } gtk_paint_box ( gtkPlus.combo_box_drop_down_toggle_button_style, window, (GtkStateType)state, shadow_type, ref area, gtkPlus.combo_box_drop_down_toggle_button, "button", x, y, width, height); GtkMisc misc = (GtkMisc)Marshal.PtrToStructure (gtkPlus.combo_box_drop_down_arrow, typeof (GtkMisc)); int extent = (int)(Math.Min (width - misc.xpad * 2, height - misc.ypad * 2) * GetWidgetStyleSingle (gtkPlus.combo_box_drop_down_arrow, "arrow-scaling")); gtk_paint_arrow ( gtkPlus.combo_box_drop_down_arrow_style, window, (GtkStateType)state, GtkShadowType.GTK_SHADOW_NONE, ref area, gtkPlus.combo_box_drop_down_arrow, "arrow", GtkArrowType.GTK_ARROW_DOWN, true, (int)Math.Floor (x + misc.xpad + (width - extent) * misc.xalign), (int)Math.Floor (y + misc.ypad + (height - extent) * misc.yalign), extent, extent); } } class ComboBoxBorderPainter : Painter { public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_shadow ( style, window, GtkStateType.GTK_STATE_NORMAL, GtkShadowType.GTK_SHADOW_IN, ref area, widget, "combobox", x, y, width, height); } } #endregion #region GroupBox class GroupBoxPainter : Painter { GtkPlusState state; public void Configure (GtkPlusState state) { this.state = state; } public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_shadow ( style, window, (GtkStateType)state, GtkShadowType.GTK_SHADOW_ETCHED_IN, ref area, widget, "frame", x, y, width, height); } } #endregion #region Header class HeaderPainter : Painter { GtkPlusState state; public void Configure (GtkPlusState state) { this.state = state; } public override void AttachStyle (WidgetType widgetType, IntPtr drawable, GtkPlus gtkPlus) { gtkPlus.tree_view_column_button_style = gtk_style_attach (gtkPlus.tree_view_column_button_style, drawable); } public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_box ( gtkPlus.tree_view_column_button_style, window, (GtkStateType)state, state == GtkPlusState.Pressed ? GtkShadowType.GTK_SHADOW_IN : GtkShadowType.GTK_SHADOW_OUT, ref area, gtkPlus.tree_view_column_button, "button", x, y, width, height); } } #endregion #region ProgressBar class ProgressBarBarPainter : Painter { public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_box ( style, window, GtkStateType.GTK_STATE_NORMAL, GtkShadowType.GTK_SHADOW_IN, ref area, widget, "trough", x, y, width, height); } } class ProgressBarChunkPainter : Painter { public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_box ( style, window, GtkStateType.GTK_STATE_PRELIGHT, GtkShadowType.GTK_SHADOW_OUT, ref area, widget, "bar", x, y, width, height); } } #endregion #region ScrollBar class ScrollBarArrowButtonPainter : Painter { GtkPlusState state; bool horizontal; bool up_or_left; public void Configure (GtkPlusState state, bool horizontal, bool upOrLeft) { this.state = state; this.horizontal = horizontal; up_or_left = upOrLeft; } public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gboolean can_focus; g_object_get (widget, "can-focus", out can_focus, IntPtr.Zero); if (can_focus) { int focus_line_width; int focus_padding; gtk_widget_style_get (widget, "focus-line-width", out focus_line_width, "focus-padding", out focus_padding, IntPtr.Zero); int focus_width = focus_line_width + focus_padding; if (horizontal) { y -= focus_width; height -= 2 * focus_width; } else { x -= focus_width; width -= 2 * focus_width; } } GtkShadowType shadow_type = state == GtkPlusState.Pressed ? GtkShadowType.GTK_SHADOW_IN : GtkShadowType.GTK_SHADOW_OUT; string detail = horizontal ? "hscrollbar" : "vscrollbar"; gtk_paint_box ( style, window, (GtkStateType)state, shadow_type, ref area, widget, detail, x, y, width, height); width /= 2; height /= 2; x += width / 2; y += height / 2; if (state == GtkPlusState.Pressed) { int arrow_displacement_x; int arrow_displacement_y; gtk_widget_style_get (widget, "arrow-displacement-x", out arrow_displacement_x, "arrow-displacement-y", out arrow_displacement_y, IntPtr.Zero); x += arrow_displacement_x; y += arrow_displacement_y; } gtk_paint_arrow ( style, window, (GtkStateType)state, shadow_type, ref area, widget, detail, horizontal ? up_or_left ? GtkArrowType.GTK_ARROW_LEFT : GtkArrowType.GTK_ARROW_RIGHT : up_or_left ? GtkArrowType.GTK_ARROW_UP : GtkArrowType.GTK_ARROW_DOWN, true, x, y, width, height); } } abstract class RangeThumbButtonPainter : Painter { GtkPlusState state; bool horizontal; protected bool Horizontal { get { return horizontal; } } public void Configure (GtkPlusState state, bool horizontal) { this.state = state; this.horizontal = horizontal; } public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_slider ( style, window, (GtkStateType)state, state == GtkPlusState.Pressed && GetWidgetStyleBoolean (widget, "activate-slider") ? GtkShadowType.GTK_SHADOW_IN : GtkShadowType.GTK_SHADOW_OUT, ref area, widget, Detail, x, y, width, height, horizontal ? GtkOrientation.GTK_ORIENTATION_HORIZONTAL : GtkOrientation.GTK_ORIENTATION_VERTICAL); } protected abstract string Detail { get; } } class ScrollBarThumbButtonPainter : RangeThumbButtonPainter { protected override string Detail { get { return "slider"; } } } class ScrollBarTrackPainter : Painter { GtkPlusState state; bool up_or_left; public void Configure (GtkPlusState state, bool upOrLeft) { this.state = state; up_or_left = upOrLeft; } public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_box ( style, window, state == GtkPlusState.Pressed ? GtkStateType.GTK_STATE_ACTIVE : GtkStateType.GTK_STATE_INSENSITIVE, GtkShadowType.GTK_SHADOW_IN, ref area, widget, GetWidgetStyleBoolean (widget, "trough-side-details") ? up_or_left ? "trough-upper" : "trough-lower" : "trough", x, y, width, height); } } #endregion #region StatusBar class StatusBarGripperPainter : Painter { public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_resize_grip ( style, window, GtkStateType.GTK_STATE_NORMAL, ref area, widget, "statusbar", GdkWindowEdge.GDK_WINDOW_EDGE_SOUTH_EAST, x, y, width, height); } } #endregion #region TabWidget class TabWidgetPanePainter : Painter { public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_box_gap ( style, window, GtkStateType.GTK_STATE_NORMAL, GtkShadowType.GTK_SHADOW_OUT, ref area, widget, "notebook", x, y, width, height, GtkPositionType.GTK_POS_TOP, 0, 0); } } class TabWidgetTabItemPainter : Painter { GtkPlusState state; public void Configure (GtkPlusState state) { this.state = state; } public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_extension ( style, window, (GtkStateType)state, GtkShadowType.GTK_SHADOW_OUT, ref area, widget, "tab", x, y, width, height, GtkPositionType.GTK_POS_BOTTOM); } } #endregion #region TextBox class TextBoxPainter : Painter { GtkPlusState state; public void Configure (GtkPlusState state) { this.state = state; } public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_shadow ( style, window, GtkStateType.GTK_STATE_NORMAL, GtkShadowType.GTK_SHADOW_IN, ref area, widget, "entry", x, y, width, height); GtkStyle style_structure = (GtkStyle)Marshal.PtrToStructure (style, typeof (GtkStyle)); x += style_structure.xthickness; y += style_structure.ythickness; width -= 2 * style_structure.xthickness; height -= 2 * style_structure.ythickness; gtk_paint_flat_box ( style, window, (GtkStateType)state, GtkShadowType.GTK_SHADOW_NONE, ref area, widget, "entry_bg", x, y, width, height); } } #endregion #region ToolBar class ToolBarPainter : Painter { public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_box ( style, window, GtkStateType.GTK_STATE_NORMAL, GetWidgetStyleShadowType (widget), ref area, widget, "toolbar", x, y, width, height); } } class ToolBarButtonPainter : Painter { GtkPlusState state; public void Configure (GtkPlusState state) { this.state = state; } public override void AttachStyle (WidgetType widgetType, IntPtr drawable, GtkPlus gtkPlus) { gtkPlus.tool_bar_button_style = gtk_style_attach (gtkPlus.tool_bar_button_style, drawable); } public override void Paint (GtkStylePointer style, GdkWindowPointer window, GdkRectangle area, GtkWidgetPointer widget, gint x, gint y, gint width, gint height, GtkPlus gtkPlus) { gtk_paint_box ( gtkPlus.tool_bar_button_style, window, (GtkStateType)state, state == GtkPlusState.Pressed ? GtkShadowType.GTK_SHADOW_IN : GtkShadowType.GTK_SHADOW_OUT, ref area, gtkPlus.tool_bar_button, "button", x, y, width, height); } } class ToolBarCheckedButtonPainter : Painter { public override void AttachStyle (WidgetType widgetType, IntPtr drawable, GtkPlus gtkPlus) { gtkPlus.tool_bar_toggle_button_style = gtk_style_attach (gtkPlus.tool_bar_toggle_button_style, drawable); } public override void Paint (GtkStylePointer style, GdkWindowPointer window, GdkRectangle area, GtkWidgetPointer widget, gint x, gint y, gint width, gint height, GtkPlus gtkPlus) { gtk_paint_box ( gtkPlus.tool_bar_toggle_button_style, window, GtkStateType.GTK_STATE_ACTIVE, GtkShadowType.GTK_SHADOW_IN, ref area, gtkPlus.tool_bar_toggle_button, "button", x, y, width, height); } } #endregion #region TrackBar class TrackBarTrackPainter : Painter { public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_box ( style, window, GtkStateType.GTK_STATE_ACTIVE, GtkShadowType.GTK_SHADOW_IN, ref area, widget, "trough", x, y, width, height); } } class TrackBarThumbPainter : RangeThumbButtonPainter { protected override string Detail { get { return Horizontal ? "hscale" : "vscale"; } } } #endregion #region TreeView class TreeViewGlyphPainter : Painter { bool closed; public void Configure (bool closed) { this.closed = closed; } public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { gtk_paint_expander ( style, window, GtkStateType.GTK_STATE_NORMAL, ref area, widget, "treeview", x + width / 2, y + height / 2, closed ? GtkExpanderStyle.GTK_EXPANDER_COLLAPSED : GtkExpanderStyle.GTK_EXPANDER_EXPANDED); } } #endregion #region UpDown class UpDownPainter : Painter { bool up; GtkPlusState state; public void Configure (bool up, GtkPlusState state) { this.up = up; this.state = state; } public override void Paint (IntPtr style, IntPtr window, GdkRectangle area, IntPtr widget, int x, int y, int width, int height, GtkPlus gtkPlus) { GtkShadowType shadow_type = GetWidgetStyleShadowType (widget); if (shadow_type != GtkShadowType.GTK_SHADOW_NONE) gtk_paint_box ( style, window, GtkStateType.GTK_STATE_NORMAL, shadow_type, ref area, widget, "spinbutton", x, y - (up ? 0 : height), width, height * 2); shadow_type = state == GtkPlusState.Pressed ? GtkShadowType.GTK_SHADOW_IN : GtkShadowType.GTK_SHADOW_OUT; gtk_paint_box ( style, window, (GtkStateType)state, shadow_type, ref area, widget, up ? "spinbutton_up" : "spinbutton_down", x, y, width, height); if (up) y += 2; height -= 2; width -= 3; x += 1; int w = width / 2; w -= w % 2 - 1; int h = (w + 1) / 2; x += (width - w) / 2; y += (height - h) / 2; height = h; width = w; gtk_paint_arrow ( style, window, (GtkStateType)state, shadow_type, ref area, widget, "spinbutton", up ? GtkArrowType.GTK_ARROW_UP : GtkArrowType.GTK_ARROW_DOWN, true, x, y, width, height); } } #endregion #endregion #region Widget types const WidgetType WidgetTypeNotNeeded = (WidgetType)0; enum WidgetType { Button, CheckBox, ComboBox, GroupBox, ProgressBar, RadioButton, #region ScrollBar HScrollBar, VScrollBar, #endregion StatusBar, TabWidget, TextBox, ToolBar, #region TrackBar HorizontalTrackBar, VerticalTrackBar, #endregion TreeView, UpDown, } #endregion #region GTK+ utility methods static GtkShadowType GetWidgetStyleShadowType (GtkWidgetPointer widget) { GtkShadowType result; gtk_widget_style_get (widget, "shadow-type", out result, IntPtr.Zero); return result; } static int GetWidgetStyleInteger (GtkWidgetPointer widget, string propertyName) { gint result; gtk_widget_style_get (widget, propertyName, out result, IntPtr.Zero); return result; } static float GetWidgetStyleSingle (GtkWidgetPointer widget, string propertyName) { gfloat result; gtk_widget_style_get (widget, propertyName, out result, IntPtr.Zero); return result; } static bool GetWidgetStyleBoolean (GtkWidgetPointer widget, string propertyName) { gboolean result; gtk_widget_style_get (widget, propertyName, out result, IntPtr.Zero); return result; } #region GetFirstChildWidgetOfType static class GetFirstChildWidgetOfType { public static GtkWidgetPointer Get (GtkContainerPointer parent, GType childType) { Type = childType; Result = GtkWidgetPointer.Zero; ContainersToSearch = new ArrayList (); ContainersToSearch.Add (parent); while (true) { ArrayList containers_to_search = ContainersToSearch; ContainersToSearch = new ArrayList (); foreach (GtkContainerPointer container in containers_to_search) { gtk_widget_realize (container); gtk_container_forall (container, Callback, gpointer.Zero); if (Result != GtkWidgetPointer.Zero) return Result; } if (ContainersToSearch.Count == 0) return GtkWidgetPointer.Zero; } } static void Callback (GtkWidgetPointer widget, gpointer data) { if (Result != IntPtr.Zero) return; if (g_type_check_instance_is_a (widget, Type)) Result = widget; else if (g_type_check_instance_is_a (widget, gtk_container_get_type ())) ContainersToSearch.Add (widget); } static GType Type; static GtkWidgetPointer Result; static ArrayList ContainersToSearch; } #endregion static GtkStylePointer GetWidgetStyle (GtkWidgetPointer widget) { return gtk_rc_get_style (widget); } #endregion #region Platform Invoke #region Library names #if VISUAL_STYLES_USE_GTKPLUS_ON_WINDOWS const string GobjectLibraryName = "libgobject-2.0-0.dll"; const string GdkLibraryName = "libgdk-win32-2.0-0.dll"; const string GdkPixbufLibraryName = "libgdk_pixbuf-2.0-0.dll"; const string GtkLibraryName = "libgtk-win32-2.0-0.dll"; #else const string GobjectLibraryName = "libgobject-2.0.so"; const string GdkLibraryName = "libgdk-x11-2.0.so"; const string GdkPixbufLibraryName = "libgdk_pixbuf-2.0.so"; const string GtkLibraryName = "libgtk-x11-2.0.so"; #endif #endregion #region GDK [DllImport (GdkLibraryName)] static extern void gdk_draw_rectangle (GdkDrawablePointer drawable, GdkGCPointer gc, gboolean filled, gint x, gint y, gint width, gint height); [DllImport (GdkLibraryName)] static extern GdkGCPointer gdk_gc_new (GdkDrawablePointer drawable); [DllImport (GdkLibraryName)] static extern void gdk_gc_set_rgb_fg_color (GdkGCPointer gc, ref GdkColor color); [DllImport (GdkLibraryName)] static extern GdkPixbufPointer gdk_pixbuf_get_from_drawable (GdkPixbufPointer dest, GdkDrawablePointer src, GdkColormapPointer cmap, int src_x, int src_y, int dest_x, int dest_y, int width, int height); [DllImport (GdkLibraryName)] static extern GdkPixmapPointer gdk_pixmap_new (GdkDrawablePointer drawable, gint width, gint height, gint depth); struct GdkColor{ public guint32 pixel; public guint16 red; public guint16 green; public guint16 blue; public GdkColor (Color value) { pixel = 0; red = (guint16)(value.R << 8); green = (guint16)(value.G << 8); blue = (guint16)(value.B << 8); } } internal struct GdkRectangle { public gint x; public gint y; public gint width; public gint height; public GdkRectangle (Rectangle value) { x = value.X; y = value.Y; width = value.Width; height = value.Height; } } #endregion #region GdkPixbuf [DllImport (GdkPixbufLibraryName)] static extern IntPtr gdk_pixbuf_get_pixels (GdkPixbufPointer pixbuf); [DllImport (GdkPixbufLibraryName)] static extern int gdk_pixbuf_get_rowstride (GdkPixbufPointer pixbuf); [DllImport (GdkPixbufLibraryName)] static extern GdkPixbufPointer gdk_pixbuf_new (GdkColorspace colorspace, gboolean has_alpha, int bits_per_sample, int width, int height); enum GdkColorspace { GDK_COLORSPACE_RGB } #endregion #region GTK #region Functions [DllImport (GtkLibraryName)] static extern gboolean gtk_init_check (ref int argc, ref string[] argv); [DllImport (GtkLibraryName)] static extern IntPtr gtk_check_version (guint required_major, guint required_minor, guint required_micro); [DllImport (GtkLibraryName)] static extern void gtk_container_add (GtkContainerPointer container, GtkWidgetPointer widget); [DllImport (GtkLibraryName)] static extern void gtk_container_forall (GtkContainerPointer container, GtkCallback callback, gpointer callback_data); [UnmanagedFunctionPointer (CallingConvention.Cdecl)] delegate void GtkCallback (GtkWidgetPointer widget, gpointer data); [DllImport (GtkLibraryName)] static extern void gtk_object_destroy (GtkObjectPointer @object); [DllImport (GtkLibraryName)] static extern GtkStylePointer gtk_rc_get_style (GtkWidgetPointer widget); [DllImport (GtkLibraryName)] static extern GtkStylePointer gtk_style_attach (GtkStylePointer style, GdkWindowPointer window); [DllImport (GtkLibraryName)] static extern void gtk_widget_realize (GtkWidgetPointer widget); [DllImport (GtkLibraryName)] static extern void gtk_widget_style_get (GtkWidgetPointer widget, string property, out gint value, IntPtr nullTerminator); [DllImport (GtkLibraryName)] static extern void gtk_widget_style_get (GtkWidgetPointer widget, string property, out gfloat value, IntPtr nullTerminator); [DllImport (GtkLibraryName)] static extern void gtk_widget_style_get (GtkWidgetPointer widget, string property1, out gint value1, string property2, out gint value2, IntPtr nullTerminator); [DllImport (GtkLibraryName)] static extern void gtk_widget_style_get (GtkWidgetPointer widget, string property, out GtkShadowType value, IntPtr nullTerminator); [DllImport (GtkLibraryName)] static extern void gtk_widget_style_get (GtkWidgetPointer widget, string property, out gboolean value, IntPtr nullTerminator); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_window_new (GtkWindowType type); [DllImport (GtkLibraryName)] static extern void gtk_window_set_default (GtkWindowPointer window, GtkWidgetPointer default_widget); [DllImport (GtkLibraryName)] static extern GtkObjectPointer gtk_adjustment_new (gdouble value, gdouble lower, gdouble upper, gdouble step_increment, gdouble page_increment, gdouble page_size); [DllImport (GtkLibraryName)] static extern GtkStylePointer gtk_widget_get_style (GtkWidgetPointer widget); [DllImport (GtkLibraryName)] static extern GtkTreeViewColumnPointer gtk_tree_view_column_new (); [DllImport (GtkLibraryName)] static extern gint gtk_tree_view_insert_column ( GtkTreeViewPointer tree_view, GtkTreeViewColumnPointer column, gint position); [DllImport (GtkLibraryName)] static extern void gtk_toolbar_insert ( GtkToolbarPointer toolbar, GtkToolItemPointer item, gint pos); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_bin_get_child (GtkBinPointer bin); #region Widget type [DllImport (GtkLibraryName)] static extern GType gtk_arrow_get_type (); [DllImport (GtkLibraryName)] static extern GType gtk_container_get_type (); [DllImport (GtkLibraryName)] static extern GType gtk_toggle_button_get_type (); #endregion #region Widget creation [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_button_new (); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_check_button_new (); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_combo_box_entry_new (); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_entry_new (); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_fixed_new (); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_frame_new (string label); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_hscale_new_with_range (gdouble min, gdouble max, gdouble step); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_hscrollbar_new (GtkAdjustmentPointer adjustment); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_notebook_new (); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_progress_bar_new (); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_radio_button_new (GSListPointer group); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_spin_button_new (GtkAdjustmentPointer adjustment, gdouble climb_rate, guint digits); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_statusbar_new (); [DllImport (GtkLibraryName)] static extern GtkToolItemPointer gtk_toggle_tool_button_new (); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_toolbar_new (); [DllImport (GtkLibraryName)] static extern GtkToolItemPointer gtk_tool_button_new ( GtkWidgetPointer icon_widget, string label); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_tree_view_new (); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_vscale_new_with_range (gdouble min, gdouble max, gdouble step); [DllImport (GtkLibraryName)] static extern GtkWidgetPointer gtk_vscrollbar_new (GtkAdjustmentPointer adjustment); #endregion #region Painting [DllImport (GtkLibraryName)] static extern void gtk_paint_arrow (GtkStylePointer style, GdkWindowPointer window, GtkStateType state_type, GtkShadowType shadow_type, ref GdkRectangle area, GtkWidgetPointer widget, string detail, GtkArrowType arrow_type, gboolean fill, gint x, gint y, gint width, gint height); [DllImport (GtkLibraryName)] static extern void gtk_paint_box (GtkStylePointer style, GdkWindowPointer window, GtkStateType state_type, GtkShadowType shadow_type, ref GdkRectangle area, GtkWidgetPointer widget, string detail, gint x, gint y, gint width, gint height); [DllImport (GtkLibraryName)] static extern void gtk_paint_box_gap ( GtkStylePointer style, GdkWindowPointer window, GtkStateType state_type, GtkShadowType shadow_type, ref GdkRectangle area, GtkWidgetPointer widget, string detail, gint x, gint y, gint width, gint height, GtkPositionType gap_side, gint gap_x, gint gap_width); [DllImport (GtkLibraryName)] static extern void gtk_paint_check (GtkStylePointer style, GdkWindowPointer window, GtkStateType state_type, GtkShadowType shadow_type, ref GdkRectangle area, GtkWidgetPointer widget, string detail, gint x, gint y, gint width, gint height); [DllImport (GtkLibraryName)] static extern void gtk_paint_expander ( GtkStylePointer style, GdkWindowPointer window, GtkStateType state_type, ref GdkRectangle area, GtkWidgetPointer widget, string detail, gint x, gint y, GtkExpanderStyle expander_style); [DllImport (GtkLibraryName)] static extern void gtk_paint_extension ( GtkStylePointer style, GdkWindowPointer window, GtkStateType state_type, GtkShadowType shadow_type, ref GdkRectangle area, GtkWidgetPointer widget, string detail, gint x, gint y, gint width, gint height, GtkPositionType gap_side); [DllImport (GtkLibraryName)] static extern void gtk_paint_flat_box ( GtkStylePointer style, GdkWindowPointer window, GtkStateType state_type, GtkShadowType shadow_type, ref GdkRectangle area, GtkWidgetPointer widget, string detail, gint x, gint y, gint width, gint height); [DllImport (GtkLibraryName)] static extern void gtk_paint_option (GtkStylePointer style, GdkWindowPointer window, GtkStateType state_type, GtkShadowType shadow_type, ref GdkRectangle area, GtkWidgetPointer widget, string detail, gint x, gint y, gint width, gint height); [DllImport (GtkLibraryName)] static extern void gtk_paint_resize_grip ( GtkStylePointer style, GdkWindowPointer window, GtkStateType state_type, ref GdkRectangle area, GtkWidgetPointer widget, string detail, GdkWindowEdge edge, gint x, gint y, gint width, gint height); [DllImport (GtkLibraryName)] static extern void gtk_paint_shadow (GtkStylePointer style, GdkWindowPointer window, GtkStateType state_type, GtkShadowType shadow_type, ref GdkRectangle area, GtkWidgetPointer widget, string detail, gint x, gint y, gint width, gint height); [DllImport (GtkLibraryName)] static extern void gtk_paint_slider ( GtkStylePointer style, GdkWindowPointer window, GtkStateType state_type, GtkShadowType shadow_type, ref GdkRectangle area, GtkWidgetPointer widget, string detail, gint x, gint y, gint width, gint height, GtkOrientation orientation); #endregion #endregion #region Enumerations internal enum GtkShadowType { GTK_SHADOW_NONE, GTK_SHADOW_IN, GTK_SHADOW_OUT, GTK_SHADOW_ETCHED_IN, GTK_SHADOW_ETCHED_OUT } enum GtkStateType { GTK_STATE_NORMAL, GTK_STATE_ACTIVE, GTK_STATE_PRELIGHT, GTK_STATE_SELECTED, GTK_STATE_INSENSITIVE } enum GtkWindowType { GTK_WINDOW_TOPLEVEL, GTK_WINDOW_POPUP } enum GtkArrowType { GTK_ARROW_UP, GTK_ARROW_DOWN, GTK_ARROW_LEFT, GTK_ARROW_RIGHT, GTK_ARROW_NONE } enum GtkOrientation { GTK_ORIENTATION_HORIZONTAL, GTK_ORIENTATION_VERTICAL } enum GtkExpanderStyle { GTK_EXPANDER_COLLAPSED, GTK_EXPANDER_SEMI_COLLAPSED, GTK_EXPANDER_SEMI_EXPANDED, GTK_EXPANDER_EXPANDED } enum GtkPositionType { GTK_POS_LEFT, GTK_POS_RIGHT, GTK_POS_TOP, GTK_POS_BOTTOM } enum GtkWidgetFlags : uint { GTK_CAN_DEFAULT = 1 << 13, } enum GdkWindowEdge { GDK_WINDOW_EDGE_NORTH_WEST, GDK_WINDOW_EDGE_NORTH, GDK_WINDOW_EDGE_NORTH_EAST, GDK_WINDOW_EDGE_WEST, GDK_WINDOW_EDGE_EAST, GDK_WINDOW_EDGE_SOUTH_WEST, GDK_WINDOW_EDGE_SOUTH, GDK_WINDOW_EDGE_SOUTH_EAST } #endregion #region Structures struct GtkStyle { #pragma warning disable 169 GObject parent_instance; [MarshalAs(UnmanagedType.ByValArray, SizeConst=5)] GdkColor[] fg; [MarshalAs(UnmanagedType.ByValArray, SizeConst=5)] GdkColor[] bg; [MarshalAs(UnmanagedType.ByValArray, SizeConst=5)] GdkColor[] light; [MarshalAs(UnmanagedType.ByValArray, SizeConst=5)] GdkColor[] dark; [MarshalAs(UnmanagedType.ByValArray, SizeConst=5)] GdkColor[] mid; [MarshalAs(UnmanagedType.ByValArray, SizeConst=5)] GdkColor[] text; [MarshalAs(UnmanagedType.ByValArray, SizeConst=5)] GdkColor[] @base; [MarshalAs(UnmanagedType.ByValArray, SizeConst=5)] GdkColor[] text_aa; GdkColor black; GdkColor white; PangoFontDescriptionPointer font_desc; #pragma warning restore 169 public gint xthickness; public gint ythickness; } struct GtkWidget { #pragma warning disable 169 GtkObject @object; guint16 private_flags; guint8 state; guint8 saved_state; string name; GtkStylePointer style; GtkRequisition requisition; #pragma warning restore 169 public GtkAllocation allocation; #pragma warning disable 169 GdkWindowPointer window; GtkWidgetPointer parent; #pragma warning restore 169 } struct GtkObject { #pragma warning disable 169 GInitiallyUnowned parent_instance; #pragma warning restore 169 public guint32 flags; } struct GtkRequisition { #pragma warning disable 169 gint width; gint height; #pragma warning restore 169 } struct GtkMisc { #pragma warning disable 169 GtkWidget widget; #pragma warning restore 169 public gfloat xalign; public gfloat yalign; public guint16 xpad; public guint16 ypad; } struct GtkTreeViewColumn { #pragma warning disable 169 GtkObject parent; GtkWidgetPointer tree_view; #pragma warning restore 169 public GtkWidgetPointer button; } #endregion #region Macros static void GTK_WIDGET_SET_FLAGS (GtkWidgetPointer wid, GtkWidgetFlags flag) { GtkObject @object = (GtkObject)Marshal.PtrToStructure (wid, typeof (GtkObject)); @object.flags |= (guint32)flag; Marshal.StructureToPtr (@object, wid, false); } #endregion #endregion #region GObject [DllImport (GobjectLibraryName)] static extern gpointer g_object_ref (gpointer @object); [DllImport (GobjectLibraryName)] static extern void g_object_unref (gpointer @object); [DllImport (GobjectLibraryName)] static extern gboolean g_type_check_instance_is_a (GTypeInstancePointer type_instance, GType iface_type); [DllImport (GobjectLibraryName)] static extern void g_object_get (gpointer @object, string property_name, out gboolean value, IntPtr nullTerminator); const int G_TYPE_FUNDAMENTAL_SHIFT = 2; enum G_TYPE { } struct GTypeInstance { #pragma warning disable 169 GTypeClassPointer g_class; #pragma warning restore 169 } internal struct GObject { #pragma warning disable 169 GTypeInstance g_type_instance; guint ref_count; GDataPointer qdata; #pragma warning restore 169 } #endregion #endregion } #region Exposed constants enum GtkPlusToggleButtonValue { Unchecked = GtkPlus.GtkShadowType.GTK_SHADOW_OUT, Mixed = GtkPlus.GtkShadowType.GTK_SHADOW_ETCHED_IN, Checked = GtkPlus.GtkShadowType.GTK_SHADOW_IN } enum GtkPlusState { Normal, Pressed, Hot, Selected, Disabled } #endregion }