gpt4 book ai didi

c++11 - 如何在 Gtk::DrawingArea 区域绘制一条新线,同时保留之前已经绘制的线?

转载 作者:行者123 更新时间:2023-12-01 12:41:12 25 4
gpt4 key购买 nike

我在 Ubuntu 12.04 LTS 32 位上使用带有 GNU 工具链和 gtkmm3 的 C++11。我一直在玩 Programming with gtkmm 3 中 gtkmm3 的一些示例。 .

基于 17.2.1.Example在那里,我继承自 Gtk::DrawingArea(此处为 MyDrawingArea)并覆盖了 on_draw() 事件处理程序,如下所示:

MyDrawingArea.hpp

...

protected:

bool on_draw ( const Cairo::RefPtr<Cairo::Context>& cr ) override;

MyDrawingArea.cpp

 bool MyDrawingArea::on_draw( const Cairo::RefPtr<Cairo::Context>& cr )
{

Gtk::Allocation allocation = get_allocation( );
const int width = allocation.get_width( );
const int height = allocation.get_height( );
int coord1{ height - 3 };
cr->set_line_width( 3.0 );

this->get_window( )->freeze_updates( );

cr->set_source_rgb( 0, 0.40, 0.60 );
cr->move_to( 0, coord1 );
cr->line_to( width, coord1 );
cr->stroke( );

cr->set_source_rgb( 1, 0.05, 1 );
cr->move_to( mXStart, coord1 );
cr->line_to( mXStart, mYAxis * 1.5 );
cr->show_text( to_string( mYAxis ) );
cr->stroke( );
mXStart += 5;

this->get_window( )->thaw_updates( );
return true;

}

我的目标是根据我在一个小测试应用程序中所做的计算绘制一个简单的条形图,想法是每次调用 on_draw() 事件时,下一个条形图将是在 mXAxis 上向右移动 5 个单位,将根据新的 mYaxis 值绘制一条垂直线,该值是根据新计算的结果计算的。

当我想重新绘制图形并触发 MyDrawingArea::on_draw() 事件时,我在计算完成并设置了新的 x 轴和 y 轴后从我的应用程序调用 MyDrawingArea.show_all() 。

然而,这并没有像我预期的那样工作:MyDrawingArea.show_all() 使整个绘图窗口无效并从头开始绘制:新图形线出现在适当的位置,但是之前的都被抹掉了。我也试过MyDrawingArea.queue_draw(),效果一样。 但我想保留以前的图表结果,以便在使用不同的值进行计算时获得计算结果的概况。

此实现还导致我的图形上的底线(我在图形上的 x 轴)- 由我的代码示例中的第一个 stroke() 调用绘制,在每次调用时重新呈现到 on_draw() - 虽然这不应该是必要的,因为这条线在 MyDrawingArea 的生命周期内持续存在 - 没有必要在每个新的 on_draw() 上使其无效然后重新绘制 事件,正如我的代码目前所做的那样,因为我还没有找到处理这个问题的方法。

我是 Cairo 的新手,所以我确定我做的这件事可能完全错了,但是明确的、面向任务的文档似乎很少——没有找到任何解释如何做的东西,尽管我我相信这很简单。

我需要做什么才能在 Gtk::DrawingArea 上绘制一条新线,同时保留在之前的遍中已经绘制的先前图形线,并建立将要绘制的图形元素在 Gtk::DrawingArea 小部件的生命周期内持续存在。显然,使用 show_all()queue_draw() 并在 on_draw() 事件中完成这一切并不是可行的方法。

最佳答案

通常,您必须绘制整个小部件,Cairo 会将绘图裁剪到预定义的脏区域。有关性能提示的“GtkWidget::draw”信号,另请参阅 GTK 引用手册:

The signal handler will get a cr with a clip region already set to the widget's dirty region, i.e. to the area that needs repainting. Complicated widgets that want to avoid redrawing themselves completely can get the full extents of the clip region with gdk_cairo_get_clip_rectangle(), or they can get a finer-grained representation of the dirty region with cairo_copy_clip_rectangle_list().

因此您可以使用 gtk_widget_queue_draw_area() 仅重绘您想要的区域。

关于c++11 - 如何在 Gtk::DrawingArea 区域绘制一条新线,同时保留之前已经绘制的线?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24530471/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com