complete shedchange text information
[lttv.git] / ltt / branches / poly / lttv / modules / guiControlFlow / Draw_Item.c
CommitLineData
cf6cb7e0 1/******************************************************************************
2 * Draw_Item.c
3 *
4 * This file contains methods responsible for drawing a generic type of data
5 * in a drawable. Doing this generically will permit user defined drawing
6 * behavior in a later time.
7 *
b782dd11 8 * This file provides an API which is meant to be reusable for all viewers that
9 * need to show information in line, icon, text, background or point form in
10 * a drawable area having time for x axis. The y axis, in the control flow
11 * viewer case, is corresponding to the different processes, but it can be
12 * reused integrally for cpu, and eventually locks, buffers, network
13 * interfaces... What will differ between the viewers is the precise
14 * information which interests us. We may think that the most useful
15 * information for control flow are some specific events, like schedule
16 * change, and processes'states. It may differ for a cpu viewer : the
17 * interesting information could be more the execution mode of each cpu.
18 * This API in meant to make viewer's writers life easier : it will become
19 * a simple choice of icons and line types for the precise information
20 * the viewer has to provide (agremented with keeping supplementary records
21 * and modifying slightly the DrawContext to suit the needs.)
22 *
f0728492 23 * We keep each data type in attributes, keys to specific information
24 * being formed from the GQuark corresponding to the information received.
25 * (facilities / facility_name / events / eventname.)
26 * (cpus/cpu_name, process_states/ps_name,
27 * execution_modes/em_name, execution_submodes/es_name).
cf6cb7e0 28 * The goal is then to provide a generic way to print information on the
29 * screen for all this different information.
30 *
31 * Information can be printed as
32 *
33 * - text (text + color + size + position (over or under line)
34 * - icon (icon filename, corresponding to a loaded icon, accessible through
35 * a GQuark. Icons are loaded statically at the guiControlFlow level during
36 * module initialization and can be added on the fly if not present in the
37 * GQuark.) The habitual place for xpm icons is in
38 * ${prefix}/share/LinuxTraceToolkit.) + position (over or under line)
39 * - line (color, width, style)
189a5d08 40 * - Arc (big points) (color, size)
cf6cb7e0 41 * - background color (color)
42 *
189a5d08 43 * An item is a leaf of the attributes tree. It is, in that case, including
44 * all kind of events categories we can have. It then associates each category
45 * with one or more actions (drawing something) or nothing.
46 *
7d5ffafa 47 * Each item has an array of hooks (hook list). Each hook represents an
48 * operation to perform. We seek the array each time we want to
a2e850ff 49 * draw an item. We execute each operation in order. An operation type
50 * is associated with each hook to permit user listing and modification
51 * of these operations. The operation type is also used to find the
52 * corresponding priority for the sorting. Operation type and priorities
53 * are enum and a static int table.
cf6cb7e0 54 *
55 * The array has to be sorted by priority each time we add a task in it.
a2e850ff 56 * A priority is associated with each operation type. It permits
cf6cb7e0 57 * to perform background color selection before line or text drawing. We also
58 * draw lines before text, so the text appears over the lines.
59 *
60 * Executing all the arrays of operations for a specific event (which
61 * implies information for state, event, cpu, execution mode and submode)
62 * has to be done in a same DrawContext. The goal there is to keep the offset
63 * of the text and icons over and under the middle line, so a specific
64 * event could be printed as ( R Si 0 for running, scheduled in, cpu 0 ),
7d5ffafa 65 * text being easy to replace with icons. The DrawContext is passed as
66 * call_data for the operation hooks.
cf6cb7e0 67 *
b782dd11 68 * We use the lttv global attributes to keep track of the loaded icons.
69 * If we need an icon, we look for it in the icons / icon name pathname.
70 * If found, we use the pointer to it. If not, we load the pixmap in
1a31868c 71 * memory and set the pointer to the GdkPixmap in the attributes. The
72 * structure pointed to contains the pixmap and the mask bitmap.
b782dd11 73 *
cf6cb7e0 74 * Author : Mathieu Desnoyers, October 2003
75 */
7d5ffafa 76
77#include <glib.h>
09e2606f 78#include <gtk/gtk.h>
79#include <gdk/gdk.h>
7d5ffafa 80#include <lttv/hook.h>
f0728492 81#include <lttv/attribute.h>
82#include <lttv/iattribute.h>
1a31868c 83#include <string.h>
7d5ffafa 84
b782dd11 85#include <lttv/processTrace.h>
86#include <lttv/state.h>
87
09e2606f 88#include "Draw_Item.h"
1a31868c 89
90
91#define MAX_PATH_LEN 256
92
b782dd11 93/* Drawing hook functions */
4c69e0cc 94gboolean draw_text( void *hook_data, void *call_data)
b782dd11 95{
96 PropertiesText *Properties = (PropertiesText*)hook_data;
97 DrawContext *Draw_Context = (DrawContext*)call_data;
09e2606f 98
99 PangoContext *context;
100 PangoLayout *layout;
50439712 101 PangoAttribute *attribute;
09e2606f 102 PangoFontDescription *FontDesc;// = pango_font_description_new();
103 gint Font_Size;
104 PangoRectangle ink_rect;
105
34b04882 106 layout = Draw_Context->pango_layout;
9f6858a9 107
09e2606f 108 context = pango_layout_get_context(layout);
109 FontDesc = pango_context_get_font_description(context);
9f6858a9 110
09e2606f 111 pango_font_description_set_size(FontDesc, Properties->size*PANGO_SCALE);
50439712 112 pango_layout_context_changed(layout);
113
09e2606f 114 pango_layout_set_text(layout, Properties->Text, -1);
115 pango_layout_get_pixel_extents(layout, &ink_rect, NULL);
116 switch(Properties->position) {
117 case OVER:
9f6858a9 118 gdk_draw_layout_with_colors(Draw_Context->drawable,
119 Draw_Context->gc,
09e2606f 120 Draw_Context->Current->modify_over->x,
121 Draw_Context->Current->modify_over->y,
9f6858a9 122 layout, Properties->foreground, Properties->background);
09e2606f 123 Draw_Context->Current->modify_over->x += ink_rect.width;
124
125 break;
126 case MIDDLE:
9f6858a9 127 gdk_draw_layout_with_colors(Draw_Context->drawable,
128 Draw_Context->gc,
09e2606f 129 Draw_Context->Current->modify_middle->x,
130 Draw_Context->Current->modify_middle->y,
9f6858a9 131 layout, Properties->foreground, Properties->background);
09e2606f 132 Draw_Context->Current->modify_middle->x += ink_rect.width;
133 break;
134 case UNDER:
9f6858a9 135 gdk_draw_layout_with_colors(Draw_Context->drawable,
136 Draw_Context->gc,
09e2606f 137 Draw_Context->Current->modify_under->x,
138 Draw_Context->Current->modify_under->y,
9f6858a9 139 layout, Properties->foreground, Properties->background);
09e2606f 140 Draw_Context->Current->modify_under->x += ink_rect.width;
141 break;
142 }
143
09e2606f 144 return 0;
b782dd11 145}
146
1a31868c 147
148/* To speed up the process, search in already loaded icons list first. Only
149 * load it if not present.
150 */
4c69e0cc 151gboolean draw_icon( void *hook_data, void *call_data)
b782dd11 152{
153 PropertiesIcon *Properties = (PropertiesIcon*)hook_data;
154 DrawContext *Draw_Context = (DrawContext*)call_data;
155
1a31868c 156 LttvIAttribute *attributes = LTTV_IATTRIBUTE(lttv_global_attributes());
157 LttvAttributeValue value;
158 gchar icon_name[MAX_PATH_LEN] = "icons/";
159 IconStruct *icon_info;
8d088fb2 160
1a31868c 161 strcat(icon_name, Properties->icon_name);
162
163 g_assert(lttv_iattribute_find_by_path(attributes, icon_name,
164 LTTV_POINTER, &value));
165 if(*(value.v_pointer) == NULL)
166 {
167 *(value.v_pointer) = icon_info = g_new(IconStruct,1);
168
169 icon_info->pixmap = gdk_pixmap_create_from_xpm(Draw_Context->drawable,
170 &icon_info->mask, NULL, Properties->icon_name);
171 }
172 else
173 {
174 icon_info = *(value.v_pointer);
175 }
176
177 gdk_gc_set_clip_mask(Draw_Context->gc, icon_info->mask);
178
09e2606f 179 switch(Properties->position) {
180 case OVER:
1a31868c 181 gdk_gc_set_clip_origin(
182 Draw_Context->gc,
183 Draw_Context->Current->modify_over->x,
184 Draw_Context->Current->modify_over->y);
09e2606f 185 gdk_draw_drawable(Draw_Context->drawable,
1a31868c 186 Draw_Context->gc,
187 icon_info->pixmap,
09e2606f 188 0, 0,
189 Draw_Context->Current->modify_over->x,
190 Draw_Context->Current->modify_over->y,
191 Properties->width, Properties->height);
192
193 Draw_Context->Current->modify_over->x += Properties->width;
194
195 break;
196 case MIDDLE:
1a31868c 197 gdk_gc_set_clip_origin(
198 Draw_Context->gc,
199 Draw_Context->Current->modify_middle->x,
200 Draw_Context->Current->modify_middle->y);
09e2606f 201 gdk_draw_drawable(Draw_Context->drawable,
1a31868c 202 Draw_Context->gc,
203 icon_info->pixmap,
09e2606f 204 0, 0,
205 Draw_Context->Current->modify_middle->x,
206 Draw_Context->Current->modify_middle->y,
207 Properties->width, Properties->height);
208
09e2606f 209 Draw_Context->Current->modify_middle->x += Properties->width;
210 break;
211 case UNDER:
1a31868c 212 gdk_gc_set_clip_origin(
213 Draw_Context->gc,
214 Draw_Context->Current->modify_under->x,
215 Draw_Context->Current->modify_under->y);
09e2606f 216 gdk_draw_drawable(Draw_Context->drawable,
1a31868c 217 Draw_Context->gc,
218 icon_info->pixmap,
09e2606f 219 0, 0,
220 Draw_Context->Current->modify_under->x,
221 Draw_Context->Current->modify_under->y,
222 Properties->width, Properties->height);
223
224 Draw_Context->Current->modify_under->x += Properties->width;
225 break;
226 }
227
1a31868c 228 gdk_gc_set_clip_origin(Draw_Context->gc, 0, 0);
229 gdk_gc_set_clip_mask(Draw_Context->gc, NULL);
09e2606f 230
231 return 0;
b782dd11 232}
233
4c69e0cc 234gboolean draw_line( void *hook_data, void *call_data)
b782dd11 235{
236 PropertiesLine *Properties = (PropertiesLine*)hook_data;
237 DrawContext *Draw_Context = (DrawContext*)call_data;
238
09e2606f 239 gdk_gc_set_foreground(Draw_Context->gc, Properties->color);
240 gdk_gc_set_line_attributes( Draw_Context->gc,
241 Properties->line_width,
242 Properties->style,
243 GDK_CAP_BUTT,
244 GDK_JOIN_MITER);
245
246 switch(Properties->position) {
247 case OVER:
248 drawing_draw_line(
249 NULL, Draw_Context->drawable,
250 Draw_Context->Previous->over->x,
251 Draw_Context->Previous->over->y,
252 Draw_Context->Current->over->x,
253 Draw_Context->Current->over->y,
254 Draw_Context->gc);
255 break;
256 case MIDDLE:
257 drawing_draw_line(
258 NULL, Draw_Context->drawable,
259 Draw_Context->Previous->middle->x,
260 Draw_Context->Previous->middle->y,
261 Draw_Context->Current->middle->x,
262 Draw_Context->Current->middle->y,
263 Draw_Context->gc);
264 break;
265 case UNDER:
266 drawing_draw_line(
267 NULL, Draw_Context->drawable,
268 Draw_Context->Previous->under->x,
269 Draw_Context->Previous->under->y,
270 Draw_Context->Current->under->x,
271 Draw_Context->Current->under->y,
272 Draw_Context->gc);
273
274 break;
275 }
276
277 return 0;
b782dd11 278}
279
4c69e0cc 280gboolean draw_arc( void *hook_data, void *call_data)
b782dd11 281{
282 PropertiesArc *Properties = (PropertiesArc*)hook_data;
283 DrawContext *Draw_Context = (DrawContext*)call_data;
284
09e2606f 285 gdk_gc_set_foreground(Draw_Context->gc, Properties->color);
286
287 switch(Properties->position) {
288 case OVER:
289 gdk_draw_arc(Draw_Context->drawable, Draw_Context->gc,
290 Properties->filled,
291 Draw_Context->Current->modify_over->x,
292 Draw_Context->Current->modify_over->y,
293 Properties->size, Properties->size, 0, 360*64);
294 Draw_Context->Current->modify_over->x += Properties->size;
09e2606f 295 break;
296 case MIDDLE:
297 gdk_draw_arc(Draw_Context->drawable, Draw_Context->gc,
298 Properties->filled,
299 Draw_Context->Current->modify_middle->x,
300 Draw_Context->Current->modify_middle->y,
301 Properties->size, Properties->size, 0, 360*64);
302 Draw_Context->Current->modify_middle->x += Properties->size;
303
304 break;
305 case UNDER:
306 gdk_draw_arc(Draw_Context->drawable, Draw_Context->gc,
307 Properties->filled,
308 Draw_Context->Current->modify_under->x,
309 Draw_Context->Current->modify_under->y,
310 Properties->size, Properties->size, 0, 360*64);
311 Draw_Context->Current->modify_under->x += Properties->size;
312
313 break;
314 }
315
316
317 return 0;
b782dd11 318}
319
4c69e0cc 320gboolean draw_bg( void *hook_data, void *call_data)
b782dd11 321{
322 PropertiesBG *Properties = (PropertiesBG*)hook_data;
323 DrawContext *Draw_Context = (DrawContext*)call_data;
324
09e2606f 325 gdk_gc_set_foreground(Draw_Context->gc, Properties->color);
326
327
328 gdk_draw_rectangle(Draw_Context->drawable, Draw_Context->gc,
329 TRUE,
330 Draw_Context->Previous->over->x,
331 Draw_Context->Previous->over->y,
332 Draw_Context->Current->over->x - Draw_Context->Previous->over->x,
333 Draw_Context->Previous->under->y);
334
335 return 0;
b782dd11 336}
337
338
This page took 0.038569 seconds and 4 git commands to generate.