强制GtkLabel剪切其居中对齐的文本

时间:2018-08-01 22:54:04

标签: label gtk text-alignment

我有一个GtkLabel,其文本将保持居中,而不论其短于或长于标签。

例如,设置了SS_CENTER样式标志的Win32静态控件的行为如下:

             ┌===========================┐
             │     Lorem ipsum dolor     │
             └===========================┘
     

-文本短于控件时;

             ┌===========================┐
  Lorem ipsum│dolor sit amet, consectetur│adipiscing
             └===========================┘
     

-文本长于控件。

N.B。:用户看到的文本的唯一部分在框架内。

我希望GtkLabel-s做同样的事情,但实际上它们以不同的方式呈现居中文本:

             ┌===========================┐
             │     Lorem ipsum dolor     │
             └===========================┘
     

-文本短于控件时;

             ┌===========================┐
             │Lorem ipsum dolor sit amet,│consectetur adipiscing
             └===========================┘
     

-文本长于控件。

如何使GtkLabel中居中的文本保持居中,即使很长?

以防万一:用户看到的实际文本以〜4 FPS更新,并且在运行时之前未知。

2 个答案:

答案 0 :(得分:2)

这可能是可行的,但是从用户体验的角度来看,显示句子的半随机部分并没有任何意义。

也许看看gtk_label_set_ellipsize? 他们使用PangoEllipsizeMode来指示要隐藏句子的哪一部分。然后,您可以隐藏标签的开头,中间或结尾部分。虽然这是一个枚举,而不是一个标志,所以您将无法隐藏开始和结束以仅显示中间部分。

所以我看到的可能的解决方案是:

  • 子类化GtkLabel并自己绘制图形,可能是使用PangoCairo
  • 或在GTK +中添加新的包装模式并实现它(在pango,PangoCairo还是两者中?)

如果这不是很强的要求,请使用更简洁的字体并使用现有的elipsize模式。

答案 1 :(得分:1)

终于解决了。

#include <gtk/gtk.h>

struct SCTX {
    GtkWidget *text;
    GdkRectangle rect;
};

void Resize(GtkWidget *view, GdkRectangle *rect, gpointer user) {
    struct SCTX *sctx = user;
    GtkRequisition requ;

    if ((sctx->rect.width  != rect->width )
    ||  (sctx->rect.height != rect->height)) {
        sctx->rect = *rect;
        gtk_widget_size_request(sctx->text, &requ);
        gtk_layout_move(view, sctx->text, (rect->width  - requ.width ) / 2,
                                          (rect->height - requ.height) / 2);
    }
}

int main(int argc, char *argv[]) {
    GtkWidget *hwnd, *view;
    struct SCTX sctx = {0};

    gtk_init(&argc, &argv);
    hwnd = gtk_window_new(GTK_WINDOW_TOPLEVEL);
    g_signal_connect(hwnd, "destroy", gtk_main_quit, 0);

    view = gtk_layout_new(0, 0);
    g_signal_connect(view, "size-allocate", G_CALLBACK(Resize), &sctx);
    gtk_widget_set_size_request(view, 320, 200);

    sctx.text = gtk_label_new("Lorem ipsum dolor sit amet, consectetur "
                              "adipiscing elit, sed do eiusmod tempor "
                              "incididunt ut labore et dolore magna aliqua");
    gtk_container_add(view, sctx.text);

    gtk_container_add(hwnd, view);
    gtk_container_set_border_width(GTK_CONTAINER(hwnd), 32);
    gtk_window_set_position(GTK_WINDOW(hwnd), GTK_WIN_POS_CENTER);
    gtk_widget_show_all(hwnd);
    gtk_main();
    return 0;
}

这里的关键部分是将GtkLayout包裹在我们的GtkLabel周围,以便前者可以将后者移动到其边界内。