2009-11-17 2 views
11

Я разработал одно всплывающее окно (без декорации), используя GTK + и инструмент для просеивания в C. Это всплывающее окно в его родительском окне при нажатии кнопки. Я хочу уничтожить или скрыть это всплывающее окно, когда пользователь выберет это окно. Пользователь может щелкнуть по родительскому окну или любому другому окну. Я попытался захватить событие GDK_FOCUS_CHANGE, но я не могу зафиксировать это событие. Есть ли способ достичь этого? Как узнать, что щелчок находится в другом окне, а затем всплывающее окно? Как ясно, что всплывающее окно потеряло фокус? Чтобы я мог скрыть это. Соответствующий код выглядит следующим образом:Как скрыть всплывающее окно Gtk, когда пользователь нажимает за окном

/* 
* Compile me with: 

gcc -o popup popup.c $(pkg-config --cflags --libs gtk+-2.0 gmodule-2.0) 
*/ 

#include <gtk/gtk.h> 

static void on_popup_clicked (GtkButton*, GtkWidget*); 
static gboolean on_popup_window_event(GtkWidget*, GdkEventExpose*); 

int main (int argc, char *argv[]) 
{ 
    GtkWidget *window, *button, *vbox; 

    gtk_init (&argc, &argv); 

    window = gtk_window_new (GTK_WINDOW_TOPLEVEL); 
    gtk_window_set_title (GTK_WINDOW (window), "Parent window"); 
    gtk_container_set_border_width (GTK_CONTAINER (window), 10); 
    gtk_widget_set_size_request (window, 300, 300); 
    gtk_window_set_position (GTK_WINDOW (window),GTK_WIN_POS_CENTER); 

    button = gtk_button_new_with_label("Pop Up"); 
    g_signal_connect (G_OBJECT (button), "clicked",G_CALLBACK (on_popup_clicked),(gpointer) window); 

    vbox = gtk_vbox_new (FALSE, 3); 
    gtk_box_pack_end(GTK_BOX (vbox), button, FALSE, FALSE, 5); 
    gtk_container_add (GTK_CONTAINER (window), vbox); 

    gtk_widget_show_all (window); 
    gtk_main(); 
    return 0; 
} 

void on_popup_clicked (GtkButton* button, GtkWidget* pWindow) 
{ 
    GtkWidget *popup_window; 
    popup_window = gtk_window_new (GTK_WINDOW_POPUP); 
    gtk_window_set_title (GTK_WINDOW (popup_window), "Pop Up window"); 
    gtk_container_set_border_width (GTK_CONTAINER (popup_window), 10); 
    gtk_window_set_resizable(GTK_WINDOW (popup_window), FALSE); 
    gtk_window_set_decorated(GTK_WINDOW (popup_window), FALSE); 
    gtk_widget_set_size_request (popup_window, 150, 150); 
    gtk_window_set_transient_for(GTK_WINDOW (popup_window),GTK_WINDOW (pWindow)); 
    gtk_window_set_position (GTK_WINDOW (popup_window),GTK_WIN_POS_CENTER); 
    g_signal_connect (G_OBJECT (button), "event", 
         G_CALLBACK (on_popup_window_event),NULL); 

    GdkColor color; 
    gdk_color_parse("#3b3131", &color); 
    gtk_widget_modify_bg(GTK_WIDGET(popup_window), GTK_STATE_NORMAL, &color); 


    gtk_widget_show_all (popup_window); 
} 

gboolean on_popup_window_event(GtkWidget *popup_window, GdkEventExpose *event) 
{ 
    if(event->type == GDK_FOCUS_CHANGE) 
     gtk_widget_hide (popup_window); 

    return FALSE; 
} 

Здесь я не в состоянии скрыть это всплывающее окно, когда пользователь нажимает на родительском окне или на другом окне. Как я могу это сделать?

Я должен придерживаться версии Gtk + 2.14.

+0

Пожалуйста, не повторяйте вопросы. Это, по-видимому, повторение (лучше представленное) http://stackoverflow.com/questions/1740947/problem-with-gtk-popup-window-focus-change-event-handling – mlibby

+0

Я позабочусь об этом в будущем. – kbalar

ответ

8

Изменения:

  • переключатель от GTK_WINDOW_POPUP к GTK_WINDOW_TOPLEVEL, нелогичным, но я не мог понять, как получить всплывающее окно, чтобы принять фокус.
  • добавить gtk_window подсказки, чтобы предотвратить всплывающие окна от показа в панели задач и пейджера
  • намеренно установить фокус на всплывающем окне
  • установить GDK_FOCUS_CHANGE_MASK на GDK_WINDOW с gtk_widget_set_events (требуется для следующего шага)
  • подключения к focus-out-event всплывающего окна
  • изменить обработчик сигнала для обработки другого сигнала

I woul d также предлагает прочитать источник GTK +, чтобы увидеть, как он обрабатывает всплывающие окна для всплывающих подсказок и меню, когда они отображаются ... но они обычно уничтожаются на основе мыши, выходящей из диапазона, а не всплывающего фокуса, как такового.


#include 

static void on_popup_clicked (GtkButton*, GtkWidget*); 
gboolean on_popup_focus_out (GtkWidget*, GdkEventFocus*, gpointer); 

int 
main (int argc, char *argv[]) 
{ 
    GtkWidget *window, *button, *vbox; 

    gtk_init (&argc, &argv); 

    window = gtk_window_new (GTK_WINDOW_TOPLEVEL); 
    gtk_window_set_title (GTK_WINDOW (window), "Parent window"); 
    gtk_container_set_border_width (GTK_CONTAINER (window), 10); 
    gtk_widget_set_size_request (window, 300, 300); 
    gtk_window_set_position (GTK_WINDOW (window), GTK_WIN_POS_CENTER); 

    button = gtk_button_new_with_label ("Pop Up"); 
    g_signal_connect (G_OBJECT (button), 
        "clicked", 
        G_CALLBACK (on_popup_clicked), 
        (gpointer) window); 

    vbox = gtk_vbox_new (FALSE, 3); 
    gtk_box_pack_end (GTK_BOX (vbox), button, FALSE, FALSE, 5); 
    gtk_container_add (GTK_CONTAINER (window), vbox); 

    gtk_widget_show_all (window); 
    gtk_main(); 
    return 0; 
} 

void 
on_popup_clicked (GtkButton* button, GtkWidget* pWindow) 
{ 
    GtkWidget *popup_window; 

    popup_window = gtk_window_new (GTK_WINDOW_TOPLEVEL); 
    gtk_window_set_title (GTK_WINDOW (popup_window), "Pop Up window"); 
    gtk_container_set_border_width (GTK_CONTAINER (popup_window), 10); 
    gtk_window_set_resizable (GTK_WINDOW (popup_window), FALSE); 
    gtk_window_set_decorated (GTK_WINDOW (popup_window), FALSE); 
    gtk_window_set_skip_taskbar_hint (GTK_WINDOW (popup_window), TRUE); 
    gtk_window_set_skip_pager_hint (GTK_WINDOW (popup_window), TRUE); 
    gtk_widget_set_size_request (popup_window, 150, 150); 
    gtk_window_set_transient_for (GTK_WINDOW (popup_window), GTK_WINDOW (pWindow)); 
    gtk_window_set_position (GTK_WINDOW (popup_window), GTK_WIN_POS_CENTER); 

    gtk_widget_set_events (popup_window, GDK_FOCUS_CHANGE_MASK); 
    g_signal_connect (G_OBJECT (popup_window), 
        "focus-out-event", 
        G_CALLBACK (on_popup_focus_out), 
        NULL); 

    GdkColor color; 
    gdk_color_parse ("#3b3131", &color); 
    gtk_widget_modify_bg (GTK_WIDGET (popup_window), GTK_STATE_NORMAL, &color); 

    gtk_widget_show_all (popup_window); 
    gtk_widget_grab_focus (popup_window); 
} 

gboolean 
on_popup_focus_out (GtkWidget *widget, 
        GdkEventFocus *event, 
        gpointer data) 
{ 
    gtk_widget_destroy (widget); 
    return TRUE; 
} 
+0

Получил работу! Спасибо, что посмотрели на это. – kbalar

+0

Хорошая сделка. Если этот ответ работает на вас, не стесняйтесь щелкнуть большой галочкой рядом с ответом.:) – mlibby

+0

Отличное решение, но есть одна проблема: главное окно визуально регистрирует изменения фокуса, окрашивая границу по-другому (Windows 7). Это не должно происходить для всплывающего окна (поддельного). Любая идея предотвратить это? – schlamar

2

Вам не нужно устанавливать фокус клавиатуры во всплывающее окно.

Вам просто нужно, чтобы захватить мыши к popup_window->window с помощью gdk_pointer_grab(...) с Истинными owner_events и GDK_BUTTON_PRESS_MASK GdkEventMask аргументов.

Затем подключите свой popup_window к "button-press-event". Внутри его обработчик скрывает/уничтожает ваш popup_window и снимает захват с использованием gdk_pointer_ungrab(...), если * координаты событий отрицательны или выше вашего размера popup_window.

1

Другой альтернативой является простое добавление кнопки для прослушивания в родительском окне. Это имеет то преимущество, что всплывающее окно по-прежнему выглядит как всплывающее окно (как родительский, так и сам может быть активным сразу)

#include <stdio.h> 
#include <gtk/gtk.h> 

static void on_popup_clicked (GtkButton*, GtkWidget*); 

gulong handler_id; 

gboolean 
on_click (GtkWidget *widget, 
       GdkEvent *event, 
       gpointer user_data) 
{ 
    g_signal_handler_disconnect (widget, handler_id); 
    gtk_widget_destroy (user_data); 
    return TRUE; 
} 


gboolean 
on_popup_focus_out (GtkWidget *widget, 
        GdkEventFocus *event, 
        gpointer data) 
{ 
    gtk_widget_destroy (widget); 
    return TRUE; 
} 


int 
main (int argc, char *argv[]) 
{ 
    GtkWidget *window, *button, *vbox; 

    gtk_init (&argc, &argv); 

    window = gtk_window_new (GTK_WINDOW_TOPLEVEL); 
    gtk_window_set_title (GTK_WINDOW (window), "Parent window"); 
    gtk_container_set_border_width (GTK_CONTAINER (window), 10); 
    gtk_widget_set_size_request (window, 300, 300); 
    gtk_window_set_position (GTK_WINDOW (window), GTK_WIN_POS_CENTER); 

    button = gtk_button_new_with_label ("Pop Up"); 
    g_signal_connect (G_OBJECT (button), 
        "clicked", 
        G_CALLBACK (on_popup_clicked), 
        (gpointer) window); 

    vbox = gtk_vbox_new (FALSE, 3); 
    gtk_box_pack_end (GTK_BOX (vbox), button, FALSE, FALSE, 5); 
    gtk_container_add (GTK_CONTAINER (window), vbox); 

    gtk_widget_show_all (window); 
    gtk_main(); 
    return 0; 
} 

void 
on_popup_clicked (GtkButton* button, GtkWidget* pWindow) 
{ 
    GtkWidget *popup_window; 

    popup_window = gtk_window_new (GTK_WINDOW_POPUP); 
    gtk_window_set_title (GTK_WINDOW (popup_window), "Pop Up window"); 
    gtk_container_set_border_width (GTK_CONTAINER (popup_window), 10); 
    gtk_window_set_resizable (GTK_WINDOW (popup_window), FALSE); 
    gtk_window_set_decorated (GTK_WINDOW (popup_window), FALSE); 
    gtk_window_set_skip_taskbar_hint (GTK_WINDOW (popup_window), TRUE); 
    gtk_window_set_skip_pager_hint (GTK_WINDOW (popup_window), TRUE); 
    gtk_widget_set_size_request (popup_window, 150, 150); 
    gtk_window_set_transient_for (GTK_WINDOW (popup_window), GTK_WINDOW (pWindow)); 
    gtk_window_set_position (GTK_WINDOW (popup_window), GTK_WIN_POS_CENTER); 

    gtk_widget_add_events (popup_window, GDK_FOCUS_CHANGE_MASK); 
    gtk_widget_add_events (pWindow, GDK_BUTTON_PRESS_MASK); 

    g_signal_connect (G_OBJECT (popup_window), 
        "focus-out-event", 
        G_CALLBACK (on_popup_focus_out), 
        NULL); 

    handler_id = g_signal_connect (G_OBJECT (pWindow), 
        "button-press-event", 
        G_CALLBACK (on_click), 
        popup_window); 

    GdkColor color; 
    gdk_color_parse ("#3b3131", &color); 
    gtk_widget_modify_bg (GTK_WIDGET (popup_window), GTK_STATE_NORMAL, &color); 

    gtk_widget_show_all (popup_window); 
    gtk_widget_grab_focus (popup_window); 
} 
Смежные вопросы