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