-1

更新:我相信这不是特定于 GtkEntry,而是键盘输入本身没有被发送到程序(不再触发键事件)。

我正在开发一个显示管理器。我正在 Arch Linux 上进行测试。这个想法是,当用户输入他们的凭据并按 Enter 键时,显示管理器会分叉并启动他们在~/.xinitrc. 我正在使用 Awesome WM 作为窗口管理器进行测试。当用户退出窗口管理器时,显示管理器应该重新获得控制权,以便用户可以再次登录。

问题是用户退出窗口管理器后,显示管理器显示,但我不能再输入 GtkEntry 框。我可以突出显示文本,甚至可以右键单击并剪切和粘贴,但我无法输入。我在下面做了一个简化的例子。请注意,如果我不启动窗口管理器,而是启动类似sleep命令的东西,那么一切正常。

#include <stdbool.h>
#include <stdlib.h>
#include <string.h>
#include <sys/wait.h>
#include <unistd.h>

#include <gtk/gtk.h>

#define ENTER_KEY    65293

static pthread_t login_thread;

static void* login_func(void *data) {
    pid_t child_pid = fork();
    if (child_pid == 0) {
        execl("/bin/bash", "/bin/bash", "--login", "/home/gulshan/.xinitrc", NULL); // This doesn't work
        // execl("/usr/bin/awesome", "/usr/bin/awesome", NULL); // This doesn't work
        // execl("/usr/bin/sleep", "/usr/bin/sleep", "5", NULL); // This works
        printf("exec error");
        exit(1);
    }

    // Wait for child process to finish
    int status;
    waitpid(child_pid, &status, 0);

    printf("Returning from login_func\n");
    return NULL;
}

static gboolean key_event(GtkWidget *widget, GdkEventKey *event) {
    if (event->keyval == ENTER_KEY) {
        pthread_create(&login_thread, NULL, login_func, (void*) widget);
    }
    return FALSE;
}

int main(int argc, char *argv[]) {
    gtk_init(&argc, &argv);

    GtkWidget *window = gtk_window_new(GTK_WINDOW_TOPLEVEL);

    gtk_container_add(GTK_CONTAINER(window), gtk_entry_new());
    gtk_widget_show_all(window);

    g_signal_connect(window, "key-release-event", G_CALLBACK(key_event), NULL);
    g_signal_connect(window, "destroy", G_CALLBACK(gtk_main_quit), NULL);
    gtk_main();

    return 0;
}

这是我编译代码的方式:

gcc `pkg-config --cflags --libs gtk+-3.0` -Wall -o test test.c

为了测试这段代码,我首先运行:

Xephyr -ac -br -noreset -screen 800x600 :1

这会在我可用于测试的窗口中启动 X 服务器。然后我开始代码

DISPLAY=:1 ./test

在 Xephyr 窗口中启动它。如果您退出显示管理器(在 Awesome 中,您可以右键单击并单击“退出”),您会看到无法再在文本框中输入内容。

4

1 回答 1

0

我认为你不应该混合线程和分叉查看线程和分叉思考两次之前使用它们

我已经像这样修改了你的代码:

#include <stdbool.h>
#include <stdlib.h>
#include <string.h>
#include <sys/wait.h>
#include <unistd.h>

#include <gtk/gtk.h>

#define ENTER_KEY    65293

static pthread_t login_thread;

static void* login_func(void *data) {
    pid_t child_pid = fork();
    if (child_pid == 0) {
        execl("/usr/bin/awesome", "/usr/bin/awesome", NULL); // This doesn't work
        printf("exec error");
        exit(1);
    }

    // Wait for child process to finish
    int status;
    waitpid(child_pid, &status, 0);

    printf("Returning from login_func\n");
    return NULL;
}

static gboolean key_event(GtkWidget *widget, GdkEventKey *event) {
    if (event->keyval == ENTER_KEY) {
      login_func((void *) widget);
    }
    return FALSE;
}

int main(int argc, char *argv[]) {
    gtk_init(&argc, &argv);

    GtkWidget *window = gtk_window_new(GTK_WINDOW_TOPLEVEL);

    gtk_container_add(GTK_CONTAINER(window), gtk_entry_new());
    gtk_widget_show_all(window);

    g_signal_connect(window, "key-release-event", G_CALLBACK(key_event), NULL);
    g_signal_connect(window, "destroy", G_CALLBACK(gtk_main_quit), NULL);
    gtk_main();

    return 0;
}

如您所见,我刚刚删除了线程。现在,当我从 Awesome 注销时,我可以输入 gtk 条目。

于 2015-02-07T20:38:05.553 回答