Приложение Gnome3 AppIndicator Vala не может показать подменю (оно сразу автозакрывается)?

Рассмотрите исходный код, вставляемый ниже как test.vala. Это - простое приложение, которое должно показать значок на главной панели/панели, когда значок нажат, это должно показать меню с одним объектом в нем (Открытый) при нажатии на Open, это должно показать подменю с несколькими объектами в нем. Я компилирую это на:

$ cat /etc/issue
Ubuntu 18.04.1 LTS \n \l
$ uname -a
Linux MyPC 4.15.0-38-generic #41-Ubuntu SMP Wed Oct 10 10:59:38 UTC 2018 x86_64 x86_64 x86_64 GNU/Linux
$ printf 'Desktop: %s\nSession: %s\n' "$XDG_CURRENT_DESKTOP" "$GDMSESSION"
Desktop: ubuntu:GNOME
Session: ubuntu
$ gnome-shell --version
GNOME Shell 3.28.3

... и я компилирую с:

valac -X -D'GETTEXT_PACKAGE="my-indicator"' -D NEWMETHOD --pkg=gtk+-3.0 --pkg appindicator3-0.1 test.vala

... и для этого необходимо будет также установить libappindicator-dev пакет.

Затем я запускаю приложение:

$ ./test 
main() ... 
Main(): ok
Creating MainWindow
^C

... и результат, который я получаю, показывают на этом, анимировал gif:

out.gif

Обратите внимание, что appindicator значок показывают (как ожидалось) при нажатии на него, меню первого уровня с "Открытым" объектом показывают (как ожидалось) - но когда я нажимаю на "Open", я действительно не получаю подменю, которое я ожидаю; вместо этого похоже, что существует попытка открыть подменю, и это сразу закрывается?

Что я должен сделать, так, чтобы это приложение открыло подменю правильно?

Вот test.vala:

// build with:
// valac -X -D'GETTEXT_PACKAGE="my-indicator"' --pkg=gtk+-3.0 --pkg appindicator3-0.1 test.vala

// "It's not possible to define a preprocessor symbol inside the Vala code (like with C). The only way to define a symbol is to feed it through the valac option -D."
// valac -X -D'GETTEXT_PACKAGE="my-indicator"' -D NEWMETHOD --pkg=gtk+-3.0 --pkg appindicator3-0.1 test.vala

// see also: https://valadoc.org/gtk+-3.0/Gtk.MenuItem.html

using GLib;
using Gtk;
using AppIndicator;

public Main App;
public const string AppName = "Test";

extern void exit(int exit_code);

public class MyIndicator: GLib.Object{

  protected Indicator indicator;
  protected string icon;
  protected string name;

  public MyIndicator(){

    App.my_indicator = this;

    this.name = "My Indicator";

    this.icon = "account-logged-in"; // looks like a checkmark
    this.indicator = new Indicator("my_indicator", icon, IndicatorCategory.APPLICATION_STATUS);
    indicator.set_status(IndicatorStatus.ACTIVE);

    var menu = new Gtk.Menu();

    // open -------------------------------------
    #if NEWMETHOD
      var item = new Gtk.MenuItem.with_label(_("Open"));
    #else
      var item = new Gtk.ImageMenuItem.with_label(_("Open"));
    #endif
    menu.append(item);
    var item_open = item;

    item.set_reserve_indicator(false);

    item.activate.connect(() => {
      var submenu = new Gtk.Menu();
      submenu.reserve_toggle_size = true;
      //var dummy_window = new Gtk.Window();
      //Gtk.Image icon = null;
      int i;
      for (i = 0; i < 10; i++) {
        #if NEWMETHOD
          var subitem = new Gtk.MenuItem.with_label ( "Exit %d".printf(i) );
        #else
          var subitem = new Gtk.ImageMenuItem.with_label ( "Exit %d".printf(i) );
        #endif
        subitem.set_reserve_indicator(true);
        submenu.append(subitem);
        subitem.activate.connect(() => {
          App.exit_app();
          exit(0);
        });
        //subitem.activate();
      }
      submenu.show_all();

      item_open.set_submenu(submenu);
    });
    item.activate(); // so it shows submenu triangle

    indicator.set_menu(menu);
    menu.show_all();
  }
}


public class Main : GLib.Object{

  public MyIndicator my_indicator;

  public static int main (string[] args) {

    stdout.printf("main() ... \n");
    stdout.flush();
    Gtk.init(ref args);
    App = new Main(args);
    bool success = App.start_application(args);
    App.exit_app();

    return (success) ? 0 : 1;
  }

  public Main(string[] args){
    stdout.printf("Main(): ok\n");
    stdout.flush();
  }

  public bool start_application(string[] args){
    stdout.printf("Creating MainWindow\n");
    stdout.flush();

    new MyIndicator(); // var ind = new MyIndicator();

    //start event loop
    Gtk.main();

    return true;
  }

  public void exit_app (){
    stdout.printf("exit_app()\n");
    stdout.flush();
    Gtk.main_quit ();
  }
}
2
задан 2 November 2018 в 18:16

1 ответ

Править: см. также https://stackoverflow.com/questions/53805975/re-creating-gtk-menu-in-event-handler-with-vala


Хорошо, добрался, это - переписало вышеупомянутый код, таким образом, создание подменю является отдельной функцией, таким образом, стало легче определить, что проблема была здесь:

item_open.activate.connect(() => {
  //~ item_open.set_submenu(createSubmenu()); // NO; having the set_submenu run in .connect, causes immediate shutdown of the created submenu!!
  stdout.printf("item.activate.connect running\n");
});
item_open.set_submenu(createSubmenu()); // is OK here
item_open.activate(); // so it shows submenu triangle

Поскольку в комментарии говорится, имея set_submenu выполненный в .connect, вызывает непосредственное завершение работы созданного подменю. Я предполагаю, что это вызвано тем, что в .connect обработчик, у нас есть некоторый "анонимный контекст" или что бы то ни было, который заставляет все локальные переменные, созданные там быть уничтоженными, когда обработчик выходит, независимо от того, если на какого-либо из тех ссылаются в другом месте или нет. Таким образом, решение состоит в том, чтобы работать createSubmenu за пределами обработчика подключений.

Обратите внимание, что, даже с этим рабочим кодом, при компиляции его я добираюсь:

$ valac -X -D'GETTEXT_PACKAGE="my-indicator"' -D NEWMETHOD --pkg=gtk+-3.0 --pkg appindicator3-0.1 test.vala
/tmp/test.vala.c: In function ‘my_indicator_createSubmenu’:
/tmp/test.vala.c:182:52: warning: passing argument 2 of ‘gtk_menu_shell_append’ from incompatible pointer type [-Wincompatible-pointer-types]
    gtk_menu_shell_append ((GtkMenuShell*) _tmp11_, _tmp12_);
                                                    ^~~~~~~
In file included from /usr/include/gtk-3.0/gtk/gtkmenu.h:33:0,
                 from /usr/include/gtk-3.0/gtk/gtklabel.h:34,
                 from /usr/include/gtk-3.0/gtk/gtkaccellabel.h:35,
                 from /usr/include/gtk-3.0/gtk/gtk.h:33,
                 from /usr/include/libappindicator3-0.1/libappindicator/app-indicator.h:33,
                 from /tmp/test.vala.c:15:
/usr/include/gtk-3.0/gtk/gtkmenushell.h:91:10: note: expected ‘GtkWidget * {aka struct _GtkWidget *}’ but argument is of type ‘GtkMenuItem * {aka struct _GtkMenuItem *}’
 void     gtk_menu_shell_append         (GtkMenuShell *menu_shell,
          ^~~~~~~~~~~~~~~~~~~~~
/tmp/test.vala.c: In function ‘my_indicator_construct’:
/tmp/test.vala.c:260:47: warning: passing argument 2 of ‘gtk_menu_shell_append’ from incompatible pointer type [-Wincompatible-pointer-types]
  gtk_menu_shell_append ((GtkMenuShell*) menu, item_open);
                                               ^~~~~~~~~
In file included from /usr/include/gtk-3.0/gtk/gtkmenu.h:33:0,
                 from /usr/include/gtk-3.0/gtk/gtklabel.h:34,
                 from /usr/include/gtk-3.0/gtk/gtkaccellabel.h:35,
                 from /usr/include/gtk-3.0/gtk/gtk.h:33,
                 from /usr/include/libappindicator3-0.1/libappindicator/app-indicator.h:33,
                 from /tmp/test.vala.c:15:
/usr/include/gtk-3.0/gtk/gtkmenushell.h:91:10: note: expected ‘GtkWidget * {aka struct _GtkWidget *}’ but argument is of type ‘GtkMenuItem * {aka struct _GtkMenuItem *}’
 void     gtk_menu_shell_append         (GtkMenuShell *menu_shell,
          ^~~~~~~~~~~~~~~~~~~~~
/tmp/test.vala.c:265:40: warning: passing argument 2 of ‘gtk_menu_item_set_submenu’ from incompatible pointer type [-Wincompatible-pointer-types]
  gtk_menu_item_set_submenu (item_open, _tmp10_);
                                        ^~~~~~~
In file included from /usr/include/gtk-3.0/gtk/gtkcheckmenuitem.h:33:0,
                 from /usr/include/gtk-3.0/gtk/gtk.h:72,
                 from /usr/include/libappindicator3-0.1/libappindicator/app-indicator.h:33,
                 from /tmp/test.vala.c:15:
/usr/include/gtk-3.0/gtk/gtkmenuitem.h:120:12: note: expected ‘GtkWidget * {aka struct _GtkWidget *}’ but argument is of type ‘GtkMenu * {aka struct _GtkMenu *}’
 void       gtk_menu_item_set_submenu          (GtkMenuItem         *menu_item,
            ^~~~~~~~~~~~~~~~~~~~~~~~~

... но я предполагаю не то, чтобы большой из проблемы. Вероятно, libappindicator3 должен быть изменен.

Так или иначе вот обновленное полное (и работающий) код test.vala:

// build with:
// valac -X -D'GETTEXT_PACKAGE="my-indicator"' --pkg=gtk+-3.0 --pkg appindicator3-0.1 test.vala

// "It's not possible to define a preprocessor symbol inside the Vala code (like with C). The only way to define a symbol is to feed it through the valac option -D."
// valac -X -D'GETTEXT_PACKAGE="my-indicator"' -D NEWMETHOD --pkg=gtk+-3.0 --pkg appindicator3-0.1 test.vala

// see also: https://valadoc.org/gtk+-3.0/Gtk.MenuItem.html

using GLib;
using Gtk;
using AppIndicator;

public Main App;
public const string AppName = "Test";

extern void exit(int exit_code);

public class MyIndicator: GLib.Object{

  protected Indicator indicator;
  protected string icon;
  protected string name;

  public Gtk.Menu createSubmenu(){
    var submenu = new Gtk.Menu();
    submenu.reserve_toggle_size = true;
    //var dummy_window = new Gtk.Window();
    //Gtk.Image icon = null;
    int i;
    for (i = 0; i < 10; i++) {
      #if NEWMETHOD
        var subitem = new Gtk.MenuItem.with_label ( "Exit %d".printf(i) );
      #else
        var subitem = new Gtk.ImageMenuItem.with_label ( "Exit %d".printf(i) );
      #endif

      subitem.set_reserve_indicator(true);
      submenu.append(subitem);
      subitem.activate.connect(() => {
        App.exit_app();
        exit(0);
      });
      //subitem.activate(); // no way, causes immediate exit!
    }

    submenu.show_all();
    return submenu;
  }
  public MyIndicator(){

    App.my_indicator = this;

    this.name = "My Indicator";

    this.icon = "account-logged-in"; // looks like a checkmark
    this.indicator = new Indicator("my_indicator", icon, IndicatorCategory.APPLICATION_STATUS);
    indicator.set_status(IndicatorStatus.ACTIVE);

    /*
    // from https://valadoc.org/gtk+-3.0/Gtk.Menu.html
        // MenuBar:
        //Gtk.MenuBar bar = new Gtk.MenuBar (); //error: Argument 1: Cannot convert from `unowned Gtk.MenuBar?' to `unowned Gtk.Menu?'
        Gtk.Menu bar = new Gtk.Menu ();
        //indicator.add (bar); // error: The name `add' does not exist in the context of `AppIndicator.Indicator'
    indicator.set_menu(bar);

        // File:
        Gtk.MenuItem item_file = new Gtk.MenuItem.with_label ("File");
        bar.add (item_file);

        Gtk.Menu filemenu = new Gtk.Menu ();
        item_file.set_submenu (filemenu);

        Gtk.MenuItem item_open = new Gtk.MenuItem.with_label ("Open");
        filemenu.add (item_open);

        // Submenu:
        Gtk.Menu submenu = new Gtk.Menu ();
        item_open.set_submenu (submenu);

        Gtk.MenuItem item_foo = new Gtk.MenuItem.with_label ("foo");
        submenu.add (item_foo);

        Gtk.MenuItem item_bar = new Gtk.MenuItem.with_label ("bar");
        submenu.add (item_bar);

    bar.show_all();
    */

    var menu = new Gtk.Menu();

    // open -------------------------------------
    #if NEWMETHOD
      var item_open = new Gtk.MenuItem.with_label(_("Open"));
    #else
      var item_open = new Gtk.ImageMenuItem.with_label(_("Open"));
    #endif
    menu.append(item_open);

    item_open.set_reserve_indicator(false);

    item_open.activate.connect(() => {
      //~ item_open.set_submenu(createSubmenu()); // NO; having the set_submenu run in .connect, causes immediate shutdown of the created submenu!!
      stdout.printf("item.activate.connect running\n");
    });
    item_open.set_submenu(createSubmenu()); // is OK here
    item_open.activate(); // so it shows submenu triangle

    indicator.set_menu(menu);
    menu.show_all();
  }
}


public class Main : GLib.Object{

  public MyIndicator my_indicator;

  public static int main (string[] args) {

    stdout.printf("main() ... \n");
    stdout.flush();
    Gtk.init(ref args);
    App = new Main(args);
    bool success = App.start_application(args);
    App.exit_app();

    return (success) ? 0 : 1;
  }

  public Main(string[] args){
    stdout.printf("Main(): ok\n");
    stdout.flush();
  }

  public bool start_application(string[] args){
    stdout.printf("Creating MainWindow\n");
    stdout.flush();

    new MyIndicator(); // var ind = new MyIndicator();

    //start event loop
    Gtk.main();

    return true;
  }

  public void exit_app (){
    stdout.printf("exit_app()\n");
    stdout.flush();
    Gtk.main_quit ();
  }
}
0
ответ дан 2 November 2018 в 18:16

Другие вопросы по тегам:

Похожие вопросы: