一个五脏俱全的Gtk小程序

refer from:http://tzc.is-programmer.com/posts/534.html


本人自从学习Gtk以来,一直没能做一个像模像样的例子,前段时间抽空写了一个小程序,大概800多行,是调用Gtk和GLib库共同编译的,贴到这里和大家分享:

 

#include<gtk/gtk.h>
#include<glib.h>
#include<stdio.h>
#include<malloc.h>
#include<math.h>

enum            /*定义图形的类型*/
{
    PIX_DRAW=1,    /*类型-点*/
    LINE_DRAW,
    CIRCLE_DRAW,
    RECT_DRAW,
    POLY_DRAW
};
typedef struct        /*定义含树视图和绘图区的结构体,以便回调函数传递参数*/
{
    GtkWidget *tree_view;
    GtkWidget *draw_area;
}tree_drawing;
typedef struct        /*定义含树视图、绘图区和文本输入框的结构体,以便回调函数传递参数*/
{
    GtkWidget *window;
    GtkWidget *view;
    GtkWidget *entry;
}wview;
typedef struct        /*为方便参数传递而创建的结构体*/
{
    GtkWidget *window;
    GtkWidget *view;
    GtkWidget *entry;
    GtkWidget *draw_area;
}wdview;
typedef struct _2D_POINT    /*定义二维数据点结构体*/
{
    float x,y;
}_2d_point;
/*------------定义全局变量--------------*/
double width=100.0,height=100.0;    /*定义系统长宽,这里与窗口窗宽无关*/
int draw_type=0;            /*定义图像类型*/
GSList *prt=NULL;            /*定义存放数据的链表*/
GtkTreeModel *model;            /*定义树视图模型*/
GtkTreeStore *tree_store;        /*定义树形存储*/
GtkTreeIter top_pix,top_line,top_circle,top_rect,top_poly,other;    /*定义树视图的项,以便添加元素*/
/*-----------函数声明部分------------------*/
GtkTreeView* create_tree(void);        /*创建树视图*/
GtkTreeModel* create_model(void);    /*创建树视图模型*/

void get_new_file(GtkWidget *button,wview *wv);/*获得文件名,并修改树视图显示*/
void create_file(GtkWidget *create_file_button,GtkWidget *view);/*创建文件菜单的回调函数*/
void get_import_file(GtkWidget* button,wdview *wdv);        /*获得导入文件名,并修改树视图和绘图区显示*/
void messagebox(const char *str);                /*消息提示框,用于提示相关信息*/
void import_file(GtkWidget *import_file_button,tree_drawing *td);/*导入文件菜单的回调函数*/
void export_file(GtkWidget *export_file_button,GtkWidget *view);/*导出文件菜单的回调函数*/

void enlarge_obj(GtkWidget *enlarge,GtkWidget *data);/*工具栏中放大按钮的回调函数,修改绘图区显示*/
void deflate_obj(GtkWidget *deflate,GtkWidget *data);/*工具栏中缩小按钮的回调函数,修改绘图区显示*/
void draw_pix(void);                    /*draw菜单下,points菜单的回调函数*/
void draw_line(void);                    /*Line菜单的回调函数*/
void draw_circle(void);                    /*Circle菜单的回调函数*/
void draw_rect(void);                    /*Rectagle菜单的回调函数*/
void draw_poly(void);                    /*Plygon菜单的回调函数*/
void draw_prt(GtkWidget *data);                /*在绘图区中按照图像类型,将链表内容绘制出来,修改绘图区显示*/
/*下面这几个函数都没实现,这里不做具体介绍*/
/*
void show_part(GtkWidget *view,GtkWidget *draw_area);
void hide_part(GtkWidget *view,GtkWidget *draw_are);
void del_part(GtkWidget *view,GtkWidget *draw_area);
void modify_part(GtkWidget *view,GtkWidget *draw_area);
GtkWidget* pop_menu(GtkWidget *draw_area);
static gboolean view_press(GtkWidget *widget,GdkEventButton *event,GtkWidget *menu);
*/
/*-----------函数定义部分------------*/
/*
GtkWidget* pop_menu(GtkWidget *draw_area)
{
    GtkWidget *file_menu;
    GtkWidget *show_item,*hide_item,*del_item,*modify_item;

    show_item=gtk_menu_item_new_with_label("show");
    hide_item=gtk_menu_item_new_with_label("hide");
    del_item=gtk_menu_item_new_with_label("delete");
    modify_item=gtk_menu_item_new_with_label("modify");

    gtk_menu_append(GTK_MENU(file_menu),show_item);
    gtk_menu_append(GTK_MENU(file_menu),hide_item);
    gtk_menu_append(GTK_MENU(file_menu),del_item);
    gtk_menu_append(GTK_MENU(file_menu),modify_item);

    g_signal_connect_swapped(G_OBJECT(show_item),"activate",G_CALLBACK(show_part),draw_area);
    g_signal_connect_swapped(G_OBJECT(hide_item),"activate",G_CALLBACK(hide_part),draw_area);
    g_signal_connect_swapped(G_OBJECT(del_item),"activate",G_CALLBACK(del_part),draw_area);
    g_signal_connect_swapped(G_OBJECT(modify_item),"activate",G_CALLBACK(modify_part),draw_area);

    return file_menu;
}
void show_part(GtkWidget *view,GtkWidget *draw_area)
{
}
void hide_part(GtkWidget *view,GtkWidget *draw_are)
{
}
void del_part(GtkWidget *view,GtkWidget *draw_area)
{
}
void modify_part(GtkWidget *view,GtkWidget *draw_area)
{
}
*/
/*上面几个函数都还没实现,由于事件问题,在这里就不具体实现了*/
/*-----------函数定义部分-------------*/
/*创建树视图,向里面添加列,并调用树视图模型创建函数完成树视图的创建,
 * 输入参数:无
 * 输出参数:树视图构件tree_view
 * 日期:2007-4-7修改*/
GtkTreeView* create_tree(void)
{
    GtkWidget *tree_view;
    GtkTreeViewColumn *col;
    GtkCellRenderer *cell;
   
    tree_view=gtk_tree_view_new();
    col=gtk_tree_view_column_new();
    gtk_tree_view_column_set_title(col,"part");
    gtk_tree_view_append_column(GTK_TREE_VIEW(tree_view),col);
    cell=gtk_cell_renderer_text_new();
        //g_object_set(cell,"editable",TRUE,NULL);
    //g_signal_connect(cell,"edited",G_CALLBACK(cell_edited),NULL);
    gtk_tree_view_column_pack_start(col,cell,TRUE);
    gtk_tree_view_column_add_attribute(col,cell,"text",0);
    model=create_model();
    gtk_tree_view_set_model(GTK_TREE_VIEW(tree_view),model);

    return GTK_TREE_VIEW(tree_view);
}
/*创建树视图模型,向tree_store里面添加条目
 * 输入参数:无
 * 输出参数:树视图模型tree_store
 * 日期:2007-4-7修改*/
GtkTreeModel* create_model()
{
    tree_store=gtk_tree_store_new(1,G_TYPE_STRING);
    //添加树的项
    gtk_tree_store_append(tree_store,&top_pix,NULL);
    gtk_tree_store_set(tree_store,&top_pix,
        0,"Points",-1);
    gtk_tree_store_append(tree_store,&top_line,NULL);
    gtk_tree_store_set(tree_store,&top_line,
        0,"Lines",-1);
    gtk_tree_store_append(tree_store,&top_circle,NULL);
    gtk_tree_store_set(tree_store,&top_circle,
        0,"Circles",-1);
    gtk_tree_store_append(tree_store,&top_rect,NULL);
    gtk_tree_store_set(tree_store,&top_rect,
        0,"Rectangles",-1);
    gtk_tree_store_append(tree_store,&top_poly,NULL);
    gtk_tree_store_set(tree_store,&top_poly,
        0,"Polygons",-1);
    gtk_tree_store_append(tree_store,&other,NULL);
    gtk_tree_store_set(tree_store,&other,
        0,"Others",-1);
    return GTK_TREE_MODEL(tree_store);

}
/*获得文本输入框内容,并将其加入到树视图中显示,并隐藏新建文件窗口
 * 输入参数:按钮构件本身button,包含文本输入框、树视图和绘图区的结构体wv
 * 输出参数:无
 * 日期:2007-4-7修改*/
void get_new_file(GtkWidget *button,wview *wv)
{
    char *str=malloc(20*sizeof(char));
    str=(char*)gtk_entry_get_text(GTK_ENTRY(wv->entry));
    GtkTreeIter child;
    gtk_tree_store_append(tree_store,&child,&other);/*向树存储类型的构件中添加条目*/
    gtk_tree_store_set(tree_store,&child,        /*设置新添加条目的文本显示*/
        0,str,-1);
    /*FILE *fp=NULL;
    if((fp=fopen(str,"w"))==NULL)
        printf("error in open file %s\n",str);
    fclose(fp);*/
    gtk_widget_hide(wv->window);
}
/*隐藏窗口,
 * 输入参数:按钮button,窗口构件window
 * 输出参数:无
 * 日期:2007-4-7修改*/
void cancel_create(GtkWidget *button,GtkWidget *window)
{
    gtk_widget_hide(window);
}
/*创建文件菜单的回调函数。用于打开一个对话框,让用户输入文件名,然后将文件名放到树视图的other条目下,创建文件,
 * 输入参数:菜单项本身create_file_item, 树视图view
 * 输出参数:无
 * 日期:2007-4-7修改*/
void create_file(GtkWidget *create_file_button,GtkWidget *view)
{
    GtkWidget *window,*entry;
    GtkWidget *label,*vbox,*hbox,*b_ok,*b_cancel;
   
    window=gtk_window_new(GTK_WINDOW_TOPLEVEL);
    gtk_widget_set_size_request(window,300,110);
    gtk_window_set_title(GTK_WINDOW(window),"create part file");
    g_signal_connect(G_OBJECT(window),"destroy",G_CALLBACK(gtk_widget_destroy),NULL);
    gtk_window_set_position(GTK_WINDOW(window),GTK_WIN_POS_CENTER);
   
    vbox=gtk_vbox_new(FALSE,15);
    gtk_container_add(GTK_CONTAINER(window),vbox);
   
    label=gtk_label_new("Please input the file name:");
    gtk_box_pack_start(GTK_BOX(vbox),label,FALSE,FALSE,0);
    entry=gtk_entry_new();
    gtk_box_pack_start(GTK_BOX(vbox),entry,FALSE,FALSE,0);
    hbox=gtk_hbox_new(FALSE,0);
    gtk_box_pack_start(GTK_BOX(vbox),hbox,FALSE,FALSE,0);

    wview *wv=malloc(sizeof(wview));
    wv->window=window;
    wv->view=view;
    wv->entry=entry;

    b_ok=gtk_button_new_with_label("  ok   ");
    g_signal_connect(G_OBJECT(b_ok),"clicked",G_CALLBACK(get_new_file),wv);
    gtk_box_pack_start(GTK_BOX(hbox),b_ok,FALSE,FALSE,50);
    b_cancel=gtk_button_new_with_label("cancel");
    g_signal_connect(G_OBJECT(b_cancel),"clicked",G_CALLBACK(cancel_create),window);
   
    gtk_box_pack_start(GTK_BOX(hbox),b_cancel,FALSE,FALSE,50);

    gtk_widget_show_all(window);
}
/*用于隐藏构件,这里是将messagebox窗口隐藏,
 * 输入参数:按钮本身button,窗口构件data
 * 输出参数:无
 * 日期:2007-4-7修改*/
void f_button(GtkWidget * button,GtkWidget * data)
{
    gtk_widget_hide(data);
}
/*一个消息提示框,弹出一个窗口,用于提示当前的错误或者其它信息
 * 输入参数:字符串str
 * 输出参数:无
 * 日期:2007-4-7修改*/
void messagebox(const char* str)
{
    GtkWidget * window,*table,*label,*button;
    /////
    window=gtk_window_new(GTK_WINDOW_TOPLEVEL);
    gtk_window_set_title(GTK_WINDOW(window),"messagebox");
    gtk_widget_set_size_request(GTK_WIDGET(window),300,100);
    gtk_window_set_position(GTK_WINDOW(window),GTK_WIN_POS_CENTER);
    g_signal_connect(G_OBJECT(window),"destroy",G_CALLBACK(gtk_widget_destroy),NULL);
    /////
    table=gtk_table_new(2,3,FALSE);
    gtk_container_add(GTK_CONTAINER(window),table);
    ///
    label=gtk_label_new(str);
    gtk_table_attach_defaults(GTK_TABLE(table),label,1,2,0,1);
    gtk_widget_show(label);
   
   
    button=gtk_button_new_with_label("     确定     ");
    gtk_table_attach(GTK_TABLE(table),button,1,2,1,2,GTK_SHRINK,GTK_FILL,0,0);
    g_signal_connect(G_OBJECT(button),"clicked",G_CALLBACK(f_button),window);
    gtk_widget_show(button);
    gtk_widget_show_all(window);
}
/*导入文件对话框ok按钮的回调函数,实现读取文件名,并在将文件内容读取到链表结束后,
 * 根据图形类型修改树视图和绘图区,将文件名添加到树视图的相应条目下。
 * 输出参数:按钮本身:button,包含标签、树视图和绘图区的一个结构体
 * 输出参数:无
 * 日期:2007-4-7修改*/
void get_import_file(GtkWidget* button,wdview *wdv)
{
    char *str=malloc(20*sizeof(char));
    str=(char*)gtk_entry_get_text(GTK_ENTRY(wdv->entry));
    char *filename=malloc(20*sizeof(char));
    sprintf(filename,"%s.124",str);
    FILE *fp=NULL;
    if((fp=fopen(filename,"r"))==NULL)
    {
    messagebox("the file you import is not exist !");
    return ;
    }
    char ch;
    _2d_point *p=malloc(sizeof(_2d_point));
    fscanf(fp,"%c",&ch);
    GtkTreeIter child;
    if(ch=='P')            //判断是哪种类型的部件,以便放到对应目录下
    {
    gtk_tree_store_append(tree_store,&child,&top_pix);
    gtk_tree_store_set(tree_store,&child,
        0,str,-1);
    draw_type=PIX_DRAW;
    }
    else if(ch=='L')
    {
    gtk_tree_store_append(tree_store,&child,&top_line);
    gtk_tree_store_set(tree_store,&child,
        0,str,-1);
    draw_type=LINE_DRAW;
    }
    else if(ch=='C')
    {
    gtk_tree_store_append(tree_store,&child,&top_circle);
    gtk_tree_store_set(tree_store,&child,
        0,str,-1);
    draw_type=CIRCLE_DRAW;
    }
    else if(ch=='R')
    {
    gtk_tree_store_append(tree_store,&child,&top_rect);
    gtk_tree_store_set(tree_store,&child,
        0,str,-1);
    draw_type=RECT_DRAW;
    }
    else if(ch=='D')
    {
    gtk_tree_store_append(tree_store,&child,&top_poly);
    gtk_tree_store_set(tree_store,&child,
        0,str,-1);
    draw_type=POLY_DRAW;
    }
    else
    {
        gtk_tree_store_append(tree_store,&child,&other);
        gtk_tree_store_set(tree_store,&child,
        0,str,-1);
    }
    GSList *iter1=prt;
    while(iter1)//删除数组中原来内容
    {
    prt=g_slist_remove_link(prt,iter1);
    g_slist_free(iter1);
    iter1=prt;
    }
    while(!feof(fp))
    {
    p=malloc(sizeof(_2d_point));
    int ret=fscanf(fp,"%f",&(p->x));
    ret=fscanf(fp,"%f",&(p->y));
    if(ret!=1)
    {
        break;
    }
    prt=g_slist_append(prt,p);
    }
    draw_prt(wdv->draw_area);
   
    gtk_widget_destroy(wdv->window);
    fclose(fp);
}
/*导入菜单的回调函数,创建一个窗口用于输入要导入的文件名,ok按钮的回调函数实现读取文件名,
 * 将文件读取到链表,并修改树视图和绘图区的显示,
 * 输入参数:菜单项本生import_file_button, 包含树视图和绘图区的结构体td
 * 输出参数:无,
 * 日期:2007-4-7修改*/
void import_file(GtkWidget *import_file_button,tree_drawing *td)
{
    GtkWidget *window,*entry;
    GtkWidget *label,*vbox,*hbox,*b_ok,*b_cancel;
   
    window=gtk_window_new(GTK_WINDOW_TOPLEVEL);
    gtk_widget_set_size_request(window,300,110);
    gtk_window_set_title(GTK_WINDOW(window),"create part file");
    g_signal_connect(G_OBJECT(window),"destroy",G_CALLBACK(gtk_widget_destroy),NULL);
    gtk_window_set_position(GTK_WINDOW(window),GTK_WIN_POS_CENTER);
   
    vbox=gtk_vbox_new(FALSE,15);
    gtk_container_add(GTK_CONTAINER(window),vbox);
   
    label=gtk_label_new("Please input the file you want to import:");
    gtk_box_pack_start(GTK_BOX(vbox),label,FALSE,FALSE,0);
    entry=gtk_entry_new();
    gtk_box_pack_start(GTK_BOX(vbox),entry,FALSE,FALSE,0);
    hbox=gtk_hbox_new(FALSE,50);
    gtk_box_pack_start(GTK_BOX(vbox),hbox,FALSE,FALSE,0);

    wdview *wdv=malloc(sizeof(wdview));
    wdv->window=window;
    wdv->view=td->tree_view;
    wdv->entry=entry;
    wdv->draw_area=td->draw_area;

    b_ok=gtk_button_new_with_label("  ok   ");
    g_signal_connect(G_OBJECT(b_ok),"clicked",G_CALLBACK(get_import_file),wdv);
    gtk_box_pack_start(GTK_BOX(hbox),b_ok,FALSE,FALSE,50);
    b_cancel=gtk_button_new_with_label("cancel");
    g_signal_connect(G_OBJECT(b_cancel),"clicked",G_CALLBACK(cancel_create),window);
   
    gtk_box_pack_start(GTK_BOX(hbox),b_cancel,FALSE,FALSE,0);

    gtk_widget_show_all(window);

}
/*导出文件的回调函数,根据当前选中的树视图中的项,将链表中的数据写到文本文件中,
 * 在文本文件中,第一行是图形的类型,后面的是数据,
 * 存储的文件名就是当前选中的那一行的文本。
 * 输入参数:菜单项本身export_file, 树视图view
 * 输出参数:无
 * 日期:2007-4-7修改*/
void export_file(GtkWidget *export_file,GtkWidget *view)
{
    GtkTreeIter iter;
    GtkTreeView *treeview=(GtkTreeView*)view;
    GtkTreeSelection *selection=gtk_tree_view_get_selection(treeview);/*获得树视图的选定行*/

    if(gtk_tree_selection_get_selected(selection,NULL,&iter))/*从选定行获得选定项*/
    {
    gint i;
    GtkTreePath *path;
    path=gtk_tree_model_get_path(model,&iter);        /*由选定项获得路径。这条在这里无用*/
    //i=gtk_tree_path_get_indeces(path)[0];
    // gtk_tree_store_set (tree_store, &iter, column,
        //                  filename, -1);
    //获得文本
    char *text=malloc(sizeof(char));
    gtk_tree_model_get(GTK_TREE_MODEL(tree_store),&iter,0,&text,-1);/*由项获得项的文本*/
    char *file_name=malloc(19*sizeof(char));
    sprintf(file_name,"%s.124",text);
    FILE *fp=NULL;
    if((fp=fopen(file_name,"w"))==NULL)
    {
        messagebox("conn't open file you saved!");
        return ;
    }
    GSList *iterator=prt;
    char type;
    if(draw_type==PIX_DRAW)
    {
        type='P';
    }
    else if(draw_type==LINE_DRAW)
    {
        type='L';
    }
    else if(draw_type==CIRCLE_DRAW)
    {
        type='C';
    }
    else if(draw_type==RECT_DRAW)
    {
        type='R';
    }
    else if(draw_type==POLY_DRAW)
    {
        type='D';
    }
    else
    {
        messagebox("you have not create file!");
        return ;
    }
    fprintf(fp,"%c\n",type);
    while(iterator)
    {
        fprintf(fp,"%f\t%f\n",((_2d_point*)(iterator->data))->x,((_2d_point*)(iterator->data))->y);
        iterator=iterator->next;
    }
    fclose(fp);
    }
    return;
}
/*绘图函数,根据width和height的值将链表中的数据进行变换到当前坐标系下,由draw_type决定绘制什么图形,
 * 输入参数:绘图区data,
 * 输出参数:无
 * 日期:2007-4-7*/
void draw_prt(GtkWidget *data)
{
    int area_w=100,area_h=100;
    area_w=data->allocation.width;            //获得窗口尺寸
    area_h=data->allocation.height;
    gdk_window_clear(data->window);
    GSList *iter1=NULL,*iter2=NULL;
    iter1=prt;
    if(iter1==NULL){
        return ;
    }
    iter2=prt->next;
    if(draw_type==PIX_DRAW)
    {
        int x,y;
        while(iter1)
        {
            x=(((_2d_point*)(iter1->data))->x)*area_w/width+area_w/2;
            y=(((_2d_point*)(iter1->data))->y)*area_h/height+area_h/2;
            gdk_draw_point(GTK_WIDGET(data)->window,GTK_WIDGET(data)->style->
                fg_gc[GTK_WIDGET_STATE (data)],x,y);
            iter1=iter1->next;
        }
    }
    if(draw_type==LINE_DRAW)
    {
        int x0,x1,y0,y1;
        while(iter2)
        {
            x0=(((_2d_point*)(iter1->data))->x)*area_w/width+area_w/2;
            y0=(((_2d_point*)(iter1->data))->y)*area_h/height+area_h/2;
            x1=(((_2d_point*)(iter2->data))->x)*area_w/width+area_w/2;
            y1=(((_2d_point*)(iter2->data))->y)*area_h/height+area_h/2;           
            gdk_draw_line(GTK_WIDGET(data)->window,GTK_WIDGET(data)->
                style->fg_gc[GTK_WIDGET_STATE (data)],x0,y0,x1,y1);
            iter1=iter2;
            iter2=iter1->next;
        }
    }
    if(draw_type==CIRCLE_DRAW)
    {
        int x0,x1,y0,y1;
        int r=0.0;
        while(iter1&&iter2)
        {
            x0=(((_2d_point*)(iter1->data))->x)*area_w/width+area_w/2;
            y0=(((_2d_point*)(iter1->data))->y)*area_h/height+area_h/2;
            x1=(((_2d_point*)(iter2->data))->x)*area_w/width+area_w/2;
            y1=(((_2d_point*)(iter2->data))->y)*area_h/height+area_h/2;           
            r=sqrt((x1-x0)*(x1-x0)+(y1-y0)*(y1-y0));
            gdk_draw_arc(GTK_WIDGET(data)->window,GTK_WIDGET(data)->style->
                    fg_gc[GTK_WIDGET_STATE (data)],FALSE,x0-r,y0-r,2*r,2*r,0,64*360);
            iter1=iter2->next;
            if(iter1)
            {
                iter2=iter1->next;
            }
        }
    }
    if(draw_type==RECT_DRAW)
    {
        int x0,x1,y0,y1;
        while(iter1&&iter2)
        {
            x0=(((_2d_point*)(iter1->data))->x)*area_w/width+area_w/2;
            y0=(((_2d_point*)(iter1->data))->y)*area_h/height+area_h/2;
            x1=(((_2d_point*)(iter2->data))->x)*area_w/width+area_w/2;
            y1=(((_2d_point*)(iter2->data))->y)*area_h/height+area_h/2;   
            int h=abs(y1-y0);
            int w=abs(x1-x0);
            x0=x0<x1?x0:x1;
            y0=y0<y1?y0:y1;   
            gdk_draw_rectangle(GTK_WIDGET(data)->window,GTK_WIDGET(data)->style->
                    fg_gc[GTK_WIDGET_STATE (data)],FALSE,x0,y0,w,h);
            iter1=iter2->next;
            if(iter1)
            {
                iter2=iter1->next;
            }
        }
    }
    if(draw_type==POLY_DRAW)
    {
        int x0,x1,y0,y1;
        while(iter2)
        {
            x0=(((_2d_point*)(iter1->data))->x)*area_w/width+area_w/2;
            y0=(((_2d_point*)(iter1->data))->y)*area_h/height+area_h/2;
            x1=(((_2d_point*)(iter2->data))->x)*area_w/width+area_w/2;
            y1=(((_2d_point*)(iter2->data))->y)*area_h/height+area_h/2;               
            gdk_draw_line(GTK_WIDGET(data)->window,GTK_WIDGET(data)->style->
                    fg_gc[GTK_WIDGET_STATE (data)],x0,y0,x1,y1);
            iter1=iter2;
            iter2=iter1->next;
        }
        x0=(((_2d_point*)(iter1->data))->x)*area_w/width+area_w/2;
        y0=(((_2d_point*)(iter1->data))->y)*area_h/height+area_h/2;
        x1=(((_2d_point*)(prt->data))->x)*area_w/width+area_w/2;
        y1=(((_2d_point*)(prt->data))->y)*area_h/height+area_h/2;
        gdk_draw_line(GTK_WIDGET(data)->window,GTK_WIDGET(data)->style->
                fg_gc[GTK_WIDGET_STATE (data)],x0,y0,x1,y1);
    }
}
/*工具栏上放大按钮的回调函数,实现将对象放大显示(通过将全局变量width和height的值改变,并修改绘图区显示),
 * 输入参数:工具栏按钮本身enlarge和绘图区构件data,
 * 输出参数:无,
 * 日期:2007-4-7修改*/
void enlarge_obj(GtkWidget *enlarge,GtkWidget *data)
{
    width=width/1.2;
    height=height/1.2;
    draw_prt(data);
}
/*工具栏上缩小按钮的回调函数,实现将对象缩小显示(通过将全局变量width和height的值改变,并修改绘图区显示),
 * 输入参数:工具栏按钮本身deflate和绘图区构件data,
 * 输出参数:无,
 * 日期:2007-4-7修改*/
void deflate_obj(GtkWidget *deflate,GtkWidget *data)
{
    width=width*1.2;
    height=height*1.2;
    draw_prt(data);
}
/*菜单Draw下的Plygon菜单项的回调函数,实现将全局变量draw_type属性为点,并将存放数据信息的链表清空。
 * 输入参数:无
 * 输出参数:无
 * 时间:2007-4-7修改*/
void draw_pix(void)
{
    //清空绘图区
    //设置图象类型
    GSList *iter1=prt;
    while(iter1)
    {
        prt=g_slist_remove_link(prt,iter1);
        g_slist_free(iter1);
        iter1=prt;
    }
    draw_type=PIX_DRAW;

}
/*菜单Draw下的Plygon菜单项的回调函数,实现将全局变量draw_type属性为线,并将存放数据信息的链表清空。
 * 输入参数:无
 * 输出参数:无
 * 时间:2007-4-7修改*/
void draw_line(void)
{
    // 清空绘图区
    GSList *iter1=prt;
        while(iter1)
    {
        prt=g_slist_remove_link(prt,iter1);
        g_slist_free(iter1);
        iter1=prt;
    }
    draw_type=LINE_DRAW;
}
/*菜单Draw下的Plygon菜单项的回调函数,实现将全局变量draw_type属性为圆,并将存放数据信息的链表清空。
 * 输入参数:无
 * 输出参数:无
 * 时间:2007-4-7修改*/
void draw_circle(void)
{
    //
    GSList *iter1=prt;
    while(iter1)
    {
        prt=g_slist_remove_link(prt,iter1);
        g_slist_free(iter1);
        iter1=prt;
    }
    draw_type=CIRCLE_DRAW;
}
/*菜单Draw下的Plygon菜单项的回调函数,实现将全局变量draw_type属性为矩形,并将存放数据信息的链表清空。
 * 输入参数:无
 * 输出参数:无
 * 时间:2007-4-7修改*/
void draw_rect(void)
{
    //
    GSList *iter1=prt;
    while(iter1)
    {
        prt=g_slist_remove_link(prt,iter1);
        g_slist_free(iter1);
        iter1=prt;
    }
    draw_type=RECT_DRAW;
}
/*菜单Draw下的Plygon菜单项的回调函数,实现将全局变量draw_type属性为多边形,并将存放数据信息的链表清空。
 * 输入参数:无
 * 输出参数:无
 * 时间:2007-4-7修改*/
void draw_poly(void)
{
    //
    GSList *iter1=prt;
    while(iter1)
    {
        prt=g_slist_remove_link(prt,iter1);
        g_slist_free(iter1);
        iter1=prt;
    }
    draw_type=POLY_DRAW;
}
/*创建File菜单
 * 输入参数:包含树视图和绘图区的结构体td,
 * 输出参数:菜单项file_item,
 * 日期:2007-4-7修改*/
GtkWidget* create_file_item(tree_drawing *td)
{
    GtkWidget *file_menu;    //定义菜单
    GtkWidget *file_item;
    GtkWidget *new_file,*import_item,*export_item,*quit_item;    //file 菜单下的菜单项
    char *path=malloc(100*sizeof(char));
    //-------------------创建file菜单----------------------------------
    file_menu=gtk_menu_new();                //创建菜单
        /* 创建菜单项 */
    new_file=gtk_menu_item_new_with_label("New");
        import_item = gtk_menu_item_new_with_label ("Import");
        export_item = gtk_menu_item_new_with_label ("Export");
        quit_item = gtk_menu_item_new_with_label ("Quit");
   
        /* 将它们加到菜单中 */
    gtk_menu_append(GTK_MENU(file_menu),new_file);
        gtk_menu_append (GTK_MENU (file_menu),import_item);
        gtk_menu_append (GTK_MENU (file_menu),export_item);
        gtk_menu_append (GTK_MENU (file_menu),quit_item);   
   
        /* 将回调函数绑定到activate信号 */
    g_signal_connect_swapped(G_OBJECT(new_file),"activate",G_CALLBACK(create_file),td->tree_view);
         g_signal_connect_swapped (G_OBJECT (import_item), "activate",G_CALLBACK (import_file),td);
         g_signal_connect_swapped (G_OBJECT (export_item), "activate", G_CALLBACK (export_file),td->tree_view);   
        g_signal_connect_swapped (G_OBJECT (quit_item), "activate",G_CALLBACK (gtk_main_quit),NULL);

    file_item = gtk_menu_item_new_with_label ("File");
    //gtk_widget_set_size_request(GTK_WIDGET(file_item),50,20);
    gtk_menu_item_set_submenu (GTK_MENU_ITEM (file_item), file_menu);

    return file_item;
}
/*创建菜单Draw
 * 输入参数:无
 * 输出参数:菜单项:draw_menu
 * 日期:2007-4-7修改*/
GtkWidget* create_draw_item(void)
{
    GtkWidget *draw_menu;    //定义菜单
    GtkWidget *draw_item;
    GtkWidget *circle_draw,*rect_draw,*polygon_draw,*line_draw,*pixel_draw;

    draw_menu=gtk_menu_new();
    pixel_draw=gtk_menu_item_new_with_label("Pixel");
    line_draw=gtk_menu_item_new_with_label("Line");
    circle_draw=gtk_menu_item_new_with_label("Circle");
    rect_draw=gtk_menu_item_new_with_label("Rectangle");
    polygon_draw=gtk_menu_item_new_with_label("Polygon");

    gtk_menu_append(GTK_MENU(draw_menu),pixel_draw);
    gtk_menu_append(GTK_MENU(draw_menu),line_draw);
    gtk_menu_append(GTK_MENU(draw_menu),circle_draw);
    gtk_menu_append(GTK_MENU(draw_menu),rect_draw);
    gtk_menu_append(GTK_MENU(draw_menu),polygon_draw);

    g_signal_connect_swapped(G_OBJECT(pixel_draw),"activate",G_CALLBACK(draw_pix),NULL);
    g_signal_connect_swapped(G_OBJECT(line_draw),"activate",G_CALLBACK(draw_line),NULL);
    g_signal_connect_swapped(G_OBJECT(circle_draw),"activate",G_CALLBACK(draw_circle),NULL);
    g_signal_connect_swapped(G_OBJECT(rect_draw),"activate",G_CALLBACK(draw_rect),NULL);
    g_signal_connect_swapped(G_OBJECT(polygon_draw),"activate",G_CALLBACK(draw_poly),NULL);

    draw_item=gtk_menu_item_new_with_label("Draw");
    //gtk_widget_set_size_request(GTK_WIDGET(draw_item),50,20);
    gtk_menu_item_set_submenu(GTK_MENU_ITEM(draw_item),draw_menu);

    return draw_item;
}
/*创建工具栏,由于在工具栏中的按钮的回调函数需要修改树视图和绘图区,
 * 所以将树视图和绘图区作为参数传递进来,以便回调函数参数传递
 * 输入参数:包含绘图区和树视图的结构体td
 * 输出参数:工具栏构件handlebox
 * 时间:2007-4-7修改*/
GtkWidget* create_toolbar(tree_drawing *td)
{
    GtkWidget *handlebox;
    GtkWidget *toolbar;
    GtkWidget *new_button,*import_button,*export_button,*quit_button;//工具按钮
    GtkWidget *zoom_out,*zoom_in;
    GtkWidget * iconw;


    handlebox=gtk_handle_box_new();
    toolbar=gtk_toolbar_new();    //设置工具栏格式
    gtk_toolbar_set_orientation (GTK_TOOLBAR (toolbar), GTK_ORIENTATION_HORIZONTAL);
    gtk_toolbar_set_style (GTK_TOOLBAR (toolbar), GTK_TOOLBAR_BOTH);
    gtk_container_set_border_width (GTK_CONTAINER (toolbar), 5);

    new_button=gtk_toolbar_append_item(GTK_TOOLBAR(toolbar),"New","Create a file","Private",NULL,GTK_SIGNAL_FUNC(create_file),td->tree_view);
    import_button=gtk_toolbar_append_item(GTK_TOOLBAR(toolbar),"Import",
            "Import file","Private",NULL,GTK_SIGNAL_FUNC(import_file),td);   
    export_button=gtk_toolbar_append_item(GTK_TOOLBAR(toolbar),"Export",
            "Export file","Private",NULL,GTK_SIGNAL_FUNC(export_file),td->tree_view);   
    iconw=gtk_image_new_from_file("enlarge.png");
    zoom_out=gtk_toolbar_append_item(GTK_TOOLBAR(toolbar),NULL,"zoom out","Private",iconw,GTK_SIGNAL_FUNC(enlarge_obj),td->draw_area);
    gtk_toolbar_append_space(GTK_TOOLBAR(toolbar));

    iconw=gtk_image_new_from_file("deflate.png");
    zoom_in=gtk_toolbar_append_item(GTK_TOOLBAR(toolbar),NULL,"zoom in","Private",iconw,GTK_SIGNAL_FUNC(deflate_obj),td->draw_area);
    quit_button=gtk_toolbar_append_item(GTK_TOOLBAR(toolbar),"Quit",
            "Quit the file","Private",NULL,GTK_SIGNAL_FUNC(gtk_main_quit),NULL);   
    gtk_container_add(GTK_CONTAINER(handlebox),toolbar);

    return handlebox;
}
/*在绘图区中点击鼠标事件的回调函数,
 * 如果点击左键则按照绘图类型将鼠标处的坐标值处理后添加到链表prt中,并调用显示函数,显示图形;
 * 如果是鼠标右键,则将绘图类型设置为FALSE
 * 输入参数:事件盒构件本身event_box,事件:event,绘图区构件:data
 * 输出参数:TURE活FALSE,这个暂时没实际意义
 * 时间:2007-4-7修改*/
static gboolean button_press_callback(GtkWidget *event_box,GdkEventButton *event,GtkWidget *data)
{
   
    if(!draw_type)
    {
        return FALSE;
    }
    /*
if(event->button==2)
{
    gtk_menu_popup()
}   
*/
    if(event->button==3)
    {
        draw_type=FALSE;
    }
    int area_w=100,area_h=100;
    area_w=data->allocation.width;            //获得窗口尺寸
    area_h=data->allocation.height;
    if(event->button==1)
    {
        _2d_point *p=malloc(sizeof(_2d_point));
        p->x=((event->x-area_w/2)*width/area_w);
        p->y=((event->y-area_h/2)*height/area_h);
        prt=g_slist_append(prt,p);
        //printf("%f %f\t,%d,%d\n",p->x,p->y,area_w,area_h);
        GSList *iter1=NULL,*iter2=NULL;
        iter1=prt;
        iter2=prt->next;
        gdk_draw_point(GTK_WIDGET(data)->window,GTK_WIDGET(data)->style->
                fg_gc[GTK_WIDGET_STATE (event_box)],event->x,event->y);
        draw_prt(data);   
    }
    //gdk_drawable_unref(GTK_WIDGET(data)->window);
    return TRUE;
}
/*
static gboolean view_press(GtkWidget *widget,GdkEventButton *event,GtkWidget *menu)
{
      if(event->button==3)
    {
        gtk_menu_popup(GTK_MENU(widget),NULL,NULL,NULL,NULL,event->button,event->time);
    }
}*/
static gboolean
expose_event(GtkWidget *widget, GdkEventExpose *event,gpointer data)
{
    draw_prt(widget);
    return TRUE;
}
gint main(gint argc,char *argv[])
{
    GtkWidget *window;
    GtkWidget *file_item,*draw_item;    /*菜单项*/
    GtkWidget *menu_bar;            /*菜单*/
    GtkWidget *tool_bar;            /*工具栏*/
    GtkWidget *vbox,*hpaned;        /*竖直盒,水平分栏窗口*/
    GtkWidget *view,*event_box,*draw_area;
    GtkWidget *handle_box;            /*手柄构建,用于存放工具按钮*/

    gtk_init(&argc,&argv);
   
    window=gtk_window_new(GTK_WINDOW_TOPLEVEL);
    gtk_window_set_title(GTK_WINDOW(window),"drawing window");
    g_signal_connect(G_OBJECT(window),"destroy",G_CALLBACK(gtk_main_quit),NULL);
    gtk_widget_set_size_request(GTK_WIDGET(window),500,400);
    gtk_container_set_border_width (GTK_CONTAINER (window), 10);
    vbox = gtk_vbox_new (FALSE, 0);            /*创建主竖直盒*/
    gtk_container_add (GTK_CONTAINER (window), vbox);
    gtk_window_set_position(GTK_WINDOW(window),GTK_WIN_POS_CENTER);/*将窗口的显示位置设定为屏幕中央*/

    draw_area=gtk_drawing_area_new();/*由于这两个构件需要作为其它构件回调函数的参数,所以先在这里创建*/
    view=(GtkWidget*)create_tree();
    tree_drawing *td=malloc(sizeof(tree_drawing));/*定义结构体,以便回调函数调用*/
    td->tree_view=view;
    td->draw_area=draw_area;
    /*======================== 添加菜单 =====================================    */
    menu_bar = gtk_menu_bar_new ();
    gtk_box_pack_start(GTK_BOX(vbox),menu_bar,FALSE,FALSE,3);
    file_item=create_file_item(td);                //调用自定义创建菜单file的函数
    gtk_menu_bar_append (GTK_MENU_BAR (menu_bar), file_item);
    draw_item=create_draw_item();
    gtk_menu_bar_append(GTK_MENU_BAR(menu_bar),draw_item);
    /*
    GtkWidget *pop_up;
    pop_up=pop_menu(draw_area);
    */
    /*========================= 添加工具栏 =================================*/
    handle_box=create_toolbar(td);            /*调用自定义函数创建工具栏,由于工具栏中的按钮需要改变绘图区和树视图,所以需要将绘图区和树视图作为参数进行传递*/
    gtk_box_pack_start(GTK_BOX(vbox),handle_box,FALSE,FALSE,3);

    /*========================= 添加树视图 ==========================================*/
    hpaned=gtk_hpaned_new();            /*创建水平分栏窗口*/
    gtk_box_pack_start(GTK_BOX(vbox),hpaned,TRUE,TRUE,3);
    //g_signal_connect_swapped(G_OBJECT(view),"event",G_CALLBACK(view_press),pop_up);//添加右键菜单

    gtk_paned_add1(GTK_PANED(hpaned),view);

    /*========================== 添加绘图区 =========================================*/
    event_box=gtk_event_box_new();            //添加eventbox以便鼠标响应
    gtk_paned_add2(GTK_PANED(hpaned),event_box);
    gtk_container_add(GTK_CONTAINER(event_box),draw_area);
   
    /*-----------test - color------*/
    GdkColor color;
    color.red=55000;
    color.blue=55000;
    color.green=55000;
    gtk_widget_modify_bg(draw_area,GTK_STATE_NORMAL,&color);    /*设置背景颜色*/
    GdkColor colorf;
    colorf.red=0;
    colorf.blue=65535;
    colorf.green=0;
    gtk_widget_modify_fg(draw_area,GTK_STATE_NORMAL,&colorf);    /*设置前景颜色*/
    g_signal_connect(G_OBJECT(event_box),"button_press_event",G_CALLBACK(button_press_callback),draw_area);
    g_signal_connect(G_OBJECT(draw_area),"expose_event",G_CALLBACK(expose_event),NULL);
    //-----------------------------

    gtk_widget_show_all(window);
    width=draw_area->allocation.width;            /*获得绘图区尺寸,并赋值给全局变量width和height*/
    height=draw_area->allocation.height;   
    gtk_main();

    return 0;
}




下面是编译命令:

$> gcc -o main `pkg-config --cflags --libs gtk+-2.0 glib-2.0` soft.c

编译完成之后将图标enlarge.png和deflate.png拷贝到同一目录下即可执行了。界面如下:

一个五脏俱全的Gtk小程序_第1张图片

关于使用方面的这里介绍一点,打开后,点击菜单Draw,然后选中对应类型就可以在绘图区绘制了,也可以导入对象,导入文件的格式全部为.124格式( :),自己随便定义的格式),若想保存,应该首先新建,然后绘制,绘制完毕之后,选中树视图中的对象名,点击export工具图标即可导出,导入时只需输入文件名,不需要扩展名(切记),在绘图时不要点击右键,右键将清空绘图。当然了,放大/缩小图标就是用来放大缩小用的。

 

你可能感兴趣的:(一个五脏俱全的Gtk小程序)