39 #include <visp3/core/vpConfig.h>
41 #if (defined(VISP_HAVE_GTK))
50 #include <visp3/core/vpDisplay.h>
51 #include <visp3/gui/vpDisplayGTK.h>
54 #include <visp3/core/vpDisplayException.h>
55 #include <visp3/core/vpImageConvert.h>
56 #include <visp3/core/vpImageTools.h>
57 #include <visp3/core/vpMath.h>
59 #ifndef DOXYGEN_SHOULD_SKIP_THIS
62 #include <gdk/gdkrgb.h>
67 class vpDisplayGTK::Impl
71 : m_widget(nullptr), m_background(nullptr), m_gc(nullptr), m_blue(), m_red(), m_yellow(), m_green(), m_cyan(), m_orange(),
72 m_white(), m_black(), m_gdkcolor(), m_lightBlue(), m_darkBlue(), m_lightRed(), m_darkRed(), m_lightGreen(),
73 m_darkGreen(), m_purple(), m_lightGray(), m_gray(), m_darkGray(), m_colormap(nullptr), m_font(nullptr), m_vectgtk(nullptr),
79 void init(
unsigned int win_width,
unsigned int win_height,
int win_x,
int win_y,
const std::string &title)
81 gint width =
static_cast<gint
>(win_width);
82 gint height =
static_cast<gint
>(win_height);
88 gtk_init(argc, &argv);
91 m_widget = gtk_window_new(GTK_WINDOW_TOPLEVEL);
93 gtk_widget_add_events(m_widget, GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK | GDK_POINTER_MOTION_MASK);
95 gtk_window_set_default_size(GTK_WINDOW(m_widget), width, height);
97 gtk_window_move(GTK_WINDOW(m_widget), win_x, win_y);
99 gtk_widget_show(m_widget);
104 m_background = gdk_pixmap_new(m_widget->window, width, height, -1);
107 m_gc = gdk_gc_new(m_widget->window);
110 m_colormap = gdk_window_get_colormap(m_widget->window);
115 gdk_color_parse(
"light blue", &m_lightBlue);
116 gdk_colormap_alloc_color(m_colormap, &m_lightBlue, FALSE, TRUE);
119 gdk_color_parse(
"blue", &m_blue);
120 gdk_colormap_alloc_color(m_colormap, &m_blue, FALSE, TRUE);
123 gdk_color_parse(
"dark blue", &m_darkBlue);
124 gdk_colormap_alloc_color(m_colormap, &m_darkBlue, FALSE, TRUE);
127 gdk_color_parse(
"#FF8C8C", &m_lightRed);
128 gdk_colormap_alloc_color(m_colormap, &m_lightRed, FALSE, TRUE);
131 gdk_color_parse(
"red", &m_red);
132 gdk_colormap_alloc_color(m_colormap, &m_red, FALSE, TRUE);
135 gdk_color_parse(
"dark red", &m_darkRed);
136 gdk_colormap_alloc_color(m_colormap, &m_darkRed, FALSE, TRUE);
139 gdk_color_parse(
"light green", &m_lightGreen);
140 gdk_colormap_alloc_color(m_colormap, &m_lightGreen, FALSE, TRUE);
143 gdk_color_parse(
"green", &m_green);
144 gdk_colormap_alloc_color(m_colormap, &m_green, FALSE, TRUE);
147 gdk_color_parse(
"dark green", &m_darkGreen);
148 gdk_colormap_alloc_color(m_colormap, &m_darkGreen, FALSE, TRUE);
151 gdk_color_parse(
"yellow", &m_yellow);
152 gdk_colormap_alloc_color(m_colormap, &m_yellow, FALSE, TRUE);
155 gdk_color_parse(
"cyan", &m_cyan);
156 gdk_colormap_alloc_color(m_colormap, &m_cyan, FALSE, TRUE);
159 gdk_color_parse(
"orange", &m_orange);
160 gdk_colormap_alloc_color(m_colormap, &m_orange, FALSE, TRUE);
163 gdk_color_parse(
"purple", &m_purple);
164 gdk_colormap_alloc_color(m_colormap, &m_purple, FALSE, TRUE);
167 gdk_color_parse(
"white", &m_white);
168 gdk_colormap_alloc_color(m_colormap, &m_white, FALSE, TRUE);
171 gdk_color_parse(
"black", &m_black);
172 gdk_colormap_alloc_color(m_colormap, &m_black, FALSE, TRUE);
175 gdk_color_parse(
"#C0C0C0", &m_lightGray);
176 gdk_colormap_alloc_color(m_colormap, &m_lightGray, FALSE, TRUE);
179 gdk_color_parse(
"#808080", &m_gray);
180 gdk_colormap_alloc_color(m_colormap, &m_gray, FALSE, TRUE);
183 gdk_color_parse(
"#404040", &m_darkGray);
184 gdk_colormap_alloc_color(m_colormap, &m_darkGray, FALSE, TRUE);
188 m_font = gdk_font_load(
"-*-times-medium-r-normal-*-16-*-*-*-*-*-*-*");
189 if (m_font ==
nullptr)
190 m_font = gdk_font_load(
"-*-courier-bold-r-normal-*-*-140-*-*-*-*-*-*");
191 if (m_font ==
nullptr)
192 m_font = gdk_font_load(
"-*-courier 10 pitch-medium-r-normal-*-16-*-*-*-*-*-*-*");
195 gdk_window_set_title(m_widget->window, title.c_str());
198 void setFont(
const std::string &fontname) { m_font = gdk_font_load((
const gchar *)fontname.c_str()); }
200 void setTitle(
const std::string &title) { gdk_window_set_title(m_widget->window, title.c_str()); }
202 void setWindowPosition(
int win_x,
int win_y) { gtk_window_move(GTK_WINDOW(m_widget), win_x, win_y); }
208 gdk_draw_gray_image(m_background, m_gc, 0, 0, width, height, GDK_RGB_DITHER_NONE, I.
bitmap, width);
213 gdk_draw_gray_image(m_background, m_gc, 0, 0, width, height, GDK_RGB_DITHER_NONE, sampled.
bitmap, width);
217 gdk_window_set_back_pixmap(m_widget->window, m_background, FALSE);
224 gdk_draw_rgb_32_image(m_background, m_gc, 0, 0, width, height, GDK_RGB_DITHER_NONE, (
unsigned char *)I.
bitmap,
230 gdk_draw_rgb_32_image(m_background, m_gc, 0, 0, width, height, GDK_RGB_DITHER_NONE,
231 (
unsigned char *)sampled.
bitmap, 4 * width);
235 gdk_window_set_back_pixmap(m_widget->window, m_background, FALSE);
240 gdk_draw_gray_image(m_background, m_gc, j_min, i_min, width, height, GDK_RGB_DITHER_NONE, I.
bitmap, width);
242 gdk_window_set_back_pixmap(m_widget->window, m_background, FALSE);
247 gdk_draw_rgb_32_image(m_background, m_gc, j_min, i_min, width, height, GDK_RGB_DITHER_NONE,
248 (
unsigned char *)I.
bitmap, width * 4);
251 gdk_window_set_back_pixmap(m_widget->window, m_background, FALSE);
256 if (m_col !=
nullptr) {
261 if (m_widget !=
nullptr) {
262 gdk_window_hide(m_widget->window);
263 gdk_window_destroy(m_widget->window);
264 gtk_widget_destroy(m_widget);
271 gdk_window_clear(m_widget->window);
278 gdk_gc_set_foreground(m_gc, m_col[color.
id]);
280 m_gdkcolor.red = 256 * color.
R;
281 m_gdkcolor.green = 256 * color.
G;
282 m_gdkcolor.blue = 256 * color.
B;
283 gdk_colormap_alloc_color(m_colormap, &m_gdkcolor, FALSE, TRUE);
284 gdk_gc_set_foreground(m_gc, &m_gdkcolor);
286 if (m_font !=
nullptr)
288 (
const gchar *)text.c_str());
290 std::cout <<
"Cannot draw string: no font is selected" << std::endl;
294 unsigned int thickness,
unsigned int scale)
297 gdk_gc_set_foreground(m_gc, m_col[color.
id]);
299 m_gdkcolor.red = 256 * color.
R;
300 m_gdkcolor.green = 256 * color.
G;
301 m_gdkcolor.blue = 256 * color.
B;
302 gdk_colormap_alloc_color(m_colormap, &m_gdkcolor, FALSE, TRUE);
303 gdk_gc_set_foreground(m_gc, &m_gdkcolor);
306 gdk_gc_set_line_attributes(m_gc,
static_cast<gint
>(thickness), GDK_LINE_SOLID, GDK_CAP_BUTT, GDK_JOIN_BEVEL);
309 gdk_draw_arc(m_background, m_gc, FALSE,
vpMath::round((center.
get_u() - radius) / scale),
310 vpMath::round((center.
get_v() - radius) / scale),
static_cast<gint
>(2. * radius / scale),
311 static_cast<gint
>(2. * radius / scale), 23040, 23040);
313 gdk_draw_arc(m_background, m_gc, TRUE,
vpMath::round((center.
get_u() - radius) / scale),
314 vpMath::round((center.
get_v() - radius) / scale),
static_cast<gint
>(2. * radius / scale),
315 static_cast<gint
>(2. * radius / scale), 23040, 23040);
322 gdk_gc_set_foreground(m_gc, m_col[color.
id]);
324 m_gdkcolor.red = 256 * color.
R;
325 m_gdkcolor.green = 256 * color.
G;
326 m_gdkcolor.blue = 256 * color.
B;
327 gdk_colormap_alloc_color(m_colormap, &m_gdkcolor, FALSE, TRUE);
328 gdk_gc_set_foreground(m_gc, &m_gdkcolor);
331 gdk_gc_set_line_attributes(m_gc,
static_cast<gint
>(thickness), GDK_LINE_ON_OFF_DASH, GDK_CAP_BUTT, GDK_JOIN_BEVEL);
334 gdk_gc_set_line_attributes(m_gc, 0, GDK_LINE_SOLID, GDK_CAP_BUTT, GDK_JOIN_BEVEL);
341 gdk_gc_set_foreground(m_gc, m_col[color.
id]);
343 m_gdkcolor.red = 256 * color.
R;
344 m_gdkcolor.green = 256 * color.
G;
345 m_gdkcolor.blue = 256 * color.
B;
346 gdk_colormap_alloc_color(m_colormap, &m_gdkcolor, FALSE, TRUE);
347 gdk_gc_set_foreground(m_gc, &m_gdkcolor);
350 gdk_gc_set_line_attributes(m_gc,
static_cast<gint
>(thickness), GDK_LINE_SOLID, GDK_CAP_BUTT, GDK_JOIN_BEVEL);
358 gdk_gc_set_foreground(m_gc, m_col[color.
id]);
360 m_gdkcolor.red = 256 * color.
R;
361 m_gdkcolor.green = 256 * color.
G;
362 m_gdkcolor.blue = 256 * color.
B;
363 gdk_colormap_alloc_color(m_colormap, &m_gdkcolor, FALSE, TRUE);
364 gdk_gc_set_foreground(m_gc, &m_gdkcolor);
367 if (thickness == 1) {
372 static_cast<gint
>(thickness),
static_cast<gint
>(thickness));
377 unsigned int thickness,
unsigned int scale)
380 gdk_gc_set_foreground(m_gc, m_col[color.
id]);
382 m_gdkcolor.red = 256 * color.
R;
383 m_gdkcolor.green = 256 * color.
G;
384 m_gdkcolor.blue = 256 * color.
B;
385 gdk_colormap_alloc_color(m_colormap, &m_gdkcolor, FALSE, TRUE);
386 gdk_gc_set_foreground(m_gc, &m_gdkcolor);
388 gdk_gc_set_line_attributes(m_gc,
static_cast<gint
>(thickness), GDK_LINE_SOLID, GDK_CAP_BUTT, GDK_JOIN_BEVEL);
393 static_cast<gint
>(h / scale));
397 static_cast<gint
>(h / scale));
400 gdk_gc_set_line_attributes(m_gc, 0, GDK_LINE_SOLID, GDK_CAP_BUTT, GDK_JOIN_BEVEL);
404 const GdkEventType &event_type)
408 GdkEvent *ev =
nullptr;
409 while ((ev = gdk_event_get())) {
410 if (ev->any.window == m_widget->window && ev->type == event_type) {
411 double u = ((GdkEventButton *)ev)->x;
412 double v = ((GdkEventButton *)ev)->y;
416 switch (
static_cast<int>(((GdkEventButton *)ev)->button)) {
436 }
while (ret ==
false && blocking ==
true);
443 ImageGtk = gdk_image_get(m_background, 0, 0, width, height);
448 guchar OctetRouge, OctetVert, OctetBleu, mask;
451 for (gint y = 0; y < height; ++y) {
452 for (gint x = 0; x < width; ++x) {
453 pixel = gdk_image_get_pixel(ImageGtk, x, y);
454 OctetBleu =
static_cast<guchar
>(pixel) & mask;
455 OctetVert =
static_cast<guchar
>(pixel >> 8) & mask;
456 OctetRouge =
static_cast<guchar
>(pixel >> 16) & mask;
457 I[y][x].R = OctetRouge;
458 I[y][x].G = OctetVert;
459 I[y][x].B = OctetBleu;
465 unsigned int getScreenDepth() {
return static_cast<unsigned int>(gdk_window_get_visual(m_widget->window)->depth); }
472 GdkEvent *ev =
nullptr;
473 while ((ev = gdk_event_get()) !=
nullptr) {
476 if (ev->any.window == m_widget->window && ev->type == GDK_KEY_PRESS) {
477 key = gdk_keyval_name(ev->key.keyval);
486 }
while (ret ==
false && blocking ==
true);
493 GdkEvent *ev =
nullptr;
494 if ((ev = gdk_event_get())) {
495 if (ev->any.window == m_widget->window && ev->type == GDK_MOTION_NOTIFY) {
496 double u = ((GdkEventMotion *)ev)->x;
497 double v = ((GdkEventMotion *)ev)->y;
511 gdk_window_get_pointer(m_widget->window, &u, &v,
nullptr);
512 ip.
set_u(
static_cast<double>(u) * scale);
513 ip.
set_v(
static_cast<double>(v) * scale);
516 void getScreenSize(
bool is_init,
unsigned int &w,
unsigned int &h)
522 gtk_init(argc, &argv);
524 GtkWidget *widget_ = gtk_window_new(GTK_WINDOW_TOPLEVEL);
525 gtk_window_set_default_size(GTK_WINDOW(widget_), 100, 100);
526 gtk_widget_show(widget_);
528 GdkScreen *screen_ = gdk_window_get_screen(widget_->window);
529 w =
static_cast<unsigned int>(gdk_screen_get_width(screen_));
530 h =
static_cast<unsigned int>(gdk_screen_get_height(screen_));
531 gtk_widget_destroy(widget_);
534 GdkScreen *screen_ = gdk_window_get_screen(m_widget->window);
535 w =
static_cast<unsigned int>(gdk_screen_get_width(screen_));
536 h =
static_cast<unsigned int>(gdk_screen_get_height(screen_));
542 GdkPixmap *m_background;
544 GdkColor m_blue, m_red, m_yellow, m_green, m_cyan, m_orange, m_white, m_black, m_gdkcolor;
545 GdkColor m_lightBlue, m_darkBlue, m_lightRed, m_darkRed, m_lightGreen, m_darkGreen, m_purple;
546 GdkColor m_lightGray, m_gray, m_darkGray;
547 GdkColormap *m_colormap;
611 init(I, win_x, win_y, win_title);
665 init(I, win_x, win_y, win_title);
754 if (!win_title.empty())
784 if (!win_title.empty())
803 const std::string &win_title)
844 if (!title.empty()) {
845 m_impl->setTitle(title);
865 m_impl->setWindowPosition(win_x, win_y);
918 int i_min = std::max<int>(
static_cast<int>(ceil(iP.
get_i() /
m_scale)), 0);
919 int j_min = std::max<int>(
static_cast<int>(ceil(iP.
get_j() /
m_scale)), 0);
921 m_impl->displayImageROI(Itemp,
static_cast<gint
>(j_min),
static_cast<gint
>(i_min),
974 int i_min = std::max<int>(
static_cast<int>(ceil(iP.
get_i() /
m_scale)), 0);
975 int j_min = std::max<int>(
static_cast<int>(ceil(iP.
get_j() /
m_scale)), 0);
977 m_impl->displayImageROI(Itemp,
static_cast<gint
>(j_min),
static_cast<gint
>(i_min),
1003 m_impl->closeDisplay();
1016 m_impl->flushDisplay();
1028 const unsigned int )
1031 m_impl->flushDisplay();
1054 unsigned int h,
unsigned int thickness)
1061 if ((std::fabs(a) > std::numeric_limits<double>::epsilon()) &&
1062 (std::fabs(b) > std::numeric_limits<double>::epsilon())) {
1105 m_impl->displayText(ip, text, color,
m_scale);
1122 unsigned int thickness)
1128 m_impl->displayCircle(center, radius, color, fill, thickness,
m_scale);
1144 double i = ip.
get_i();
1145 double j = ip.
get_j();
1148 ip1.
set_i(i - size / 2);
1150 ip2.
set_i(i + size / 2);
1155 ip1.
set_j(j - size / 2);
1157 ip2.
set_j(j + size / 2);
1173 unsigned int thickness)
1180 m_impl->displayDotLine(ip1, ip2, color, thickness,
m_scale);
1194 unsigned int thickness)
1200 m_impl->displayLine(ip1, ip2, color, thickness,
m_scale);
1216 m_impl->displayPoint(ip, color, thickness,
m_scale);
1237 bool fill,
unsigned int thickness)
1243 m_impl->displayRectangle(topLeft, w, h, color, fill, thickness,
m_scale);
1263 bool fill,
unsigned int thickness)
1272 m_impl->displayRectangle(topLeft, w, h, color, fill, thickness,
m_scale);
1300 m_impl->displayRectangle(topLeft, w, h, color, fill, thickness,
m_scale);
1329 ret = m_impl->getClick(ip, button, blocking,
m_scale, GDK_BUTTON_PRESS);
1359 ret = m_impl->getClick(ip, button, blocking,
m_scale, GDK_BUTTON_PRESS);
1391 ret = m_impl->getClick(ip, button, blocking,
m_scale, GDK_BUTTON_PRESS);
1427 ret = m_impl->getClick(ip, button, blocking,
m_scale, GDK_BUTTON_RELEASE);
1443 m_impl->getImage(I,
static_cast<gint
>(
m_width),
static_cast<gint
>(
m_height));
1457 unsigned int depth = m_impl->getScreenDepth();
1483 ret = m_impl->getKeyboardEvent(key, blocking);
1515 ret = m_impl->getKeyboardEvent(key, blocking);
1540 ret = m_impl->getPointerMotionEvent(ip,
m_scale);
1561 m_impl->getPointerPosition(ip,
m_scale);
1586 unsigned int width, height;
1596 unsigned int width, height;
1603 #elif !defined(VISP_BUILD_SHARED_LIBS)
1605 void dummy_vpDisplayGTK() { };
Class to define RGB colors available for display functionalities.
Error that can be emitted by the vpDisplay class and its derivatives.
@ notInitializedError
Display not initialized.
void displayRectangle(const vpImagePoint &topLeft, unsigned int width, unsigned int height, const vpColor &color, bool fill=false, unsigned int thickness=1) VP_OVERRIDE
virtual ~vpDisplayGTK() VP_OVERRIDE
void displayCircle(const vpImagePoint ¢er, unsigned int radius, const vpColor &color, bool fill=false, unsigned int thickness=1) VP_OVERRIDE
void getScreenSize(unsigned int &screen_width, unsigned int &screen_height) VP_OVERRIDE
void getImage(vpImage< vpRGBa > &I) VP_OVERRIDE
Get the window pixmap and put it in vpRGBa image.
unsigned int getScreenWidth() VP_OVERRIDE
bool getClick(bool blocking=true) VP_OVERRIDE
unsigned int getScreenHeight() VP_OVERRIDE
void displayImageROI(const vpImage< unsigned char > &I, const vpImagePoint &iP, unsigned int width, unsigned int height) VP_OVERRIDE
void setFont(const std::string &fontname) VP_OVERRIDE
void displayImage(const vpImage< vpRGBa > &I) VP_OVERRIDE
void flushDisplay() VP_OVERRIDE
void displayLine(const vpImagePoint &ip1, const vpImagePoint &ip2, const vpColor &color, unsigned int thickness=1) VP_OVERRIDE
void setTitle(const std::string &win_title) VP_OVERRIDE
void displayArrow(const vpImagePoint &ip1, const vpImagePoint &ip2, const vpColor &color=vpColor::white, unsigned int w=4, unsigned int h=2, unsigned int thickness=1) VP_OVERRIDE
bool getPointerMotionEvent(vpImagePoint &ip) VP_OVERRIDE
void setWindowPosition(int win_x, int win_y) VP_OVERRIDE
unsigned int getScreenDepth()
get the window depth (8,16,24,32)
void closeDisplay() VP_OVERRIDE
void init(vpImage< unsigned char > &I, int win_x=-1, int win_y=-1, const std::string &win_title="") VP_OVERRIDE
bool getPointerPosition(vpImagePoint &ip) VP_OVERRIDE
void displayCross(const vpImagePoint &ip, unsigned int size, const vpColor &color, unsigned int thickness=1) VP_OVERRIDE
bool getKeyboardEvent(bool blocking=true) VP_OVERRIDE
bool getClickUp(vpImagePoint &ip, vpMouseButton::vpMouseButtonType &button, bool blocking=true) VP_OVERRIDE
void displayDotLine(const vpImagePoint &ip1, const vpImagePoint &ip2, const vpColor &color, unsigned int thickness=1) VP_OVERRIDE
void displayPoint(const vpImagePoint &ip, const vpColor &color, unsigned int thickness=1) VP_OVERRIDE
void clearDisplay(const vpColor &color=vpColor::white) VP_OVERRIDE
void displayText(const vpImagePoint &ip, const std::string &text, const vpColor &color=vpColor::green) VP_OVERRIDE
void flushDisplayROI(const vpImagePoint &iP, unsigned int width, unsigned int height) VP_OVERRIDE
Class that defines generic functionalities for display.
int m_windowXPosition
display position
int m_windowYPosition
display position
bool m_displayHasBeenInitialized
display has been initialized
void setScale(vpScaleType scaleType, unsigned int width, unsigned int height)
Class that defines a 2D point in an image. This class is useful for image processing and stores only ...
static double distance(const vpImagePoint &iP1, const vpImagePoint &iP2)
void subsample(unsigned int v_scale, unsigned int h_scale, vpImage< Type > &sampled) const
unsigned int getWidth() const
void resize(unsigned int h, unsigned int w)
resize the image : Image initialization
Type * bitmap
points toward the bitmap
unsigned int getHeight() const
static double sqr(double x)
static int round(double x)
unsigned char B
Blue component.
unsigned char R
Red component.
unsigned char G
Green component.
Defines a rectangle in the plane.
vpImagePoint getTopLeft() const
VISP_EXPORT int wait(double t0, double t)