首页 > 代码库 > 使用XCB编写X Window程序(04):在窗口中绘制文字

使用XCB编写X Window程序(04):在窗口中绘制文字

   在前面的几节中,我展示了使用XCB创建窗口、在窗口中画图以及捕获并处理事件。在这一篇中,我将展示在窗口中绘制文字。绘制文字当然离不开字体,所以我还会简单地探讨一下X Server的核心字体系统。老规矩,先上代码和运行效果图,代码如下:

 1 #include <stdlib.h> 2 #include <stdio.h> 3 #include <string.h> 4 #include <inttypes.h> 5 #include <xcb/xcb.h> 6  7  8 #define    WIDTH 600 9 #define    HEIGHT 40010 #define TEST_COOKIE(fn,errMessage) 11     cookie=fn; 12     error=xcb_request_check(connection,cookie); 13     if(error){ 14         fprintf(stderr, "Error: %s : %"PRIu8"\n", errMessage, error->error_code); 15     } 16 17 static void drawText(xcb_connection_t *connection, xcb_screen_t *screen, xcb_window_t window,18         int16_t x, int16_t y, const char *font_name, const char *string){19     /*cookie and error, for TEST_COOKIE */20     xcb_void_cookie_t cookie;21     xcb_generic_error_t *error;22 23     xcb_font_t font = xcb_generate_id(connection);24     TEST_COOKIE(xcb_open_font_checked(connection, font, strlen(font_name), font_name), "Can‘t open font");25     xcb_gcontext_t gc = xcb_generate_id(connection);26     uint32_t mask = XCB_GC_FOREGROUND | XCB_GC_BACKGROUND | XCB_GC_FONT;27     uint32_t values[3] = {screen->black_pixel, screen->white_pixel, font};28     TEST_COOKIE(xcb_create_gc_checked(connection, gc, window, mask, values), "Can‘t create GC")29     /*draw the text*/30     TEST_COOKIE(xcb_image_text_8_checked(connection, strlen(string), window, gc, x, y, string), "Can‘t draw text");31     /*close the font*/32     xcb_close_font(connection, font);33     xcb_free_gc(connection, gc);34 }35 36 int main()37 {38     xcb_connection_t *connection = xcb_connect(NULL, NULL);39     xcb_screen_t *screen = xcb_setup_roots_iterator( xcb_get_setup(connection)).data;40 41     /*Create the window*/42     xcb_window_t window = xcb_generate_id(connection);43     uint32_t mask = XCB_CW_BACK_PIXEL | XCB_CW_EVENT_MASK;44     uint32_t values[2];45     values[0] = screen->white_pixel;46     values[1] = XCB_EVENT_MASK_KEY_RELEASE | XCB_EVENT_MASK_BUTTON_PRESS |47         XCB_EVENT_MASK_EXPOSURE | XCB_EVENT_MASK_POINTER_MOTION;48     xcb_create_window(connection, screen->root_depth,49             window, screen->root, 20, 200,50             WIDTH, HEIGHT,51             0, XCB_WINDOW_CLASS_INPUT_OUTPUT,52             screen->root_visual,53             mask, values);54     xcb_map_window(connection, window);55     xcb_flush(connection);56 57     /*event loop*/58     xcb_generic_event_t *event;59     while((event = xcb_wait_for_event(connection))){60         switch(event->response_type & ~0x80){61             case XCB_EXPOSE:62                 {63                     drawText(connection, screen, window, 10, 20,"fixed", "Press ESC key to exit.");64                     drawText(connection, screen, window, 10, 40, "Dejavu Sans Mono", "Press ESC key to exit.");65                     drawText(connection, screen, window, 10, 60, "文泉驿正黑", "按ESC键退出。");66                     break;67                 }68             case XCB_KEY_RELEASE:69                 {70                     xcb_key_release_event_t *kr = (xcb_key_release_event_t*)event;71                     switch(kr->detail){72                         case 9: /* ESC */73                             {74                                 free(event);75                                 xcb_disconnect(connection);76                                 return 0;77                             }78                     }79                 }80         }81         free(event);82     }83     return 0;84 }

运行效果如下图:

 

  我承认,这段代码写得有点乱,不过说明怎么在窗口中绘制文字已经足够了。在窗口中绘制文字只需要用到如下函数:

  1、使用xcb_generate_id()生成一个ID,用来表示一个字体;

  2、使用xcb_open_font()或者xcb_open_font_checked()函数打开一个字体;

  3、使用前面打开的字体创建一个gc;创建gc前面已经讲过了,只需要调用xcb_generate_id()和xcb_create_gc()或者xcb_create_gc_checked()函数即可,不过在这里,要注意创建gc时的mask和values值,mask需要XCB_GC_FOREGROUND | XCB_GC_BACKGROUND | XCB_GC_FONT,分别代表前景色、背景色和字体,在values中设置相应的值;

  4、使用xcb_image_text_8()或者xcb_image_text_8_checked()函数在窗口上绘制文字。

 

  现在再回过头来看代码,是不是觉得豁然开朗?就这么几个API而已。而且可以看出,在XCB中,每一个有可能出错的函数都有另外一个加上了_checked的版本,带_checked的版本会返回一个cookie,通过xcb_request_check()函数可以检查该cookie,从而判断该函数调用是否成功。使用C语言的API就是这一点很麻烦,经常要去检查函数调用是否成功,所以使用C语言写程序经常会写成这样:

int error;error = some_function(...);if(error){    printf("...some message...");    ...}error = another_function(...);if(error){    printf("....another message...");    ...}

  这实在是太痛苦了。好在C语言中还有宏这么一个东西,所以我就在程序的开头定义了一个宏TEST_COOKIE,专门用来干这个事。宏定义得不好,还有改进的余地,不过已经让我的代码写起来和读起来都舒服了不少。

  在程序中,我使用了三个drawText,要求程序分别用fixed字体、Dejavu San Mono字体和文泉驿正黑字体绘制三行文字,但是最终的结果只有fixed那一行成功绘制,剩下的两行都在控制台显示有错误信息,都表示字体打不开、gc创建不了、所以文字当然绘制不了。之所以出现这种情况,就是因为这里选择的字体只能是X Server核心字体系统里安装的字体。我在《Linux江湖05:Linux桌面系统字体配置要略(下)》里面写过,Xorg中有两种字体系统,一种是X Server核心字体系统,另一种是Xft。虽然X Server核心字体系统经过多年的发展,功能已经很完善了,能比较完美地支持TrueType字体,但是Xorg官方反复建议大家不要使用核心字体系统,而是推荐大家使用Xft。可能是因为X Server核心字体系统对字体的安装和管理都比较麻烦吧。使用xlsfonts命令可以列出X Server中安装有哪些字体,如下图:

  可以看到每一个字体的信息都又臭又长,只有最后三行短一点,所以我才在程序中选择了最短的fixed字体作为示例。而且X Server核心字体系统安装字体时需要使用mkfontdir、mkfontscale等极其麻烦的命令。

  最后有一点想不通的是,为什么xcb_image_text_8()函数后面有一个8呢?是不是还有xcb_image_text_16()和xcb_image_text_32()这样的函数呢?没关系,使用Ctrl+]跳到XCB的头文件里面看一下。如下图:

  里面的注释还是很详细的。特别是下图的最后一句:

最后一句再次强调X core fonts是过时的、不建议使用的,再次推荐大家使用Xft。我在写这一系列XCB文章的时候就说过,使用XCB只是为了学习,学习X协议的底层知识,学习GUI系统的构建原理等。真正写应用程序的时候,是没有谁使用XCB这么底层的库的。而高层次的库中需要绘制文字时,估计早就都是Xft一统天下了,所以并不需要我们费太多的心。

 

(京山游侠于2014-07-15发布于博客园,转载请注明出处。)