changeset 89:9f6c2c92be2b

Table, TableColumn and TableItem
author Frank Benoit <benoit@tionex.de>
date Wed, 16 Jan 2008 16:28:05 +0100
parents fe5c7d89efac
children 9ba02d7fb226
files dwt/internal/gtk/OS.d dwt/widgets/Display.d dwt/widgets/List.d dwt/widgets/Table.d dwt/widgets/TableColumn.d dwt/widgets/TableItem.d dwt/widgets/Widget.d
diffstat 7 files changed, 5514 insertions(+), 39 deletions(-) [+]
line wrap: on
line diff
--- a/dwt/internal/gtk/OS.d	Wed Jan 16 15:35:57 2008 +0100
+++ b/dwt/internal/gtk/OS.d	Wed Jan 16 16:28:05 2008 +0100
@@ -2369,61 +2369,61 @@
         return gdk_pixbuf_get_type();
     }
 
-    static bool GTK_IS_BUTTON( GTypeInstance* arg0 )
+    static bool GTK_IS_BUTTON( void* arg0 )
     {
         lock.lock();
         scope(exit) lock.unlock();
-        return  cast(bool)g_type_check_instance_is_a( arg0, gtk_button_get_type() );
+        return  cast(bool)g_type_check_instance_is_a( cast(GTypeInstance*)arg0, gtk_button_get_type() );
     }
 
-    static bool GTK_IS_WINDOW( GTypeInstance* arg0 )
+    static bool GTK_IS_WINDOW( void* arg0 )
     {
         lock.lock();
         scope(exit) lock.unlock();
-        return cast(bool)g_type_check_instance_is_a( arg0, gtk_window_get_type());
+        return cast(bool)g_type_check_instance_is_a( cast(GTypeInstance*)arg0, gtk_window_get_type());
     }
 
-    static bool GTK_IS_CELL_RENDERER_PIXBUF( GTypeInstance* arg0 )
+    static bool GTK_IS_CELL_RENDERER_PIXBUF( void* arg0 )
     {
         lock.lock();
         scope(exit) lock.unlock();
-        return cast(bool)g_type_check_instance_is_a( arg0, gtk_cell_renderer_pixbuf_get_type());
+        return cast(bool)g_type_check_instance_is_a( cast(GTypeInstance*)arg0, gtk_cell_renderer_pixbuf_get_type());
     }
 
-    static bool GTK_IS_CELL_RENDERER_TEXT( GTypeInstance* arg0 )
+    static bool GTK_IS_CELL_RENDERER_TEXT( void* arg0 )
     {
         lock.lock();
         scope(exit) lock.unlock();
-        return cast(bool)g_type_check_instance_is_a( arg0, gtk_cell_renderer_text_get_type());
+        return cast(bool)g_type_check_instance_is_a( cast(GTypeInstance*)arg0, gtk_cell_renderer_text_get_type());
     }
 
-    static bool GTK_IS_CELL_RENDERER_TOGGLE( GTypeInstance* arg0 )
+    static bool GTK_IS_CELL_RENDERER_TOGGLE( void* arg0 )
     {
         lock.lock();
         scope(exit) lock.unlock();
-        return cast(bool)g_type_check_instance_is_a( arg0, gtk_cell_renderer_toggle_get_type ());
+        return cast(bool)g_type_check_instance_is_a( cast(GTypeInstance*)arg0, gtk_cell_renderer_toggle_get_type ());
     }
 
-    static bool GTK_IS_CONTAINER( GTypeInstance* arg0 )
+    static bool GTK_IS_CONTAINER( void* arg0 )
     {
         lock.lock();
         scope(exit) lock.unlock();
-        return cast(bool)g_type_check_instance_is_a( arg0, gtk_container_get_type () );
+        return cast(bool)g_type_check_instance_is_a( cast(GTypeInstance*)arg0, gtk_container_get_type () );
     }
 
 
-    static bool GTK_IS_IMAGE_MENU_ITEM( GTypeInstance* arg0 )
+    static bool GTK_IS_IMAGE_MENU_ITEM( void* arg0 )
     {
         lock.lock();
         scope(exit) lock.unlock();
-        return cast(bool)g_type_check_instance_is_a( arg0, gtk_image_menu_item_get_type ());
+        return cast(bool)g_type_check_instance_is_a( cast(GTypeInstance*)arg0, gtk_image_menu_item_get_type ());
     }
 
-    static bool GTK_IS_PLUG( GTypeInstance* arg0 )
+    static bool GTK_IS_PLUG( void* arg0 )
     {
         lock.lock();
         scope(exit) lock.unlock();
-        return cast(bool)g_type_check_instance_is_a( arg0, gtk_plug_get_type () );
+        return cast(bool)g_type_check_instance_is_a( cast(GTypeInstance*)arg0, gtk_plug_get_type () );
     }
 
     // Should use d char[] instead for next two methods? - JJR
--- a/dwt/widgets/Display.d	Wed Jan 16 15:35:57 2008 +0100
+++ b/dwt/widgets/Display.d	Wed Jan 16 16:28:05 2008 +0100
@@ -222,12 +222,14 @@
     int treeSelectionLength;
 
     /* Set direction callback */
-    CallbackData setDirectionCallbackData;
+    CallbackData setDirectionProcCallbackData;
 
     /* Get all children callback */
     CallbackData allChildrenProcCallbackData;
     GList* allChildren;
 
+    CallbackData cellDataProcCallbackData;
+
     /* Settings callbacks */
     GtkWidget* shellHandle;
     bool settingsChanged, runSettingsFld;
@@ -720,10 +722,34 @@
     }
 }
 
-int /*long*/ cellDataProc (int /*long*/ tree_column, int /*long*/ cell, int /*long*/ tree_model, int /*long*/ iter, GtkWidget* data) {
-    Widget widget = getWidget (data);
-    if (widget is null) return 0;
-    return widget.cellDataProc (tree_column, cell, tree_model, iter, data);
+void doCellDataProc( GtkWidget* widget, GtkTreeViewColumn *tree_column, GtkCellRenderer *cell_renderer ){
+    cellDataProcCallbackData.display = this;
+    cellDataProcCallbackData.data = widget;
+    OS.gtk_tree_view_column_set_cell_data_func ( tree_column, cell_renderer, &cellDataProcFunc, &cellDataProcCallbackData, null );
+}
+
+private static extern(C) void cellDataProcFunc (
+    GtkTreeViewColumn *tree_column,
+    GtkCellRenderer *cell,
+    GtkTreeModel *tree_model,
+    GtkTreeIter *iter,
+    void* data)
+{
+    version(LOG) Stderr.formatln( "Display {}:", __LINE__ ).flush;
+    CallbackData* cbdata = cast(CallbackData*)data;
+    return cbdata.display.cellDataProc( tree_column, cell, tree_model, iter, cbdata.data );
+}
+
+void cellDataProc(
+    GtkTreeViewColumn *tree_column,
+    GtkCellRenderer *cell,
+    GtkTreeModel *tree_model,
+    GtkTreeIter *iter,
+    void* data)
+{
+    Widget widget = getWidget (cast(GtkWidget*)data);
+    if (widget is null) return;
+    widget.cellDataProc (tree_column, cell, tree_model, iter, data);
 }
 
 protected void checkDevice () {
@@ -1309,7 +1335,15 @@
     klass.render = &rendererRenderProcFunc;
     klass.get_size = &rendererGetSizeProcFunc;
 }
-private static extern(C) void rendererGetSizeProcFunc(GtkCellRenderer *cell, GtkWidget *handle, GdkRectangle *cell_area, int *x_offset, int *y_offset, int *width, int *height) {
+private static extern(C) void rendererGetSizeProcFunc(
+    GtkCellRenderer      *cell,
+    GtkWidget            *handle,
+    GdkRectangle         *cell_area,
+    int                  *x_offset,
+    int                  *y_offset,
+    int                  *width,
+    int                  *height)
+{
     version(LOG) Stderr.formatln( "Display {}:", __LINE__ ).flush;
     Display display = getCurrent ();
     Widget widget = display.getWidget (handle);
@@ -3502,11 +3536,19 @@
     this.data = data;
 }
 
-private static extern(C) int /*long*/ setDirectionProcFunc (GtkWidget* widget, void* direction) {
+
+void doSetDirectionProc( GtkWidget* widget, int direction ){
+    setDirectionProcCallbackData.display = this;
+    setDirectionProcCallbackData.data = cast(void*)direction;
+    OS.gtk_container_forall (cast(GtkContainer*)widget, cast(GtkCallback)&setDirectionProcFunc, &setDirectionProcCallbackData);
+}
+
+private static extern(C) int /*long*/ setDirectionProcFunc (GtkWidget* widget, void* data) {
     version(LOG) Stderr.formatln( "Display {}:", __LINE__ ).flush;
-    return setDirectionProc( widget, cast(int)direction );
+    CallbackData* cbdata = cast(CallbackData*)data;
+    return cbdata.display.setDirectionProc( widget, cast(int)cbdata.data );
 }
-static int /*long*/ setDirectionProc (GtkWidget* widget, int /*long*/ direction) {
+int /*long*/ setDirectionProc (GtkWidget* widget, int /*long*/ direction) {
     OS.gtk_widget_set_direction (widget,  direction);
     if (OS.GTK_IS_CONTAINER (cast(GTypeInstance*)widget)) {
         OS.gtk_container_forall (cast(GtkContainer*)widget, cast(GtkCallback)&setDirectionProcFunc, cast(void*)direction);
--- a/dwt/widgets/List.d	Wed Jan 16 15:35:57 2008 +0100
+++ b/dwt/widgets/List.d	Wed Jan 16 16:28:05 2008 +0100
@@ -802,9 +802,8 @@
     return OS.GTK_VERSION < OS.buildVERSION (2, 6, 5) ? 1 : result;
 }
 
-override int /*long*/ gtk_row_activated (int /*long*/ tree, int /*long*/ path, int /*long*/ column) {
+override void gtk_row_activated (GtkTreeView* tree, GtkTreePath* path, GtkTreeViewColumn* column){
     postEvent (DWT.DefaultSelection);
-    return 0;
 }
 
 void hookEvents () {
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/dwt/widgets/Table.d	Wed Jan 16 16:28:05 2008 +0100
@@ -0,0 +1,3464 @@
+/*******************************************************************************
+ * Copyright (c) 2000, 2007 IBM Corporation and others.
+ * All rights reserved. This program and the accompanying materials
+ * are made available under the terms of the Eclipse Public License v1.0
+ * which accompanies this distribution, and is available at
+ * http://www.eclipse.org/legal/epl-v10.html
+ *
+ * Contributors:
+ *     IBM Corporation - initial API and implementation
+ *******************************************************************************/
+module dwt.widgets.Table;
+
+
+
+import dwt.DWT;
+import dwt.DWTException;
+import dwt.events.SelectionEvent;
+import dwt.events.SelectionListener;
+import dwt.graphics.Color;
+import dwt.graphics.Font;
+import dwt.graphics.GC;
+import dwt.graphics.Image;
+import dwt.graphics.Point;
+import dwt.graphics.Rectangle;
+import dwt.internal.gtk.OS;
+import dwt.widgets.Composite;
+import dwt.widgets.TableItem;
+import dwt.widgets.TableColumn;
+import dwt.widgets.Listener;
+import dwt.widgets.ImageList;
+import dwt.widgets.Shell;
+import dwt.widgets.Decorations;
+import dwt.widgets.Menu;
+import dwt.widgets.Display;
+import dwt.widgets.ScrollBar;
+import dwt.widgets.Event;
+import dwt.widgets.Control;
+import dwt.widgets.TypedListener;
+
+static import tango.stdc.stringz;
+import tango.util.Convert;
+
+/**
+ * Instances of this class implement a selectable user interface
+ * object that displays a list of images and strings and issues
+ * notification when selected.
+ * <p>
+ * The item children that may be added to instances of this class
+ * must be of type <code>TableItem</code>.
+ * </p><p>
+ * Style <code>VIRTUAL</code> is used to create a <code>Table</code> whose
+ * <code>TableItem</code>s are to be populated by the client on an on-demand basis
+ * instead of up-front.  This can provide significant performance improvements for
+ * tables that are very large or for which <code>TableItem</code> population is
+ * expensive (for example, retrieving values from an external source).
+ * </p><p>
+ * Here is an example of using a <code>Table</code> with style <code>VIRTUAL</code>:
+ * <code><pre>
+ *  final Table table = new Table (parent, DWT.VIRTUAL | DWT.BORDER);
+ *  table.setItemCount (1000000);
+ *  table.addListener (DWT.SetData, new Listener () {
+ *      public void handleEvent (Event event) {
+ *          TableItem item = (TableItem) event.item;
+ *          int index = table.indexOf (item);
+ *          item.setText ("Item " + index);
+ *          System.out.println (item.getText ());
+ *      }
+ *  });
+ * </pre></code>
+ * </p><p>
+ * Note that although this class is a subclass of <code>Composite</code>,
+ * it does not make sense to add <code>Control</code> children to it,
+ * or set a layout on it.
+ * </p><p>
+ * <dl>
+ * <dt><b>Styles:</b></dt>
+ * <dd>SINGLE, MULTI, CHECK, FULL_SELECTION, HIDE_SELECTION, VIRTUAL</dd>
+ * <dt><b>Events:</b></dt>
+ * <dd>Selection, DefaultSelection, SetData, MeasureItem, EraseItem, PaintItem</dd>
+ * </dl>
+ * </p><p>
+ * Note: Only one of the styles SINGLE, and MULTI may be specified.
+ * </p><p>
+ * IMPORTANT: This class is <em>not</em> intended to be subclassed.
+ * </p>
+ */
+public class Table : Composite {
+    CallbackData treeSelectionCallbackData;
+    GtkWidget* modelHandle, checkRenderer;
+    int itemCount, columnCount, lastIndexOf, sortDirection;
+    GtkWidget* ignoreCell;
+    TableItem [] items;
+    TableColumn [] columns;
+    TableItem currentItem;
+    TableColumn sortColumn;
+    ImageList imageList, headerImageList;
+    bool firstCustomDraw;
+    int drawState, drawFlags;
+    GdkColor* drawForeground;
+    bool ownerDraw, ignoreSize;
+
+    static const int CHECKED_COLUMN = 0;
+    static const int GRAYED_COLUMN = 1;
+    static const int FOREGROUND_COLUMN = 2;
+    static const int BACKGROUND_COLUMN = 3;
+    static const int FONT_COLUMN = 4;
+    static const int FIRST_COLUMN = FONT_COLUMN + 1;
+    static const int CELL_PIXBUF = 0;
+    static const int CELL_TEXT = 1;
+    static const int CELL_FOREGROUND = 2;
+    static const int CELL_BACKGROUND = 3;
+    static const int CELL_FONT = 4;
+    static const int CELL_TYPES = CELL_FONT + 1;
+
+/**
+ * Constructs a new instance of this class given its parent
+ * and a style value describing its behavior and appearance.
+ * <p>
+ * The style value is either one of the style constants defined in
+ * class <code>DWT</code> which is applicable to instances of this
+ * class, or must be built by <em>bitwise OR</em>'ing together
+ * (that is, using the <code>int</code> "|" operator) two or more
+ * of those <code>DWT</code> style constants. The class description
+ * lists the style constants that are applicable to the class.
+ * Style bits are also inherited from superclasses.
+ * </p>
+ *
+ * @param parent a composite control which will be the parent of the new instance (cannot be null)
+ * @param style the style of control to construct
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the parent is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the parent</li>
+ *    <li>ERROR_INVALID_SUBCLASS - if this class is not an allowed subclass</li>
+ * </ul>
+ *
+ * @see DWT#SINGLE
+ * @see DWT#MULTI
+ * @see DWT#CHECK
+ * @see DWT#FULL_SELECTION
+ * @see DWT#HIDE_SELECTION
+ * @see DWT#VIRTUAL
+ * @see Widget#checkSubclass
+ * @see Widget#getStyle
+ */
+public this (Composite parent, int style) {
+    super (parent, checkStyle (style));
+}
+
+override void _addListener (int eventType, Listener listener) {
+    super._addListener (eventType, listener);
+    if (!ownerDraw) {
+        switch (eventType) {
+            case DWT.MeasureItem:
+            case DWT.EraseItem:
+            case DWT.PaintItem:
+                ownerDraw = true;
+                recreateRenderers ();
+                break;
+        }
+    }
+}
+
+TableItem _getItem (int index) {
+    if ((style & DWT.VIRTUAL) is 0) return items [index];
+    if (items [index] !is null) return items [index];
+    return items [index] = new TableItem (this, DWT.NONE, index, false);
+}
+
+static int checkStyle (int style) {
+    /*
+    * To be compatible with Windows, force the H_SCROLL
+    * and V_SCROLL style bits.  On Windows, it is not
+    * possible to create a table without scroll bars.
+    */
+    style |= DWT.H_SCROLL | DWT.V_SCROLL;
+    /* GTK is always FULL_SELECTION */
+    style |= DWT.FULL_SELECTION;
+    return checkBits (style, DWT.SINGLE, DWT.MULTI, 0, 0, 0, 0);
+}
+
+override void cellDataProc (
+    GtkTreeViewColumn *tree_column,
+    GtkCellRenderer *cell,
+    GtkTreeModel *tree_model,
+    GtkTreeIter *iter,
+    void* data)
+{
+    if (cell is cast(GtkCellRenderer*)ignoreCell) return;
+    auto path = OS.gtk_tree_model_get_path (tree_model, iter);
+    auto index = OS.gtk_tree_path_get_indices (path);
+    TableItem item = _getItem (index[0]);
+    OS.gtk_tree_path_free (path);
+    if (item !is null) OS.g_object_set_qdata (cast(GObject*)cell, Display.SWT_OBJECT_INDEX2, item.handle);
+    bool isPixbuf = OS.GTK_IS_CELL_RENDERER_PIXBUF (cast(GTypeInstance*)cell);
+    if (!(isPixbuf || OS.GTK_IS_CELL_RENDERER_TEXT (cast(GTypeInstance*)cell))) return;
+    int modelIndex = -1;
+    bool customDraw = false;
+    if (columnCount is 0) {
+        modelIndex = Table.FIRST_COLUMN;
+        customDraw = firstCustomDraw;
+    } else {
+        TableColumn column = cast(TableColumn) display.getWidget (cast(GtkWidget*)tree_column);
+        if (column !is null) {
+            modelIndex = column.modelIndex;
+            customDraw = column.customDraw;
+        }
+    }
+    if (modelIndex is -1) return 0;
+    bool setData = false;
+    if ((style & DWT.VIRTUAL) !is 0) {
+        /*
+        * Feature in GTK.  On GTK before 2.4, fixed_height_mode is not
+        * supported, and the tree asks for the data of all items.  The
+        * fix is to only provide the data if the row is visible.
+        */
+        if (OS.GTK_VERSION < OS.buildVERSION (2, 3, 2)) {
+            OS.gtk_widget_realize (handle);
+            GdkRectangle visible;
+            OS.gtk_tree_view_get_visible_rect (handle, &visible);
+            GdkRectangle area;
+            path = OS.gtk_tree_model_get_path (tree_model, iter);
+            OS.gtk_tree_view_get_cell_area (handle, path, tree_column, &area);
+            OS.gtk_tree_path_free (path);
+            if (area.y + area.height < 0 || area.y + visible.y > visible.y + visible.height) {
+                /* Give an image from the image list to make sure the row has
+                * the correct height.
+                */
+                if (imageList !is null && imageList.pixbufs.length > 0) {
+                    if (isPixbuf) OS.g_object_set1 (cell, OS.pixbuf.ptr, cast(int)imageList.pixbufs [0] );
+                }
+                return 0;
+            }
+        }
+        if (!item.cached) {
+            lastIndexOf = index[0];
+            setData = checkData (item);
+        }
+    }
+    void* ptr;
+    if (setData) {
+        if (isPixbuf) {
+            OS.gtk_tree_model_get1 (tree_model, iter, modelIndex + CELL_PIXBUF, &ptr);
+            OS.g_object_set1 (cell, OS.pixbuf.ptr, cast(int)ptr);
+        } else {
+            OS.gtk_tree_model_get1 (tree_model, iter, modelIndex + CELL_TEXT, &ptr);
+            if (ptr !is null) {
+                OS.g_object_set1 (cell, OS.text.ptr, cast(int)ptr);
+                OS.g_free (ptr);
+            }
+        }
+    }
+    if (customDraw) {
+        /*
+        * Bug on GTK. Gtk renders the background on top of the checkbox and pixbuf.
+        * This only happens in version 2.2.1 and earlier. The fix is not to set the background.
+        */
+        if (OS.GTK_VERSION > OS.buildVERSION (2, 2, 1)) {
+            if (!ownerDraw) {
+                ptr = null;
+                OS.gtk_tree_model_get1 (tree_model, iter, modelIndex + CELL_BACKGROUND, &ptr);
+                if (ptr !is null) {
+                    OS.g_object_set1 (cell, OS.cell_background_gdk.ptr, cast(int)ptr);
+                }
+            }
+        }
+        if (!isPixbuf) {
+            ptr = null;
+            OS.gtk_tree_model_get1 (tree_model, iter, modelIndex + CELL_FOREGROUND, &ptr);
+            if (ptr !is null) {
+                OS.g_object_set1 (cell, OS.foreground_gdk.ptr, cast(int)ptr);
+            }
+            ptr = null;
+            OS.gtk_tree_model_get1 (tree_model, iter, modelIndex + CELL_FONT, &ptr);
+            if (ptr !is null) {
+                OS.g_object_set1 (cell, OS.font_desc.ptr, cast(int)ptr);
+            }
+        }
+    }
+    if (setData) {
+        ignoreCell = cast(GtkWidget*)cell;
+        setScrollWidth (tree_column, item);
+        ignoreCell = null;
+    }
+    return;
+}
+
+bool checkData (TableItem item) {
+    if (item.cached) return true;
+    if ((style & DWT.VIRTUAL) !is 0) {
+        item.cached = true;
+        Event event = new Event ();
+        event.item = item;
+        event.index = indexOf (item);
+        int mask = OS.G_SIGNAL_MATCH_DATA | OS.G_SIGNAL_MATCH_ID;
+        int signal_id = OS.g_signal_lookup (OS.row_changed.ptr, OS.gtk_tree_model_get_type ());
+        OS.g_signal_handlers_block_matched (modelHandle, mask, signal_id, 0, null, null, handle);
+        currentItem = item;
+        sendEvent (DWT.SetData, event);
+        //widget could be disposed at this point
+        currentItem = null;
+        if (isDisposed ()) return false;
+        OS.g_signal_handlers_unblock_matched (modelHandle, mask, signal_id, 0, null, null, handle);
+        if (item.isDisposed ()) return false;
+    }
+    return true;
+}
+
+protected void checkSubclass () {
+    if (!isValidSubclass ()) error (DWT.ERROR_INVALID_SUBCLASS);
+}
+
+/**
+ * Adds the listener to the collection of listeners who will
+ * be notified when the user changes the receiver's selection, by sending
+ * it one of the messages defined in the <code>SelectionListener</code>
+ * interface.
+ * <p>
+ * When <code>widgetSelected</code> is called, the item field of the event object is valid.
+ * If the receiver has the <code>DWT.CHECK</code> style and the check selection changes,
+ * the event object detail field contains the value <code>DWT.CHECK</code>.
+ * <code>widgetDefaultSelected</code> is typically called when an item is double-clicked.
+ * The item field of the event object is valid for default selection, but the detail field is not used.
+ * </p>
+ *
+ * @param listener the listener which should be notified when the user changes the receiver's selection
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see SelectionListener
+ * @see #removeSelectionListener
+ * @see SelectionEvent
+ */
+public void addSelectionListener (SelectionListener listener) {
+    checkWidget ();
+    if (listener is null) error (DWT.ERROR_NULL_ARGUMENT);
+    TypedListener typedListener = new TypedListener (listener);
+    addListener (DWT.Selection, typedListener);
+    addListener (DWT.DefaultSelection, typedListener);
+}
+
+int calculateWidth (GtkTreeViewColumn* column, GtkTreeIter* iter) {
+    OS.gtk_tree_view_column_cell_set_cell_data (column, modelHandle, iter, false, false);
+    /*
+    * Bug in GTK.  The width calculated by gtk_tree_view_column_cell_get_size()
+    * always grows in size regardless of the text or images in the table.
+    * The fix is to determine the column width from the cell renderers.
+    */
+    // Code intentionally commented
+    //int [] width = new int [1];
+    //OS.gtk_tree_view_column_cell_get_size (column, null, null, null, width, null);
+    //return width [0];
+
+    int width = 0;
+    int w;
+    OS.gtk_widget_style_get1(handle, OS.focus_line_width.ptr, &w );
+    width += 2 * w;
+    auto list = OS.gtk_tree_view_column_get_cell_renderers (column);
+    if (list is null) return 0;
+    auto temp = list;
+    while (temp !is null) {
+        auto renderer = OS.g_list_data (temp);
+        if (renderer !is null) {
+            OS.gtk_cell_renderer_get_size (renderer, handle, null, null, null, &w, null);
+            width += w;
+        }
+        temp = OS.g_list_next (temp);
+    }
+    OS.g_list_free (list);
+    return width;
+}
+
+/**
+ * Clears the item at the given zero-relative index in the receiver.
+ * The text, icon and other attributes of the item are set to the default
+ * value.  If the table was created with the <code>DWT.VIRTUAL</code> style,
+ * these attributes are requested again as needed.
+ *
+ * @param index the index of the item to clear
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_RANGE - if the index is not between 0 and the number of elements in the list minus 1 (inclusive)</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see DWT#VIRTUAL
+ * @see DWT#SetData
+ *
+ * @since 3.0
+ */
+public void clear (int index) {
+    checkWidget ();
+    if (!(0 <= index && index < itemCount)) {
+        error(DWT.ERROR_INVALID_RANGE);
+    }
+    TableItem item = items [index];
+    if (item !is null) item.clear ();
+}
+
+/**
+ * Removes the items from the receiver which are between the given
+ * zero-relative start and end indices (inclusive).  The text, icon
+ * and other attributes of the items are set to their default values.
+ * If the table was created with the <code>DWT.VIRTUAL</code> style,
+ * these attributes are requested again as needed.
+ *
+ * @param start the start index of the item to clear
+ * @param end the end index of the item to clear
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_RANGE - if either the start or end are not between 0 and the number of elements in the list minus 1 (inclusive)</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see DWT#VIRTUAL
+ * @see DWT#SetData
+ *
+ * @since 3.0
+ */
+public void clear (int start, int end) {
+    checkWidget ();
+    if (start > end) return;
+    if (!(0 <= start && start <= end && end < itemCount)) {
+        error (DWT.ERROR_INVALID_RANGE);
+    }
+    if (start is 0 && end is itemCount - 1) {
+        clearAll();
+    } else {
+        for (int i=start; i<=end; i++) {
+            TableItem item = items [i];
+            if (item !is null) item.clear();
+        }
+    }
+}
+
+/**
+ * Clears the items at the given zero-relative indices in the receiver.
+ * The text, icon and other attributes of the items are set to their default
+ * values.  If the table was created with the <code>DWT.VIRTUAL</code> style,
+ * these attributes are requested again as needed.
+ *
+ * @param indices the array of indices of the items
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_RANGE - if the index is not between 0 and the number of elements in the list minus 1 (inclusive)</li>
+ *    <li>ERROR_NULL_ARGUMENT - if the indices array is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see DWT#VIRTUAL
+ * @see DWT#SetData
+ *
+ * @since 3.0
+ */
+public void clear (int [] indices) {
+    checkWidget ();
+    if (indices is null) error (DWT.ERROR_NULL_ARGUMENT);
+    if (indices.length is 0) return;
+    for (int i=0; i<indices.length; i++) {
+        if (!(0 <= indices [i] && indices [i] < itemCount)) {
+            error (DWT.ERROR_INVALID_RANGE);
+        }
+    }
+    for (int i=0; i<indices.length; i++) {
+        TableItem item = items [indices [i]];
+        if (item !is null) item.clear();
+    }
+}
+
+/**
+ * Clears all the items in the receiver. The text, icon and other
+ * attributes of the items are set to their default values. If the
+ * table was created with the <code>DWT.VIRTUAL</code> style, these
+ * attributes are requested again as needed.
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see DWT#VIRTUAL
+ * @see DWT#SetData
+ *
+ * @since 3.0
+ */
+public void clearAll () {
+    checkWidget ();
+    for (int i=0; i<itemCount; i++) {
+        TableItem item = items [i];
+        if (item !is null) item.clear();
+    }
+}
+
+public Point computeSize (int wHint, int hHint, bool changed) {
+    checkWidget ();
+    if (wHint !is DWT.DEFAULT && wHint < 0) wHint = 0;
+    if (hHint !is DWT.DEFAULT && hHint < 0) hHint = 0;
+    Point size = computeNativeSize (handle, wHint, hHint, changed);
+    Rectangle trim = computeTrim (0, 0, size.x, size.y);
+    size.x = trim.width;
+    size.y = trim.height;
+    return size;
+}
+
+void createColumn (TableColumn column, int index) {
+    int modelIndex = FIRST_COLUMN;
+    if (columnCount !is 0) {
+        int modelLength = OS.gtk_tree_model_get_n_columns (modelHandle);
+        bool [] usedColumns = new bool [modelLength];
+        for (int i=0; i<columnCount; i++) {
+            int columnIndex = columns [i].modelIndex;
+            for (int j = 0; j < CELL_TYPES; j++) {
+                usedColumns [columnIndex + j] = true;
+            }
+        }
+        while (modelIndex < modelLength) {
+            if (!usedColumns [modelIndex]) break;
+            modelIndex++;
+        }
+        if (modelIndex is modelLength) {
+            auto oldModel = modelHandle;
+            auto types = getColumnTypes (columnCount + 4); // grow by 4 rows at a time
+            auto newModel = OS.gtk_list_store_newv (types.length, types.ptr);
+            if (newModel is null) error (DWT.ERROR_NO_HANDLES);
+            void* ptr;
+            for (int i=0; i<itemCount; i++) {
+                GtkTreeIter* newItem = new GtkTreeIter;
+                OS.gtk_list_store_append (newModel, newItem);
+                TableItem item = items [i];
+                if (item !is null) {
+                    auto oldItem = item.handle;
+                    for (int j=0; j<modelLength; j++) {
+                        OS.gtk_tree_model_get1 (oldModel, oldItem, j, &ptr);
+                        OS.gtk_list_store_set1 (newModel, newItem, j, ptr);
+                        if (types [j] is OS.G_TYPE_STRING ()) OS.g_free ((ptr));
+                    }
+                    OS.gtk_list_store_remove (oldModel, oldItem);
+                    OS.g_free (oldItem);
+                    item.handle = cast(GtkWidget*)newItem;
+                } else {
+                    delete (newItem);
+                }
+            }
+            OS.gtk_tree_view_set_model (handle, newModel);
+            OS.g_object_unref (oldModel);
+            modelHandle = cast(GtkWidget*)newModel;
+        }
+    }
+    auto columnHandle = OS.gtk_tree_view_column_new ();
+    if (columnHandle is null) error (DWT.ERROR_NO_HANDLES);
+    if (index is 0 && columnCount > 0) {
+        TableColumn checkColumn = columns [0];
+        createRenderers (cast(GtkTreeViewColumn*)checkColumn.handle, checkColumn.modelIndex, false, checkColumn.style);
+    }
+    createRenderers ( columnHandle, modelIndex, index is 0, column is null ? 0 : column.style);
+    /*
+    * Use GTK_TREE_VIEW_COLUMN_GROW_ONLY on GTK versions < 2.3.2
+    * because fixed_height_mode is not supported.
+    */
+    bool useVirtual = (style & DWT.VIRTUAL) !is 0 && OS.GTK_VERSION >= OS.buildVERSION (2, 3, 2);
+    if (!useVirtual && columnCount is 0) {
+        OS.gtk_tree_view_column_set_sizing (columnHandle, OS.GTK_TREE_VIEW_COLUMN_GROW_ONLY);
+    } else {
+        OS.gtk_tree_view_column_set_sizing (columnHandle, OS.GTK_TREE_VIEW_COLUMN_FIXED);
+        if (columnCount !is 0) OS.gtk_tree_view_column_set_visible (columnHandle, false);
+    }
+    OS.gtk_tree_view_column_set_resizable (columnHandle, true);
+    OS.gtk_tree_view_column_set_clickable (columnHandle, true);
+    OS.gtk_tree_view_column_set_min_width (columnHandle, 0);
+    OS.gtk_tree_view_insert_column (handle, columnHandle, index);
+    if (column !is null) {
+        column.handle = cast(GtkWidget*)columnHandle;
+        column.modelIndex = modelIndex;
+    }
+    /* Disable searching when using VIRTUAL */
+    if ((style & DWT.VIRTUAL) !is 0) {
+        /*
+        * Bug in GTK. Until GTK 2.6.5, calling gtk_tree_view_set_enable_search(FALSE)
+        * would prevent the user from being able to type in text to search the tree.
+        * After 2.6.5, GTK introduced Ctrl+F as being the key binding for interactive
+        * search. This meant that even if FALSE was passed to enable_search, the user
+        * can still bring up the search pop up using the keybinding. GTK also introduced
+        * the notion of passing a -1 to gtk_set_search_column to disable searching
+        * (including the search key binding).  The fix is to use the right calls
+        * for the right version.
+        */
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 6, 5)) {
+            OS.gtk_tree_view_set_search_column (handle, -1);
+        } else {
+            OS.gtk_tree_view_set_enable_search (handle, false);
+        }
+    } else {
+        /* Set the search column whenever the model changes */
+        int firstColumn = columnCount is 0 ? FIRST_COLUMN : columns [0].modelIndex;
+        OS.gtk_tree_view_set_search_column (handle, firstColumn + CELL_TEXT);
+    }
+}
+
+void createHandle (int index) {
+    state |= HANDLE;
+    fixedHandle = cast(GtkWidget*)OS.g_object_new (display.gtk_fixed_get_type (), null);
+    if (fixedHandle is null) error (DWT.ERROR_NO_HANDLES);
+    OS.gtk_fixed_set_has_window (fixedHandle, true);
+    scrolledHandle = cast(GtkWidget*)OS.gtk_scrolled_window_new (null, null);
+    if (scrolledHandle is null) error (DWT.ERROR_NO_HANDLES);
+    auto types = getColumnTypes (1);
+    modelHandle = cast(GtkWidget*)OS.gtk_list_store_newv (types.length, types.ptr);
+    if (modelHandle is null) error (DWT.ERROR_NO_HANDLES);
+    handle = cast(GtkWidget*)OS.gtk_tree_view_new_with_model (modelHandle);
+    if (handle is null) error (DWT.ERROR_NO_HANDLES);
+    if ((style & DWT.CHECK) !is 0) {
+        checkRenderer = cast(GtkWidget*)OS.gtk_cell_renderer_toggle_new ();
+        if (checkRenderer is null) error (DWT.ERROR_NO_HANDLES);
+        OS.g_object_ref (checkRenderer);
+    }
+    createColumn (null, 0);
+    OS.gtk_container_add (fixedHandle, scrolledHandle);
+    OS.gtk_container_add (scrolledHandle, handle);
+
+    int mode = (style & DWT.MULTI) !is 0 ? OS.GTK_SELECTION_MULTIPLE : OS.GTK_SELECTION_BROWSE;
+    auto selectionHandle = OS.gtk_tree_view_get_selection (handle);
+    OS.gtk_tree_selection_set_mode (selectionHandle, mode);
+    OS.gtk_tree_view_set_headers_visible (handle, false);
+    int hsp = (style & DWT.H_SCROLL) !is 0 ? OS.GTK_POLICY_AUTOMATIC : OS.GTK_POLICY_NEVER;
+    int vsp = (style & DWT.V_SCROLL) !is 0 ? OS.GTK_POLICY_AUTOMATIC : OS.GTK_POLICY_NEVER;
+    OS.gtk_scrolled_window_set_policy (scrolledHandle, hsp, vsp);
+    if ((style & DWT.BORDER) !is 0) OS.gtk_scrolled_window_set_shadow_type (scrolledHandle, OS.GTK_SHADOW_ETCHED_IN);
+    /* Disable searching when using VIRTUAL */
+    if ((style & DWT.VIRTUAL) !is 0) {
+        /* The fixed_height_mode property only exists in GTK 2.3.2 and greater */
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 3, 2)) {
+            OS.g_object_set1 (handle, OS.fixed_height_mode.ptr, true);
+        }
+        /*
+        * Bug in GTK. Until GTK 2.6.5, calling gtk_tree_view_set_enable_search(FALSE)
+        * would prevent the user from being able to type in text to search the tree.
+        * After 2.6.5, GTK introduced Ctrl+F as being the key binding for interactive
+        * search. This meant that even if FALSE was passed to enable_search, the user
+        * can still bring up the search pop up using the keybinding. GTK also introduced
+        * the notion of passing a -1 to gtk_set_search_column to disable searching
+        * (including the search key binding).  The fix is to use the right calls
+        * for the right version.
+        */
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 6, 5)) {
+            OS.gtk_tree_view_set_search_column (handle, -1);
+        } else {
+            OS.gtk_tree_view_set_enable_search (handle, false);
+        }
+    }
+}
+
+void createItem (TableColumn column, int index) {
+    if (!(0 <= index && index <= columnCount)) error (DWT.ERROR_INVALID_RANGE);
+    if (columnCount is 0) {
+        column.handle = cast(GtkWidget*)OS.gtk_tree_view_get_column (handle, 0);
+        OS.gtk_tree_view_column_set_sizing (column.handle, OS.GTK_TREE_VIEW_COLUMN_FIXED);
+        OS.gtk_tree_view_column_set_visible (column.handle, false);
+        column.modelIndex = FIRST_COLUMN;
+        createRenderers (cast(GtkTreeViewColumn*)column.handle, column.modelIndex, true, column.style);
+        column.customDraw = firstCustomDraw;
+        firstCustomDraw = false;
+    } else {
+        createColumn (column, index);
+    }
+    auto boxHandle = OS.gtk_hbox_new (false, 3);
+    if (boxHandle is null) error (DWT.ERROR_NO_HANDLES);
+    auto labelHandle = OS.gtk_label_new_with_mnemonic (null);
+    if (labelHandle is null) error (DWT.ERROR_NO_HANDLES);
+    auto imageHandle = OS.gtk_image_new ();
+    if (imageHandle is null) error (DWT.ERROR_NO_HANDLES);
+    OS.gtk_container_add (boxHandle, imageHandle);
+    OS.gtk_container_add (boxHandle, labelHandle);
+    OS.gtk_widget_show (boxHandle);
+    OS.gtk_widget_show (labelHandle);
+    column.labelHandle = labelHandle;
+    column.imageHandle = imageHandle;
+    OS.gtk_tree_view_column_set_widget (column.handle, boxHandle);
+    auto widget = OS.gtk_widget_get_parent (boxHandle);
+    while (widget !is handle) {
+        if (OS.GTK_IS_BUTTON (cast(GTypeInstance*)widget)) {
+            column.buttonHandle = widget;
+            break;
+        }
+        widget = OS.gtk_widget_get_parent (widget);
+    }
+    if (columnCount is columns.length) {
+        TableColumn [] newColumns = new TableColumn [columns.length + 4];
+        System.arraycopy (columns, 0, newColumns, 0, columns.length);
+        columns = newColumns;
+    }
+    System.arraycopy (columns, index, columns, index + 1, columnCount++ - index);
+    columns [index] = column;
+    if ((state & FONT) !is 0) {
+        column.setFontDescription (getFontDescription ());
+    }
+    if (columnCount >= 1) {
+        for (int i=0; i<itemCount; i++) {
+            TableItem item = items [i];
+            if (item !is null) {
+                Font [] cellFont = item.cellFont;
+                if (cellFont !is null) {
+                    Font [] temp = new Font [columnCount];
+                    System.arraycopy (cellFont, 0, temp, 0, index);
+                    System.arraycopy (cellFont, index, temp, index+1, columnCount-index-1);
+                    item.cellFont = temp;
+                }
+            }
+        }
+    }
+}
+
+void createItem (TableItem item, int index) {
+    if (!(0 <= index && index <= itemCount)) error (DWT.ERROR_INVALID_RANGE);
+    if (itemCount is items.length) {
+        int length = drawCount is 0 ? items.length + 4 : Math.max (4, items.length * 3 / 2);
+        TableItem [] newItems = new TableItem [length];
+        System.arraycopy (items, 0, newItems, 0, items.length);
+        items = newItems;
+    }
+    item.handle = cast(GtkWidget*) new GtkTreeIter;
+    if (item.handle is null) error (DWT.ERROR_NO_HANDLES);
+    /*
+    * Feature in GTK.  It is much faster to append to a list store
+    * than to insert at the end using gtk_list_store_insert().
+    */
+    if (index is itemCount) {
+        OS.gtk_list_store_append (modelHandle, item.handle);
+    } else {
+        OS.gtk_list_store_insert (modelHandle, item.handle, index);
+    }
+    System.arraycopy (items, index, items, index + 1, itemCount++ - index);
+    items [index] = item;
+}
+
+void createRenderers (GtkTreeViewColumn* columnHandle, int modelIndex, bool check, int columnStyle) {
+    OS.gtk_tree_view_column_clear (columnHandle);
+    if ((style & DWT.CHECK) !is 0 && check) {
+        OS.gtk_tree_view_column_pack_start (columnHandle, checkRenderer, false);
+        OS.gtk_tree_view_column_add_attribute (columnHandle, checkRenderer, OS.active.ptr, CHECKED_COLUMN);
+        /*
+        * Feature in GTK. The inconsistent property only exists in GTK 2.2.x.
+        */
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 2, 0)) {
+            OS.gtk_tree_view_column_add_attribute (columnHandle, checkRenderer, OS.inconsistent.ptr, GRAYED_COLUMN);
+        }
+        /*
+        * Bug in GTK. GTK renders the background on top of the checkbox.
+        * This only happens in version 2.2.1 and earlier. The fix is not to set the background.
+        */
+        if (OS.GTK_VERSION > OS.buildVERSION (2, 2, 1)) {
+            if (!ownerDraw) OS.gtk_tree_view_column_add_attribute (columnHandle, checkRenderer, OS.cell_background_gdk.ptr, BACKGROUND_COLUMN);
+        }
+        if (ownerDraw) {
+            display.doCellDataProc( handle, cast(GtkTreeViewColumn*)columnHandle, cast(GtkCellRenderer*)checkRenderer );
+            OS.g_object_set_qdata (cast(GObject*)checkRenderer, Display.SWT_OBJECT_INDEX1, columnHandle);
+        }
+    }
+    auto pixbufRenderer = ownerDraw ? cast(GtkCellRenderer*)OS.g_object_new (display.gtk_cell_renderer_pixbuf_get_type (), null) : OS.gtk_cell_renderer_pixbuf_new ();
+    if (pixbufRenderer is null) error (DWT.ERROR_NO_HANDLES);
+    auto textRenderer = ownerDraw ? cast(GtkCellRenderer*)OS.g_object_new (display.gtk_cell_renderer_text_get_type (), null) : OS.gtk_cell_renderer_text_new ();
+    if (textRenderer is null) error (DWT.ERROR_NO_HANDLES);
+
+    if (ownerDraw) {
+        OS.g_object_set_qdata (cast(GObject*)pixbufRenderer, Display.SWT_OBJECT_INDEX1, columnHandle);
+        OS.g_object_set_qdata (cast(GObject*)textRenderer, Display.SWT_OBJECT_INDEX1, columnHandle);
+    }
+
+    /*
+    * Feature in GTK.  When a tree view column contains only one activatable
+    * cell renderer such as a toggle renderer, mouse clicks anywhere in a cell
+    * activate that renderer. The workaround is to set a second cell renderer
+    * to be activatable.
+    */
+    if ((style & DWT.CHECK) !is 0 && check) {
+        OS.g_object_set1 (pixbufRenderer, OS.mode.ptr, OS.GTK_CELL_RENDERER_MODE_ACTIVATABLE);
+    }
+
+    /* Set alignment */
+    if ((columnStyle & DWT.RIGHT) !is 0) {
+        OS.g_object_set1(textRenderer, OS.xalign.ptr, cast(int)cast(void*)1f);
+        OS.gtk_tree_view_column_pack_end (columnHandle, textRenderer, true);
+        OS.gtk_tree_view_column_pack_end (columnHandle, pixbufRenderer, false);
+        OS.gtk_tree_view_column_set_alignment (columnHandle, 1f);
+    } else if ((columnStyle & DWT.CENTER) !is 0) {
+        OS.g_object_set1(textRenderer, OS.xalign.ptr, cast(int)cast(void*)0.5f);
+        OS.gtk_tree_view_column_pack_start (columnHandle, pixbufRenderer, false);
+        OS.gtk_tree_view_column_pack_end (columnHandle, textRenderer, true);
+        OS.gtk_tree_view_column_set_alignment (columnHandle, 0.5f);
+    } else {
+        OS.gtk_tree_view_column_pack_start (columnHandle, pixbufRenderer, false);
+        OS.gtk_tree_view_column_pack_start (columnHandle, textRenderer, true);
+        OS.gtk_tree_view_column_set_alignment (columnHandle, 0f);
+    }
+
+    /* Add attributes */
+    OS.gtk_tree_view_column_add_attribute (columnHandle, pixbufRenderer, OS.pixbuf.ptr, modelIndex + CELL_PIXBUF);
+    /*
+     * Bug on GTK. Gtk renders the background on top of the pixbuf.
+     * This only happens in version 2.2.1 and earlier. The fix is not to set the background.
+     */
+    if (OS.GTK_VERSION > OS.buildVERSION (2, 2, 1)) {
+        if (!ownerDraw) {
+            OS.gtk_tree_view_column_add_attribute (columnHandle, pixbufRenderer, OS.cell_background_gdk.ptr, BACKGROUND_COLUMN);
+            OS.gtk_tree_view_column_add_attribute (columnHandle, textRenderer, OS.cell_background_gdk.ptr, BACKGROUND_COLUMN);
+        }
+    }
+    OS.gtk_tree_view_column_add_attribute (columnHandle, textRenderer, OS.text.ptr, modelIndex + CELL_TEXT);
+    OS.gtk_tree_view_column_add_attribute (columnHandle, textRenderer, OS.foreground_gdk.ptr, FOREGROUND_COLUMN);
+    OS.gtk_tree_view_column_add_attribute (columnHandle, textRenderer, OS.font_desc.ptr, FONT_COLUMN);
+
+    bool customDraw = firstCustomDraw;
+    if (columnCount !is 0) {
+        for (int i=0; i<columnCount; i++) {
+            if (columns [i].handle is cast(GtkWidget*)columnHandle) {
+                customDraw = columns [i].customDraw;
+                break;
+            }
+        }
+    }
+    if ((style & DWT.VIRTUAL) !is 0 || customDraw || ownerDraw) {
+        display.doCellDataProc( handle, cast(GtkTreeViewColumn*)columnHandle, cast(GtkCellRenderer*)textRenderer );
+        display.doCellDataProc( handle, cast(GtkTreeViewColumn*)columnHandle, cast(GtkCellRenderer*)pixbufRenderer );
+    }
+}
+
+override void createWidget (int index) {
+    super.createWidget (index);
+    items = new TableItem [4];
+    columns = new TableColumn [4];
+    itemCount = columnCount = 0;
+}
+
+GdkColor* defaultBackground () {
+    return display.COLOR_LIST_BACKGROUND;
+}
+
+GdkColor* defaultForeground () {
+    return display.COLOR_LIST_FOREGROUND;
+}
+
+override void deregister () {
+    super.deregister ();
+    display.removeWidget ( cast(GtkWidget*)OS.gtk_tree_view_get_selection (handle));
+    if (checkRenderer !is null) display.removeWidget (checkRenderer);
+}
+
+/**
+ * Deselects the item at the given zero-relative index in the receiver.
+ * If the item at the index was already deselected, it remains
+ * deselected. Indices that are out of range are ignored.
+ *
+ * @param index the index of the item to deselect
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void deselect (int index) {
+    checkWidget();
+    if (index < 0 || index >= itemCount) return;
+    bool fixColumn = showFirstColumn ();
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    OS.gtk_tree_selection_unselect_iter (selection, _getItem (index).handle);
+    OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    if (fixColumn) hideFirstColumn ();
+}
+
+/**
+ * Deselects the items at the given zero-relative indices in the receiver.
+ * If the item at the given zero-relative index in the receiver
+ * is selected, it is deselected.  If the item at the index
+ * was not selected, it remains deselected.  The range of the
+ * indices is inclusive. Indices that are out of range are ignored.
+ *
+ * @param start the start index of the items to deselect
+ * @param end the end index of the items to deselect
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void deselect (int start, int end) {
+    checkWidget();
+    bool fixColumn = showFirstColumn ();
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    for (int index=start; index<=end; index++) {
+        if (index < 0 || index >= itemCount) continue;
+        OS.gtk_tree_selection_unselect_iter (selection, _getItem (index).handle);
+    }
+    OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    if (fixColumn) hideFirstColumn ();
+}
+
+/**
+ * Deselects the items at the given zero-relative indices in the receiver.
+ * If the item at the given zero-relative index in the receiver
+ * is selected, it is deselected.  If the item at the index
+ * was not selected, it remains deselected. Indices that are out
+ * of range and duplicate indices are ignored.
+ *
+ * @param indices the array of indices for the items to deselect
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the set of indices is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void deselect (int [] indices) {
+    checkWidget();
+    if (indices is null) error (DWT.ERROR_NULL_ARGUMENT);
+    bool fixColumn = showFirstColumn ();
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    for (int i=0; i<indices.length; i++) {
+        int index = indices[i];
+        if (index < 0 || index >= itemCount) continue;
+        OS.gtk_tree_selection_unselect_iter (selection, _getItem (index).handle);
+    }
+    OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    if (fixColumn) hideFirstColumn ();
+}
+
+/**
+ * Deselects all selected items in the receiver.
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void deselectAll () {
+    checkWidget();
+    bool fixColumn = showFirstColumn ();
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    OS.gtk_tree_selection_unselect_all (selection);
+    OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    if (fixColumn) hideFirstColumn ();
+}
+
+void destroyItem (TableColumn column) {
+    int index = 0;
+    while (index < columnCount) {
+        if (columns [index] is column) break;
+        index++;
+    }
+    if (index is columnCount) return;
+    auto columnHandle = column.handle;
+    if (columnCount is 1) {
+        firstCustomDraw = column.customDraw;
+    }
+    System.arraycopy (columns, index + 1, columns, index, --columnCount - index);
+    columns [columnCount] = null;
+    OS.gtk_tree_view_remove_column (handle, columnHandle);
+    if (columnCount is 0) {
+        auto oldModel = modelHandle;
+        auto types = getColumnTypes (1);
+        auto newModel = OS.gtk_list_store_newv (types.length, types.ptr);
+        if (newModel is null) error (DWT.ERROR_NO_HANDLES);
+        void* ptr;
+        for (int i=0; i<itemCount; i++) {
+            GtkTreeIter* newItem = new GtkTreeIter;
+            if (newItem is null) error (DWT.ERROR_NO_HANDLES);
+            OS.gtk_list_store_append (newModel, newItem);
+            TableItem item = items [i];
+            if (item !is null) {
+                auto oldItem = item.handle;
+                for (int j=0; j<FIRST_COLUMN; j++) {
+                    OS.gtk_tree_model_get1 (oldModel, oldItem, j, &ptr);
+                    OS.gtk_list_store_set1 (newModel, newItem, j, ptr);
+                }
+                OS.gtk_tree_model_get1 (oldModel, oldItem, column.modelIndex + CELL_PIXBUF, &ptr);
+                OS.gtk_list_store_set1 (newModel, newItem, FIRST_COLUMN + CELL_PIXBUF, ptr);
+                OS.gtk_tree_model_get1 (oldModel, oldItem, column.modelIndex + CELL_TEXT, &ptr);
+                OS.gtk_list_store_set1 (newModel, newItem, FIRST_COLUMN + CELL_TEXT, ptr );
+                OS.g_free (ptr);
+                OS.gtk_tree_model_get1 (oldModel, oldItem, column.modelIndex + CELL_FOREGROUND, &ptr);
+                OS.gtk_list_store_set1 (newModel, newItem, FIRST_COLUMN + CELL_FOREGROUND, ptr);
+                OS.gtk_tree_model_get1 (oldModel, oldItem, column.modelIndex + CELL_BACKGROUND, &ptr);
+                OS.gtk_list_store_set1 (newModel, newItem, FIRST_COLUMN + CELL_BACKGROUND, ptr);
+                OS.gtk_tree_model_get1 (oldModel, oldItem, column.modelIndex + CELL_FONT, &ptr);
+                OS.gtk_list_store_set1 (newModel, newItem, FIRST_COLUMN + CELL_FONT, ptr);
+                OS.gtk_list_store_remove (oldModel, oldItem);
+                OS.g_free (oldItem);
+                item.handle = cast(GtkWidget*) newItem;
+            } else {
+                delete newItem;
+            }
+        }
+        OS.gtk_tree_view_set_model (handle, newModel);
+        OS.g_object_unref (oldModel);
+        modelHandle = cast(GtkWidget*)newModel;
+        createColumn (null, 0);
+    } else {
+        for (int i=0; i<itemCount; i++) {
+            TableItem item = items [i];
+            if (item !is null) {
+                auto iter = item.handle;
+                int modelIndex = column.modelIndex;
+                OS.gtk_list_store_set1 (modelHandle, iter, modelIndex + CELL_PIXBUF, null);
+                OS.gtk_list_store_set1 (modelHandle, iter, modelIndex + CELL_TEXT, null);
+                OS.gtk_list_store_set1 (modelHandle, iter, modelIndex + CELL_FOREGROUND, null);
+                OS.gtk_list_store_set1 (modelHandle, iter, modelIndex + CELL_BACKGROUND, null);
+                OS.gtk_list_store_set1 (modelHandle, iter, modelIndex + CELL_FONT, null);
+
+                Font [] cellFont = item.cellFont;
+                if (cellFont !is null) {
+                    if (columnCount is 0) {
+                        item.cellFont = null;
+                    } else {
+                        Font [] temp = new Font [columnCount];
+                        System.arraycopy (cellFont, 0, temp, 0, index);
+                        System.arraycopy (cellFont, index + 1, temp, index, columnCount - index);
+                        item.cellFont = temp;
+                    }
+                }
+            }
+        }
+        if (index is 0) {
+            TableColumn checkColumn = columns [0];
+            createRenderers (cast(GtkTreeViewColumn*)checkColumn.handle, checkColumn.modelIndex, true, checkColumn.style);
+        }
+    }
+    /* Disable searching when using VIRTUAL */
+    if ((style & DWT.VIRTUAL) !is 0) {
+        /*
+        * Bug in GTK. Until GTK 2.6.5, calling gtk_tree_view_set_enable_search(FALSE)
+        * would prevent the user from being able to type in text to search the tree.
+        * After 2.6.5, GTK introduced Ctrl+F as being the key binding for interactive
+        * search. This meant that even if FALSE was passed to enable_search, the user
+        * can still bring up the search pop up using the keybinding. GTK also introduced
+        * the notion of passing a -1 to gtk_set_search_column to disable searching
+        * (including the search key binding).  The fix is to use the right calls
+        * for the right version.
+        */
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 6, 5)) {
+            OS.gtk_tree_view_set_search_column (handle, -1);
+        } else {
+            OS.gtk_tree_view_set_enable_search (handle, false);
+        }
+    } else {
+        /* Set the search column whenever the model changes */
+        int firstColumn = columnCount is 0 ? FIRST_COLUMN : columns [0].modelIndex;
+        OS.gtk_tree_view_set_search_column (handle, firstColumn + CELL_TEXT);
+    }
+}
+
+void destroyItem (TableItem item) {
+    int index = 0;
+    while (index < itemCount) {
+        if (items [index] is item) break;
+        index++;
+    }
+    if (index is itemCount) return;
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    OS.gtk_list_store_remove (modelHandle, item.handle);
+    OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    System.arraycopy (items, index + 1, items, index, --itemCount - index);
+    items [itemCount] = null;
+    if (itemCount is 0) resetCustomDraw ();
+}
+
+override bool dragDetect (int x, int y, bool filter, bool* consume) {
+    bool selected = false;
+    if (filter) {
+        GtkTreeViewColumn * path;
+        if (OS.gtk_tree_view_get_path_at_pos (handle, x, y, cast(void**)&path, null, null, null)) {
+            if (path !is null) {
+                auto selection = OS.gtk_tree_view_get_selection (handle);
+                if (OS.gtk_tree_selection_path_is_selected (selection, path)) selected = true;
+                OS.gtk_tree_path_free (path);
+            }
+        } else {
+            return false;
+        }
+    }
+    bool dragDetect = super.dragDetect (x, y, filter, consume);
+    if (dragDetect && selected && consume !is null) consume[0] = true;
+    return dragDetect;
+}
+
+GdkDrawable* eventWindow () {
+    return paintWindow ();
+}
+
+override void fixChildren (Shell newShell, Shell oldShell, Decorations newDecorations, Decorations oldDecorations, Menu [] menus) {
+    super.fixChildren (newShell, oldShell, newDecorations, oldDecorations, menus);
+    for (int i=0; i<columnCount; i++) {
+        TableColumn column = columns [i];
+        if (column.toolTipText !is null) {
+            column.setToolTipText(oldShell, null);
+            column.setToolTipText(newShell, column.toolTipText);
+        }
+    }
+}
+
+GdkColor* getBackgroundColor () {
+    return getBaseColor ();
+}
+
+public Rectangle getClientArea () {
+    checkWidget ();
+    forceResize ();
+    OS.gtk_widget_realize (handle);
+    auto fixedWindow = OS.GTK_WIDGET_WINDOW (fixedHandle);
+    auto binWindow = OS.gtk_tree_view_get_bin_window (handle);
+    int binX, binY;
+    OS.gdk_window_get_origin (binWindow, &binX, &binY);
+    int fixedX, fixedY;
+    OS.gdk_window_get_origin (fixedWindow, &fixedX, &fixedY);
+    auto clientHandle = clientHandle ();
+    int width = (state & ZERO_WIDTH) !is 0 ? 0 : OS.GTK_WIDGET_WIDTH (clientHandle);
+    int height = (state & ZERO_HEIGHT) !is 0 ? 0 : OS.GTK_WIDGET_HEIGHT (clientHandle);
+    return new Rectangle (fixedX - binX, fixedY - binY, width, height);
+}
+
+/**
+ * Returns the column at the given, zero-relative index in the
+ * receiver. Throws an exception if the index is out of range.
+ * Columns are returned in the order that they were created.
+ * If no <code>TableColumn</code>s were created by the programmer,
+ * this method will throw <code>ERROR_INVALID_RANGE</code> despite
+ * the fact that a single column of data may be visible in the table.
+ * This occurs when the programmer uses the table like a list, adding
+ * items but never creating a column.
+ *
+ * @param index the index of the column to return
+ * @return the column at the given index
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_RANGE - if the index is not between 0 and the number of elements in the list minus 1 (inclusive)</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#getColumnOrder()
+ * @see Table#setColumnOrder(int[])
+ * @see TableColumn#getMoveable()
+ * @see TableColumn#setMoveable(bool)
+ * @see DWT#Move
+ */
+public TableColumn getColumn (int index) {
+    checkWidget();
+    if (!(0 <= index && index < columnCount)) error (DWT.ERROR_INVALID_RANGE);
+    return columns [index];
+}
+
+/**
+ * Returns the number of columns contained in the receiver.
+ * If no <code>TableColumn</code>s were created by the programmer,
+ * this value is zero, despite the fact that visually, one column
+ * of items may be visible. This occurs when the programmer uses
+ * the table like a list, adding items but never creating a column.
+ *
+ * @return the number of columns
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int getColumnCount () {
+    checkWidget();
+    return columnCount;
+}
+
+uint[] getColumnTypes (int columnCount) {
+    uint[] types = new uint [FIRST_COLUMN + (columnCount * CELL_TYPES)];
+    // per row data
+    types [CHECKED_COLUMN] = OS.G_TYPE_BOOLEAN ();
+    types [GRAYED_COLUMN] = OS.G_TYPE_BOOLEAN ();
+    types [FOREGROUND_COLUMN] = OS.GDK_TYPE_COLOR ();
+    types [BACKGROUND_COLUMN] = OS.GDK_TYPE_COLOR ();
+    types [FONT_COLUMN] = OS.PANGO_TYPE_FONT_DESCRIPTION ();
+    // per cell data
+    for (int i=FIRST_COLUMN; i<types.length; i+=CELL_TYPES) {
+        types [i + CELL_PIXBUF] = OS.GDK_TYPE_PIXBUF ();
+        types [i + CELL_TEXT] = OS.G_TYPE_STRING ();
+        types [i + CELL_FOREGROUND] = OS.GDK_TYPE_COLOR ();
+        types [i + CELL_BACKGROUND] = OS.GDK_TYPE_COLOR ();
+        types [i + CELL_FONT] = OS.PANGO_TYPE_FONT_DESCRIPTION ();
+    }
+    return types;
+}
+
+/**
+ * Returns an array of zero-relative integers that map
+ * the creation order of the receiver's items to the
+ * order in which they are currently being displayed.
+ * <p>
+ * Specifically, the indices of the returned array represent
+ * the current visual order of the items, and the contents
+ * of the array represent the creation order of the items.
+ * </p><p>
+ * Note: This is not the actual structure used by the receiver
+ * to maintain its list of items, so modifying the array will
+ * not affect the receiver.
+ * </p>
+ *
+ * @return the current visual order of the receiver's items
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#setColumnOrder(int[])
+ * @see TableColumn#getMoveable()
+ * @see TableColumn#setMoveable(bool)
+ * @see DWT#Move
+ *
+ * @since 3.1
+ */
+public int [] getColumnOrder () {
+    checkWidget ();
+    if (columnCount is 0) return new int [0];
+    auto list = OS.gtk_tree_view_get_columns (handle);
+    if (list is null) return new int [0];
+    int  i = 0, count = OS.g_list_length (list);
+    int [] order = new int [count];
+    auto temp = list;
+    while (temp !is null) {
+        auto column = OS.g_list_data (temp);
+        if (column !is null) {
+            for (int j=0; j<columnCount; j++) {
+                if (columns [j].handle is column) {
+                    order [i++] = j;
+                    break;
+                }
+            }
+        }
+        temp = OS.g_list_next (temp);
+    }
+    OS.g_list_free (list);
+    return order;
+}
+
+/**
+ * Returns an array of <code>TableColumn</code>s which are the
+ * columns in the receiver.  Columns are returned in the order
+ * that they were created.  If no <code>TableColumn</code>s were
+ * created by the programmer, the array is empty, despite the fact
+ * that visually, one column of items may be visible. This occurs
+ * when the programmer uses the table like a list, adding items but
+ * never creating a column.
+ * <p>
+ * Note: This is not the actual structure used by the receiver
+ * to maintain its list of items, so modifying the array will
+ * not affect the receiver.
+ * </p>
+ *
+ * @return the items in the receiver
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#getColumnOrder()
+ * @see Table#setColumnOrder(int[])
+ * @see TableColumn#getMoveable()
+ * @see TableColumn#setMoveable(bool)
+ * @see DWT#Move
+ */
+public TableColumn [] getColumns () {
+    checkWidget();
+    TableColumn [] result = new TableColumn [columnCount];
+    System.arraycopy (columns, 0, result, 0, columnCount);
+    return result;
+}
+
+TableItem getFocusItem () {
+    GtkTreeViewColumn * path;
+    OS.gtk_tree_view_get_cursor (handle, cast(void**)&path, null);
+    if (path is null) return null;
+    TableItem item = null;
+    auto indices = OS.gtk_tree_path_get_indices (path);
+    if (indices !is null) {
+        item = _getItem (indices [0]);
+    }
+    OS.gtk_tree_path_free (path);
+    return item;
+}
+
+GdkColor* getForegroundColor () {
+    return getTextColor ();
+}
+
+/**
+ * Returns the width in pixels of a grid line.
+ *
+ * @return the width of a grid line in pixels
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int getGridLineWidth () {
+    checkWidget();
+    return 0;
+}
+
+/**
+ * Returns the height of the receiver's header
+ *
+ * @return the height of the header or zero if the header is not visible
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 2.0
+ */
+public int getHeaderHeight () {
+    checkWidget ();
+    if (!OS.gtk_tree_view_get_headers_visible (handle)) return 0;
+    if (columnCount > 0) {
+        GtkRequisition requisition;
+        int height = 0;
+        for (int i=0; i<columnCount; i++) {
+            auto buttonHandle = columns [i].buttonHandle;
+            if (buttonHandle !is null) {
+                OS.gtk_widget_size_request (buttonHandle, &requisition);
+                height = Math.max (height, requisition.height);
+            }
+        }
+        return height;
+    }
+    OS.gtk_widget_realize (handle);
+    auto fixedWindow = OS.GTK_WIDGET_WINDOW (fixedHandle);
+    auto binWindow = OS.gtk_tree_view_get_bin_window (handle);
+    int binY;
+    OS.gdk_window_get_origin (binWindow, null, &binY);
+    int fixedY;
+    OS.gdk_window_get_origin (fixedWindow, null, &fixedY);
+    return binY - fixedY;
+}
+
+/**
+ * Returns <code>true</code> if the receiver's header is visible,
+ * and <code>false</code> otherwise.
+ * <p>
+ * If one of the receiver's ancestors is not visible or some
+ * other condition makes the receiver not visible, this method
+ * may still indicate that it is considered visible even though
+ * it may not actually be showing.
+ * </p>
+ *
+ * @return the receiver's header's visibility state
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public bool getHeaderVisible () {
+    checkWidget();
+    return cast(bool) OS.gtk_tree_view_get_headers_visible (handle);
+}
+
+/**
+ * Returns the item at the given, zero-relative index in the
+ * receiver. Throws an exception if the index is out of range.
+ *
+ * @param index the index of the item to return
+ * @return the item at the given index
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_RANGE - if the index is not between 0 and the number of elements in the list minus 1 (inclusive)</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public TableItem getItem (int index) {
+    checkWidget();
+    if (!(0 <= index && index < itemCount)) error (DWT.ERROR_INVALID_RANGE);
+    return _getItem (index);
+}
+
+/**
+ * Returns the item at the given point in the receiver
+ * or null if no such item exists. The point is in the
+ * coordinate system of the receiver.
+ * <p>
+ * The item that is returned represents an item that could be selected by the user.
+ * For example, if selection only occurs in items in the first column, then null is
+ * returned if the point is outside of the item.
+ * Note that the DWT.FULL_SELECTION style hint, which specifies the selection policy,
+ * determines the extent of the selection.
+ * </p>
+ *
+ * @param point the point used to locate the item
+ * @return the item at the given point, or null if the point is not in a selectable item
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the point is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public TableItem getItem (Point point) {
+    checkWidget();
+    if (point is null) error (DWT.ERROR_NULL_ARGUMENT);
+    void* path;
+    OS.gtk_widget_realize (handle);
+    if (!OS.gtk_tree_view_get_path_at_pos (handle, point.x, point.y, &path, null, null, null)) return null;
+    if (path is null) return null;
+    auto indices = OS.gtk_tree_path_get_indices (path );
+    TableItem item = null;
+    if (indices !is null) {
+        item = _getItem (indices [0]);
+    }
+    OS.gtk_tree_path_free (path );
+    return item;
+}
+
+/**
+ * Returns the number of items contained in the receiver.
+ *
+ * @return the number of items
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int getItemCount () {
+    checkWidget ();
+    return itemCount;
+}
+
+/**
+ * Returns the height of the area which would be used to
+ * display <em>one</em> of the items in the receiver's.
+ *
+ * @return the height of one item
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int getItemHeight () {
+    checkWidget();
+    if (itemCount is 0) {
+        auto column = OS.gtk_tree_view_get_column (handle, 0);
+        int w, h;
+        ignoreSize = true;
+        OS.gtk_tree_view_column_cell_get_size (column, null, null, null, &w, &h);
+        ignoreSize = false;
+        return h;
+    } else {
+        int height = 0;
+        GtkTreeIter* iter = new GtkTreeIter;
+        OS.gtk_tree_model_get_iter_first (modelHandle, iter);
+        int columnCount = Math.max (1, this.columnCount);
+        for (int i=0; i<columnCount; i++) {
+            auto column = OS.gtk_tree_view_get_column (handle, i);
+            OS.gtk_tree_view_column_cell_set_cell_data (column, modelHandle, iter, false, false);
+            int w, h;
+            OS.gtk_tree_view_column_cell_get_size (column, null, null, null, &w, &h);
+            height = Math.max (height, h );
+        }
+        delete iter;
+        return height;
+    }
+}
+
+/**
+ * Returns a (possibly empty) array of <code>TableItem</code>s which
+ * are the items in the receiver.
+ * <p>
+ * Note: This is not the actual structure used by the receiver
+ * to maintain its list of items, so modifying the array will
+ * not affect the receiver.
+ * </p>
+ *
+ * @return the items in the receiver
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public TableItem [] getItems () {
+    checkWidget();
+    TableItem [] result = new TableItem [itemCount];
+    if ((style & DWT.VIRTUAL) !is 0) {
+        for (int i=0; i<itemCount; i++) {
+            result [i] = _getItem (i);
+        }
+    } else {
+        System.arraycopy (items, 0, result, 0, itemCount);
+    }
+    return result;
+}
+
+/**
+ * Returns <code>true</code> if the receiver's lines are visible,
+ * and <code>false</code> otherwise.
+ * <p>
+ * If one of the receiver's ancestors is not visible or some
+ * other condition makes the receiver not visible, this method
+ * may still indicate that it is considered visible even though
+ * it may not actually be showing.
+ * </p>
+ *
+ * @return the visibility state of the lines
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public bool getLinesVisible() {
+    checkWidget();
+    return cast(bool)OS.gtk_tree_view_get_rules_hint (handle);
+}
+
+GtkCellRendererPixbuf* getPixbufRenderer (GtkTreeViewColumn* column) {
+    auto list = OS.gtk_tree_view_column_get_cell_renderers (column);
+    if (list is null) return null;
+    int count = OS.g_list_length (list);
+    GtkCellRendererPixbuf* pixbufRenderer;
+    int i = 0;
+    while (i < count) {
+        auto renderer = OS.g_list_nth_data (list, i);
+         if (OS.GTK_IS_CELL_RENDERER_PIXBUF (cast(GTypeInstance*)renderer)) {
+            pixbufRenderer = cast(GtkCellRendererPixbuf*)renderer;
+            break;
+        }
+        i++;
+    }
+    OS.g_list_free (list);
+    return pixbufRenderer;
+}
+
+/**
+ * Returns an array of <code>TableItem</code>s that are currently
+ * selected in the receiver. The order of the items is unspecified.
+ * An empty array indicates that no items are selected.
+ * <p>
+ * Note: This is not the actual structure used by the receiver
+ * to maintain its selection, so modifying the array will
+ * not affect the receiver.
+ * </p>
+ * @return an array representing the selection
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public TableItem [] getSelection () {
+    checkWidget();
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    if (OS.GTK_VERSION < OS.buildVERSION (2, 2, 0)) {
+        display.treeSelectionLength  = 0;
+        display.treeSelection = new int [itemCount];
+        display.doTreeSelectionProcConnect( &treeSelectionCallbackData, handle, selection );
+        TableItem [] result = new TableItem [display.treeSelectionLength];
+        for (int i=0; i<result.length; i++) result [i] = _getItem (display.treeSelection [i]);
+        return result;
+    }
+    /*
+    * Bug in GTK.  gtk_tree_selection_get_selected_rows() segmentation faults
+    * in versions smaller than 2.2.4 if the model is NULL.  The fix is
+    * to give a valid pointer instead.
+    */
+    int dummy;
+    void* model = OS.GTK_VERSION < OS.buildVERSION (2, 2, 4) ? &dummy : null;
+    auto list = OS.gtk_tree_selection_get_selected_rows (selection, &model);
+    if (list !is null) {
+        int count = OS.g_list_length (list);
+        int [] treeSelection = new int [count];
+        int length_ = 0;
+        for (int i=0; i<count; i++) {
+            auto data = OS.g_list_nth_data (list, i);
+            auto indices = OS.gtk_tree_path_get_indices (data);
+            if (indices !is null) {
+                treeSelection [length_] = indices [0];
+                length_++;
+            }
+        }
+        OS.g_list_free (list);
+        TableItem [] result = new TableItem [length_];
+        for (int i=0; i<result.length; i++) result [i] = _getItem (treeSelection [i]);
+        return result;
+    }
+    return null;
+}
+
+/**
+ * Returns the number of selected items contained in the receiver.
+ *
+ * @return the number of selected items
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int getSelectionCount () {
+    checkWidget();
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    if (OS.GTK_VERSION < OS.buildVERSION (2, 2, 0)) {
+        display.treeSelectionLength = 0;
+        display.treeSelection = null;
+        display.doTreeSelectionProcConnect( &treeSelectionCallbackData, handle, selection );
+        return display.treeSelectionLength;
+    }
+    return OS.gtk_tree_selection_count_selected_rows (selection);
+}
+
+/**
+ * Returns the zero-relative index of the item which is currently
+ * selected in the receiver, or -1 if no item is selected.
+ *
+ * @return the index of the selected item
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int getSelectionIndex () {
+    checkWidget();
+    if (OS.GTK_VERSION < OS.buildVERSION (2, 2, 0)) {
+        display.treeSelectionLength  = 0;
+        display.treeSelection = new int [itemCount];
+        auto selection = OS.gtk_tree_view_get_selection (handle);
+        display.doTreeSelectionProcConnect( &treeSelectionCallbackData, handle, selection );
+        if (display.treeSelectionLength is 0) return -1;
+        return display.treeSelection [0];
+    }
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    /*
+    * Bug in GTK.  gtk_tree_selection_get_selected_rows() segmentation faults
+    * in versions smaller than 2.2.4 if the model is NULL.  The fix is
+    * to give a valid pointer instead.
+    */
+    int dummy;
+    void* model = OS.GTK_VERSION < OS.buildVERSION (2, 2, 4) ? &dummy : null;
+    auto list = OS.gtk_tree_selection_get_selected_rows (selection, &model);
+    if (list !is null) {
+        int count = OS.g_list_length (list);
+        int index;
+        for (int i=0; i<count; i++) {
+            auto data = OS.g_list_nth_data (list, i);
+            auto indices = OS.gtk_tree_path_get_indices (data);
+            if (indices !is null) {
+                index = indices[0];
+                break;
+            }
+        }
+        OS.g_list_free (list);
+        return index;
+    }
+    return -1;
+}
+
+/**
+ * Returns the zero-relative indices of the items which are currently
+ * selected in the receiver. The order of the indices is unspecified.
+ * The array is empty if no items are selected.
+ * <p>
+ * Note: This is not the actual structure used by the receiver
+ * to maintain its selection, so modifying the array will
+ * not affect the receiver.
+ * </p>
+ * @return the array of indices of the selected items
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int [] getSelectionIndices () {
+    checkWidget();
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    if (OS.GTK_VERSION < OS.buildVERSION (2, 2, 0)) {
+        display.treeSelectionLength  = 0;
+        display.treeSelection = new int [itemCount];
+        display.doTreeSelectionProcConnect( &treeSelectionCallbackData, handle, selection );
+        if (display.treeSelectionLength is display.treeSelection.length) return display.treeSelection;
+        int [] result = new int [display.treeSelectionLength];
+        System.arraycopy (display.treeSelection, 0, result, 0, display.treeSelectionLength);
+        return result;
+    }
+    /*
+    * Bug in GTK.  gtk_tree_selection_get_selected_rows() segmentation faults
+    * in versions smaller than 2.2.4 if the model is NULL.  The fix is
+    * to give a valid pointer instead.
+    */
+    int dummy;
+    void* model = OS.GTK_VERSION < OS.buildVERSION (2, 2, 4) ? &dummy : null;
+    auto list = OS.gtk_tree_selection_get_selected_rows (selection, &model);
+    if (list !is null) {
+        int count = OS.g_list_length (list);
+        int [] treeSelection = new int [count];
+        int length_ = 0;
+        for (int i=0; i<count; i++) {
+            auto data = OS.g_list_nth_data (list, i);
+            auto indices = OS.gtk_tree_path_get_indices (data);
+            if (indices !is null) {
+                treeSelection [length_] = indices [0];
+                length_++;
+            }
+        }
+        OS.g_list_free (list);
+        int [] result = new int [length_];
+        System.arraycopy (treeSelection, 0, result, 0, length_);
+        return result;
+    }
+    return null;
+}
+
+/**
+ * Returns the column which shows the sort indicator for
+ * the receiver. The value may be null if no column shows
+ * the sort indicator.
+ *
+ * @return the sort indicator
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see #setSortColumn(TableColumn)
+ *
+ * @since 3.2
+ */
+public TableColumn getSortColumn () {
+    checkWidget ();
+    return sortColumn;
+}
+
+/**
+ * Returns the direction of the sort indicator for the receiver.
+ * The value will be one of <code>UP</code>, <code>DOWN</code>
+ * or <code>NONE</code>.
+ *
+ * @return the sort direction
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see #setSortDirectioncast(int)
+ *
+ * @since 3.2
+ */
+public int getSortDirection () {
+    checkWidget ();
+    return sortDirection;
+}
+
+GtkCellRendererText* getTextRenderer (GtkTreeViewColumn* column) {
+    auto list = OS.gtk_tree_view_column_get_cell_renderers (column);
+    if (list is null) return null;
+    int count = OS.g_list_length (list);
+    GtkCellRendererText* textRenderer;
+    int i = 0;
+    while (i < count) {
+        auto renderer = OS.g_list_nth_data (list, i);
+         if (OS.GTK_IS_CELL_RENDERER_TEXT (cast(GTypeInstance*)renderer)) {
+            textRenderer = cast(GtkCellRendererText*)renderer;
+            break;
+        }
+        i++;
+    }
+    OS.g_list_free (list);
+    return textRenderer;
+}
+
+/**
+ * Returns the zero-relative index of the item which is currently
+ * at the top of the receiver. This index can change when items are
+ * scrolled or new items are added or removed.
+ *
+ * @return the index of the top item
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int getTopIndex () {
+    checkWidget();
+    void* path;
+    OS.gtk_widget_realize (handle);
+    if (!OS.gtk_tree_view_get_path_at_pos (handle, 1, 1, &path, null, null, null)) return 0;
+    if (path is null) return 0;
+    auto indices = OS.gtk_tree_path_get_indices (path);
+    int index;
+    if (indices !is null) index = indices[0];
+    OS.gtk_tree_path_free (path);
+    return index;
+}
+
+override int gtk_button_press_event (GtkWidget* widget, GdkEventButton* gdkEvent) {
+    if (gdkEvent.window !is OS.gtk_tree_view_get_bin_window (handle)) return 0;
+    auto result = super.gtk_button_press_event (widget, gdkEvent);
+    if (result !is 0) return result;
+    /*
+    * Feature in GTK.  In a multi-select table view, when multiple items are already
+    * selected, the selection state of the item is toggled and the previous selection
+    * is cleared. This is not the desired behaviour when bringing up a popup menu.
+    * Also, when an item is reselected with the right button, the tree view issues
+    * an unwanted selection event. The workaround is to detect that case and not
+    * run the default handler when the item is already part of the current selection.
+    */
+    int button = gdkEvent.button;
+    if (button is 3 && gdkEvent.type is OS.GDK_BUTTON_PRESS) {
+        void* path;
+        if (OS.gtk_tree_view_get_path_at_pos (handle, cast(int)gdkEvent.x, cast(int)gdkEvent.y, &path, null, null, null)) {
+            if (path !is null) {
+                auto selection = OS.gtk_tree_view_get_selection (handle);
+                if (OS.gtk_tree_selection_path_is_selected (selection, path)) result = 1;
+                OS.gtk_tree_path_free (path);
+            }
+        }
+    }
+
+    /*
+    * Feature in GTK.  When the user clicks in a single selection GtkTreeView
+    * and there are no selected items, the first item is selected automatically
+    * before the click is processed, causing two selection events.  The is fix
+    * is the set the cursor item to be same as the clicked item to stop the
+    * widget from automatically selecting the first item.
+    */
+    if ((style & DWT.SINGLE) !is 0 && getSelectionCount () is 0) {
+        void* path;
+        if (OS.gtk_tree_view_get_path_at_pos (handle, cast(int)gdkEvent.x, cast(int)gdkEvent.y, &path, null, null, null)) {
+            if (path !is null) {
+                auto selection = OS.gtk_tree_view_get_selection (handle);
+                OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+                OS.gtk_tree_view_set_cursor (handle, path, null, false);
+                OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+                OS.gtk_tree_path_free (path);
+            }
+        }
+    }
+    /*
+    * Bug in GTK. GTK segments fault, if the GtkTreeView widget is
+    * not in focus and all items in the widget are disposed before
+    * it finishes processing a button press.  The fix is to give
+    * focus to the widget before it starts processing the event.
+    */
+    if (!OS.GTK_WIDGET_HAS_FOCUS (handle)) {
+        OS.gtk_widget_grab_focus (handle);
+    }
+    return result;
+}
+
+override int gtk_button_release_event (GtkWidget* widget, GdkEventButton* event) {
+    auto window = OS.GDK_EVENT_WINDOW (event);
+    if (window !is OS.gtk_tree_view_get_bin_window (handle)) return 0;
+    return super.gtk_button_release_event (widget, event);
+}
+
+override int gtk_changed (GtkWidget* widget) {
+    TableItem item = getFocusItem ();
+    if (item !is null) {
+        Event event = new Event ();
+        event.item = item;
+        postEvent (DWT.Selection, event);
+    }
+    return 0;
+}
+
+override int gtk_key_press_event (GtkWidget* widget, GdkEventKey* keyEvent) {
+    auto result = super.gtk_key_press_event (widget, keyEvent);
+    if (result !is 0) return result;
+    if (OS.GTK_VERSION < OS.buildVERSION (2, 2 ,0)) {
+        /*
+        * Feature in GTK 2.0.x.  When an item is default selected using
+        * the return key, GTK does not issue notification. The fix is
+        * to issue this notification when the return key is pressed.
+        */
+        int key = keyEvent.keyval;
+        switch (key) {
+            case OS.GDK_Return:
+            case OS.GDK_KP_Enter: {
+                Event event = new Event ();
+                event.item = getFocusItem ();
+                postEvent (DWT.DefaultSelection, event);
+                break;
+            }
+        }
+    }
+    return result;
+}
+
+override int gtk_popup_menu (GtkWidget* widget) {
+    auto result = super.gtk_popup_menu (widget);
+    /*
+    * Bug in GTK.  The context menu for the typeahead in GtkTreeViewer
+    * opens in the bottom right corner of the screen when Shift+F10
+    * is pressed and the typeahead window was not visible.  The fix is
+    * to prevent the context menu from opening by stopping the default
+    * handler.
+    *
+    * NOTE: The bug only happens in GTK 2.6.5 and lower.
+    */
+    return OS.GTK_VERSION < OS.buildVERSION (2, 6, 5) ? 1 : result;
+}
+
+override int gtk_motion_notify_event (GtkWidget* widget, GdkEventMotion* event) {
+    auto window = OS.GDK_EVENT_WINDOW (event);
+    if (window !is OS.gtk_tree_view_get_bin_window (handle)) return 0;
+    return super.gtk_motion_notify_event (widget, event);
+}
+
+override void gtk_row_activated (GtkTreeView* tree, GtkTreePath* path, GtkTreeViewColumn* column) {
+    TableItem item = null;
+    auto indices = OS.gtk_tree_path_get_indices (path);
+    if (indices !is null) {
+        item = _getItem (indices [0]);
+    }
+    Event event = new Event ();
+    event.item = item;
+    postEvent (DWT.DefaultSelection, event);
+}
+
+override int gtk_toggled (int /*long*/ renderer, char* pathStr) {
+    auto path = OS.gtk_tree_path_new_from_string (pathStr);
+    if (path is null) return 0;
+    auto indices = OS.gtk_tree_path_get_indices (path);
+    if (indices !is null) {
+        TableItem item = _getItem (indices [0]);
+        item.setChecked (!item.getChecked ());
+        Event event = new Event ();
+        event.detail = DWT.CHECK;
+        event.item = item;
+        postEvent (DWT.Selection, event);
+    }
+    OS.gtk_tree_path_free (path);
+    return 0;
+}
+
+override void gtk_widget_size_request (GtkWidget* widget, GtkRequisition* requisition) {
+    /*
+     * Bug in GTK.  For some reason, gtk_widget_size_request() fails
+     * to include the height of the tree view items when there are
+     * no columns visible.  The fix is to temporarily make one column
+     * visible.
+     */
+    if (columnCount is 0) {
+        super.gtk_widget_size_request (widget, requisition);
+        return;
+    }
+    auto columns = OS.gtk_tree_view_get_columns (handle), list = columns;
+    bool fixVisible = columns !is null;
+    while (list !is null) {
+        auto column = OS.g_list_data (list);
+        if (OS.gtk_tree_view_column_get_visible (column)) {
+            fixVisible = false;
+            break;
+        }
+        list = OS.g_list_next (list);
+    }
+    void* columnHandle;
+    if (fixVisible) {
+        columnHandle = OS.g_list_data (columns);
+        OS.gtk_tree_view_column_set_visible (columnHandle, true);
+    }
+    super.gtk_widget_size_request (widget, requisition);
+    if (fixVisible) {
+        OS.gtk_tree_view_column_set_visible (columnHandle, false);
+    }
+    if (columns !is null) OS.g_list_free (columns);
+}
+
+void hideFirstColumn () {
+    auto firstColumn = OS.gtk_tree_view_get_column (handle, 0);
+    OS.gtk_tree_view_column_set_visible (firstColumn, false);
+}
+
+override void hookEvents () {
+    super.hookEvents ();
+    auto selection = OS.gtk_tree_view_get_selection(handle);
+    OS.g_signal_connect_closure (selection, OS.changed.ptr, display.closures [CHANGED], false);
+    OS.g_signal_connect_closure (handle, OS.row_activated.ptr, display.closures [ROW_ACTIVATED], false);
+    if (checkRenderer !is null) {
+        OS.g_signal_connect_closure (checkRenderer, OS.toggled.ptr, display.closures [TOGGLED], false);
+    }
+}
+
+/**
+ * Searches the receiver's list starting at the first column
+ * (index 0) until a column is found that is equal to the
+ * argument, and returns the index of that column. If no column
+ * is found, returns -1.
+ *
+ * @param column the search column
+ * @return the index of the column
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the column is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int indexOf (TableColumn column) {
+    checkWidget();
+    if (column is null) error (DWT.ERROR_NULL_ARGUMENT);
+    for (int i=0; i<columnCount; i++) {
+        if (columns [i] is column) return i;
+    }
+    return -1;
+}
+
+/**
+ * Searches the receiver's list starting at the first item
+ * (index 0) until an item is found that is equal to the
+ * argument, and returns the index of that item. If no item
+ * is found, returns -1.
+ *
+ * @param item the search item
+ * @return the index of the item
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the item is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int indexOf (TableItem item) {
+    checkWidget();
+    if (item is null) error (DWT.ERROR_NULL_ARGUMENT);
+    if (1 <= lastIndexOf && lastIndexOf < itemCount - 1) {
+        if (items [lastIndexOf] is item) return lastIndexOf;
+        if (items [lastIndexOf + 1] is item) return ++lastIndexOf;
+        if (items [lastIndexOf - 1] is item) return --lastIndexOf;
+    }
+    if (lastIndexOf < itemCount / 2) {
+        for (int i=0; i<itemCount; i++) {
+            if (items [i] is item) return lastIndexOf = i;
+        }
+    } else {
+        for (int i=itemCount - 1; i>=0; --i) {
+            if (items [i] is item) return lastIndexOf = i;
+        }
+    }
+    return -1;
+}
+
+/**
+ * Returns <code>true</code> if the item is selected,
+ * and <code>false</code> otherwise.  Indices out of
+ * range are ignored.
+ *
+ * @param index the index of the item
+ * @return the selection state of the item at the index
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public bool isSelected (int index) {
+    checkWidget();
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    char* buffer = tango.stdc.stringz.toStringz( to!(char[])(index));
+    auto path = OS.gtk_tree_path_new_from_string (buffer);
+    bool answer = cast(bool)OS.gtk_tree_selection_path_is_selected (selection, path);
+    OS.gtk_tree_path_free (path);
+    return answer;
+}
+
+alias Composite.mnemonicHit mnemonicHit;
+bool mnemonicHit (char key) {
+    for (int i=0; i<columnCount; i++) {
+        auto labelHandle = columns [i].labelHandle;
+        if (labelHandle !is null && mnemonicHit (labelHandle, key)) return true;
+    }
+    return false;
+}
+
+alias Composite.mnemonicMatch mnemonicMatch;
+bool mnemonicMatch (char key) {
+    for (int i=0; i<columnCount; i++) {
+        auto labelHandle = columns [i].labelHandle;
+        if (labelHandle !is null && mnemonicMatch (labelHandle, key)) return true;
+    }
+    return false;
+}
+
+GdkDrawable* paintWindow () {
+    OS.gtk_widget_realize (handle);
+    return OS.gtk_tree_view_get_bin_window (handle);
+}
+
+void recreateRenderers () {
+    if (checkRenderer !is null) {
+        display.removeWidget (checkRenderer);
+        OS.g_object_unref (checkRenderer);
+        checkRenderer = ownerDraw ? cast(GtkWidget*)OS.g_object_new (display.gtk_cell_renderer_toggle_get_type(), null) : cast(GtkWidget*)OS.gtk_cell_renderer_toggle_new ();
+        if (checkRenderer is null) error (DWT.ERROR_NO_HANDLES);
+        OS.g_object_ref (checkRenderer);
+        display.addWidget (checkRenderer, this);
+        OS.g_signal_connect_closure (checkRenderer, OS.toggled.ptr, display.closures [TOGGLED], false);
+    }
+    if (columnCount is 0) {
+        createRenderers (OS.gtk_tree_view_get_column (handle, 0), Table.FIRST_COLUMN, true, 0);
+    } else {
+        for (int i = 0; i < columnCount; i++) {
+            TableColumn column = columns [i];
+            createRenderers (cast(GtkTreeViewColumn*)column.handle, column.modelIndex, i is 0, column.style);
+        }
+    }
+}
+
+void redrawBackgroundImage () {
+    Control control = findBackgroundControl ();
+    if (control !is null && control.backgroundImage !is null) {
+        redrawWidget (0, 0, 0, 0, true, false, false);
+    }
+}
+
+override void register () {
+    super.register ();
+    display.addWidget ( cast(GtkWidget*)OS.gtk_tree_view_get_selection (handle), this);
+    if (checkRenderer !is null) display.addWidget (checkRenderer, this);
+}
+
+override void releaseChildren (bool destroy) {
+    if (items !is null) {
+        for (int i=0; i<itemCount; i++) {
+            TableItem item = items [i];
+            if (item !is null && !item.isDisposed ()) {
+                item.release (false);
+            }
+        }
+        items = null;
+    }
+    if (columns !is null) {
+        for (int i=0; i<columnCount; i++) {
+            TableColumn column = columns [i];
+            if (column !is null && !column.isDisposed ()) {
+                column.release (false);
+            }
+        }
+        columns = null;
+    }
+    super.releaseChildren (destroy);
+}
+
+override void releaseWidget () {
+    super.releaseWidget ();
+    if (modelHandle !is null) OS.g_object_unref (modelHandle);
+    modelHandle = null;
+    if (checkRenderer !is null) OS.g_object_unref (checkRenderer);
+    checkRenderer = null;
+    if (imageList !is null) imageList.dispose ();
+    if (headerImageList !is null) headerImageList.dispose ();
+    imageList = headerImageList = null;
+    currentItem = null;
+}
+
+/**
+ * Removes the item from the receiver at the given
+ * zero-relative index.
+ *
+ * @param index the index for the item
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_RANGE - if the index is not between 0 and the number of elements in the list minus 1 (inclusive)</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void remove (int index) {
+    checkWidget();
+    if (!(0 <= index && index < itemCount)) error (DWT.ERROR_ITEM_NOT_REMOVED);
+    GtkTreeIter iter;
+    TableItem item = items [index];
+    bool disposed = false;
+    if (item !is null) {
+        disposed = item.isDisposed ();
+        if (!disposed) {
+            iter = *cast(GtkTreeIter*)item.handle;
+            item.release (false);
+        }
+    } else {
+        OS.gtk_tree_model_iter_nth_child (modelHandle, &iter, null, index);
+    }
+    if (!disposed) {
+        auto selection = OS.gtk_tree_view_get_selection (handle);
+        OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+        OS.gtk_list_store_remove (modelHandle, &iter);
+        OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+        System.arraycopy (items, index + 1, items, index, --itemCount - index);
+        items [itemCount] = null;
+    }
+}
+
+/**
+ * Removes the items from the receiver which are
+ * between the given zero-relative start and end
+ * indices (inclusive).
+ *
+ * @param start the start of the range
+ * @param end the end of the range
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_RANGE - if either the start or end are not between 0 and the number of elements in the list minus 1 (inclusive)</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void remove (int start, int end) {
+    checkWidget();
+    if (start > end) return;
+    if (!(0 <= start && start <= end && end < itemCount)) {
+        error (DWT.ERROR_INVALID_RANGE);
+    }
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    GtkTreeIter iter;
+    OS.gtk_tree_model_iter_nth_child (modelHandle, &iter, null, start);
+    int index = start;
+    while (index <= end) {
+        TableItem item = items [index];
+        if (item !is null && !item.isDisposed ()) item.release (false);
+        OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+        OS.gtk_list_store_remove (modelHandle, &iter);
+        OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+        index++;
+    }
+    System.arraycopy (items, index, items, start, itemCount - index);
+    for (int i=itemCount-(index-start); i<itemCount; i++) items [i] = null;
+    itemCount = itemCount - (index - start);
+}
+
+/**
+ * Removes the items from the receiver's list at the given
+ * zero-relative indices.
+ *
+ * @param indices the array of indices of the items
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_RANGE - if the index is not between 0 and the number of elements in the list minus 1 (inclusive)</li>
+ *    <li>ERROR_NULL_ARGUMENT - if the indices array is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void remove (int [] indices) {
+    checkWidget();
+    if (indices is null) error (DWT.ERROR_NULL_ARGUMENT);
+    if (indices.length is 0) return;
+    int [] newIndices = new int [indices.length];
+    System.arraycopy (indices, 0, newIndices, 0, indices.length);
+    sort (newIndices);
+    int start = newIndices [newIndices.length - 1], end = newIndices [0];
+    if (!(0 <= start && start <= end && end < itemCount)) {
+        error (DWT.ERROR_INVALID_RANGE);
+    }
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    int last = -1;
+    GtkTreeIter iter;
+    for (int i=0; i<newIndices.length; i++) {
+        int index = newIndices [i];
+        if (index !is last) {
+            TableItem item = items [index];
+            bool disposed = false;
+            if (item !is null) {
+                disposed = item.isDisposed ();
+                if (!disposed) {
+                    iter = *cast(GtkTreeIter*) item.handle;
+                    item.release (false);
+                }
+            } else {
+                OS.gtk_tree_model_iter_nth_child (modelHandle, &iter, null, index);
+            }
+            if (!disposed) {
+                OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+                OS.gtk_list_store_remove (modelHandle, &iter);
+                OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+                System.arraycopy (items, index + 1, items, index, --itemCount - index);
+                items [itemCount] = null;
+            }
+            last = index;
+        }
+    }
+}
+
+/**
+ * Removes all of the items from the receiver.
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void removeAll () {
+    checkWidget();
+    int index = itemCount - 1;
+    while (index >= 0) {
+        TableItem item = items [index];
+        if (item !is null && !item.isDisposed ()) item.release (false);
+        --index;
+    }
+    items = new TableItem [4];
+    itemCount = 0;
+
+    /*
+    * Bug in GTK.  In version 2.3.2, when the property fixed-height-mode
+    * is set and there are items in the list, OS.gtk_list_store_clear()
+    * segment faults.  The fix is to create a new empty model instead.
+    */
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+//  OS.gtk_list_store_clear (modelHandle);
+    auto oldModel = modelHandle;
+    auto types = getColumnTypes (Math.max (1,columnCount));
+    auto newModel = OS.gtk_list_store_newv (types.length, types.ptr);
+    if (newModel is null) error (DWT.ERROR_NO_HANDLES);
+    OS.gtk_tree_view_set_model (handle, newModel);
+    OS.g_object_unref (oldModel);
+    modelHandle = cast(GtkWidget*)newModel;
+    OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+
+    resetCustomDraw ();
+    /* Disable searching when using VIRTUAL */
+    if ((style & DWT.VIRTUAL) !is 0) {
+        /*
+        * Bug in GTK. Until GTK 2.6.5, calling gtk_tree_view_set_enable_search(FALSE)
+        * would prevent the user from being able to type in text to search the tree.
+        * After 2.6.5, GTK introduced Ctrl+F as being the key binding for interactive
+        * search. This meant that even if FALSE was passed to enable_search, the user
+        * can still bring up the search pop up using the keybinding. GTK also introduced
+        * the notion of passing a -1 to gtk_set_search_column to disable searching
+        * (including the search key binding).  The fix is to use the right calls
+        * for the right version.
+        */
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 6, 5)){
+            OS.gtk_tree_view_set_search_column (handle, -1);
+        } else {
+            OS.gtk_tree_view_set_enable_search (handle, false);
+        }
+    } else {
+        /* Set the search column whenever the model changes */
+        int firstColumn = columnCount is 0 ? FIRST_COLUMN : columns [0].modelIndex;
+        OS.gtk_tree_view_set_search_column (handle, firstColumn + CELL_TEXT);
+    }
+}
+
+/**
+ * Removes the listener from the collection of listeners who will
+ * be notified when the user changes the receiver's selection.
+ *
+ * @param listener the listener which should no longer be notified
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see SelectionListener
+ * @see #addSelectionListener(SelectionListener)
+ */
+public void removeSelectionListener(SelectionListener listener) {
+    checkWidget();
+    if (listener is null) error (DWT.ERROR_NULL_ARGUMENT);
+    if (eventTable is null) return;
+    eventTable.unhook (DWT.Selection, listener);
+    eventTable.unhook (DWT.DefaultSelection,listener);
+}
+
+override void rendererGetSizeProc (
+    GtkCellRenderer      *cell,
+    GtkWidget            *widget,
+    GdkRectangle         *cell_area,
+    int                  *x_offset,
+    int                  *y_offset,
+    int                  *width,
+    int                  *height)
+{
+    auto g_class = OS.g_type_class_peek_parent (OS.G_OBJECT_GET_CLASS (cell));
+    GtkCellRendererClass* klass = cast(GtkCellRendererClass*)g_class;
+    klass.get_size( cell, handle, cell_area, x_offset, y_offset, width, height);
+    if (!ignoreSize && OS.GTK_IS_CELL_RENDERER_TEXT (cell)) {
+        auto iter = OS.g_object_get_qdata (cast(GObject*)cell, Display.SWT_OBJECT_INDEX2);
+        TableItem item = null;
+        if (iter !is null) {
+            auto path = OS.gtk_tree_model_get_path (modelHandle, iter);
+            auto buffer = OS.gtk_tree_path_get_indices (path);
+            int index = buffer [0];
+            item = _getItem (index);
+            OS.gtk_tree_path_free (path);
+        }
+        if (item !is null) {
+            int columnIndex = 0;
+            if (columnCount > 0) {
+                auto columnHandle = OS.g_object_get_qdata (cast(GObject*)cell, Display.SWT_OBJECT_INDEX1);
+                for (int i = 0; i < columnCount; i++) {
+                    if (columns [i].handle is columnHandle) {
+                        columnIndex = i;
+                        break;
+                    }
+                }
+            }
+            if (hooks (DWT.MeasureItem)) {
+                int contentWidth, contentHeight;
+                if (width !is null) contentWidth = *width;
+                if (height !is null) contentHeight = *height;
+                Image image = item.getImage (columnIndex);
+                int imageWidth = 0;
+                if (image !is null) {
+                    Rectangle bounds = image.getBounds ();
+                    imageWidth = bounds.width;
+                }
+                contentWidth += imageWidth;
+                GC gc = new GC (this);
+                gc.setFont (item.getFont (columnIndex));
+                Event event = new Event ();
+                event.item = item;
+                event.index = columnIndex;
+                event.gc = gc;
+                event.width = contentWidth;
+                event.height = contentHeight;
+                sendEvent (DWT.MeasureItem, event);
+                gc.dispose ();
+                contentWidth = event.width - imageWidth;
+                contentHeight = event.height;
+                if (width !is null) *width = contentWidth;
+                if (height !is null) *height = contentHeight;
+            }
+        }
+    }
+}
+
+override void rendererRenderProc (
+    GtkCellRenderer * cell,
+    GdkDrawable * window,
+    GtkWidget * widget,
+    GdkRectangle *background_area,
+    GdkRectangle *cell_area,
+    GdkRectangle *expose_area,
+    int flags)
+{
+    TableItem item = null;
+    auto iter = OS.g_object_get_qdata (cast(GObject*)cell, Display.SWT_OBJECT_INDEX2);
+    if (iter !is null) {
+        auto path = OS.gtk_tree_model_get_path (modelHandle, iter);
+        auto buffer = OS.gtk_tree_path_get_indices (path);
+        int index = buffer [0];
+        item = _getItem (index);
+        OS.gtk_tree_path_free (path);
+    }
+    auto columnHandle = OS.g_object_get_qdata (cast(GObject*)cell, Display.SWT_OBJECT_INDEX1);
+    int columnIndex = 0;
+    if (columnCount > 0) {
+        for (int i = 0; i < columnCount; i++) {
+            if (columns [i].handle is columnHandle) {
+                columnIndex = i;
+                break;
+            }
+        }
+    }
+    if (item !is null) {
+        if (OS.GTK_IS_CELL_RENDERER_TOGGLE (cell) || (OS.GTK_IS_CELL_RENDERER_PIXBUF (cell) && (columnIndex !is 0 || (style & DWT.CHECK) is 0))) {
+            drawFlags = cast(int)/*64*/flags;
+            drawState = DWT.FOREGROUND;
+            void* ptr;
+            OS.gtk_tree_model_get1 (modelHandle, item.handle, Table.BACKGROUND_COLUMN, &ptr);
+            if (ptr is null) {
+                int modelIndex = columnCount is 0 ? Table.FIRST_COLUMN : columns [columnIndex].modelIndex;
+                OS.gtk_tree_model_get1 (modelHandle, item.handle, modelIndex + Table.CELL_BACKGROUND, &ptr);
+            }
+            if (ptr !is null) drawState |= DWT.BACKGROUND;
+            if ((flags & OS.GTK_CELL_RENDERER_SELECTED) !is 0) drawState |= DWT.SELECTED;
+            if ((flags & OS.GTK_CELL_RENDERER_FOCUSED) !is 0) drawState |= DWT.FOCUSED;
+
+            GdkRectangle rect;
+            auto path = OS.gtk_tree_model_get_path (modelHandle, iter);
+            OS.gtk_tree_view_get_background_area (handle, path, columnHandle, &rect);
+            OS.gtk_tree_path_free (path);
+
+            if ((drawState & DWT.SELECTED) is 0) {
+                Control control = findBackgroundControl ();
+                if (control !is null && control.backgroundImage !is null) {
+                    OS.gdk_window_clear_area (window, rect.x, rect.y, rect.width, rect.height);
+                }
+            }
+
+            if (hooks (DWT.EraseItem)) {
+                bool wasSelected = false;
+                if ((drawState & DWT.SELECTED) !is 0) {
+                    wasSelected = true;
+                    OS.gdk_window_clear_area (window, rect.x, rect.y, rect.width, rect.height);
+                }
+                GC gc = new GC (this);
+                if ((drawState & DWT.SELECTED) !is 0) {
+                    gc.setBackground (display.getSystemColor (DWT.COLOR_LIST_SELECTION));
+                    gc.setForeground (display.getSystemColor (DWT.COLOR_LIST_SELECTION_TEXT));
+                } else {
+                    gc.setBackground (item.getBackground (columnIndex));
+                    gc.setForeground (item.getForeground (columnIndex));
+                }
+                gc.setFont (item.getFont (columnIndex));
+                gc.setClipping (rect.x, rect.y, rect.width, rect.height);
+                Event event = new Event ();
+                event.item = item;
+                event.index = columnIndex;
+                event.gc = gc;
+                event.x = rect.x;
+                event.y = rect.y;
+                event.width = rect.width;
+                event.height = rect.height;
+                event.detail = drawState;
+                sendEvent (DWT.EraseItem, event);
+                drawForeground = null;
+                drawState = event.doit ? event.detail : 0;
+                drawFlags &= ~(OS.GTK_CELL_RENDERER_FOCUSED | OS.GTK_CELL_RENDERER_SELECTED);
+                if ((drawState & DWT.SELECTED) !is 0) drawFlags |= OS.GTK_CELL_RENDERER_SELECTED;
+                if ((drawState & DWT.FOCUSED) !is 0) drawFlags |= OS.GTK_CELL_RENDERER_FOCUSED;
+                if ((drawState & DWT.SELECTED) !is 0) {
+                    auto style = OS.gtk_widget_get_style (widget);
+                    //TODO - parity and sorted
+                    OS.gtk_paint_flat_box (style, window, OS.GTK_STATE_SELECTED, OS.GTK_SHADOW_NONE, &rect, widget, "cell_odd".ptr, rect.x, rect.y, rect.width, rect.height);
+                } else {
+                    if (wasSelected) drawForeground = gc.getForeground ().handle;
+                }
+                gc.dispose();
+            }
+        }
+    }
+    int /*long*/ result = 0;
+    if ((drawState & DWT.BACKGROUND) !is 0 && (drawState & DWT.SELECTED) is 0) {
+        GC gc = new GC (this);
+        gc.setBackground (item.getBackground (columnIndex));
+        gc.fillRectangle (background_area.x, background_area.y, background_area.width, background_area.height);
+        gc.dispose ();
+    }
+    if ((drawState & DWT.FOREGROUND) !is 0 || OS.GTK_IS_CELL_RENDERER_TOGGLE (cell)) {
+        auto g_class = OS.g_type_class_peek_parent (OS.G_OBJECT_GET_CLASS (cell));
+        GtkCellRendererClass* klass = cast(GtkCellRendererClass*)g_class;
+        if (drawForeground !is null && OS.GTK_IS_CELL_RENDERER_TEXT (cell)) {
+            OS.g_object_set1 (cell, OS.foreground_gdk.ptr, cast(int)drawForeground);
+        }
+        klass.render( cell, window, widget, background_area, cell_area, expose_area, drawFlags);
+    }
+    if (item !is null) {
+        if (OS.GTK_IS_CELL_RENDERER_TEXT (cell)) {
+            if (hooks (DWT.PaintItem)) {
+                GdkRectangle rect;
+                auto path = OS.gtk_tree_model_get_path (modelHandle, iter);
+                OS.gtk_tree_view_get_background_area (handle, path, columnHandle, &rect);
+                OS.gtk_tree_path_free (path);
+                ignoreSize = true;
+                int contentX, contentWidth;
+                OS.gtk_cell_renderer_get_size (cell, handle, null, null, null, &contentWidth, null);
+                OS.gtk_tree_view_column_cell_get_position (columnHandle, cell, &contentX, null);
+                ignoreSize = false;
+                Image image = item.getImage (columnIndex);
+                int imageWidth = 0;
+                if (image !is null) {
+                    Rectangle bounds = image.getBounds ();
+                    imageWidth = bounds.width;
+                }
+                contentX -= imageWidth;
+                contentWidth += imageWidth;
+                GC gc = new GC (this);
+                if ((drawState & DWT.SELECTED) !is 0) {
+                    gc.setBackground (display.getSystemColor (DWT.COLOR_LIST_SELECTION));
+                    gc.setForeground (display.getSystemColor (DWT.COLOR_LIST_SELECTION_TEXT));
+                } else {
+                    gc.setBackground (item.getBackground (columnIndex));
+                    Color foreground = drawForeground !is null ? Color.gtk_new (display, drawForeground) : item.getForeground (columnIndex);
+                    gc.setForeground (foreground);
+                }
+                gc.setFont (item.getFont (columnIndex));
+                gc.setClipping (rect.x, rect.y, rect.width, rect.height);
+                Event event = new Event ();
+                event.item = item;
+                event.index = columnIndex;
+                event.gc = gc;
+                event.x = rect.x + contentX;
+                event.y = rect.y;
+                event.width = contentWidth;
+                event.height = rect.height;
+                event.detail = drawState;
+                sendEvent (DWT.PaintItem, event);
+                gc.dispose();
+            }
+        }
+    }
+}
+
+void resetCustomDraw () {
+    if ((style & DWT.VIRTUAL) !is 0 || ownerDraw) return;
+    int end = Math.max (1, columnCount);
+    for (int i=0; i<end; i++) {
+        bool customDraw = columnCount !is 0 ? columns [i].customDraw : firstCustomDraw;
+        if (customDraw) {
+            auto column = OS.gtk_tree_view_get_column (handle, i);
+            auto textRenderer = getTextRenderer (column);
+            display.doCellDataProc( null, cast(GtkTreeViewColumn*)column, cast(GtkCellRenderer*)textRenderer );
+            if (columnCount !is 0) columns [i].customDraw = false;
+        }
+    }
+    firstCustomDraw = false;
+}
+
+/**
+ * Selects the item at the given zero-relative index in the receiver.
+ * If the item at the index was already selected, it remains
+ * selected. Indices that are out of range are ignored.
+ *
+ * @param index the index of the item to select
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void select (int index) {
+    checkWidget();
+    if (!(0 <= index && index < itemCount))  return;
+    bool fixColumn = showFirstColumn ();
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    TableItem item = _getItem (index);
+    OS.gtk_tree_selection_select_iter (selection, item.handle);
+    if ((style & DWT.SINGLE) !is 0) {
+        auto path = OS.gtk_tree_model_get_path (modelHandle, item.handle);
+        OS.gtk_tree_view_set_cursor (handle, path, null, false);
+        OS.gtk_tree_path_free (path);
+    }
+    OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    if (fixColumn) hideFirstColumn ();
+}
+
+/**
+ * Selects the items in the range specified by the given zero-relative
+ * indices in the receiver. The range of indices is inclusive.
+ * The current selection is not cleared before the new items are selected.
+ * <p>
+ * If an item in the given range is not selected, it is selected.
+ * If an item in the given range was already selected, it remains selected.
+ * Indices that are out of range are ignored and no items will be selected
+ * if start is greater than end.
+ * If the receiver is single-select and there is more than one item in the
+ * given range, then all indices are ignored.
+ * </p>
+ *
+ * @param start the start of the range
+ * @param end the end of the range
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#setSelection(int,int)
+ */
+public void select (int start, int end) {
+    checkWidget ();
+    if (end < 0 || start > end || ((style & DWT.SINGLE) !is 0 && start !is end)) return;
+    if (itemCount is 0 || start >= itemCount) return;
+    start = Math.max (0, start);
+    end = Math.min (end, itemCount - 1);
+    bool fixColumn = showFirstColumn ();
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    for (int index=start; index<=end; index++) {
+        TableItem item = _getItem (index);
+        OS.gtk_tree_selection_select_iter (selection, item.handle);
+        if ((style & DWT.SINGLE) !is 0) {
+            auto path = OS.gtk_tree_model_get_path (modelHandle, item.handle);
+            OS.gtk_tree_view_set_cursor (handle, path, null, false);
+            OS.gtk_tree_path_free (path);
+        }
+    }
+    OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    if (fixColumn) hideFirstColumn ();
+}
+
+/**
+ * Selects the items at the given zero-relative indices in the receiver.
+ * The current selection is not cleared before the new items are selected.
+ * <p>
+ * If the item at a given index is not selected, it is selected.
+ * If the item at a given index was already selected, it remains selected.
+ * Indices that are out of range and duplicate indices are ignored.
+ * If the receiver is single-select and multiple indices are specified,
+ * then all indices are ignored.
+ * </p>
+ *
+ * @param indices the array of indices for the items to select
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the array of indices is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#setSelection(int[])
+ */
+public void select (int [] indices) {
+    checkWidget ();
+    if (indices is null) error (DWT.ERROR_NULL_ARGUMENT);
+    int length = indices.length;
+    if (length is 0 || ((style & DWT.SINGLE) !is 0 && length > 1)) return;
+    bool fixColumn = showFirstColumn ();
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    for (int i=0; i<length; i++) {
+        int index = indices [i];
+        if (!(0 <= index && index < itemCount)) continue;
+        TableItem item = _getItem (index);
+        OS.gtk_tree_selection_select_iter (selection, item.handle);
+        if ((style & DWT.SINGLE) !is 0) {
+            auto path = OS.gtk_tree_model_get_path (modelHandle, item.handle);
+            OS.gtk_tree_view_set_cursor (handle, path, null, false);
+            OS.gtk_tree_path_free (path);
+        }
+    }
+    OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    if (fixColumn) hideFirstColumn ();
+}
+
+/**
+ * Selects all of the items in the receiver.
+ * <p>
+ * If the receiver is single-select, do nothing.
+ * </p>
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void selectAll () {
+    checkWidget();
+    if ((style & DWT.SINGLE) !is 0) return;
+    bool fixColumn = showFirstColumn ();
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    OS.gtk_tree_selection_select_all (selection);
+    OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    if (fixColumn) hideFirstColumn ();
+}
+
+void selectFocusIndex (int index) {
+    /*
+    * Note that this method both selects and sets the focus to the
+    * specified index, so any previous selection in the list will be lost.
+    * gtk does not provide a way to just set focus to a specified list item.
+    */
+    if (!(0 <= index && index < itemCount))  return;
+    TableItem item = _getItem (index);
+    auto path = OS.gtk_tree_model_get_path (modelHandle, item.handle);
+    auto selection = OS.gtk_tree_view_get_selection (handle);
+    OS.g_signal_handlers_block_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    OS.gtk_tree_view_set_cursor (handle, path, null, false);
+    /*
+    * Bug in GTK. For some reason, when an event loop is run from
+    * within a key pressed handler and a dialog is displayed that
+    * contains a GtkTreeView,  gtk_tree_view_set_cursor() does
+    * not set the cursor or select the item.  The fix is to select the
+    * item with gtk_tree_selection_select_iter() as well.
+    *
+    * NOTE: This happens in GTK 2.2.1 and is fixed in GTK 2.2.4.
+    */
+    OS.gtk_tree_selection_select_iter (selection, item.handle);
+    OS.g_signal_handlers_unblock_matched (selection, OS.G_SIGNAL_MATCH_DATA, 0, 0, null, null, udCHANGED);
+    OS.gtk_tree_path_free (path);
+}
+
+override void setBackgroundColor (GdkColor* color) {
+    super.setBackgroundColor (color);
+    OS.gtk_widget_modify_base (handle, 0, color);
+}
+
+override void setBackgroundPixmap (GdkPixmap* pixmap) {
+    super.setBackgroundPixmap (pixmap);
+    auto window = paintWindow ();
+    if (window !is null) OS.gdk_window_set_back_pixmap (window, null, true);
+}
+
+override int setBounds (int x, int y, int width, int height, bool move, bool resize) {
+    int result = super.setBounds (x, y, width, height, move, resize);
+    /*
+    * Bug on GTK.  The tree view sometimes does not get a paint
+    * event or resizes to a one pixel square when resized in a new
+    * shell that is not visible after any event loop has been run.  The
+    * problem is intermittent. It doesn't seem to happen the first time
+    * a new shell is created. The fix is to ensure the tree view is realized
+    * after it has been resized.
+    */
+    OS.gtk_widget_realize (handle);
+    /*
+    * Bug in GTK.  An empty GtkTreeView fails to repaint the focus rectangle
+    * correctly when resized on versions before 2.6.0.  The fix is to force
+    * the widget to redraw.
+    */
+    if (OS.GTK_VERSION < OS.buildVERSION (2, 6, 0) && itemCount is 0) redraw (false);
+    return result;
+}
+
+/**
+ * Sets the order that the items in the receiver should
+ * be displayed in to the given argument which is described
+ * in terms of the zero-relative ordering of when the items
+ * were added.
+ *
+ * @param order the new order to display the items
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the item order is null</li>
+ *    <li>ERROR_INVALID_ARGUMENT - if the item order is not the same length as the number of items</li>
+ * </ul>
+ *
+ * @see Table#getColumnOrder()
+ * @see TableColumn#getMoveable()
+ * @see TableColumn#setMoveable(bool)
+ * @see DWT#Move
+ *
+ * @since 3.1
+ */
+public void setColumnOrder (int [] order) {
+    checkWidget ();
+    if (order is null) error (DWT.ERROR_NULL_ARGUMENT);
+    if (columnCount is 0) {
+        if (order.length > 0) error (DWT.ERROR_INVALID_ARGUMENT);
+        return;
+    }
+    if (order.length !is columnCount) error (DWT.ERROR_INVALID_ARGUMENT);
+    bool [] seen = new bool [columnCount];
+    for (int i = 0; i<order.length; i++) {
+        int index = order [i];
+        if (index < 0 || index >= columnCount) error (DWT.ERROR_INVALID_RANGE);
+        if (seen [index]) error (DWT.ERROR_INVALID_ARGUMENT);
+        seen [index] = true;
+    }
+    for (int i=0; i<order.length; i++) {
+        auto column = columns [order [i]].handle;
+        auto baseColumn = i is 0 ? null : columns [order [i-1]].handle;
+        OS.gtk_tree_view_move_column_after (handle, column, baseColumn);
+    }
+}
+
+override void setFontDescription (PangoFontDescription* font) {
+    super.setFontDescription (font);
+    TableColumn[] columns = getColumns ();
+    for (int i = 0; i < columns.length; i++) {
+        if (columns[i] !is null) {
+            columns[i].setFontDescription (font);
+        }
+    }
+}
+
+/**
+ * Marks the receiver's header as visible if the argument is <code>true</code>,
+ * and marks it invisible otherwise.
+ * <p>
+ * If one of the receiver's ancestors is not visible or some
+ * other condition makes the receiver not visible, marking
+ * it visible may not actually cause it to be displayed.
+ * </p>
+ *
+ * @param show the new visibility state
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void setHeaderVisible (bool show) {
+    checkWidget ();
+    OS.gtk_tree_view_set_headers_visible (handle, show);
+}
+
+/**
+ * Sets the number of items contained in the receiver.
+ *
+ * @param count the number of items
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.0
+ */
+public void setItemCount (int count) {
+    checkWidget ();
+    count = Math.max (0, count);
+    if (count is itemCount) return;
+    bool isVirtual = (style & DWT.VIRTUAL) !is 0;
+    if (!isVirtual) setRedraw (false);
+    remove (count, itemCount - 1);
+    int length = Math.max (4, (count + 3) / 4 * 4);
+    TableItem [] newItems = new TableItem [length];
+    System.arraycopy (items, 0, newItems, 0, itemCount);
+    items = newItems;
+    if (isVirtual) {
+        GtkTreeIter iter;
+        for (int i=itemCount; i<count; i++) {
+            OS.gtk_list_store_append (modelHandle, &iter);
+        }
+        itemCount = count;
+    } else {
+        for (int i=itemCount; i<count; i++) {
+            new TableItem (this, DWT.NONE, i, true);
+        }
+    }
+    if (!isVirtual) setRedraw (true);
+}
+
+/**
+ * Marks the receiver's lines as visible if the argument is <code>true</code>,
+ * and marks it invisible otherwise.
+ * <p>
+ * If one of the receiver's ancestors is not visible or some
+ * other condition makes the receiver not visible, marking
+ * it visible may not actually cause it to be displayed.
+ * </p>
+ *
+ * @param show the new visibility state
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void setLinesVisible (bool show) {
+    checkWidget();
+    OS.gtk_tree_view_set_rules_hint (handle, show);
+}
+
+override void setParentBackground () {
+    super.setParentBackground ();
+    auto window = paintWindow ();
+    if (window !is null) OS.gdk_window_set_back_pixmap (window, null, true);
+}
+
+void setParentWindow (GtkWidget* widget) {
+    auto window = eventWindow ();
+    OS.gtk_widget_set_parent_window (widget, window);
+}
+
+override public void setRedraw (bool redraw) {
+    checkWidget();
+    super.setRedraw (redraw);
+    if (redraw && drawCount is 0) {
+        /* Resize the item array to match the item count */
+        if (items.length > 4 && items.length - itemCount > 3) {
+            int length = Math.max (4, (itemCount + 3) / 4 * 4);
+            TableItem [] newItems = new TableItem [length];
+            System.arraycopy (items, 0, newItems, 0, itemCount);
+            items = newItems;
+        }
+    }
+}
+
+void setScrollWidth (GtkTreeViewColumn* column, TableItem item) {
+    if (columnCount !is 0 || currentItem is item) return;
+    /*
+    * Use GTK_TREE_VIEW_COLUMN_GROW_ONLY on GTK versions < 2.3.2
+    * because fixed_height_mode is not supported.
+    */
+    if (((style & DWT.VIRTUAL) !is 0) && OS.GTK_VERSION < OS.buildVERSION (2, 3, 2)) return;
+    int width = OS.gtk_tree_view_column_get_fixed_width (column);
+    int itemWidth = calculateWidth (column, cast(GtkTreeIter*)item.handle);
+    if (width < itemWidth) {
+        OS.gtk_tree_view_column_set_fixed_width (column, itemWidth);
+    }
+}
+
+/**
+ * Sets the column used by the sort indicator for the receiver. A null
+ * value will clear the sort indicator.  The current sort column is cleared
+ * before the new column is set.
+ *
+ * @param column the column used by the sort indicator or <code>null</code>
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_ARGUMENT - if the column is disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.2
+ */
+public void setSortColumn (TableColumn column) {
+    checkWidget ();
+    if (column !is null && column.isDisposed ()) error (DWT.ERROR_INVALID_ARGUMENT);
+    if (sortColumn !is null && !sortColumn.isDisposed()) {
+        OS.gtk_tree_view_column_set_sort_indicator (sortColumn.handle, false);
+    }
+    sortColumn = column;
+    if (sortColumn !is null && sortDirection !is DWT.NONE) {
+        OS.gtk_tree_view_column_set_sort_indicator (sortColumn.handle, true);
+        OS.gtk_tree_view_column_set_sort_order (sortColumn.handle, sortDirection is DWT.DOWN ? 0 : 1);
+    }
+}
+
+/**
+ * Sets the direction of the sort indicator for the receiver. The value
+ * can be one of <code>UP</code>, <code>DOWN</code> or <code>NONE</code>.
+ *
+ * @param direction the direction of the sort indicator
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.2
+ */
+public void setSortDirection  (int direction) {
+    checkWidget ();
+    if (direction !is DWT.UP && direction !is DWT.DOWN && direction !is DWT.NONE) return;
+    sortDirection = direction;
+    if (sortColumn is null || sortColumn.isDisposed ()) return;
+    if (sortDirection is DWT.NONE) {
+        OS.gtk_tree_view_column_set_sort_indicator (sortColumn.handle, false);
+    } else {
+        OS.gtk_tree_view_column_set_sort_indicator (sortColumn.handle, true);
+        OS.gtk_tree_view_column_set_sort_order (sortColumn.handle, sortDirection is DWT.DOWN ? 0 : 1);
+    }
+}
+
+/**
+ * Selects the item at the given zero-relative index in the receiver.
+ * The current selection is first cleared, then the new item is selected.
+ *
+ * @param index the index of the item to select
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#deselectAll()
+ * @see Table#selectcast(int)
+ */
+public void setSelection (int index) {
+    checkWidget ();
+    bool fixColumn = showFirstColumn ();
+    deselectAll ();
+    selectFocusIndex (index);
+    showSelection ();
+    if (fixColumn) hideFirstColumn ();
+}
+
+/**
+ * Selects the items in the range specified by the given zero-relative
+ * indices in the receiver. The range of indices is inclusive.
+ * The current selection is cleared before the new items are selected.
+ * <p>
+ * Indices that are out of range are ignored and no items will be selected
+ * if start is greater than end.
+ * If the receiver is single-select and there is more than one item in the
+ * given range, then all indices are ignored.
+ * </p>
+ *
+ * @param start the start index of the items to select
+ * @param end the end index of the items to select
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#deselectAll()
+ * @see Table#select(int,int)
+ */
+public void setSelection (int start, int end) {
+    checkWidget ();
+    deselectAll();
+    if (end < 0 || start > end || ((style & DWT.SINGLE) !is 0 && start !is end)) return;
+    if (itemCount is 0 || start >= itemCount) return;
+    bool fixColumn = showFirstColumn ();
+    start = Math.max (0, start);
+    end = Math.min (end, itemCount - 1);
+    selectFocusIndex (start);
+    if ((style & DWT.MULTI) !is 0) {
+        select (start, end);
+    }
+    showSelection ();
+    if (fixColumn) hideFirstColumn ();
+}
+
+/**
+ * Selects the items at the given zero-relative indices in the receiver.
+ * The current selection is cleared before the new items are selected.
+ * <p>
+ * Indices that are out of range and duplicate indices are ignored.
+ * If the receiver is single-select and multiple indices are specified,
+ * then all indices are ignored.
+ * </p>
+ *
+ * @param indices the indices of the items to select
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the array of indices is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#deselectAll()
+ * @see Table#select(int[])
+ */
+public void setSelection (int [] indices) {
+    checkWidget ();
+    if (indices is null) error (DWT.ERROR_NULL_ARGUMENT);
+    deselectAll ();
+    int length = indices.length;
+    if (length is 0 || ((style & DWT.SINGLE) !is 0 && length > 1)) return;
+    bool fixColumn = showFirstColumn ();
+    selectFocusIndex (indices [0]);
+    if ((style & DWT.MULTI) !is 0) {
+        select (indices);
+    }
+    showSelection ();
+    if (fixColumn) hideFirstColumn ();
+}
+
+/**
+ * Sets the receiver's selection to the given item.
+ * The current selection is cleared before the new item is selected.
+ * <p>
+ * If the item is not in the receiver, then it is ignored.
+ * </p>
+ *
+ * @param item the item to select
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the item is null</li>
+ *    <li>ERROR_INVALID_ARGUMENT - if the item has been disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.2
+ */
+public void setSelection (TableItem item) {
+    if (item is null) error (DWT.ERROR_NULL_ARGUMENT);
+    setSelection ( [item] );
+}
+
+
+/**
+ * Sets the receiver's selection to be the given array of items.
+ * The current selection is cleared before the new items are selected.
+ * <p>
+ * Items that are not in the receiver are ignored.
+ * If the receiver is single-select and multiple items are specified,
+ * then all items are ignored.
+ * </p>
+ *
+ * @param items the array of items
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the array of items is null</li>
+ *    <li>ERROR_INVALID_ARGUMENT - if one of the items has been disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#deselectAll()
+ * @see Table#select(int[])
+ * @see Table#setSelection(int[])
+ */
+public void setSelection (TableItem [] items) {
+    checkWidget ();
+    if (items is null) error (DWT.ERROR_NULL_ARGUMENT);
+    bool fixColumn = showFirstColumn ();
+    deselectAll ();
+    int length = items.length;
+    if (!(length is 0 || ((style & DWT.SINGLE) !is 0 && length > 1))) {
+        bool first = true;
+        for (int i = 0; i < length; i++) {
+            int index = indexOf (items [i]);
+            if (index !is -1) {
+                if (first) {
+                    first = false;
+                    selectFocusIndex (index);
+                } else {
+                    select (index);
+                }
+            }
+        }
+        showSelection ();
+    }
+    if (fixColumn) hideFirstColumn ();
+}
+
+/**
+ * Sets the zero-relative index of the item which is currently
+ * at the top of the receiver. This index can change when items
+ * are scrolled or new items are added and removed.
+ *
+ * @param index the index of the top item
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void setTopIndex (int index) {
+    checkWidget();
+    if (!(0 <= index && index < itemCount)) return;
+    auto path = OS.gtk_tree_model_get_path (modelHandle, _getItem (index).handle);
+    OS.gtk_tree_view_scroll_to_cell (handle, path, null, true, 0f, 0f);
+    if (OS.GTK_VERSION < OS.buildVERSION (2, 8, 0)) {
+        /*
+        * Bug in GTK.  According to the documentation, gtk_tree_view_scroll_to_cell
+        * should vertically scroll the cell to the top if use_align is true and row_align is 0.
+        * However, prior to version 2.8 it does not scroll at all.  The fix is to determine
+        * the new location and use gtk_tree_view_scroll_to_point.
+        * If the widget is a pinhead, calling gtk_tree_view_scroll_to_point
+        * will have no effect. Therefore, it is still neccessary to call
+        * gtk_tree_view_scroll_to_cell.
+        */
+        OS.gtk_widget_realize (handle);
+        GdkRectangle cellRect;
+        OS.gtk_tree_view_get_cell_area (handle, path, null, &cellRect);
+        int tx, ty;
+        OS.gtk_tree_view_widget_to_tree_coords(handle, cellRect.x, cellRect.y, &tx, &ty);
+        OS.gtk_tree_view_scroll_to_point (handle, -1, ty);
+    }
+    OS.gtk_tree_path_free (path);
+}
+
+/**
+ * Shows the column.  If the column is already showing in the receiver,
+ * this method simply returns.  Otherwise, the columns are scrolled until
+ * the column is visible.
+ *
+ * @param column the column to be shown
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the column is null</li>
+ *    <li>ERROR_INVALID_ARGUMENT - if the column has been disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.0
+ */
+public void showColumn (TableColumn column) {
+    checkWidget ();
+    if (column is null) error (DWT.ERROR_NULL_ARGUMENT);
+    if (column.isDisposed()) error(DWT.ERROR_INVALID_ARGUMENT);
+    if (column.parent !is this) return;
+    /*
+    * This code is intentionally commented. According to the
+    * documentation, gtk_tree_view_scroll_to_cell should scroll the
+    * minimum amount to show the column but instead it scrolls strangely.
+    */
+    //OS.gtk_tree_view_scroll_to_cell (handle, 0, column.handle, false, 0, 0);
+    OS.gtk_widget_realize (handle);
+    GdkRectangle cellRect;
+    OS.gtk_tree_view_get_cell_area (handle, null, column.handle, &cellRect);
+    GdkRectangle visibleRect;
+    OS.gtk_tree_view_get_visible_rect (handle, &visibleRect);
+    if (cellRect.x < visibleRect.x) {
+        OS.gtk_tree_view_scroll_to_point (handle, cellRect.x, -1);
+    } else {
+        int width = Math.min (visibleRect.width, cellRect.width);
+        if (cellRect.x + width > visibleRect.x + visibleRect.width) {
+            int tree_x = cellRect.x + width - visibleRect.width;
+            OS.gtk_tree_view_scroll_to_point (handle, tree_x, -1);
+        }
+    }
+}
+
+bool showFirstColumn () {
+    /*
+    * Bug in GTK.  If no columns are visible, changing the selection
+    * will fail.  The fix is to temporarily make a column visible.
+    */
+    int columnCount = Math.max (1, this.columnCount);
+    for (int i=0; i<columnCount; i++) {
+        auto column = OS.gtk_tree_view_get_column (handle, i);
+        if (OS.gtk_tree_view_column_get_visible (column)) return false;
+    }
+    auto firstColumn = OS.gtk_tree_view_get_column (handle, 0);
+    OS.gtk_tree_view_column_set_visible (firstColumn, true);
+    return true;
+}
+
+/**
+ * Shows the item.  If the item is already showing in the receiver,
+ * this method simply returns.  Otherwise, the items are scrolled until
+ * the item is visible.
+ *
+ * @param item the item to be shown
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the item is null</li>
+ *    <li>ERROR_INVALID_ARGUMENT - if the item has been disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#showSelection()
+ */
+public void showItem (TableItem item) {
+    checkWidget ();
+    if (item is null) error (DWT.ERROR_NULL_ARGUMENT);
+    if (item.isDisposed()) error (DWT.ERROR_INVALID_ARGUMENT);
+    if (item.parent !is this) return;
+    showItem (cast(GtkTreeIter*)item.handle);
+}
+
+void showItem (GtkTreeIter* iter) {
+    auto path = OS.gtk_tree_model_get_path (modelHandle, iter);
+    /*
+    * This code intentionally commented.
+    * Bug in GTK.  According to the documentation, gtk_tree_view_scroll_to_cell
+    * should scroll the minimum amount to show the cell if use_align is false.
+    * However, what actually happens is the cell is scrolled to the top.
+    * The fix is to determine the new location and use gtk_tree_view_scroll_to_point.
+    * If the widget is a pinhead, calling gtk_tree_view_scroll_to_point
+    * will have no effect. Therefore, it is still neccessary to
+    * call gtk_tree_view_scroll_to_cell.
+    */
+//  OS.gtk_tree_view_scroll_to_cell (handle, path, 0, false, 0, 0);
+    OS.gtk_widget_realize (handle);
+    GdkRectangle visibleRect;
+    OS.gtk_tree_view_get_visible_rect (handle, &visibleRect);
+    GdkRectangle cellRect;
+    OS.gtk_tree_view_get_cell_area (handle, path, null, &cellRect);
+    int tx, ty;
+    OS.gtk_tree_view_widget_to_tree_coords(handle, cellRect.x, cellRect.y, &tx, &ty);
+    if (ty < visibleRect.y ) {
+        OS.gtk_tree_view_scroll_to_cell (handle, path, null, true, 0f, 0f);
+        OS.gtk_tree_view_scroll_to_point (handle, -1, ty);
+    } else {
+        int height = Math.min (visibleRect.height, cellRect.height);
+        if (ty + height > visibleRect.y + visibleRect.height) {
+            OS.gtk_tree_view_scroll_to_cell (handle, path, null, true, 1f, 0f);
+            ty += cellRect.height - visibleRect.height;
+            OS.gtk_tree_view_scroll_to_point (handle, -1, ty);
+        }
+    }
+    OS.gtk_tree_path_free (path);
+}
+
+/**
+ * Shows the selection.  If the selection is already showing in the receiver,
+ * this method simply returns.  Otherwise, the items are scrolled until
+ * the selection is visible.
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#showItem(TableItem)
+ */
+public void showSelection () {
+    checkWidget();
+    TableItem [] selection = getSelection ();
+    if (selection.length is 0) return;
+    TableItem item = selection [0];
+    showItem (cast(GtkTreeIter*)item.handle);
+}
+
+override void treeSelectionProc (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, int[] selection, int length_) {
+    if (selection !is null) {
+        auto indices = OS.gtk_tree_path_get_indices (path);
+        if (indices !is null) {
+            selection [length_] = indices [0];
+        }
+    }
+}
+
+override void updateScrollBarValue (ScrollBar bar) {
+    super.updateScrollBarValue (bar);
+    /*
+    *  Bug in GTK. Scrolling changes the XWindow position
+    * and makes the child widgets appear to scroll even
+    * though when queried their position is unchanged.
+    * The fix is to queue a resize event for each child to
+    * force the position to be corrected.
+    */
+    auto parentHandle = parentingHandle ();
+    auto list = OS.gtk_container_get_children (parentHandle);
+    if (list is null) return;
+    auto temp = list;
+    while (temp !is null) {
+        auto widget = OS.g_list_data (temp);
+        if (widget !is null) OS.gtk_widget_queue_resize  (widget);
+        temp = OS.g_list_next (temp);
+    }
+    OS.g_list_free (list);
+}
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/dwt/widgets/TableColumn.d	Wed Jan 16 16:28:05 2008 +0100
@@ -0,0 +1,674 @@
+/*******************************************************************************
+ * Copyright (c) 2000, 2007 IBM Corporation and others.
+ * All rights reserved. This program and the accompanying materials
+ * are made available under the terms of the Eclipse Public License v1.0
+ * which accompanies this distribution, and is available at
+ * http://www.eclipse.org/legal/epl-v10.html
+ *
+ * Contributors:
+ *     IBM Corporation - initial API and implementation
+ *******************************************************************************/
+module dwt.widgets.TableColumn;
+
+
+
+import dwt.DWT;
+import dwt.DWTException;
+import dwt.events.ControlListener;
+import dwt.events.SelectionEvent;
+import dwt.events.SelectionListener;
+import dwt.graphics.Image;
+import dwt.internal.gtk.OS;
+import dwt.widgets.Item;
+import dwt.widgets.Table;
+import dwt.widgets.Shell;
+import dwt.widgets.TypedListener;
+import dwt.widgets.TableItem;
+import dwt.widgets.ImageList;
+
+import Math = tango.math.Math;
+
+/**
+ * Instances of this class represent a column in a table widget.
+ * <p><dl>
+ * <dt><b>Styles:</b></dt>
+ * <dd>LEFT, RIGHT, CENTER</dd>
+ * <dt><b>Events:</b></dt>
+ * <dd> Move, Resize, Selection</dd>
+ * </dl>
+ * </p><p>
+ * Note: Only one of the styles LEFT, RIGHT and CENTER may be specified.
+ * </p><p>
+ * IMPORTANT: This class is <em>not</em> intended to be subclassed.
+ * </p>
+ */
+public class TableColumn : Item {
+    GtkWidget* labelHandle, imageHandle, buttonHandle;
+    Table parent;
+    int modelIndex, lastButton, lastTime, lastX, lastWidth;
+    bool customDraw, useFixedWidth;
+    char[] toolTipText;
+
+/**
+ * Constructs a new instance of this class given its parent
+ * (which must be a <code>Table</code>) and a style value
+ * describing its behavior and appearance. The item is added
+ * to the end of the items maintained by its parent.
+ * <p>
+ * The style value is either one of the style constants defined in
+ * class <code>DWT</code> which is applicable to instances of this
+ * class, or must be built by <em>bitwise OR</em>'ing together
+ * (that is, using the <code>int</code> "|" operator) two or more
+ * of those <code>DWT</code> style constants. The class description
+ * lists the style constants that are applicable to the class.
+ * Style bits are also inherited from superclasses.
+ * </p>
+ *
+ * @param parent a composite control which will be the parent of the new instance (cannot be null)
+ * @param style the style of control to construct
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the parent is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the parent</li>
+ *    <li>ERROR_INVALID_SUBCLASS - if this class is not an allowed subclass</li>
+ * </ul>
+ *
+ * @see DWT#LEFT
+ * @see DWT#RIGHT
+ * @see DWT#CENTER
+ * @see Widget#checkSubclass
+ * @see Widget#getStyle
+ */
+public this (Table parent, int style) {
+    super (parent, checkStyle (style));
+    this.parent = parent;
+    createWidget (parent.getColumnCount ());
+}
+
+/**
+ * Constructs a new instance of this class given its parent
+ * (which must be a <code>Table</code>), a style value
+ * describing its behavior and appearance, and the index
+ * at which to place it in the items maintained by its parent.
+ * <p>
+ * The style value is either one of the style constants defined in
+ * class <code>DWT</code> which is applicable to instances of this
+ * class, or must be built by <em>bitwise OR</em>'ing together
+ * (that is, using the <code>int</code> "|" operator) two or more
+ * of those <code>DWT</code> style constants. The class description
+ * lists the style constants that are applicable to the class.
+ * Style bits are also inherited from superclasses.
+ * </p>
+ *
+ * @param parent a composite control which will be the parent of the new instance (cannot be null)
+ * @param style the style of control to construct
+ * @param index the zero-relative index to store the receiver in its parent
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the parent is null</li>
+ *    <li>ERROR_INVALID_RANGE - if the index is not between 0 and the number of elements in the parent (inclusive)</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the parent</li>
+ *    <li>ERROR_INVALID_SUBCLASS - if this class is not an allowed subclass</li>
+ * </ul>
+ *
+ * @see DWT#LEFT
+ * @see DWT#RIGHT
+ * @see DWT#CENTER
+ * @see Widget#checkSubclass
+ * @see Widget#getStyle
+ */
+public this (Table parent, int style, int index) {
+    super (parent, checkStyle (style));
+    this.parent = parent;
+    createWidget (index);
+}
+
+/**
+ * Adds the listener to the collection of listeners who will
+ * be notified when the control is moved or resized, by sending
+ * it one of the messages defined in the <code>ControlListener</code>
+ * interface.
+ *
+ * @param listener the listener which should be notified
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see ControlListener
+ * @see #removeControlListener
+ */
+public void addControlListener(ControlListener listener) {
+    checkWidget();
+    if (listener is null) error (DWT.ERROR_NULL_ARGUMENT);
+    TypedListener typedListener = new TypedListener (listener);
+    addListener (DWT.Resize,typedListener);
+    addListener (DWT.Move,typedListener);
+}
+
+/**
+ * Adds the listener to the collection of listeners who will
+ * be notified when the control is selected by the user, by sending
+ * it one of the messages defined in the <code>SelectionListener</code>
+ * interface.
+ * <p>
+ * <code>widgetSelected</code> is called when the column header is selected.
+ * <code>widgetDefaultSelected</code> is not called.
+ * </p>
+ *
+ * @param listener the listener which should be notified when the control is selected by the user
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see SelectionListener
+ * @see #removeSelectionListener
+ * @see SelectionEvent
+ */
+public void addSelectionListener (SelectionListener listener) {
+    checkWidget();
+    if (listener is null) error (DWT.ERROR_NULL_ARGUMENT);
+    TypedListener typedListener = new TypedListener (listener);
+    addListener (DWT.Selection,typedListener);
+    addListener (DWT.DefaultSelection,typedListener);
+}
+
+static int checkStyle (int style) {
+    return checkBits (style, DWT.LEFT, DWT.CENTER, DWT.RIGHT, 0, 0, 0);
+}
+
+protected void checkSubclass () {
+    if (!isValidSubclass ()) error (DWT.ERROR_INVALID_SUBCLASS);
+}
+
+void createWidget (int index) {
+    parent.createItem (this, index);
+    setOrientation ();
+    hookEvents ();
+    register ();
+    text = "";
+}
+
+void deregister() {
+    super.deregister ();
+    display.removeWidget (handle);
+    if (buttonHandle !is null) display.removeWidget (buttonHandle);
+    if (labelHandle !is null) display.removeWidget (labelHandle);
+}
+
+void destroyWidget () {
+    parent.destroyItem (this);
+    releaseHandle ();
+}
+
+/**
+ * Returns a value which describes the position of the
+ * text or image in the receiver. The value will be one of
+ * <code>LEFT</code>, <code>RIGHT</code> or <code>CENTER</code>.
+ *
+ * @return the alignment
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int getAlignment () {
+    checkWidget();
+    if ((style & DWT.LEFT) !is 0) return DWT.LEFT;
+    if ((style & DWT.CENTER) !is 0) return DWT.CENTER;
+    if ((style & DWT.RIGHT) !is 0) return DWT.RIGHT;
+    return DWT.LEFT;
+}
+
+/**
+ * Gets the moveable attribute. A column that is
+ * not moveable cannot be reordered by the user
+ * by dragging the header but may be reordered
+ * by the programmer.
+ *
+ * @return the moveable attribute
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#getColumnOrder()
+ * @see Table#setColumnOrder(int[])
+ * @see TableColumn#setMoveable(bool)
+ * @see DWT#Move
+ *
+ * @since 3.1
+ */
+public bool getMoveable() {
+    checkWidget();
+    return cast(bool)OS.gtk_tree_view_column_get_reorderable (handle);
+}
+
+/**
+ * Returns the receiver's parent, which must be a <code>Table</code>.
+ *
+ * @return the receiver's parent
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public Table getParent () {
+    checkWidget();
+    return parent;
+}
+
+/**
+ * Gets the resizable attribute. A column that is
+ * not resizable cannot be dragged by the user but
+ * may be resized by the programmer.
+ *
+ * @return the resizable attribute
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public bool getResizable () {
+    checkWidget();
+    return cast(bool)OS.gtk_tree_view_column_get_resizable (handle);
+}
+
+/**
+ * Returns the receiver's tool tip text, or null if it has
+ * not been set.
+ *
+ * @return the receiver's tool tip text
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.2
+ */
+public char[] getToolTipText () {
+    checkWidget();
+    return toolTipText;
+}
+
+/**
+ * Gets the width of the receiver.
+ *
+ * @return the width
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int getWidth () {
+    checkWidget();
+    if (!OS.gtk_tree_view_column_get_visible (handle)) {
+        return 0;
+    }
+    if (useFixedWidth) return OS.gtk_tree_view_column_get_fixed_width (handle);
+    return OS.gtk_tree_view_column_get_width (handle);
+}
+
+override int gtk_clicked (GtkWidget* widget) {
+    /*
+    * There is no API to get a double click on a table column.  Normally, when
+    * the mouse is double clicked, this is indicated by GDK_2BUTTON_PRESS
+    * but the table column sends the click signal on button release.  The fix is to
+    * test for double click by remembering the last click time and mouse button
+    * and testing for the double click interval.
+    */
+    bool doubleClick = false;
+    bool postEvent_ = true;
+    auto eventPtr = OS.gtk_get_current_event ();
+    if (eventPtr !is null) {
+        GdkEventButton* gdkEvent = cast(GdkEventButton*)eventPtr;
+        switch (gdkEvent.type) {
+            case OS.GDK_BUTTON_RELEASE: {
+                int clickTime = display.getDoubleClickTime ();
+                int eventTime = gdkEvent.time, eventButton = gdkEvent.button;
+                if (lastButton is eventButton && lastTime !is 0 && Math.abs (lastTime - eventTime) <= clickTime) {
+                    doubleClick = true;
+                }
+                lastTime = eventTime is 0 ? 1: eventTime;
+                lastButton = eventButton;
+                break;
+            }
+            case OS.GDK_MOTION_NOTIFY: {
+                /*
+                * Bug in GTK.  Dragging a column in a GtkTreeView causes a clicked
+                * signal to be emitted even though the mouse button was never released.
+                * The fix to ignore the signal if the current GDK event is a motion notify.
+                * The GTK bug was fixed in version 2.6
+                */
+                if (OS.GTK_VERSION < OS.buildVERSION (2, 6, 0)) postEvent_ = false;
+                break;
+            }
+            default:
+        }
+        OS.gdk_event_free (eventPtr);
+    }
+    if (postEvent_) postEvent (doubleClick ? DWT.DefaultSelection : DWT.Selection);
+    return 0;
+}
+
+override int gtk_mnemonic_activate (GtkWidget* widget, int /*long*/ arg1) {
+    return parent.gtk_mnemonic_activate (widget, arg1);
+}
+
+override int gtk_size_allocate (GtkWidget* widget, int /*long*/ allocation) {
+    useFixedWidth = false;
+    int x = OS.GTK_WIDGET_X (widget);
+    int width = OS.GTK_WIDGET_WIDTH (widget);
+    if (x !is lastX) {
+        lastX = x;
+        sendEvent (DWT.Move);
+    }
+    if (width !is lastWidth) {
+        lastWidth = width;
+        sendEvent (DWT.Resize);
+    }
+    return 0;
+}
+
+void hookEvents () {
+    super.hookEvents ();
+    OS.g_signal_connect_closure (handle, OS.clicked.ptr, display.closures [CLICKED], false);
+    if (buttonHandle !is null) OS.g_signal_connect_closure_by_id (buttonHandle, display.signalIds [SIZE_ALLOCATE], 0, display.closures [SIZE_ALLOCATE], false);
+    if (labelHandle !is null) OS.g_signal_connect_closure_by_id (labelHandle, display.signalIds [MNEMONIC_ACTIVATE], 0, display.closures [MNEMONIC_ACTIVATE], false);
+}
+
+/**
+ * Causes the receiver to be resized to its preferred size.
+ * For a composite, this involves computing the preferred size
+ * from its layout, if there is one.
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ */
+public void pack () {
+    checkWidget();
+    int width = 0;
+    if (buttonHandle !is null) {
+        GtkRequisition requisition;
+        OS.gtk_widget_size_request (buttonHandle, &requisition);
+        width = requisition.width;
+    }
+    if ((parent.style & DWT.VIRTUAL) !is 0) {
+        for (int i=0; i<parent.items.length; i++) {
+            TableItem item = parent.items [i];
+            if (item !is null && item.cached) {
+                width = Math.max (width, parent.calculateWidth ( cast(GtkTreeViewColumn*)handle, cast(GtkTreeIter*)item.handle));
+            }
+        }
+    } else {
+        GtkTreeIter iter;
+        if (OS.gtk_tree_model_get_iter_first (parent.modelHandle, &iter)) {
+            do {
+                width = Math.max (width, parent.calculateWidth (cast(GtkTreeViewColumn*)handle, &iter));
+            } while (OS.gtk_tree_model_iter_next(parent.modelHandle, &iter));
+        }
+    }
+    setWidth(width);
+}
+
+void register () {
+    super.register ();
+    display.addWidget (handle, this);
+    if (buttonHandle !is null) display.addWidget (buttonHandle, this);
+    if (labelHandle !is null) display.addWidget (labelHandle, this);
+}
+
+void releaseHandle () {
+    super.releaseHandle ();
+    handle = buttonHandle = labelHandle = imageHandle = null;
+    modelIndex = -1;
+    parent = null;
+}
+
+void releaseParent () {
+    super.releaseParent ();
+    if (parent.sortColumn is this) {
+        parent.sortColumn = null;
+    }
+}
+
+/**
+ * Removes the listener from the collection of listeners who will
+ * be notified when the control is moved or resized.
+ *
+ * @param listener the listener which should no longer be notified
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see ControlListener
+ * @see #addControlListener
+ */
+public void removeControlListener (ControlListener listener) {
+    checkWidget();
+    if (listener is null) error (DWT.ERROR_NULL_ARGUMENT);
+    if (eventTable is null) return;
+    eventTable.unhook (DWT.Move, listener);
+    eventTable.unhook (DWT.Resize, listener);
+}
+
+/**
+ * Removes the listener from the collection of listeners who will
+ * be notified when the control is selected by the user.
+ *
+ * @param listener the listener which should no longer be notified
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the listener is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see SelectionListener
+ * @see #addSelectionListener
+ */
+public void removeSelectionListener(SelectionListener listener) {
+    checkWidget();
+    if (listener is null) error (DWT.ERROR_NULL_ARGUMENT);
+    if (eventTable is null) return;
+    eventTable.unhook (DWT.Selection, listener);
+    eventTable.unhook (DWT.DefaultSelection,listener);
+}
+
+/**
+ * Controls how text and images will be displayed in the receiver.
+ * The argument should be one of <code>LEFT</code>, <code>RIGHT</code>
+ * or <code>CENTER</code>.
+ *
+ * @param alignment the new alignment
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void setAlignment (int alignment) {
+    checkWidget();
+    if ((alignment & (DWT.LEFT | DWT.RIGHT | DWT.CENTER)) is 0) return;
+    int index = parent.indexOf (this);
+    if (index is -1 || index is 0) return;
+    style &= ~(DWT.LEFT | DWT.RIGHT | DWT.CENTER);
+    style |= alignment & (DWT.LEFT | DWT.RIGHT | DWT.CENTER);
+    parent.createRenderers ( cast(GtkTreeViewColumn*)handle, modelIndex, index is 0, style);
+}
+
+void setFontDescription (PangoFontDescription* font) {
+    OS.gtk_widget_modify_font (labelHandle, font);
+    OS.gtk_widget_modify_font (imageHandle, font);
+}
+
+public void setImage (Image image) {
+    checkWidget ();
+    super.setImage (image);
+    if (image !is null) {
+        ImageList headerImageList = parent.headerImageList;
+        if (headerImageList is null) {
+            headerImageList = parent.headerImageList = new ImageList ();
+        }
+        int imageIndex = headerImageList.indexOf (image);
+        if (imageIndex is -1) imageIndex = headerImageList.add (image);
+        auto pixbuf = headerImageList.getPixbuf (imageIndex);
+        OS.gtk_image_set_from_pixbuf (imageHandle, pixbuf);
+        OS.gtk_widget_show (imageHandle);
+    } else {
+        OS.gtk_image_set_from_pixbuf (imageHandle, null);
+        OS.gtk_widget_hide (imageHandle);
+    }
+}
+
+/**
+ * Sets the resizable attribute.  A column that is
+ * resizable can be resized by the user dragging the
+ * edge of the header.  A column that is not resizable
+ * cannot be dragged by the user but may be resized
+ * by the programmer.
+ *
+ * @param resizable the resize attribute
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void setResizable (bool resizable) {
+    checkWidget();
+    OS.gtk_tree_view_column_set_resizable (handle, resizable);
+}
+
+/**
+ * Sets the moveable attribute.  A column that is
+ * moveable can be reordered by the user by dragging
+ * the header. A column that is not moveable cannot be
+ * dragged by the user but may be reordered
+ * by the programmer.
+ *
+ * @param moveable the moveable attribute
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @see Table#setColumnOrder(int[])
+ * @see Table#getColumnOrder()
+ * @see TableColumn#getMoveable()
+ * @see DWT#Move
+ *
+ * @since 3.1
+ */
+public void setMoveable (bool moveable) {
+    checkWidget();
+    OS.gtk_tree_view_column_set_reorderable (handle, moveable);
+}
+
+void setOrientation() {
+    if ((parent.style & DWT.RIGHT_TO_LEFT) !is 0) {
+        if (buttonHandle !is null) {
+            OS.gtk_widget_set_direction (buttonHandle, OS.GTK_TEXT_DIR_RTL);
+            display.doSetDirectionProc( buttonHandle, OS.GTK_TEXT_DIR_RTL);
+        }
+    }
+}
+
+public void setText (char[] string) {
+    checkWidget();
+    if (string is null) error (DWT.ERROR_NULL_ARGUMENT);
+    super.setText (string);
+    char [] chars = fixMnemonic (string);
+    char* buffer = tango.stdc.stringz.toStringz( chars );
+    OS.gtk_label_set_text_with_mnemonic (labelHandle, buffer);
+    if (string.length !is 0) {
+        OS.gtk_widget_show (labelHandle);
+    } else {
+        OS.gtk_widget_hide (labelHandle);
+    }
+}
+
+/**
+ * Sets the receiver's tool tip text to the argument, which
+ * may be null indicating that no tool tip text should be shown.
+ *
+ * @param string the new tool tip text (or null)
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.2
+ */
+public void setToolTipText (char[] string) {
+    checkWidget();
+    Shell shell = parent._getShell ();
+    setToolTipText (shell, string);
+    toolTipText = string;
+}
+
+void setToolTipText (Shell shell, char[] newString) {
+    shell.setToolTipText (buttonHandle, newString);
+}
+
+/**
+ * Sets the width of the receiver.
+ *
+ * @param width the new width
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void setWidth (int width) {
+    checkWidget();
+    if (width < 0) return;
+    if (width is lastWidth) return;
+    if (width > 0) {
+        useFixedWidth = true;
+        OS.gtk_tree_view_column_set_fixed_width (handle, width);
+    }
+    /*
+     * Bug in GTK.  For some reason, calling gtk_tree_view_column_set_visible()
+     * when the parent is not realized fails to show the column. The fix is to
+     * ensure that the table has been realized.
+     */
+    if (width !is 0) OS.gtk_widget_realize (parent.handle);
+    OS.gtk_tree_view_column_set_visible (handle, width !is 0);
+    lastWidth = width;
+    sendEvent (DWT.Resize);
+}
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/dwt/widgets/TableItem.d	Wed Jan 16 16:28:05 2008 +0100
@@ -0,0 +1,1278 @@
+/*******************************************************************************
+ * Copyright (c) 2000, 2007 IBM Corporation and others.
+ * All rights reserved. This program and the accompanying materials
+ * are made available under the terms of the Eclipse Public License v1.0
+ * which accompanies this distribution, and is available at
+ * http://www.eclipse.org/legal/epl-v10.html
+ *
+ * Contributors:
+ *     IBM Corporation - initial API and implementation
+ *******************************************************************************/
+module dwt.widgets.TableItem;
+
+
+
+import dwt.DWT;
+import dwt.DWTException;
+import dwt.graphics.Color;
+import dwt.graphics.Font;
+import dwt.graphics.Image;
+import dwt.graphics.Rectangle;
+import dwt.internal.gtk.OS;
+import dwt.widgets.Item;
+import dwt.widgets.Table;
+import dwt.widgets.ImageList;
+
+import Math = tango.math.Math;
+
+class TreeItem {
+    const int EXPANDER_EXTRA_PADDING = 4;
+}
+/**
+ * Instances of this class represent a selectable user interface object
+ * that represents an item in a table.
+ * <dl>
+ * <dt><b>Styles:</b></dt>
+ * <dd>(none)</dd>
+ * <dt><b>Events:</b></dt>
+ * <dd>(none)</dd>
+ * </dl>
+ * <p>
+ * IMPORTANT: This class is <em>not</em> intended to be subclassed.
+ * </p>
+ */
+public class TableItem : Item {
+    Table parent;
+    Font font;
+    Font[] cellFont;
+    bool cached, grayed;
+
+/**
+ * Constructs a new instance of this class given its parent
+ * (which must be a <code>Table</code>), a style value
+ * describing its behavior and appearance, and the index
+ * at which to place it in the items maintained by its parent.
+ * <p>
+ * The style value is either one of the style constants defined in
+ * class <code>DWT</code> which is applicable to instances of this
+ * class, or must be built by <em>bitwise OR</em>'ing together
+ * (that is, using the <code>int</code> "|" operator) two or more
+ * of those <code>DWT</code> style constants. The class description
+ * lists the style constants that are applicable to the class.
+ * Style bits are also inherited from superclasses.
+ * </p>
+ *
+ * @param parent a composite control which will be the parent of the new instance (cannot be null)
+ * @param style the style of control to construct
+ * @param index the zero-relative index to store the receiver in its parent
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the parent is null</li>
+ *    <li>ERROR_INVALID_RANGE - if the index is not between 0 and the number of elements in the parent (inclusive)</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the parent</li>
+ *    <li>ERROR_INVALID_SUBCLASS - if this class is not an allowed subclass</li>
+ * </ul>
+ *
+ * @see DWT
+ * @see Widget#checkSubclass
+ * @see Widget#getStyle
+ */
+public this (Table parent, int style, int index) {
+    this (parent, style, index, true);
+}
+
+/**
+ * Constructs a new instance of this class given its parent
+ * (which must be a <code>Table</code>) and a style value
+ * describing its behavior and appearance. The item is added
+ * to the end of the items maintained by its parent.
+ * <p>
+ * The style value is either one of the style constants defined in
+ * class <code>DWT</code> which is applicable to instances of this
+ * class, or must be built by <em>bitwise OR</em>'ing together
+ * (that is, using the <code>int</code> "|" operator) two or more
+ * of those <code>DWT</code> style constants. The class description
+ * lists the style constants that are applicable to the class.
+ * Style bits are also inherited from superclasses.
+ * </p>
+ *
+ * @param parent a composite control which will be the parent of the new instance (cannot be null)
+ * @param style the style of control to construct
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the parent is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the parent</li>
+ *    <li>ERROR_INVALID_SUBCLASS - if this class is not an allowed subclass</li>
+ * </ul>
+ *
+ * @see DWT
+ * @see Widget#checkSubclass
+ * @see Widget#getStyle
+ */
+public this (Table parent, int style) {
+    this (parent, style, checkNull (parent).getItemCount (), true);
+}
+
+
+this (Table parent, int style, int index, bool create) {
+    super (parent, style);
+    this.parent = parent;
+    if (create) {
+        parent.createItem (this, index);
+    } else {
+        GtkTreeIter handle;
+        OS.gtk_tree_model_iter_nth_child (parent.modelHandle, &handle, null, index);
+    }
+}
+
+static Table checkNull (Table control) {
+    if (control is null) DWT.error (DWT.ERROR_NULL_ARGUMENT);
+    return control;
+}
+
+protected void checkSubclass () {
+    if (!isValidSubclass ()) error (DWT.ERROR_INVALID_SUBCLASS);
+}
+
+void clear () {
+    if (parent.currentItem is this) return;
+    if (cached || (parent.style & DWT.VIRTUAL) is 0) {
+        int columnCount = OS.gtk_tree_model_get_n_columns (parent.modelHandle);
+        for (int i=0; i<columnCount; i++) {
+            OS.gtk_list_store_set1 (parent.modelHandle, handle, i, null);
+        }
+        /*
+        * Bug in GTK.  When using fixed-height-mode,
+        * row changes do not cause the row to be repainted.  The fix is to
+        * invalidate the row when it is cleared.
+        */
+        if ((parent.style & DWT.VIRTUAL) !is 0) {
+            if (OS.GTK_VERSION >= OS.buildVERSION (2, 3, 2) && OS.GTK_VERSION < OS.buildVERSION (2, 6, 3)) {
+                redraw ();
+            }
+        }
+    }
+    cached = false;
+    font = null;
+    cellFont = null;
+}
+
+void destroyWidget () {
+    parent.destroyItem (this);
+    releaseHandle ();
+}
+
+/**
+ * Returns the receiver's background color.
+ *
+ * @return the background color
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 2.0
+ */
+public Color getBackground () {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    void* ptr;
+    OS.gtk_tree_model_get1 (parent.modelHandle, handle, Table.BACKGROUND_COLUMN, &ptr);
+    if (ptr is null) return parent.getBackground ();
+    GdkColor* gdkColor = cast(GdkColor*)ptr;
+    return Color.gtk_new (display, gdkColor);
+}
+
+/**
+ * Returns a rectangle describing the receiver's size and location
+ * relative to its parent.
+ *
+ * @return the receiver's bounding rectangle
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.2
+ */
+public Rectangle getBounds () {
+    // TODO fully test on early and later versions of GTK
+    // shifted a bit too far right on later versions of GTK - however, old Tree also had this problem
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    auto parentHandle = parent.handle;
+    auto column = OS.gtk_tree_view_get_column (parentHandle, 0);
+    if (column is null) return new Rectangle (0, 0, 0, 0);
+    auto textRenderer = parent.getTextRenderer (column);
+    auto pixbufRenderer = parent.getPixbufRenderer (column);
+    if (textRenderer is null || pixbufRenderer is null)  return new Rectangle (0, 0, 0, 0);
+
+    auto path = OS.gtk_tree_model_get_path (parent.modelHandle, handle);
+    OS.gtk_widget_realize (parentHandle);
+
+    bool isExpander = OS.gtk_tree_model_iter_n_children (parent.modelHandle, handle) > 0;
+    bool isExpanded = cast(bool)OS.gtk_tree_view_row_expanded (parentHandle, path);
+    OS.gtk_tree_view_column_cell_set_cell_data (column, parent.modelHandle, handle, isExpander, isExpanded);
+
+    GdkRectangle rect;
+    OS.gtk_tree_view_get_cell_area (parentHandle, path, column, &rect);
+    OS.gtk_tree_path_free (path);
+    int right = rect.x + rect.width;
+
+    int x, w;
+    parent.ignoreSize = true;
+    OS.gtk_cell_renderer_get_size (textRenderer, parentHandle, null, null, null, &w, null);
+    parent.ignoreSize = false;
+    rect.width = w;
+    int buffer;
+    if (OS.gtk_tree_view_get_expander_column (parentHandle) is column) {
+        OS.gtk_widget_style_get1 (parentHandle, OS.expander_size.ptr, &buffer);
+        rect.x += buffer + TreeItem.EXPANDER_EXTRA_PADDING;
+    }
+    OS.gtk_widget_style_get1 (parentHandle, OS.horizontal_separator.ptr, &buffer);
+    int horizontalSeparator = buffer;
+    rect.x += horizontalSeparator;
+
+    if (OS.GTK_VERSION >= OS.buildVERSION (2, 1, 3)) {
+        OS.gtk_tree_view_column_cell_get_position (column, textRenderer, &x, null);
+        rect.x += x;
+    } else {
+        if ((parent.style & DWT.CHECK) !is 0) {
+            OS.gtk_cell_renderer_get_size (parent.checkRenderer, parentHandle, null, null, null, &w, null);
+            rect.x += w + horizontalSeparator;
+        }
+        OS.gtk_cell_renderer_get_size (pixbufRenderer, parentHandle, null, null, null, &w, null);
+        rect.x += w + horizontalSeparator;
+    }
+    if (parent.columnCount > 0) {
+        if (rect.x + rect.width > right) {
+            rect.width = Math.max (0, right - rect.x);
+        }
+    }
+    return new Rectangle (rect.x, rect.y, rect.width + 1, rect.height + 1);
+}
+
+/**
+ * Returns the background color at the given column index in the receiver.
+ *
+ * @param index the column index
+ * @return the background color
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.0
+ */
+public Color getBackground (int index) {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    int count = Math.max (1, parent.columnCount);
+    if (0 > index || index > count - 1) return getBackground ();
+    void* ptr;
+    int modelIndex = parent.columnCount is 0 ? Table.FIRST_COLUMN : parent.columns [index].modelIndex;
+    OS.gtk_tree_model_get1 (parent.modelHandle, handle, modelIndex + Table.CELL_BACKGROUND, &ptr);
+    if (ptr is null) return getBackground ();
+    GdkColor* gdkColor = cast(GdkColor*)ptr;
+    return Color.gtk_new (display, gdkColor);
+}
+
+/**
+ * Returns a rectangle describing the receiver's size and location
+ * relative to its parent at a column in the table.
+ *
+ * @param index the index that specifies the column
+ * @return the receiver's bounding column rectangle
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public Rectangle getBounds (int index) {
+    checkWidget();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    auto parentHandle = parent.handle;
+    GtkTreeViewColumn* column;
+    if (index >= 0 && index < parent.columnCount) {
+        column = cast(GtkTreeViewColumn*)parent.columns [index].handle;
+    } else {
+        column = OS.gtk_tree_view_get_column (parentHandle, index);
+    }
+    if (column is null) return new Rectangle (0, 0, 0, 0);
+    auto path = OS.gtk_tree_model_get_path (parent.modelHandle, handle);
+    OS.gtk_widget_realize (parentHandle);
+    GdkRectangle rect;
+    OS.gtk_tree_view_get_cell_area (parentHandle, path, column, &rect);
+    OS.gtk_tree_path_free (path);
+
+    if (index is 0 && (parent.style & DWT.CHECK) !is 0) {
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 1, 3)) {
+            int x, w;
+            OS.gtk_tree_view_column_cell_get_position (column, parent.checkRenderer, &x, &w);
+            rect.x += x + w;
+            rect.width -= x + w;
+        } else {
+            int w;
+            OS.gtk_cell_renderer_get_size (parent.checkRenderer, parentHandle, null, null, null, &w, null);
+            int buffer;
+            OS.gtk_widget_style_get1 (parentHandle, OS.horizontal_separator.ptr, &buffer);
+            rect.x += w + buffer;
+            rect.width -= w + buffer;
+        }
+    }
+    return new Rectangle (rect.x, rect.y, rect.width + 1, rect.height + 1);
+}
+
+/**
+ * Returns <code>true</code> if the receiver is checked,
+ * and false otherwise.  When the parent does not have
+ * the <code>CHECK</code> style, return false.
+ *
+ * @return the checked state of the checkbox
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public bool getChecked () {
+    checkWidget();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    if ((parent.style & DWT.CHECK) is 0) return false;
+    void* ptr;
+    OS.gtk_tree_model_get1 (parent.modelHandle, handle, Table.CHECKED_COLUMN, &ptr);
+    return ptr !is null;
+}
+
+/**
+ * Returns the font that the receiver will use to paint textual information for this item.
+ *
+ * @return the receiver's font
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.0
+ */
+public Font getFont () {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    return font !is null ? font : parent.getFont ();
+}
+
+/**
+ * Returns the font that the receiver will use to paint textual information
+ * for the specified cell in this item.
+ *
+ * @param index the column index
+ * @return the receiver's font
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.0
+ */
+public Font getFont (int index) {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    int count = Math.max (1, parent.columnCount);
+    if (0 > index || index > count - 1) return getFont ();
+    if (cellFont is null || cellFont [index] is null) return getFont ();
+    return cellFont [index];
+}
+
+/**
+ * Returns the foreground color that the receiver will use to draw.
+ *
+ * @return the receiver's foreground color
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 2.0
+ */
+public Color getForeground () {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    void* ptr;
+    OS.gtk_tree_model_get1 (parent.modelHandle, handle, Table.FOREGROUND_COLUMN, &ptr);
+    if (ptr is null) return parent.getForeground ();
+    GdkColor* gdkColor = cast(GdkColor*)ptr;
+    return Color.gtk_new (display, gdkColor);
+}
+
+/**
+ *
+ * Returns the foreground color at the given column index in the receiver.
+ *
+ * @param index the column index
+ * @return the foreground color
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.0
+ */
+public Color getForeground (int index) {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    int count = Math.max (1, parent.columnCount);
+    if (0 > index || index > count - 1) return getForeground ();
+    void* ptr;
+    int modelIndex =  parent.columnCount is 0 ? Table.FIRST_COLUMN : parent.columns [index].modelIndex;
+    OS.gtk_tree_model_get1 (parent.modelHandle, handle, modelIndex + Table.CELL_FOREGROUND, &ptr);
+    if (ptr is null) return getForeground ();
+    GdkColor* gdkColor = cast(GdkColor*)ptr;
+    return Color.gtk_new (display, gdkColor);
+}
+
+/**
+ * Returns <code>true</code> if the receiver is grayed,
+ * and false otherwise. When the parent does not have
+ * the <code>CHECK</code> style, return false.
+ *
+ * @return the grayed state of the checkbox
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public bool getGrayed () {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    if ((parent.style & DWT.CHECK) is 0) return false;
+    return grayed;
+}
+
+public Image getImage () {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    return getImage (0);
+}
+
+/**
+ * Returns the image stored at the given column index in the receiver,
+ * or null if the image has not been set or if the column does not exist.
+ *
+ * @param index the column index
+ * @return the image stored at the given column index in the receiver
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public Image getImage (int index) {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    int count = Math.max (1, parent.getColumnCount ());
+    if (0 > index || index > count - 1) return null;
+    void* ptr;
+    int modelIndex = parent.columnCount is 0 ? Table.FIRST_COLUMN : parent.columns [index].modelIndex;
+    OS.gtk_tree_model_get1 (parent.modelHandle, handle, modelIndex + Table.CELL_PIXBUF, &ptr);
+    if (ptr is null) return null;
+    ImageList imageList = parent.imageList;
+    int imageIndex = imageList.indexOf (ptr);
+    if (imageIndex is -1) return null;
+    return imageList.get (imageIndex);
+}
+
+/**
+ * Returns a rectangle describing the size and location
+ * relative to its parent of an image at a column in the
+ * table.  An empty rectangle is returned if index exceeds
+ * the index of the table's last column.
+ *
+ * @param index the index that specifies the column
+ * @return the receiver's bounding image rectangle
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public Rectangle getImageBounds (int index) {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    auto parentHandle = parent.handle;
+    GtkTreeViewColumn* column;
+    if (index >= 0 && index < parent.columnCount) {
+        column = cast(GtkTreeViewColumn*)parent.columns [index].handle;
+    } else {
+        column = OS.gtk_tree_view_get_column (parentHandle, index);
+    }
+    if (column is null) return new Rectangle (0, 0, 0, 0);
+    auto pixbufRenderer = parent.getPixbufRenderer (column);
+    if (pixbufRenderer is null)  return new Rectangle (0, 0, 0, 0);
+    GdkRectangle rect;
+    auto path = OS.gtk_tree_model_get_path (parent.modelHandle, handle);
+    OS.gtk_widget_realize (parentHandle);
+    OS.gtk_tree_view_get_cell_area (parentHandle, path, column, &rect);
+    OS.gtk_tree_path_free (path);
+
+    /*
+    * The OS call gtk_cell_renderer_get_size() provides the width of image to be drawn
+    * by the cell renderer.  If there is no image in the cell, the width is zero.  If the table contains
+    * images of varying widths, gtk_cell_renderer_get_size() will return the width of the image,
+    * not the width of the area in which the image is drawn.
+    * New API was added in GTK 2.1.3 for determining the full width of the renderer area.
+    * For earlier versions of GTK, the result is only correct if all rows have images of the same
+    * width.
+    */
+    if (OS.GTK_VERSION >= OS.buildVERSION (2, 1, 3)) {
+        int x, w;
+        OS.gtk_tree_view_column_cell_get_position (column, pixbufRenderer, &x, &w);
+        rect.x += x;
+        rect.width = w;
+    } else {
+        int w;
+        OS.gtk_tree_view_column_cell_set_cell_data (column, parent.modelHandle, handle, false, false);
+        OS.gtk_cell_renderer_get_size (pixbufRenderer, parentHandle, null, null, null, &w, null);
+        rect.width = w;
+    }
+    return new Rectangle (rect.x, rect.y, rect.width, rect.height + 1);
+}
+
+/**
+ * Gets the image indent.
+ *
+ * @return the indent
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public int getImageIndent () {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    /* Image indent is not supported on GTK */
+    return 0;
+}
+
+char[] getNameText () {
+    if ((parent.style & DWT.VIRTUAL) !is 0) {
+        if (!cached) return "*virtual*"; //$NON-NLS-1$
+    }
+    return super.getNameText ();
+}
+
+/**
+ * Returns the receiver's parent, which must be a <code>Table</code>.
+ *
+ * @return the receiver's parent
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public Table getParent () {
+    checkWidget ();
+    return parent;
+}
+
+public char[] getText () {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    return getText (0);
+}
+
+/**
+ * Returns the text stored at the given column index in the receiver,
+ * or empty string if the text has not been set.
+ *
+ * @param index the column index
+ * @return the text stored at the given column index in the receiver
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public char[] getText (int index) {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    int count = Math.max (1, parent.getColumnCount ());
+    if (0 > index || index > count - 1) return "";
+    char* ptr;
+    int modelIndex = parent.columnCount is 0 ? Table.FIRST_COLUMN : parent.columns [index].modelIndex;
+    OS.gtk_tree_model_get1 (parent.modelHandle, handle, modelIndex + Table.CELL_TEXT, cast(void**) &ptr);
+    if (ptr is null) return "";
+    char[] res = tango.stdc.stringz.fromUtf8z( ptr ).dup;
+    OS.g_free (ptr);
+    return res;
+}
+
+/**
+ * Returns a rectangle describing the size and location
+ * relative to its parent of the text at a column in the
+ * table.  An empty rectangle is returned if index exceeds
+ * the index of the table's last column.
+ *
+ * @param index the index that specifies the column
+ * @return the receiver's bounding text rectangle
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.3
+ */
+public Rectangle getTextBounds (int index) {
+    checkWidget ();
+    if (!parent.checkData (this)) error (DWT.ERROR_WIDGET_DISPOSED);
+    int count = Math.max (1, parent.getColumnCount ());
+    if (0 > index || index > count - 1) return new Rectangle (0, 0, 0, 0);
+    // TODO fully test on early and later versions of GTK
+    // shifted a bit too far right on later versions of GTK - however, old Tree also had this problem
+    auto parentHandle = parent.handle;
+    GtkTreeViewColumn* column;
+    if (index >= 0 && index < parent.columnCount) {
+        column = cast(GtkTreeViewColumn*)parent.columns [index].handle;
+    } else {
+        column = OS.gtk_tree_view_get_column (parentHandle, index);
+    }
+    if (column is null) return new Rectangle (0, 0, 0, 0);
+    auto textRenderer = parent.getTextRenderer (column);
+    auto pixbufRenderer = parent.getPixbufRenderer (column);
+    if (textRenderer is null || pixbufRenderer is null)  return new Rectangle (0, 0, 0, 0);
+
+    auto path = OS.gtk_tree_model_get_path (parent.modelHandle, handle);
+    OS.gtk_widget_realize (parentHandle);
+
+    bool isExpander = OS.gtk_tree_model_iter_n_children (parent.modelHandle, handle) > 0;
+    bool isExpanded = cast(bool)OS.gtk_tree_view_row_expanded (parentHandle, path);
+    OS.gtk_tree_view_column_cell_set_cell_data (column, parent.modelHandle, handle, isExpander, isExpanded);
+
+    GdkRectangle rect;
+    OS.gtk_tree_view_get_cell_area (parentHandle, path, column, &rect);
+    OS.gtk_tree_path_free (path);
+    int right = rect.x + rect.width;
+
+    int x, w;
+    parent.ignoreSize = true;
+    OS.gtk_cell_renderer_get_size (textRenderer, parentHandle, null, null, null, &w, null);
+    parent.ignoreSize = false;
+    int buffer;
+    if (OS.gtk_tree_view_get_expander_column (parentHandle) is column) {
+        OS.gtk_widget_style_get1 (parentHandle, OS.expander_size.ptr, &buffer);
+        rect.x += buffer + TreeItem.EXPANDER_EXTRA_PADDING;
+    }
+    OS.gtk_widget_style_get1 (parentHandle, OS.horizontal_separator.ptr, &buffer);
+    int horizontalSeparator = buffer;
+    rect.x += horizontalSeparator;
+
+    if (OS.GTK_VERSION >= OS.buildVERSION (2, 1, 3)) {
+        OS.gtk_tree_view_column_cell_get_position (column, textRenderer, &x, null);
+        rect.x += x;
+    } else {
+        if ((parent.style & DWT.CHECK) !is 0) {
+            OS.gtk_cell_renderer_get_size (parent.checkRenderer, parentHandle, null, null, null, &w, null);
+            rect.x += w + horizontalSeparator;
+        }
+        OS.gtk_cell_renderer_get_size (pixbufRenderer, parentHandle, null, null, null, &w, null);
+        rect.x += w + horizontalSeparator;
+    }
+    if (parent.columnCount > 0) {
+        if (rect.x + rect.width > right) {
+            rect.width = Math.max (0, right - rect.x);
+        }
+    }
+    return new Rectangle (rect.x, rect.y, rect.width + 1, rect.height + 1);
+}
+
+void redraw () {
+    if ((OS.GTK_WIDGET_FLAGS (parent.handle) & OS.GTK_REALIZED) !is 0) {
+        auto parentHandle = parent.handle;
+        auto path = OS.gtk_tree_model_get_path (parent.modelHandle, handle);
+        GdkRectangle rect;
+        OS.gtk_tree_view_get_cell_area (parentHandle, path, null, &rect);
+        OS.gtk_tree_path_free (path);
+        auto window = OS.gtk_tree_view_get_bin_window (parentHandle);
+        rect.x = 0;
+        int w, h;
+        OS.gdk_drawable_get_size (window, &w, &h);
+        rect.width = w;
+        OS.gdk_window_invalidate_rect (window, &rect, false);
+    }
+}
+
+void releaseHandle () {
+    if (handle !is null) OS.g_free (handle);
+    handle = null;
+    super.releaseHandle ();
+    parent = null;
+}
+
+void releaseWidget () {
+    super.releaseWidget ();
+    font = null;
+    cellFont = null;
+}
+
+/**
+ * Sets the receiver's background color to the color specified
+ * by the argument, or to the default system color for the item
+ * if the argument is null.
+ *
+ * @param color the new color (or null)
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_ARGUMENT - if the argument has been disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 2.0
+ */
+public void setBackground (Color color) {
+    checkWidget ();
+    if (color !is null && color.isDisposed ()) {
+        DWT.error (DWT.ERROR_INVALID_ARGUMENT);
+    }
+    GdkColor* gdkColor = color !is null ? color.handle : null;
+    OS.gtk_list_store_set1 (parent.modelHandle, handle, Table.BACKGROUND_COLUMN, gdkColor);
+    /*
+    * Bug in GTK.  When using fixed-height-mode,
+    * row changes do not cause the row to be repainted.  The fix is to
+    * invalidate the row when it is cleared.
+    */
+    if ((parent.style & DWT.VIRTUAL) !is 0) {
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 3, 2) && OS.GTK_VERSION < OS.buildVERSION (2, 6, 3)) {
+            redraw ();
+        }
+    }
+    cached = true;
+}
+
+/**
+ * Sets the background color at the given column index in the receiver
+ * to the color specified by the argument, or to the default system color for the item
+ * if the argument is null.
+ *
+ * @param index the column index
+ * @param color the new color (or null)
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_ARGUMENT - if the argument has been disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.0
+ */
+public void setBackground (int index, Color color) {
+    checkWidget ();
+    if (color !is null && color.isDisposed ()) {
+        DWT.error (DWT.ERROR_INVALID_ARGUMENT);
+    }
+    int count = Math.max (1, parent.getColumnCount ());
+    if (0 > index || index > count - 1) return;
+    int modelIndex = parent.columnCount is 0 ? Table.FIRST_COLUMN : parent.columns [index].modelIndex;
+    GdkColor* gdkColor = color !is null ? color.handle : null;
+    OS.gtk_list_store_set1 (parent.modelHandle, handle, modelIndex + Table.CELL_BACKGROUND, gdkColor);
+    /*
+    * Bug in GTK.  When using fixed-height-mode,
+    * row changes do not cause the row to be repainted.  The fix is to
+    * invalidate the row when it is cleared.
+    */
+    if ((parent.style & DWT.VIRTUAL) !is 0) {
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 3, 2) && OS.GTK_VERSION < OS.buildVERSION (2, 6, 3)) {
+            redraw ();
+        }
+    }
+    cached = true;
+
+    if (color !is null) {
+        bool customDraw = (parent.columnCount is 0)  ? parent.firstCustomDraw : parent.columns [index].customDraw;
+        if (!customDraw) {
+            if ((parent.style & DWT.VIRTUAL) is 0) {
+                auto parentHandle = parent.handle;
+                GtkTreeViewColumn* column;
+                if (parent.columnCount > 0) {
+                    column = cast(GtkTreeViewColumn*)parent.columns [index].handle;
+                } else {
+                    column = OS.gtk_tree_view_get_column (parentHandle, index);
+                }
+                if (column is null) return;
+                auto textRenderer = parent.getTextRenderer (column);
+                auto imageRenderer = parent.getPixbufRenderer (column);
+                display.doCellDataProc( parentHandle, column, cast(GtkCellRenderer*)textRenderer );
+                display.doCellDataProc( parentHandle, column, cast(GtkCellRenderer*)imageRenderer );
+            }
+            if (parent.columnCount is 0) {
+                parent.firstCustomDraw = true;
+            } else {
+                parent.columns [index].customDraw = true;
+            }
+        }
+    }
+}
+
+/**
+ * Sets the checked state of the checkbox for this item.  This state change
+ * only applies if the Table was created with the DWT.CHECK style.
+ *
+ * @param checked the new checked state of the checkbox
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void setChecked (bool checked) {
+    checkWidget();
+    if ((parent.style & DWT.CHECK) is 0) return;
+    OS.gtk_list_store_set1 (parent.modelHandle, handle, Table.CHECKED_COLUMN, cast(void*)cast(int)checked);
+    /*
+    * GTK+'s "inconsistent" state does not match DWT's concept of grayed.  To
+    * show checked+grayed differently from unchecked+grayed, we must toggle the
+    * grayed state on check and uncheck.
+    */
+    OS.gtk_list_store_set1 (parent.modelHandle, handle, Table.GRAYED_COLUMN, cast(void*)cast(int)( !checked ? false : grayed));
+    cached = true;
+}
+
+/**
+ * Sets the font that the receiver will use to paint textual information
+ * for this item to the font specified by the argument, or to the default font
+ * for that kind of control if the argument is null.
+ *
+ * @param font the new font (or null)
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_ARGUMENT - if the argument has been disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.0
+ */
+public void setFont (Font font){
+    checkWidget ();
+    if (font !is null && font.isDisposed ()) {
+        DWT.error (DWT.ERROR_INVALID_ARGUMENT);
+    }
+    if (this.font is font) return;
+    if (this.font !is null && this.font ==/*eq*/font ) return;
+    this.font = font;
+    auto fontHandle = font !is null ? font.handle : null;
+    OS.gtk_list_store_set1 (parent.modelHandle, handle, Table.FONT_COLUMN, fontHandle);
+    /*
+    * Bug in GTK.  When using fixed-height-mode,
+    * row changes do not cause the row to be repainted.  The fix is to
+    * invalidate the row when it is cleared.
+    */
+    if ((parent.style & DWT.VIRTUAL) !is 0) {
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 3, 2) && OS.GTK_VERSION < OS.buildVERSION (2, 6, 3)) {
+            redraw ();
+        }
+    }
+    cached = true;
+}
+
+/**
+ * Sets the font that the receiver will use to paint textual information
+ * for the specified cell in this item to the font specified by the
+ * argument, or to the default font for that kind of control if the
+ * argument is null.
+ *
+ * @param index the column index
+ * @param font the new font (or null)
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_ARGUMENT - if the argument has been disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.0
+ */
+public void setFont (int index, Font font) {
+    checkWidget ();
+    if (font !is null && font.isDisposed ()) {
+        DWT.error (DWT.ERROR_INVALID_ARGUMENT);
+    }
+    int count = Math.max (1, parent.getColumnCount ());
+    if (0 > index || index > count - 1) return;
+    if (cellFont is null) {
+        cellFont = new Font [count];
+    }
+    if (cellFont [index] is font) return;
+    if (cellFont [index] !is null && cellFont [index] ==/*eq*/ font) return;
+    cellFont [index] = font;
+
+    int modelIndex = parent.columnCount is 0 ? Table.FIRST_COLUMN : parent.columns [index].modelIndex;
+    auto fontHandle  = font !is null ? font.handle : null;
+    OS.gtk_list_store_set1 (parent.modelHandle, handle, modelIndex + Table.CELL_FONT, fontHandle);
+    /*
+    * Bug in GTK.  When using fixed-height-mode,
+    * row changes do not cause the row to be repainted.  The fix is to
+    * invalidate the row when it is cleared.
+    */
+    if ((parent.style & DWT.VIRTUAL) !is 0) {
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 3, 2) && OS.GTK_VERSION < OS.buildVERSION (2, 6, 3)) {
+            redraw ();
+        }
+    }
+    cached = true;
+
+    if (font !is null) {
+        bool customDraw = (parent.columnCount is 0)  ? parent.firstCustomDraw : parent.columns [index].customDraw;
+        if (!customDraw) {
+            if ((parent.style & DWT.VIRTUAL) is 0) {
+                auto parentHandle = parent.handle;
+                GtkTreeViewColumn* column;
+                if (parent.columnCount > 0) {
+                    column = cast(GtkTreeViewColumn*)parent.columns [index].handle;
+                } else {
+                    column = OS.gtk_tree_view_get_column (parentHandle, index);
+                }
+                if (column is null) return;
+                auto textRenderer = parent.getTextRenderer (column);
+                auto imageRenderer = parent.getPixbufRenderer (column);
+                display.doCellDataProc( parentHandle, column, cast(GtkCellRenderer*)textRenderer );
+                display.doCellDataProc( parentHandle, column, cast(GtkCellRenderer*)imageRenderer );
+            }
+            if (parent.columnCount is 0) {
+                parent.firstCustomDraw = true;
+            } else {
+                parent.columns [index].customDraw = true;
+            }
+        }
+    }
+}
+
+/**
+ * Sets the receiver's foreground color to the color specified
+ * by the argument, or to the default system color for the item
+ * if the argument is null.
+ *
+ * @param color the new color (or null)
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_ARGUMENT - if the argument has been disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 2.0
+ */
+public void setForeground (Color color){
+    checkWidget ();
+    if (color !is null && color.isDisposed ()) {
+        DWT.error (DWT.ERROR_INVALID_ARGUMENT);
+    }
+    GdkColor* gdkColor = color !is null ? color.handle : null;
+    OS.gtk_list_store_set1 (parent.modelHandle, handle, Table.FOREGROUND_COLUMN, gdkColor);
+    /*
+    * Bug in GTK.  When using fixed-height-mode,
+    * row changes do not cause the row to be repainted.  The fix is to
+    * invalidate the row when it is cleared.
+    */
+    if ((parent.style & DWT.VIRTUAL) !is 0) {
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 3, 2) && OS.GTK_VERSION < OS.buildVERSION (2, 6, 3)) {
+            redraw ();
+        }
+    }
+    cached = true;
+}
+
+/**
+ * Sets the foreground color at the given column index in the receiver
+ * to the color specified by the argument, or to the default system color for the item
+ * if the argument is null.
+ *
+ * @param index the column index
+ * @param color the new color (or null)
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_ARGUMENT - if the argument has been disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @since 3.0
+ */
+public void setForeground (int index, Color color){
+    checkWidget ();
+    if (color !is null && color.isDisposed ()) {
+        DWT.error (DWT.ERROR_INVALID_ARGUMENT);
+    }
+    int count = Math.max (1, parent.getColumnCount ());
+    if (0 > index || index > count - 1) return;
+    int modelIndex = parent.columnCount is 0 ? Table.FIRST_COLUMN : parent.columns [index].modelIndex;
+    GdkColor* gdkColor = color !is null ? color.handle : null;
+    OS.gtk_list_store_set1 (parent.modelHandle, handle, modelIndex + Table.CELL_FOREGROUND, gdkColor);
+    /*
+    * Bug in GTK.  When using fixed-height-mode,
+    * row changes do not cause the row to be repainted.  The fix is to
+    * invalidate the row when it is cleared.
+    */
+    if ((parent.style & DWT.VIRTUAL) !is 0) {
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 3, 2) && OS.GTK_VERSION < OS.buildVERSION (2, 6, 3)) {
+            redraw ();
+        }
+    }
+    cached = true;
+
+    if (color !is null) {
+        bool customDraw = (parent.columnCount is 0)  ? parent.firstCustomDraw : parent.columns [index].customDraw;
+        if (!customDraw) {
+            if ((parent.style & DWT.VIRTUAL) is 0) {
+                auto parentHandle = parent.handle;
+                GtkTreeViewColumn* column;
+                if (parent.columnCount > 0) {
+                    column = cast(GtkTreeViewColumn*)parent.columns [index].handle;
+                } else {
+                    column = OS.gtk_tree_view_get_column (parentHandle, index);
+                }
+                if (column is null) return;
+                auto textRenderer = parent.getTextRenderer (column);
+                auto imageRenderer = parent.getPixbufRenderer (column);
+                display.doCellDataProc( parentHandle, column, cast(GtkCellRenderer*)textRenderer );
+                display.doCellDataProc( parentHandle, column, cast(GtkCellRenderer*)imageRenderer );
+            }
+            if (parent.columnCount is 0) {
+                parent.firstCustomDraw = true;
+            } else {
+                parent.columns [index].customDraw = true;
+            }
+        }
+    }
+}
+
+/**
+ * Sets the grayed state of the checkbox for this item.  This state change
+ * only applies if the Table was created with the DWT.CHECK style.
+ *
+ * @param grayed the new grayed state of the checkbox;
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void setGrayed (bool grayed) {
+    checkWidget();
+    if ((parent.style & DWT.CHECK) is 0) return;
+    this.grayed = grayed;
+    /*
+    * GTK+'s "inconsistent" state does not match DWT's concept of grayed.
+    * Render checked+grayed as "inconsistent", unchecked+grayed as blank.
+    */
+    void* ptr;
+    OS.gtk_tree_model_get1 (parent.modelHandle, handle, Table.CHECKED_COLUMN, &ptr);
+    OS.gtk_list_store_set1 (parent.modelHandle, handle, Table.GRAYED_COLUMN, cast(void*)cast(int)( ptr is null ? false : grayed));
+    cached = true;
+}
+
+/**
+ * Sets the receiver's image at a column.
+ *
+ * @param index the column index
+ * @param image the new image
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_ARGUMENT - if the image has been disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void setImage (int index, Image image) {
+    checkWidget ();
+    if (image !is null && image.isDisposed ()) {
+        error(DWT.ERROR_INVALID_ARGUMENT);
+    }
+    int count = Math.max (1, parent.getColumnCount ());
+    if (0 > index || index > count - 1) return;
+    void* pixbuf;
+    if (image !is null) {
+        ImageList imageList = parent.imageList;
+        if (imageList is null) imageList = parent.imageList = new ImageList ();
+        int imageIndex = imageList.indexOf (image);
+        if (imageIndex is -1) imageIndex = imageList.add (image);
+        pixbuf = imageList.getPixbuf (imageIndex);
+    }
+    int modelIndex = parent.columnCount is 0 ? Table.FIRST_COLUMN : parent.columns [index].modelIndex;
+    OS.gtk_list_store_set1 (parent.modelHandle, handle, modelIndex + Table.CELL_PIXBUF, pixbuf);
+    /*
+    * Bug in GTK.  When using fixed-height-mode,
+    * row changes do not cause the row to be repainted.  The fix is to
+    * invalidate the row when it is cleared.
+    */
+    if ((parent.style & DWT.VIRTUAL) !is 0) {
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 3, 2) && OS.GTK_VERSION < OS.buildVERSION (2, 6, 3)) {
+            redraw ();
+        }
+    }
+    /*
+     * Bug in GTK.  When in fixed height mode, GTK does not recalculate the cell renderer width
+     * when the image is changed in the model.  The fix is to force it to recalculate the width if
+     * more space is required.
+     */
+    if ((parent.style & DWT.VIRTUAL) !is 0 && parent.currentItem is null) {
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 3, 2)) {
+            if (image !is null) {
+                auto parentHandle = parent.handle;
+                auto column = OS.gtk_tree_view_get_column (parentHandle, index);
+                int w;
+                auto pixbufRenderer = parent.getPixbufRenderer(column);
+                OS.gtk_tree_view_column_cell_get_position (column, pixbufRenderer, null, &w);
+                if (w < image.getBounds().width) {
+                    /*
+                    * There is no direct way to clear the cell renderer width so we
+                    * are relying on the fact that it is done as part of modifying
+                    * the style.
+                    */
+                    auto style = OS.gtk_widget_get_modifier_style (parentHandle);
+                    OS.gtk_widget_modify_style (parentHandle, style);
+                }
+            }
+        }
+    }
+    cached = true;
+}
+
+public void setImage (Image image) {
+    checkWidget ();
+    setImage (0, image);
+}
+
+/**
+ * Sets the image for multiple columns in the table.
+ *
+ * @param images the array of new images
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the array of images is null</li>
+ *    <li>ERROR_INVALID_ARGUMENT - if one of the images has been disposed</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void setImage (Image [] images) {
+    checkWidget ();
+    if (images is null) error (DWT.ERROR_NULL_ARGUMENT);
+    for (int i=0; i<images.length; i++) {
+        setImage (i, images [i]);
+    }
+}
+
+/**
+ * Sets the indent of the first column's image, expressed in terms of the image's width.
+ *
+ * @param indent the new indent
+ *
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ *
+ * @deprecated this functionality is not supported on most platforms
+ */
+public void setImageIndent (int indent) {
+    checkWidget ();
+    if (indent < 0) return;
+    /* Image indent is not supported on GTK */
+    cached = true;
+}
+
+/**
+ * Sets the receiver's text at a column
+ *
+ * @param index the column index
+ * @param string the new text
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the text is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void setText (int index, char[] string) {
+    checkWidget ();
+    if (string is null) error (DWT.ERROR_NULL_ARGUMENT);
+    int count = Math.max (1, parent.getColumnCount ());
+    if (0 > index || index > count - 1) return;
+    char* buffer = tango.stdc.stringz.toStringz( string );
+    int modelIndex = parent.columnCount is 0 ? Table.FIRST_COLUMN : parent.columns [index].modelIndex;
+    OS.gtk_list_store_set1 (parent.modelHandle, handle, modelIndex + Table.CELL_TEXT, buffer);
+    /*
+    * Bug in GTK.  When using fixed-height-mode,
+    * row changes do not cause the row to be repainted.  The fix is to
+    * invalidate the row when it is cleared.
+    */
+    if ((parent.style & DWT.VIRTUAL) !is 0) {
+        if (OS.GTK_VERSION >= OS.buildVERSION (2, 3, 2) && OS.GTK_VERSION < OS.buildVERSION (2, 6, 3)) {
+            redraw ();
+        }
+    }
+    cached = true;
+}
+
+public void setText (char[] string) {
+    checkWidget ();
+    setText (0, string);
+}
+
+/**
+ * Sets the text for multiple columns in the table.
+ *
+ * @param strings the array of new strings
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_NULL_ARGUMENT - if the text is null</li>
+ * </ul>
+ * @exception DWTException <ul>
+ *    <li>ERROR_WIDGET_DISPOSED - if the receiver has been disposed</li>
+ *    <li>ERROR_THREAD_INVALID_ACCESS - if not called from the thread that created the receiver</li>
+ * </ul>
+ */
+public void setText (char[] [] strings) {
+    checkWidget ();
+    if (strings is null) error (DWT.ERROR_NULL_ARGUMENT);
+    for (int i=0; i<strings.length; i++) {
+        char[] string = strings [i];
+        if (string !is null) setText (i, string);
+    }
+}
+}
--- a/dwt/widgets/Widget.d	Wed Jan 16 15:35:57 2008 +0100
+++ b/dwt/widgets/Widget.d	Wed Jan 16 16:28:05 2008 +0100
@@ -363,8 +363,13 @@
     return style;
 }
 
-int /*long*/ cellDataProc (int /*long*/ tree_column, int /*long*/ cell, int /*long*/ tree_model, int /*long*/ iter, GtkWidget* data) {
-    return 0;
+void cellDataProc (
+    GtkTreeViewColumn *tree_column,
+    GtkCellRenderer *cell,
+    GtkTreeModel *tree_model,
+    GtkTreeIter *iter,
+    void* data)
+{
 }
 
 void checkOpen () {
@@ -794,8 +799,7 @@
     return 0;
 }
 
-int /*long*/ gtk_row_activated (int /*long*/ tree, int /*long*/ path, int /*long*/ column) {
-    return 0;
+void gtk_row_activated (GtkTreeView* tree, GtkTreePath* path, GtkTreeViewColumn* column) {
 }
 
 int /*long*/ gtk_scroll_child (GtkWidget* widget, int /*long*/ scrollType, int /*long*/ horizontal) {
@@ -846,7 +850,7 @@
     return 0;
 }
 
-int /*long*/ gtk_toggled (int /*long*/ renderer, int /*long*/ pathStr) {
+int /*long*/ gtk_toggled (int /*long*/ renderer, char* pathStr) {
     return 0;
 }
 
@@ -1144,12 +1148,26 @@
     eventTable.unhook (eventType, handler);
 }
 
-void rendererGetSizeProc(GtkCellRenderer *cell, GtkWidget *handle, GdkRectangle *cell_area, int *x_offset, int *y_offset, int *width, int *height) {
-    return 0;
+void rendererGetSizeProc (
+    GtkCellRenderer      *cell,
+    GtkWidget            *widget,
+    GdkRectangle         *cell_area,
+    int                  *x_offset,
+    int                  *y_offset,
+    int                  *width,
+    int                  *height)
+{
 }
 
-void rendererRenderProc (GtkCellRenderer * cell, GdkDrawable * window, GtkWidget * handle, GdkRectangle *background_area, GdkRectangle *cell_area, GdkRectangle *expose_area, int flags){
-    return 0;
+void rendererRenderProc (
+    GtkCellRenderer * cell,
+    GdkDrawable * window,
+    GtkWidget * widget,
+    GdkRectangle *background_area,
+    GdkRectangle *cell_area,
+    GdkRectangle *expose_area,
+    int flags)
+{
 }
 
 /**
@@ -1512,7 +1530,6 @@
 }
 
 void treeSelectionProc (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, int[] selection, int length) {
-    return 0;
 }
 
 bool translateTraversal (int event) {
@@ -1676,7 +1693,7 @@
             return gtk_style_set (handle, arg0);
         case TOGGLED:
             trace( "TOGGLED" );
-            return gtk_toggled (cast(int)handle, arg0);
+            return gtk_toggled (cast(int)handle, cast(char*)arg0);
         case UNMAP_EVENT:
             trace( "UNMAP_EVENT" );
             return gtk_unmap_event (handle, arg0);
@@ -1706,7 +1723,8 @@
             return gtk_delete_text (handle, arg0, arg1);
         case ROW_ACTIVATED:
             trace( "ROW_ACTIVATED" );
-            return gtk_row_activated (cast(int)handle, arg0, arg1);
+            gtk_row_activated (cast(GtkTreeView*)handle, cast(GtkTreePath*)arg0, cast(GtkTreeViewColumn*)arg1);
+            return 0;
         case SCROLL_CHILD:
             trace( "SCROLL_CHILD" );
             return gtk_scroll_child (handle, arg0, arg1);