前言

我发现在Gtk+这种风格叫Note Book,在C#中它们被称作 TabPage 。

笔记本 Notebooks

笔记本控件(The NoteBook Widget)是互相重叠的页面集合,每一页都包含不同的信息,且一次只有一个页面是可见的。

笔记本控件在GUI(图形用户接口)编程中很常用,如果我们想要显示大量的相似信息,同时把它们分别显示时,使用这种控件是一个很好的方法。

创建笔记本控件

第一个你要知道的函数调用,你可能已经猜到了,是用来创建一个新的笔记本控件。

GtkWidget *gtk_notebook_new( void );

一旦创建了笔记本控件,就可以使用一系列的函数操作该控件。下面将对它们进行分别讨论。

设置页标签方向

先看一下怎样定位页面指示器—或称页标签,可以有四种位置:上、下、左或右。

void gtk_notebook_set_tab_pos( GtkNotebook     *notebook,
                               GtkPositionType  pos ); 

GtkPositionType参数可以取以下几个值,从字面上很容易理解它们的含义:

  GTK_POS_LEFT
  GTK_POS_RIGHT
  GTK_POS_TOP
  GTK_POS_BOTTOM 

GTK_POS_TOP是缺省值。

追加页标签

下面看一下怎样向笔记本中添加页面。有三种方法向笔记本中添加页面。前两种方法是非常相似的。

void gtk_notebook_append_page( GtkNotebook *notebook,
                               GtkWidget   *child,
                               GtkWidget   *tab_label );

void gtk_notebook_prepend_page( GtkNotebook *notebook,
                                GtkWidget   *child,
                                GtkWidget   *tab_label );

参数:

  • child
    该参数是放在笔记本页面里的子控件;
    child控件必须另外创建,一般是一个包含一套选项设置的容器控件,比如一个表格;
  • tab_label
    该参数是要添加的页面的标签;

这些函数通过向插入页面到笔记本的后端(append)或前端(prepend)来添加页面。

插入标签页

最后一个添加页面的函数与前两个函数类似,不过允许指定页面插入的位置。

void gtk_notebook_insert_page( GtkNotebook *notebook,
                               GtkWidget   *child,
                               GtkWidget   *tab_label,
                               gint         position ); 

其中的参数与appendprepend函数一样,还包含一个额外参数,position。该参数指定页面应该插入到哪一页。注意,第一页位置为0。

删除标签页

前面介绍了怎样添加一个页面,下面介绍怎样从笔记本中删除一个页面。

void gtk_notebook_remove_page( GtkNotebook *notebook,
                               gint         page_num );

这个函数从notebook指定的笔记本中删除由page_num参数指定的页面。

获取当前的标签页

用这个函数找出笔记本的当前页面:

gint gtk_notebook_get_current_page( GtkNotebook *notebook );

下面两个函数将笔记本的页面向前或向后移动。

前后切换标签页

对要操作的笔记本控件使用以下函数就可以了。注意:当笔记本正在最后一页时,调用 gtk_notebook_next_page() 函数,笔记本会跳到第一页。同样,如果笔记本在第一页,调用了函数 gtk_notebook_prev_page(),笔记本控件会跳到最后一页。

void gtk_notebook_next_page( GtkNoteBook *notebook );

void gtk_notebook_prev_page( GtkNoteBook *notebook );

直接选择标签页

下面这个函数设置“活动”页面。比如你想笔记本的第5页被打开,你将使用这个函数。不使用这个函数时笔记本默认显示第一页。

void gtk_notebook_set_current_page( GtkNotebook *notebook,
                                    gint         page_num );

显示或隐藏标签页

下面两个函数分别显示或隐藏笔记本的页标签以及它的边框。

void gtk_notebook_set_show_tabs( GtkNotebook *notebook,
                                 gboolean     show_tabs );

void gtk_notebook_set_show_border( GtkNotebook *notebook,
                                   gboolean     show_border );

允许显示箭头按钮滚动标签页

如果页面较多,标签页在页面上排列不下时,可以用下面这个函数。它允许用两个箭头按钮来滚动标签页。

void gtk_notebook_set_scrollable( GtkNotebook *notebook,
                                  gboolean     scrollable );

show_tabs, show_border和scrollable参数可以为 TRUEFALSE

示例

功能

这个小程序创建了一个含一个笔记本控件和6个按钮的窗口。笔记本包含11页,由三种方式添加进来:追加、插入、前插。点击按钮可以改变页标签的位置,显示/隐藏页标签和边框,删除一页,向前或向后移动标签页,以及退出程序。

笔记本 - 图1

源码

#include <stdio.h>
#include <gtk/gtk.h>

/* 这个函数旋转页标签的位置 */
void rotate_book( GtkButton   *button,
                  GtkNotebook *notebook )
{
    gtk_notebook_set_tab_pos (notebook, (notebook->tab_pos + 1) % 4);
}

/* 显示/隐藏页标签和边框 */
void tabsborder_book( GtkButton   *button,
                      GtkNotebook *notebook )
{
    gint tval = FALSE;
    gint bval = FALSE;
    if (notebook->show_tabs == 0)
    {
        tval = TRUE;
    }
    if (notebook->show_border == 0)
    {
        bval = TRUE;
    }

    gtk_notebook_set_show_tabs (notebook, tval);
    gtk_notebook_set_show_border (notebook, bval);
}

/* 从笔记本上删除一个页面 */
void remove_book( GtkButton   *button,
                  GtkNotebook *notebook )
{
    gint page;

    page = gtk_notebook_get_current_page (notebook);
    gtk_notebook_remove_page (notebook, page);
    /* 需要刷新控件 -- 这会迫使控件重绘自身。 */
    gtk_widget_queue_draw (GTK_WIDGET (notebook));
}

gint delete( GtkWidget *widget,
             GtkWidget *event,
             gpointer   data )
{
    gtk_main_quit ();
    return FALSE;
}

int main( int argc,
          char *argv[] )
{
    GtkWidget *window;
    GtkWidget *button;
    GtkWidget *table;
    GtkWidget *notebook;
    GtkWidget *frame;
    GtkWidget *label;
    GtkWidget *checkbutton;
    int i;
    char bufferf[32];
    char bufferl[32];

    gtk_init (&argc, &argv);

    window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
    gtk_window_set_title (GTK_WINDOW (window), "www.softool.cn - notebook");
    g_signal_connect (G_OBJECT (window), "delete_event",
                        G_CALLBACK (delete), NULL);
    gtk_container_set_border_width (GTK_CONTAINER (window), 10);

    //新建一个 table 控件:
    table = gtk_table_new (3, 6, FALSE);
    gtk_container_add (GTK_CONTAINER (window), table);

    /* 创建一个新的笔记本,将标签页放在顶部 */
    notebook = gtk_notebook_new ();
    gtk_notebook_set_tab_pos (GTK_NOTEBOOK (notebook), GTK_POS_TOP);
    gtk_table_attach_defaults (GTK_TABLE (table), notebook, 0, 6, 0, 1);
    gtk_widget_show (notebook);

    /* 在笔记本标签页后面,再追加几个页面 */
    for (i = 0; i < 5; i++)
    {
        sprintf(bufferf, "Append Frame %d", i + 1);
        sprintf(bufferl, "Page %d", i + 1);

        frame = gtk_frame_new (bufferf);
        gtk_container_set_border_width (GTK_CONTAINER (frame), 10);
        gtk_widget_set_size_request (frame, 100, 75);
        gtk_widget_show (frame);

        label = gtk_label_new (bufferf);
        gtk_container_add (GTK_CONTAINER (frame), label);
        gtk_widget_show (label);

        label = gtk_label_new (bufferl);
        gtk_notebook_append_page (GTK_NOTEBOOK (notebook), frame, label);
    }

    /* 在指定位置添加页面 */
    checkbutton = gtk_check_button_new_with_label ("Check me please!");
    gtk_widget_set_size_request (checkbutton, 100, 75);
    gtk_widget_show (checkbutton);
    //
    label = gtk_label_new ("Add page");
    //插入标签页:
    gtk_notebook_insert_page (GTK_NOTEBOOK (notebook), checkbutton, label, 2);

    /* 最后再向笔记本前插页面 */
    for (i = 0; i < 5; i++)
    {
        sprintf (bufferf, "Prepend Frame %d", i + 1);
        sprintf (bufferl, "PPage %d", i + 1);

        frame = gtk_frame_new (bufferf);
        gtk_container_set_border_width (GTK_CONTAINER (frame), 10);
        gtk_widget_set_size_request (frame, 100, 75);
        gtk_widget_show (frame);

        label = gtk_label_new (bufferf);
        gtk_container_add (GTK_CONTAINER (frame), label);
        gtk_widget_show (label);

        label = gtk_label_new (bufferl);
        gtk_notebook_prepend_page (GTK_NOTEBOOK (notebook), frame, label);
    }

    /* 设置起始页(第4页) */
    gtk_notebook_set_current_page (GTK_NOTEBOOK (notebook), 3);

    /* 创建一排按钮 */
    button = gtk_button_new_with_label ("close");
    g_signal_connect_swapped (G_OBJECT (button), "clicked",
                                G_CALLBACK (delete), NULL);
    gtk_table_attach_defaults (GTK_TABLE (table), button, 0, 1, 1, 2);
    gtk_widget_show (button);

    button = gtk_button_new_with_label ("next page");
    g_signal_connect_swapped (G_OBJECT (button), "clicked",
                                G_CALLBACK (gtk_notebook_next_page),
                                notebook);
    gtk_table_attach_defaults (GTK_TABLE (table), button, 1, 2, 1, 2);
    gtk_widget_show (button);

    button = gtk_button_new_with_label ("prev page");
    g_signal_connect_swapped (G_OBJECT (button), "clicked",
                                G_CALLBACK (gtk_notebook_prev_page),
                                notebook);
    gtk_table_attach_defaults (GTK_TABLE (table), button, 2, 3, 1, 2);
    gtk_widget_show (button);

    button = gtk_button_new_with_label ("tab position");
    g_signal_connect (G_OBJECT (button), "clicked",
                        G_CALLBACK (rotate_book),
                        notebook);
    gtk_table_attach_defaults (GTK_TABLE (table), button, 3, 4, 1, 2);
    gtk_widget_show (button);

    button = gtk_button_new_with_label ("tabs/border on/off");
    g_signal_connect (G_OBJECT (button), "clicked",
                      G_CALLBACK (tabsborder_book),
                      notebook);
    gtk_table_attach_defaults (GTK_TABLE (table), button, 4, 5, 1, 2);
    gtk_widget_show (button);

    button = gtk_button_new_with_label ("remove page");
    g_signal_connect (G_OBJECT (button), "clicked",
                      G_CALLBACK (remove_book),
                      notebook);
    gtk_table_attach_defaults (GTK_TABLE (table), button, 5, 6, 1, 2);
    gtk_widget_show (button);

    gtk_widget_show (table);
    gtk_widget_show (window);

    gtk_main ();

    return 0;
}