zoukankan      html  css  js  c++  java
  • GTK之treeview的手记1

    GtkTreeView 构件是一个高级的构件,利用他你就可以制作出漂亮的普通列表或者是树状的列表。这个构件里可以包含一或者多行。他的构架呢?正是采用了大名鼎鼎的MVC (Model View Controller) 设计框架。也就是说数据和显示方式是进行了一种分离的操作。

    于是在GtktreeView构件中确实还有着其他几个独立的对象结构(objects)。

    其中 GtkCellRenderer 就决定了在GtkTreeViewColumn. 中的数据究竟是如何来进行显示呈现的。

    GtkListStoreGtkTreeStore 的功能为体现模型(model)的作用。

    也就是说他们是用来处理和分析将要在GtkTreeView显示的数据的。

    GtkTreeIter 则是一个数据结构被用于在GtkTreeView构件中,对行中的数据进行操作。

    GtkTreeSelection 则是用来处理选项的。

    效果如下

     代码如下

    #include <gtk/gtk.h>
    
    enum
    {
        LIST_ITEM = 0,
        N_COLUMNS
    };
    
    void init_list(GtkWidget *list)
    {
    
        GtkCellRenderer *renderer;
        GtkTreeViewColumn *column;
        GtkListStore *store;
    
        renderer = gtk_cell_renderer_text_new ();
        column = gtk_tree_view_column_new_with_attributes("List Items",
                 renderer, "text", LIST_ITEM, NULL);
        gtk_tree_view_append_column(GTK_TREE_VIEW(list), column);
    
        store = gtk_list_store_new(N_COLUMNS, G_TYPE_STRING);
    
        gtk_tree_view_set_model(GTK_TREE_VIEW(list),
                                GTK_TREE_MODEL(store));
    
        g_object_unref(store);
    }
    
    void add_to_list(GtkWidget *list, const gchar *str)
    {
    
        GtkListStore *store;
        GtkTreeIter iter;
    
        store = GTK_LIST_STORE(gtk_tree_view_get_model
                               (GTK_TREE_VIEW(list)));
    
        gtk_list_store_append(store, &iter);
        gtk_list_store_set(store, &iter, LIST_ITEM, str, -1);
    }
    
    
    void on_changed(GtkWidget *widget, gpointer label)
    {
    
        GtkTreeIter iter;
        GtkTreeModel *model;
        gchar *value;
    
        if (gtk_tree_selection_get_selected(
                    GTK_TREE_SELECTION(widget), &model, &iter))
        {
    
            gtk_tree_model_get(model, &iter, LIST_ITEM, &value,  -1);
            gtk_label_set_text(GTK_LABEL(label), value);
            g_free(value);
        }
    }
    
    int main(int argc, char *argv[])
    {
    
        GtkWidget *window;
        GtkWidget *list;
    
        GtkWidget *vbox;
        GtkWidget *label;
        GtkTreeSelection *selection;
    
        gtk_init(&argc, &argv);
    
        window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
        list = gtk_tree_view_new();
    
        gtk_window_set_title(GTK_WINDOW(window), "List view");
        gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);//设置为居中。
        gtk_container_set_border_width(GTK_CONTAINER(window), 10);
        gtk_widget_set_size_request(window, 270, 250);
    
        gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(list), FALSE);
    
        vbox = gtk_vbox_new(FALSE, 0);
    
        gtk_box_pack_start(GTK_BOX(vbox), list, TRUE, TRUE, 5);
    
        label = gtk_label_new("");
        gtk_box_pack_start(GTK_BOX(vbox), label, FALSE, FALSE, 5);
    
        gtk_container_add(GTK_CONTAINER(window), vbox);
    
        init_list(list);
        add_to_list(list, "Aliens");
        add_to_list(list, "Leon");
        add_to_list(list, "The Verdict");
        add_to_list(list, "North Face");
        add_to_list(list, "Der Untergang");
    
        selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(list));
    
        g_signal_connect(selection, "changed",
                         G_CALLBACK(on_changed), label);
    
        g_signal_connect(G_OBJECT (window), "destroy",
                         G_CALLBACK(gtk_main_quit), NULL);
    
        gtk_widget_show_all(window);
    
        gtk_main();
    
        return 0;
    }

    在我们上面的这个示例代码中,我们将向大家展示的是5个条目并布置于GtkTreeView 构件中。我们首先在window中放置一个GtkVBox 构件。 在这个 GtkVBox 构件中含有两个构件:GtkTreeViewGtkLabel

    list = gtk_tree_view_new();
    gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(list), FALSE);
     init_list(list);

    调用list()函数,初始化构件list。

     renderer = gtk_cell_renderer_text_new();
     column = gtk_tree_view_column_new_with_attributes("List Items",
             renderer, "text", LIST_ITEM, NULL);
     gtk_tree_view_append_column(GTK_TREE_VIEW(list), column);

    在初始化函数中,我们生成了只有一栏的GtkTreeView。

     store = gtk_list_store_new(N_COLUMNS, G_TYPE_STRING);
    
     gtk_tree_view_set_model(GTK_TREE_VIEW(list), 
         GTK_TREE_MODEL(store));

    接下来我们又生成了一个GtkListStore 构件(a model) 然后把它与list 构件绑定。

     g_object_unref(store);

    这个 model 被自动的销毁,以释放内存空间。

    add_to_list(list, "Aliens");

    上面就是在调用add_to_list()函数,实现向list 中在增加一个选项的功能。

     store = GTK_LIST_STORE(gtk_tree_view_get_model
         (GTK_TREE_VIEW(list)));
    
     gtk_list_store_append(store, &iter);
     gtk_list_store_set(store, &iter, LIST_ITEM, str, -1);

    在函数add_to_list() 中,我们利用系统函数gtk_tree_view_get_model()来获得model。我们生成新的一行并把行中的数据交给model处理,这里正是借助GtkTreeIter来完成这个功能。

     selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(list));

    GtkTreeSelection际上并不需要明确生成。在这里,我们是利用 GtkTreeView构件自动来生成。来帮助完成这项工作的正如你所见到的是系统函数gtk_tree_view_get_selection()

    好了再练习一下

    增加一栏

    代码如下

    #include <gtk/gtk.h>
    
    enum
    {
        LIST_ITEM = 0,
        LIST_AGE,
        N_COLUMNS
    };
    
    
    void init_list(GtkWidget *list)
    {
        //要想让视图显示出数据,必须建立GtkCellRenderer与GtkTreeViewColumn
        GtkCellRenderer *renderer;
        GtkTreeViewColumn *column;
        GtkListStore *store;
       
       //建立一个GtkCellRenderer 
        renderer = gtk_cell_renderer_text_new ();    
        //建立一个带标题的列 并且将renderer放入其中使其能显示内容
        column = gtk_tree_view_column_new_with_attributes("List Items", renderer, "text", LIST_ITEM, NULL);    
        //将列加入gtk_tree_view
        gtk_tree_view_append_column(GTK_TREE_VIEW(list), column);
        
        
       //建立一个GtkCellRenderer 
        renderer = gtk_cell_renderer_text_new ();    
        //g_object_set (G_OBJECT (renderer), "xalign", 1.0, NULL);//居右
        //建立一个带标题的列 并且将renderer放入其中使其能显示内容
        column = gtk_tree_view_column_new_with_attributes("List age", renderer, "text", LIST_AGE, NULL);    
        //将列加入gtk_tree_view
        gtk_tree_view_append_column(GTK_TREE_VIEW(list), column);
        
        
    
        store = gtk_list_store_new(N_COLUMNS, G_TYPE_STRING,G_TYPE_INT);
        
       
        //关联视图与模型
        gtk_tree_view_set_model(GTK_TREE_VIEW(list), GTK_TREE_MODEL(store));
    
        //将数据模型交给视图管理,视图销毁时数据会被一同销毁
        g_object_unref(store);
    }
    
    void add_to_list(GtkWidget *list, const gchar *str ,gint age)
    {
    
        GtkListStore *store;
        GtkTreeIter iter;
    
        store = GTK_LIST_STORE(gtk_tree_view_get_model
                               (GTK_TREE_VIEW(list)));
    
        gtk_list_store_append(store, &iter);
        gtk_list_store_set(store, &iter, LIST_ITEM, str,LIST_AGE,age, -1);
    }
    
    
    
    void on_changed(GtkWidget *widget, gpointer label)
    {
    
        GtkTreeIter iter;
        GtkTreeModel *model;
        gchar *value;
       
        //获得treeview中选中的一行的GtkTreeIter
        if (gtk_tree_selection_get_selected( GTK_TREE_SELECTION(widget), &model, &iter))
        {
    
            gtk_tree_model_get(model, &iter, LIST_ITEM, &value,  -1);
            gtk_label_set_text(GTK_LABEL(label), value);
            g_free(value);
        }
    }
    
    int main(int argc, char *argv[])
    {
    
        GtkWidget *window;
        GtkWidget *list;
    
        GtkWidget *vbox;
        GtkWidget *label;
        GtkTreeSelection *selection;
    
        gtk_init(&argc, &argv);
    
        window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
        list = gtk_tree_view_new();
    
        gtk_window_set_title(GTK_WINDOW(window), "List view");
        gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
        gtk_container_set_border_width(GTK_CONTAINER(window), 10);
        gtk_widget_set_size_request(window, 270, 250);
    
        //设置标题的可见性状态。
        gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(list), TRUE );
    
        vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);//gtk_vbox_new(FALSE, 0);
    
        gtk_box_pack_start(GTK_BOX(vbox), list, TRUE, TRUE, 5);
    
        label = gtk_label_new("");
        gtk_box_pack_start(GTK_BOX(vbox), label, FALSE, FALSE, 5);
    
        gtk_container_add(GTK_CONTAINER(window), vbox);
    
        init_list(list);
        add_to_list(list, "Aliens"        ,10 );
        add_to_list(list,  "Leon"         ,2 );
        add_to_list(list,  "The Verdict"  ,30 );
        add_to_list(list,  "North Face"   ,4 );
        add_to_list(list,  "Der Untergang",50 );
    
        
        selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(list));
    
        g_signal_connect(selection, "changed",
                         G_CALLBACK(on_changed), label);
    
        g_signal_connect(G_OBJECT (window), "destroy",
                         G_CALLBACK(gtk_main_quit), NULL);
    
        gtk_widget_show_all(window);
    
        gtk_main();
    
        return 0;
    }
  • 相关阅读:
    代码控制数据流量开关
    用wifi来调试应用程序
    详细解读LruCache类
    修改博客园默认的代码字体大小
    通过Gson解析Json数据
    Docker、Kubernetes的 CICD实现思路
    React中路由传参及接收参数的方式
    微信小程序开发工具调试没问题,真机调试Provisional headers are shown
    物联网卡三码
    【微信开发】-- 企业转账到用户
  • 原文地址:https://www.cnblogs.com/libra13179/p/12601798.html
Copyright © 2011-2022 走看看