如何解决如何使GtkScrolledWindow与GtkOverlay内部的GtkFixed一起使用
基本问题是这样:
我有一堆小部件,用户可以随意移动。这是通过使用GtkOverlay作为外部容器来实现的,在内部容器中是一堆GtkFixed,每个容器都包含相应的小部件:
- GtkOverlay
- Gtk已修复
- 小部件A
- Gtk已修复
- 小部件B
- ...
- Gtk已修复
外部GtkOverlay的原因是GtkFixed缺少对z-index的支持。现在,每个GtkFixed都覆盖了整个区域。因此,为了能够激活当前位于其下方的窗口小部件,必须在GtkOverlay的每个子级上设置pass-through属性。该系统适用于叶子小部件GtkScale和GtkDrawingArea。但是,如果我在GTK中的GtkScrolledWindow内放东西,则假定GtkScrolledWindow位于GtkFixed的左上角,这通常是错误的。取消设置传递属性可以解决此问题,但随后其他小部件将无法访问。有什么解决方法吗?
我认为适当的解决方案是让GtkOverlay的每个孩子都不要占据整个区域。这样就不需要传递,但是我认为用GTK不可能做到这一点(或者,如果您编写自己的布局容器,则其效果就像带有自由浮动层的网页一样。
最小无效示例:
//@ {
//@ "targets":[{"name":"minimal-nonworking-example-so-64090741","type":"application","pkgconfig_libs":["gtk+-3.0"]}]
//@ }
#include <gtk/gtk.h>
#include <string>
#include <vector>
#include <algorithm>
// To be able to feed the list box
template<class Callback>
void readlines(FILE* src,Callback&& cb)
{
std::string buffer;
while(true)
{
auto ch_in = getc(src);
if(ch_in == -1)
{
if(buffer.size() != 0)
{
cb(buffer);
}
return;
}
if(ch_in=='\n')
{
cb(buffer);
buffer.clear();
}
else
{
buffer+=ch_in;
}
}
}
int main(int argc,char** argv)
{
gtk_init(&argc,&argv);
auto mainwin = GTK_WINDOW(gtk_window_new(GTK_WINDOW_TOPLEVEL));
gtk_window_set_title(mainwin,"ListBoxTest");
gtk_window_set_default_size(mainwin,800,500);
auto overlay = GTK_OVERLAY(gtk_overlay_new());
gtk_container_add(GTK_CONTAINER(mainwin),GTK_WIDGET(overlay));
gtk_widget_set_size_request(GTK_WIDGET(overlay),500,300);
auto frame = GTK_FRAME(gtk_frame_new(nullptr));
auto fixed = GTK_FIXED(gtk_fixed_new());
gtk_fixed_put(fixed,GTK_WIDGET(frame),60,60);
gtk_overlay_add_overlay(overlay,GTK_WIDGET(fixed));
// {
// Without this line,GtkScrolledWindow works as expected,but if there were more layers
// other layers would become inaccessible
gtk_overlay_set_overlay_pass_through(overlay,GTK_WIDGET(fixed),TRUE);
// }
gtk_overlay_reorder_overlay(overlay,-1);
auto scrolled_window = GTK_SCROLLED_WINDOW(gtk_scrolled_window_new(nullptr,nullptr));
gtk_container_add(GTK_CONTAINER(frame),GTK_WIDGET(scrolled_window));
gtk_widget_set_size_request(GTK_WIDGET(frame),128,128);
auto listbox = GTK_LIST_BOX(gtk_list_box_new());
gtk_container_add(GTK_CONTAINER(scrolled_window),GTK_WIDGET(listbox));
gtk_widget_show_all(GTK_WIDGET(mainwin));
readlines(stdin,[listbox](auto&& item) {
auto label = GTK_LABEL(gtk_label_new(item.c_str()));
gtk_label_set_justify(label,GTK_JUSTIFY_LEFT);
gtk_widget_set_halign(GTK_WIDGET(label),GTK_ALIGN_START);
gtk_list_box_insert(listbox,GTK_WIDGET(label),-1);
});
gtk_widget_show_all(GTK_WIDGET(fixed));
gtk_main();
gtk_widget_destroy(GTK_WIDGET(mainwin));
return 0;
}
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。