Mercurial > pidgin.yaz
annotate finch/libgnt/gnttextview.c @ 16015:de619a9523f9
sf patch #1694728, from Andrew Gaul
plug 2 memleaks when setting icons. One of them
was leaking the size of the icon for Yahoo!, the other
was leaking a patch to an icon on the local file
system for oscar. Fantastico
author | Mark Doliner <mark@kingant.net> |
---|---|
date | Sun, 08 Apr 2007 22:29:42 +0000 |
parents | f00f2e283ffb |
children | 1f24fd9a6237 |
rev | line source |
---|---|
15818 | 1 #include "gnttextview.h" |
2 #include "gntutils.h" | |
3 | |
4 #include <string.h> | |
5 | |
6 enum | |
7 { | |
8 SIGS = 1, | |
9 }; | |
10 | |
11 typedef struct | |
12 { | |
13 GntTextFormatFlags tvflag; | |
14 chtype flags; | |
15 int start; | |
16 int end; /* This is the next byte of the last character of this segment */ | |
17 } GntTextSegment; | |
18 | |
19 typedef struct | |
20 { | |
21 GList *segments; /* A list of GntTextSegments */ | |
22 int length; /* The current length of the line so far (ie. onscreen width) */ | |
23 gboolean soft; /* TRUE if it's an overflow from prev. line */ | |
24 } GntTextLine; | |
25 | |
26 typedef struct | |
27 { | |
28 char *name; | |
29 int start; | |
30 int end; | |
31 } GntTextTag; | |
32 | |
33 static GntWidgetClass *parent_class = NULL; | |
34 | |
35 static gchar *select_start; | |
36 static gchar *select_end; | |
37 static gboolean double_click; | |
38 | |
39 static void | |
40 gnt_text_view_draw(GntWidget *widget) | |
41 { | |
42 GntTextView *view = GNT_TEXT_VIEW(widget); | |
43 int i = 0; | |
44 GList *lines; | |
45 int rows, scrcol; | |
46 | |
15931
f00f2e283ffb
Some define changes. This helps in generating the python bindings.
Sadrul Habib Chowdhury <imadil@gmail.com>
parents:
15818
diff
changeset
|
47 wbkgd(widget->window, COLOR_PAIR(GNT_COLOR_NORMAL)); |
15818 | 48 werase(widget->window); |
49 | |
50 for (i = 0, lines = view->list; i < widget->priv.height && lines; i++, lines = lines->next) | |
51 { | |
52 GList *iter; | |
53 GntTextLine *line = lines->data; | |
54 | |
55 wmove(widget->window, widget->priv.height - 1 - i, 0); | |
56 | |
57 for (iter = line->segments; iter; iter = iter->next) | |
58 { | |
59 GntTextSegment *seg = iter->data; | |
60 char *end = view->string->str + seg->end; | |
61 char back = *end; | |
62 chtype fl = seg->flags; | |
63 *end = '\0'; | |
64 if (select_start < view->string->str + seg->start && select_end > view->string->str + seg->end) { | |
65 fl |= A_REVERSE; | |
66 wattrset(widget->window, fl); | |
67 wprintw(widget->window, "%s", (view->string->str + seg->start)); | |
68 } else if (select_start && select_end && | |
69 ((select_start >= view->string->str + seg->start && select_start <= view->string->str + seg->end) || | |
70 (select_end <= view->string->str + seg->end && select_start <= view->string->str + seg->start))) { | |
71 char *cur = view->string->str + seg->start; | |
72 while (*cur != '\0') { | |
73 gchar *last = g_utf8_next_char(cur); | |
74 gchar *str; | |
75 if (cur >= select_start && cur <= select_end) | |
76 fl |= A_REVERSE; | |
77 else | |
78 fl = seg->flags; | |
79 str = g_strndup(cur, last - cur); | |
80 wattrset(widget->window, fl); | |
81 waddstr(widget->window, str); | |
82 g_free(str); | |
83 cur = g_utf8_next_char(cur); | |
84 } | |
85 } else { | |
86 wattrset(widget->window, fl); | |
87 wprintw(widget->window, "%s", (view->string->str + seg->start)); | |
88 } | |
89 *end = back; | |
90 } | |
91 wattroff(widget->window, A_UNDERLINE | A_BLINK | A_REVERSE); | |
92 whline(widget->window, ' ', widget->priv.width - line->length - 1); | |
93 } | |
94 | |
95 scrcol = widget->priv.width - 1; | |
96 rows = widget->priv.height - 2; | |
97 if (rows > 0) | |
98 { | |
99 int total = g_list_length(g_list_first(view->list)); | |
100 int showing, position, up, down; | |
101 | |
102 showing = rows * rows / total + 1; | |
103 showing = MIN(rows, showing); | |
104 | |
105 total -= rows; | |
106 up = g_list_length(lines); | |
107 down = total - up; | |
108 | |
109 position = (rows - showing) * up / MAX(1, up + down); | |
110 position = MAX((lines != NULL), position); | |
111 | |
112 if (showing + position > rows) | |
113 position = rows - showing; | |
114 | |
115 if (showing + position == rows && view->list && view->list->prev) | |
116 position = MAX(1, rows - 1 - showing); | |
117 else if (showing + position < rows && view->list && !view->list->prev) | |
118 position = rows - showing; | |
119 | |
120 mvwvline(widget->window, position + 1, scrcol, | |
121 ACS_CKBOARD | COLOR_PAIR(GNT_COLOR_HIGHLIGHT_D), showing); | |
122 } | |
123 | |
124 mvwaddch(widget->window, 0, scrcol, | |
125 (lines ? ACS_UARROW : ' ') | COLOR_PAIR(GNT_COLOR_HIGHLIGHT_D)); | |
126 mvwaddch(widget->window, widget->priv.height - 1, scrcol, | |
127 ((view->list && view->list->prev) ? ACS_DARROW : ' ') | | |
128 COLOR_PAIR(GNT_COLOR_HIGHLIGHT_D)); | |
129 | |
130 GNTDEBUG; | |
131 } | |
132 | |
133 static void | |
134 gnt_text_view_size_request(GntWidget *widget) | |
135 { | |
136 if (!GNT_WIDGET_IS_FLAG_SET(widget, GNT_WIDGET_MAPPED)) | |
137 { | |
138 gnt_widget_set_size(widget, 64, 20); | |
139 } | |
140 } | |
141 | |
142 static void | |
143 gnt_text_view_map(GntWidget *widget) | |
144 { | |
145 if (widget->priv.width == 0 || widget->priv.height == 0) | |
146 gnt_widget_size_request(widget); | |
147 GNTDEBUG; | |
148 } | |
149 | |
150 static gboolean | |
151 gnt_text_view_key_pressed(GntWidget *widget, const char *text) | |
152 { | |
153 return FALSE; | |
154 } | |
155 | |
156 static void | |
157 free_text_segment(gpointer data, gpointer null) | |
158 { | |
159 GntTextSegment *seg = data; | |
160 g_free(seg); | |
161 } | |
162 | |
163 static void | |
164 free_text_line(gpointer data, gpointer null) | |
165 { | |
166 GntTextLine *line = data; | |
167 g_list_foreach(line->segments, free_text_segment, NULL); | |
168 g_list_free(line->segments); | |
169 g_free(line); | |
170 } | |
171 | |
172 static void | |
173 free_tag(gpointer data, gpointer null) | |
174 { | |
175 GntTextTag *tag = data; | |
176 g_free(tag->name); | |
177 g_free(tag); | |
178 } | |
179 | |
180 static void | |
181 gnt_text_view_destroy(GntWidget *widget) | |
182 { | |
183 GntTextView *view = GNT_TEXT_VIEW(widget); | |
184 view->list = g_list_first(view->list); | |
185 g_list_foreach(view->list, free_text_line, NULL); | |
186 g_list_free(view->list); | |
187 g_list_foreach(view->tags, free_tag, NULL); | |
188 g_list_free(view->tags); | |
189 g_string_free(view->string, TRUE); | |
190 } | |
191 | |
192 static char * | |
193 gnt_text_view_get_p(GntTextView *view, int x, int y) | |
194 { | |
195 int i = 0; | |
196 GntWidget *wid = GNT_WIDGET(view); | |
197 GntTextLine *line; | |
198 GList *lines; | |
199 GList *segs; | |
200 GntTextSegment *seg; | |
201 gchar *pos; | |
202 | |
203 y = wid->priv.height - y; | |
204 if (g_list_length(view->list) < y) { | |
205 x = 0; | |
206 y = g_list_length(view->list) - 1; | |
207 } | |
208 | |
209 lines = g_list_nth(view->list, y - 1); | |
210 if (!lines) | |
211 return NULL; | |
212 do { | |
213 line = lines->data; | |
214 lines = lines->next; | |
215 } while (line && !line->segments && lines); | |
216 | |
217 if (!line || !line->segments) /* no valid line */ | |
218 return NULL; | |
219 segs = line->segments; | |
220 seg = (GntTextSegment *)segs->data; | |
221 pos = view->string->str + seg->start; | |
222 x = MIN(x, line->length); | |
223 while (++i <= x) { | |
224 gunichar *u; | |
225 pos = g_utf8_next_char(pos); | |
226 u = g_utf8_to_ucs4(pos, -1, NULL, NULL, NULL); | |
227 if (u && g_unichar_iswide(*u)) | |
228 i++; | |
229 g_free(u); | |
230 } | |
231 return pos; | |
232 } | |
233 | |
234 static GString * | |
235 select_word_text(GntTextView *view, gchar *c) | |
236 { | |
237 gchar *start = c; | |
238 gchar *end = c; | |
239 gchar *t, *endsize; | |
240 while ((t = g_utf8_prev_char(start))) { | |
241 if (!g_ascii_isspace(*t)) { | |
242 if (start == view->string->str) | |
243 break; | |
244 start = t; | |
245 } else | |
246 break; | |
247 } | |
248 while ((t = g_utf8_next_char(end))) { | |
249 if (!g_ascii_isspace(*t)) | |
250 end = t; | |
251 else | |
252 break; | |
253 } | |
254 select_start = start; | |
255 select_end = end; | |
256 endsize = g_utf8_next_char(select_end); /* End at the correct byte */ | |
257 return g_string_new_len(start, endsize - start); | |
258 } | |
259 | |
260 static gboolean too_slow(gpointer n) | |
261 { | |
262 double_click = FALSE; | |
263 return FALSE; | |
264 } | |
265 | |
266 static gboolean | |
267 gnt_text_view_clicked(GntWidget *widget, GntMouseEvent event, int x, int y) | |
268 { | |
269 if (event == GNT_MOUSE_SCROLL_UP) { | |
270 gnt_text_view_scroll(GNT_TEXT_VIEW(widget), -1); | |
271 } else if (event == GNT_MOUSE_SCROLL_DOWN) { | |
272 gnt_text_view_scroll(GNT_TEXT_VIEW(widget), 1); | |
273 } else if (event == GNT_LEFT_MOUSE_DOWN) { | |
274 select_start = gnt_text_view_get_p(GNT_TEXT_VIEW(widget), x - widget->priv.x, y - widget->priv.y); | |
275 g_timeout_add(500, too_slow, NULL); | |
276 } else if (event == GNT_MOUSE_UP) { | |
277 if (select_start) { | |
278 GString *clip; | |
279 select_end = gnt_text_view_get_p(GNT_TEXT_VIEW(widget), x - widget->priv.x, y - widget->priv.y); | |
280 if (select_end < select_start) { | |
281 gchar *t = select_start; | |
282 select_start = select_end; | |
283 select_end = t; | |
284 } | |
285 if (select_start == select_end) { | |
286 if (double_click) { | |
287 clip = select_word_text(GNT_TEXT_VIEW(widget), select_start); | |
288 double_click = FALSE; | |
289 } else { | |
290 double_click = TRUE; | |
291 select_start = 0; | |
292 select_end = 0; | |
293 gnt_widget_draw(widget); | |
294 return TRUE; | |
295 } | |
296 } else { | |
297 gchar *endsize = g_utf8_next_char(select_end); /* End at the correct byte */ | |
298 clip = g_string_new_len(select_start, endsize - select_start); | |
299 } | |
300 gnt_widget_draw(widget); | |
301 gnt_set_clipboard_string(clip->str); | |
302 g_string_free(clip, TRUE); | |
303 } | |
304 } else | |
305 return FALSE; | |
306 return TRUE; | |
307 } | |
308 | |
309 static void | |
310 gnt_text_view_reflow(GntTextView *view) | |
311 { | |
312 /* This is pretty ugly, and inefficient. Someone do something about it. */ | |
313 GntTextLine *line; | |
314 GList *back, *iter, *list; | |
315 GString *string; | |
316 int pos = 0; /* no. of 'real' lines */ | |
317 | |
318 list = view->list; | |
319 while (list->prev) { | |
320 line = list->data; | |
321 if (!line->soft) | |
322 pos++; | |
323 list = list->prev; | |
324 } | |
325 | |
326 back = g_list_last(view->list); | |
327 view->list = NULL; | |
328 | |
329 string = view->string; | |
330 view->string = NULL; | |
331 gnt_text_view_clear(view); | |
332 | |
333 view->string = g_string_set_size(view->string, string->len); | |
334 view->string->len = 0; | |
335 GNT_WIDGET_SET_FLAGS(GNT_WIDGET(view), GNT_WIDGET_DRAWING); | |
336 | |
337 for (; back; back = back->prev) { | |
338 line = back->data; | |
339 if (back->next && !line->soft) { | |
340 gnt_text_view_append_text_with_flags(view, "\n", GNT_TEXT_FLAG_NORMAL); | |
341 } | |
342 | |
343 for (iter = line->segments; iter; iter = iter->next) { | |
344 GntTextSegment *seg = iter->data; | |
345 char *start = string->str + seg->start; | |
346 char *end = string->str + seg->end; | |
347 char back = *end; | |
348 *end = '\0'; | |
349 gnt_text_view_append_text_with_flags(view, start, seg->tvflag); | |
350 *end = back; | |
351 } | |
352 free_text_line(line, NULL); | |
353 } | |
354 g_list_free(list); | |
355 | |
356 list = view->list = g_list_first(view->list); | |
357 /* Go back to the line that was in view before resizing started */ | |
358 while (pos--) { | |
359 while (((GntTextLine*)list->data)->soft) | |
360 list = list->next; | |
361 list = list->next; | |
362 } | |
363 view->list = list; | |
364 GNT_WIDGET_UNSET_FLAGS(GNT_WIDGET(view), GNT_WIDGET_DRAWING); | |
365 if (GNT_WIDGET(view)->window) | |
366 gnt_widget_draw(GNT_WIDGET(view)); | |
367 g_string_free(string, TRUE); | |
368 } | |
369 | |
370 static void | |
371 gnt_text_view_size_changed(GntWidget *widget, int w, int h) | |
372 { | |
373 if (w != widget->priv.width) { | |
374 gnt_text_view_reflow(GNT_TEXT_VIEW(widget)); | |
375 } | |
376 } | |
377 | |
378 static void | |
379 gnt_text_view_class_init(GntTextViewClass *klass) | |
380 { | |
381 parent_class = GNT_WIDGET_CLASS(klass); | |
382 parent_class->destroy = gnt_text_view_destroy; | |
383 parent_class->draw = gnt_text_view_draw; | |
384 parent_class->map = gnt_text_view_map; | |
385 parent_class->size_request = gnt_text_view_size_request; | |
386 parent_class->key_pressed = gnt_text_view_key_pressed; | |
387 parent_class->clicked = gnt_text_view_clicked; | |
388 parent_class->size_changed = gnt_text_view_size_changed; | |
389 | |
390 GNTDEBUG; | |
391 } | |
392 | |
393 static void | |
394 gnt_text_view_init(GTypeInstance *instance, gpointer class) | |
395 { | |
396 GntWidget *widget = GNT_WIDGET(instance); | |
397 | |
398 GNT_WIDGET_SET_FLAGS(GNT_WIDGET(instance), GNT_WIDGET_GROW_Y | GNT_WIDGET_GROW_X); | |
399 | |
400 widget->priv.minw = 5; | |
401 widget->priv.minh = 2; | |
402 GNTDEBUG; | |
403 } | |
404 | |
405 /****************************************************************************** | |
406 * GntTextView API | |
407 *****************************************************************************/ | |
408 GType | |
409 gnt_text_view_get_gtype(void) | |
410 { | |
411 static GType type = 0; | |
412 | |
413 if(type == 0) | |
414 { | |
415 static const GTypeInfo info = { | |
416 sizeof(GntTextViewClass), | |
417 NULL, /* base_init */ | |
418 NULL, /* base_finalize */ | |
419 (GClassInitFunc)gnt_text_view_class_init, | |
420 NULL, /* class_finalize */ | |
421 NULL, /* class_data */ | |
422 sizeof(GntTextView), | |
423 0, /* n_preallocs */ | |
424 gnt_text_view_init, /* instance_init */ | |
425 NULL /* value_table */ | |
426 }; | |
427 | |
428 type = g_type_register_static(GNT_TYPE_WIDGET, | |
429 "GntTextView", | |
430 &info, 0); | |
431 } | |
432 | |
433 return type; | |
434 } | |
435 | |
436 GntWidget *gnt_text_view_new() | |
437 { | |
15931
f00f2e283ffb
Some define changes. This helps in generating the python bindings.
Sadrul Habib Chowdhury <imadil@gmail.com>
parents:
15818
diff
changeset
|
438 GntWidget *widget = g_object_new(GNT_TYPE_TEXT_VIEW, NULL); |
15818 | 439 GntTextView *view = GNT_TEXT_VIEW(widget); |
440 GntTextLine *line = g_new0(GntTextLine, 1); | |
441 | |
442 GNT_WIDGET_SET_FLAGS(widget, GNT_WIDGET_NO_BORDER | GNT_WIDGET_NO_SHADOW); | |
443 | |
444 view->string = g_string_new(NULL); | |
445 view->list = g_list_append(view->list, line); | |
446 | |
447 return widget; | |
448 } | |
449 | |
450 void gnt_text_view_append_text_with_flags(GntTextView *view, const char *text, GntTextFormatFlags flags) | |
451 { | |
452 gnt_text_view_append_text_with_tag(view, text, flags, NULL); | |
453 } | |
454 | |
455 void gnt_text_view_append_text_with_tag(GntTextView *view, const char *text, | |
456 GntTextFormatFlags flags, const char *tagname) | |
457 { | |
458 GntWidget *widget = GNT_WIDGET(view); | |
459 int fl = 0; | |
460 const char *start, *end; | |
461 GList *list = view->list; | |
462 GntTextLine *line; | |
463 int len; | |
464 | |
465 if (text == NULL || *text == '\0') | |
466 return; | |
467 | |
468 fl = gnt_text_format_flag_to_chtype(flags); | |
469 | |
470 len = view->string->len; | |
471 view->string = g_string_append(view->string, text); | |
472 | |
473 if (tagname) { | |
474 GntTextTag *tag = g_new0(GntTextTag, 1); | |
475 tag->name = g_strdup(tagname); | |
476 tag->start = len; | |
477 tag->end = view->string->len; | |
478 view->tags = g_list_append(view->tags, tag); | |
479 } | |
480 | |
481 view->list = g_list_first(view->list); | |
482 | |
483 start = end = view->string->str + len; | |
484 | |
485 while (*start) { | |
486 GntTextSegment *seg = NULL; | |
487 | |
488 if (*end == '\n' || *end == '\r') { | |
489 end++; | |
490 start = end; | |
491 gnt_text_view_next_line(view); | |
492 view->list = g_list_first(view->list); | |
493 continue; | |
494 } | |
495 | |
496 line = view->list->data; | |
497 if (line->length == widget->priv.width - 1) { | |
498 /* The last added line was exactly the same width as the widget */ | |
499 line = g_new0(GntTextLine, 1); | |
500 line->soft = TRUE; | |
501 view->list = g_list_prepend(view->list, line); | |
502 } | |
503 | |
504 if ((end = strchr(start, '\n')) != NULL || | |
505 (end = strchr(start, '\r')) != NULL) { | |
506 len = gnt_util_onscreen_width(start, end - 1); | |
507 if (len >= widget->priv.width - line->length - 1) { | |
508 end = NULL; | |
509 } | |
510 } | |
511 | |
512 if (end == NULL) | |
513 end = gnt_util_onscreen_width_to_pointer(start, | |
514 widget->priv.width - line->length - 1, &len); | |
515 | |
516 /* Try to append to the previous segment if possible */ | |
517 if (line->segments) { | |
518 seg = g_list_last(line->segments)->data; | |
519 if (seg->flags != fl) | |
520 seg = NULL; | |
521 } | |
522 | |
523 if (seg == NULL) { | |
524 seg = g_new0(GntTextSegment, 1); | |
525 seg->start = start - view->string->str; | |
526 seg->tvflag = flags; | |
527 seg->flags = fl; | |
528 line->segments = g_list_append(line->segments, seg); | |
529 } | |
530 seg->end = end - view->string->str; | |
531 line->length += len; | |
532 | |
533 start = end; | |
534 if (*end && *end != '\n' && *end != '\r') { | |
535 line = g_new0(GntTextLine, 1); | |
536 line->soft = TRUE; | |
537 view->list = g_list_prepend(view->list, line); | |
538 } | |
539 } | |
540 | |
541 view->list = list; | |
542 | |
543 gnt_widget_draw(widget); | |
544 } | |
545 | |
546 void gnt_text_view_scroll(GntTextView *view, int scroll) | |
547 { | |
548 if (scroll == 0) | |
549 { | |
550 view->list = g_list_first(view->list); | |
551 } | |
552 else if (scroll > 0) | |
553 { | |
554 GList *list = g_list_nth_prev(view->list, scroll); | |
555 if (list == NULL) | |
556 list = g_list_first(view->list); | |
557 view->list = list; | |
558 } | |
559 else if (scroll < 0) | |
560 { | |
561 GList *list = g_list_nth(view->list, -scroll); | |
562 if (list == NULL) | |
563 list = g_list_last(view->list); | |
564 view->list = list; | |
565 } | |
566 | |
567 gnt_widget_draw(GNT_WIDGET(view)); | |
568 } | |
569 | |
570 void gnt_text_view_next_line(GntTextView *view) | |
571 { | |
572 GntTextLine *line = g_new0(GntTextLine, 1); | |
573 GList *list = view->list; | |
574 | |
575 view->list = g_list_prepend(g_list_first(view->list), line); | |
576 view->list = list; | |
577 gnt_widget_draw(GNT_WIDGET(view)); | |
578 } | |
579 | |
580 chtype gnt_text_format_flag_to_chtype(GntTextFormatFlags flags) | |
581 { | |
582 chtype fl = 0; | |
583 | |
584 if (flags & GNT_TEXT_FLAG_BOLD) | |
585 fl |= A_BOLD; | |
586 if (flags & GNT_TEXT_FLAG_UNDERLINE) | |
587 fl |= A_UNDERLINE; | |
588 if (flags & GNT_TEXT_FLAG_BLINK) | |
589 fl |= A_BLINK; | |
590 | |
591 if (flags & GNT_TEXT_FLAG_DIM) | |
592 fl |= (A_DIM | COLOR_PAIR(GNT_COLOR_DISABLED)); | |
593 else if (flags & GNT_TEXT_FLAG_HIGHLIGHT) | |
594 fl |= (A_DIM | COLOR_PAIR(GNT_COLOR_HIGHLIGHT)); | |
595 else | |
596 fl |= COLOR_PAIR(GNT_COLOR_NORMAL); | |
597 | |
598 return fl; | |
599 } | |
600 | |
601 void gnt_text_view_clear(GntTextView *view) | |
602 { | |
603 GntTextLine *line; | |
604 | |
605 g_list_foreach(view->list, free_text_line, NULL); | |
606 g_list_free(view->list); | |
607 view->list = NULL; | |
608 | |
609 line = g_new0(GntTextLine, 1); | |
610 view->list = g_list_append(view->list, line); | |
611 if (view->string) | |
612 g_string_free(view->string, TRUE); | |
613 view->string = g_string_new(NULL); | |
614 | |
615 if (GNT_WIDGET(view)->window) | |
616 gnt_widget_draw(GNT_WIDGET(view)); | |
617 } | |
618 | |
619 int gnt_text_view_get_lines_below(GntTextView *view) | |
620 { | |
621 int below = 0; | |
622 GList *list = view->list; | |
623 while ((list = list->prev)) | |
624 ++below; | |
625 return below; | |
626 } | |
627 | |
628 int gnt_text_view_get_lines_above(GntTextView *view) | |
629 { | |
630 int above = 0; | |
631 GList *list = view->list; | |
632 list = g_list_nth(view->list, GNT_WIDGET(view)->priv.height); | |
633 if (!list) | |
634 return 0; | |
635 while ((list = list->next)) | |
636 ++above; | |
637 return above; | |
638 } | |
639 | |
640 /** | |
641 * XXX: There are quite possibly more than a few bugs here. | |
642 */ | |
643 int gnt_text_view_tag_change(GntTextView *view, const char *name, const char *text, gboolean all) | |
644 { | |
645 GList *alllines = g_list_first(view->list); | |
646 GList *list, *next, *iter, *inext; | |
647 const int text_length = text ? strlen(text) : 0; | |
648 int count = 0; | |
649 for (list = view->tags; list; list = next) { | |
650 GntTextTag *tag = list->data; | |
651 next = list->next; | |
652 if (strcmp(tag->name, name) == 0) { | |
653 int change; | |
654 char *before, *after; | |
655 | |
656 count++; | |
657 | |
658 before = g_strndup(view->string->str, tag->start); | |
659 after = g_strdup(view->string->str + tag->end); | |
660 change = (tag->end - tag->start) - text_length; | |
661 | |
662 g_string_printf(view->string, "%s%s%s", before, text ? text : "", after); | |
663 g_free(before); | |
664 g_free(after); | |
665 | |
666 /* Update the offsets of the next tags */ | |
667 for (iter = next; iter; iter = iter->next) { | |
668 GntTextTag *t = iter->data; | |
669 t->start -= change; | |
670 t->end -= change; | |
671 } | |
672 | |
673 /* Update the offsets of the segments */ | |
674 for (iter = alllines; iter; iter = inext) { | |
675 GList *segs, *snext; | |
676 GntTextLine *line = iter->data; | |
677 inext = iter->next; | |
678 for (segs = line->segments; segs; segs = snext) { | |
679 GntTextSegment *seg = segs->data; | |
680 snext = segs->next; | |
681 if (seg->start >= tag->end) { | |
682 /* The segment is somewhere after the tag */ | |
683 seg->start -= change; | |
684 seg->end -= change; | |
685 } else if (seg->end <= tag->start) { | |
686 /* This segment is somewhere in front of the tag */ | |
687 } else if (seg->start >= tag->start) { | |
688 /* This segment starts in the middle of the tag */ | |
689 if (text == NULL) { | |
690 free_text_segment(seg, NULL); | |
691 line->segments = g_list_delete_link(line->segments, segs); | |
692 if (line->segments == NULL) { | |
693 free_text_line(line, NULL); | |
694 if (view->list == iter) { | |
695 if (inext) | |
696 view->list = inext; | |
697 else | |
698 view->list = iter->prev; | |
699 } | |
700 alllines = g_list_delete_link(alllines, iter); | |
701 } | |
702 } else { | |
703 /* XXX: (null) */ | |
704 seg->start = tag->start; | |
705 seg->end = tag->end - change; | |
706 } | |
707 line->length -= change; | |
708 /* XXX: Make things work if the tagged text spans over several lines. */ | |
709 } else { | |
710 /* XXX: handle the rest of the conditions */ | |
711 g_printerr("WTF! This needs to be handled properly!!\n"); | |
712 } | |
713 } | |
714 } | |
715 if (text == NULL) { | |
716 /* Remove the tag */ | |
717 view->tags = g_list_delete_link(view->tags, list); | |
718 free_tag(tag, NULL); | |
719 } else { | |
720 tag->end -= change; | |
721 } | |
722 if (!all) | |
723 break; | |
724 } | |
725 } | |
726 return count; | |
727 } | |
728 |