更新:我相信这不是特定于 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 中,您可以右键单击并单击“退出”),您会看到无法再在文本框中输入内容。