考虑下面粘贴的源代码test.vala
。这是一个简单的应用程序,它应该在顶部栏/面板上显示一个图标,当单击该图标时,它应该显示一个包含一个项目的菜单(打开),当您单击打开时,它应该显示一个包含多个项目的子菜单. 我编译这个:
$ 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 上:
请注意,显示了 appindicator 图标(如预期的那样),单击它时会显示带有“打开”项的一级菜单(如预期的那样) - 但是当我单击“打开”时,我并没有真正得到子菜单我预计; 相反,它似乎试图打开子菜单,然后立即关闭?
我需要做什么才能让这个应用程序正确打开子菜单?
这里是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 ();
}
}
编辑:另见https://stackoverflow.com/questions/53805975/re-creating-gtk-menu-in-event-handler-with-vala
好的,明白了 - 重写了上面的代码,因此子菜单创建是一个单独的函数,因此更容易识别问题出在此处:
正如评论所说,
set_submenu
运行.connect,
会导致创建的子菜单立即关闭。我想这是因为在.connect
处理程序内部,我们有一些“匿名上下文”或其他任何东西,这会导致在处理程序退出时在那里创建的所有局部变量都被销毁,无论是否在其他地方引用了这些变量。所以解决方案是createSubmenu
在连接处理程序的外部运行。请注意,即使使用此工作代码,在编译时我也会得到:
...但我想这不是什么大问题。可能需要更改 libappindicator3。
无论如何,这里是完整的更新(和工作)代码
test.vala
: