使用pthread时无法打开新窗口

时间:2019-05-20 16:36:18

标签: c multithreading pthreads gtk glade

我正在尝试制作一个聊天应用程序(IRC),并且在使用gui时,我无法显示新窗口并无法同时运行客户端,因此服务器可以在控制台上完美运行。如果我尝试在没有服务器的情况下运行新窗口,那么它将正常运行。我想打开一个新窗口并运行客户端,以便它可以在最近打开的窗口上创建文本。

void on_login_clicked(int argc, char *argv[])
{ 
    char userlogin[20];
    char userpass[20];
    sprintf(userlogin,"%s",user_text );
    sprintf(userpass,"%s",pass_text );

    printf ("Entry contents: %s\n", userlogin);
    printf ("Entry contents: %s\n", userpass);

    struct userData *acc = malloc(sizeof(struct userData));
    char ipadd[] = "0.0.0.0";

    acc = login(userlogin, userpass);
    if(acc == NULL)
        errx(1,"Your account does not exist!");


    system("clear");

    struct sentDATA *data = malloc(sizeof(struct sentDATA));
    data->acc = acc;
    data->adressIP = ipadd;


    gtkstartnewwindow(argc,argv, data);
}

void gtkstartnewwindow(int argc, char *argv[],struct sentDATA *data)
{


    GtkWidget *window;
    GtkWidget *mainBox;
    GtkWidget *grid;
    GtkWidget *topBox;
    GtkWidget *botBox;
    GtkWidget *header;

    GtkWidget *name;
    GtkWidget *entry;   
    GtkWidget *message;
    GtkWidget *connectBtn;
    GtkWidget *imgBtn;
    GtkWidget *callBtn;

    connected = FALSE;

    gtk_init(&argc, &argv);

    window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
    gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
    gtk_widget_set_size_request (window, 400, 500);

    gtk_window_set_title(GTK_WINDOW(window), "Chat client");

    mainBox = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0);

    header = gtk_header_bar_new();

    name = gtk_label_new("");
    gtk_header_bar_set_custom_title(GTK_HEADER_BAR(header), name);

    connectBtn = gtk_button_new_with_label("Connect");
    gtk_widget_set_size_request(connectBtn, 70, 30);
    imgBtn = gtk_button_new_with_label("Send an image");
    gtk_widget_set_size_request(imgBtn, 70, 30);
    callBtn = gtk_button_new_with_label("Call someone");
    gtk_widget_set_size_request(callBtn, 70, 30);
    gtk_header_bar_pack_end(GTK_HEADER_BAR(header), imgBtn);
    gtk_header_bar_pack_start(GTK_HEADER_BAR(header), connectBtn);

    gtk_box_pack_start(GTK_BOX(mainBox), header, FALSE, FALSE, 0);

    grid = gtk_grid_new();
    gtk_grid_set_column_spacing (GTK_GRID(grid), 15);
    gtk_grid_set_row_spacing(GTK_GRID(grid), 5);

    gtk_container_set_border_width(GTK_CONTAINER(grid), 15);

    topBox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 3);
    gtk_widget_set_hexpand(topBox, TRUE);


    chat = gtk_text_view_new();
    gtk_text_view_set_editable(GTK_TEXT_VIEW(chat), FALSE);
    gtk_text_view_set_cursor_visible(GTK_TEXT_VIEW(chat), FALSE);
    gtk_widget_set_vexpand (chat, TRUE);
    gtk_widget_set_hexpand (chat, TRUE);
    gtk_grid_attach(GTK_GRID(grid), chat, 0, 0, 1, 1);


    botBox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 3);
    message = gtk_text_view_new();
    gtk_widget_set_size_request(message, 70, 30);
    gtk_widget_set_hexpand (message, TRUE); 


    gtk_grid_attach(GTK_GRID(grid), botBox, 0, 2, 2, 1);
    gtk_widget_set_vexpand (grid, TRUE);
    gtk_widget_set_hexpand (grid, TRUE);
    gtk_widget_set_halign (grid, GTK_ALIGN_FILL);
    gtk_widget_set_valign (grid, GTK_ALIGN_FILL);

    gtk_box_pack_start(GTK_BOX(mainBox), grid, TRUE, TRUE, 0);
    gtk_container_add(GTK_CONTAINER(window), mainBox);

         entry = gtk_entry_new ();
    gtk_entry_set_max_length (GTK_ENTRY (entry), 50);
    g_signal_connect (entry, "activate",G_CALLBACK (enter_callback), entry);
    gtk_entry_set_text (GTK_ENTRY (entry), "hello");    
    g_signal_connect(imgBtn, "clicked", G_CALLBACK (on_open_image), NULL);
    gtk_grid_attach(GTK_GRID(grid), entry, 0 , 1, 2, 1);

    //g_signal_connect(G_OBJECT(window), "destroy", G_CALLBACK(gtk_main_quit), G_OBJECT(window));
    gboolean runtime = TRUE;
    g_signal_connect(G_OBJECT(window), "destroy", G_CALLBACK(exit_app), &runtime);

    gtk_widget_show_all(window);

    g_message("GUI started");


    while (gtk_main_iteration_do(FALSE)) {
        if (!runtime)
            break;
        //other callback handling

        //this loop needs to be running infinitely, 
        //if you need to wait in your program anywhere, 
        //(and it cannot be done only once before the loop)
        //we will need to make it into threads

    }

    pthread_t thr1;
    pthread_create(&thr1,NULL,launchClient,data);   
    pthread_join(thr1,NULL);
}



void recieved_text (gchar *m) {
    GtkTextIter e;

    GtkTextBuffer *chatBuf = gtk_text_view_get_buffer(GTK_TEXT_VIEW(chat));
    gtk_text_buffer_get_end_iter(chatBuf, &e);
    gtk_text_buffer_insert(chatBuf, &e, m, -1);
    gtk_text_buffer_insert(chatBuf, &e, "\n", 1);
} 


launchClient也有几个pthread。 有什么办法可以同时运行launchClient和新窗口? 此外,要从launchClient运行recieved_text吗?

谢谢

1 个答案:

答案 0 :(得分:0)

在GTK中,您只能从主线程中仅 更新GUI(即与小部件相关的API)。您可以从其他线程中执行任何操作,但是每当需要更新用户界面时,都应通知主线程,并且仅在其中更新GUI。

通知步骤非常容易执行,因为底层的GLib API是线程感知的,即GLib在请求时负责锁定/解锁内部数据。有很多方法可以做到:我所知道的最简单的方法是使用g_idle_addgdk_threads_add_idle只是出于传统原因而保留的包装)。

this answer中,我提供了一个基本示例,其中包含100个并发线程,它们可以更新单个窗口小部件。