-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathgtk_treeview.c
799 lines (658 loc) · 27.2 KB
/
gtk_treeview.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
#include "gtk_treeview.h"
/** given inst use tree iter to set focus on current view
* (widget is textview associated with window clicked inside)
*/
/* TODO - check on using app->einst[app->focused]->view instead of
* passing widget - cleaner? (no not really, getting buf from focused view or
* calling gtk_text_view_get_buffer is a wash )
*
* GtkWidget *view = app->einst[app->focused]->view;
* gpointer buf = gtk_text_view_get_buffer (GTK_TEXT_VIEW(view));
*/
void tree_set_selection (GtkWidget *widget, gpointer data)
{
mainwin_t *app = data;
GtkTreeModel *model = NULL;
GtkTreeSelection *selection;
GtkTreeIter iter;
gpointer buf = gtk_text_view_get_buffer (GTK_TEXT_VIEW(widget));
// GtkWidget *view = app->einst[app->focused]->view;
// gpointer buf = gtk_text_view_get_buffer (GTK_TEXT_VIEW(view));
gboolean valid, found = FALSE;
/* initialize selection to current */
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW(app->treeview));
/* get treemodel, validate */
model = gtk_tree_view_get_model(GTK_TREE_VIEW(app->treeview));
if (!model) {
g_print ("error: tree_view_get_model - failed.\n");
return;
}
/* get first iter, iterate model to locate current inst, set selection */
valid = gtk_tree_model_get_iter_first (model, &iter);
while (valid) { /* loop over each entry in model */
kinst_t *inst = NULL;
#ifdef DEBUG
/* add printing of COLNAME for debugging */
gchar *str = NULL;
gtk_tree_model_get (model, &iter, /* get name & inst */
COLNAME, &str, COLINST, &inst, -1);
/* compare pointer to sourceview with buf from textview (widget) */
if ((gpointer)(inst->buf) == buf) {
gtk_tree_selection_select_iter (selection, &iter);
g_free (str);
found = TRUE;
break;
}
g_free (str);
#else
gtk_tree_model_get (model, &iter, COLINST, &inst, -1); /* get inst */
/* compare pointer to sourceview with buf from textview (widget) */
if ((gpointer)(inst->buf) == buf) {
gtk_tree_selection_select_iter (selection, &iter);
found = TRUE;
break;
}
#endif
valid = gtk_tree_model_iter_next (model, &iter);
}
if (!found) /* validate buffer found or warn */
g_warning ("tree_set_selection inst not found.");
}
/** get GtkTreeIter from app struct corresponding to focused textview
* caller is responsible for calling g_slice_free() on returned iter.
*/
GtkTreeIter *tree_get_iter_from_view (gpointer data)
{
mainwin_t *app = data;
GtkTreeModel *model = NULL;
GtkTreeIter *iter = NULL;
GtkWidget *view = app->einst[app->focused]->view;
gpointer buf = gtk_text_view_get_buffer (GTK_TEXT_VIEW(view));
gboolean valid;
/* get treemodel, validate */
model = gtk_tree_view_get_model(GTK_TREE_VIEW(app->treeview));
if (!model) {
g_print ("error: tree_view_get_model - failed.\n");
return NULL;
}
iter = g_slice_new (GtkTreeIter);
if (!iter) {
g_warning ("g_slice_new (GtkTreeIter) failed.");
return NULL;
}
/* get first iter, return in validate */
valid = gtk_tree_model_get_iter_first (model, iter);
while (valid) { /* loop over each entry in model */
gchar *str = NULL;
kinst_t *inst = NULL;
/* TODO - remove COLNAME & str after debugging done */
gtk_tree_model_get (model, iter, /* get name & inst */
COLNAME, &str, COLINST, &inst, -1);
/* compare pointer to sourceview with buf from textview (widget) */
if ((gpointer)(inst->buf) == buf) {
g_free (str);
return iter;
}
g_free (str);
valid = gtk_tree_model_iter_next (model, iter);
}
g_warning ("tree_set_selection inst not found.");
return NULL;
}
/** parse number 'n' from "Untitled(n)" to allows untitled bitfield update */
gint untitled_get_no (const gchar *name)
{
gchar *p, *endptr;
const gchar *u = "Untitled";
gint i;
gint64 tmp;
for (i = 0; u[i]; i++) /* validate name begins "Untitled" */
if (name[i] != u[i])
return -1;
if (!name[i]) /* if at name end, name is "Untitled" */
return 0;
p = (gchar *)(name + i);
while (*p && (*p < '0' || '9' < *p))
p++;
// g_print ("converting: '%s'\n", p);
errno = 0;
tmp = g_ascii_strtoll (p, &endptr, 0);
if (endptr == p || errno || tmp < G_MININT || G_MAXINT < tmp)
return -1;
return (gint)tmp;
}
/** get treemodel COLNAME for selected entry
* (user responsible for calling g_free on allocated return)
*/
gchar *treeview_getname (gpointer data)
{
mainwin_t *app = data;
GtkTreeModel *model;
GtkTreeIter iter;
GtkTreeSelection *selection;
gchar *name = NULL;
/* get selection from treeview */
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW(app->treeview));
/* get model & iter from selection */
if (gtk_tree_selection_get_selected (selection, &model, &iter))
gtk_tree_model_get (model, &iter, COLNAME, &name, -1);
return name;
}
/** given inst, iterate through tree to find inst and set COLNAME
* to inst->fname
*/
void treeview_setname (gpointer data, kinst_t *inst)
{
mainwin_t *app = data;
GtkTreeStore *treestore = GTK_TREE_STORE(app->treemodel);
GtkTreeIter *iter = tree_get_iter_from_view (data);
if (!iter) {
g_warning ("treeview_setname() error: iter not found.");
return;
}
gtk_tree_store_set (treestore, iter, COLNAME, inst->fname, -1);
/* set window title */
gtkate_window_set_title (NULL, app);
}
/** check COLNAME entry in tree, and if "Untitled(n)" clear
* bit in nuntitled bitfield. note: only used when changing
* display COLNAME in treeview, otherwise bitfield will be
* out of sync with names displayed in tree.
* (i.e. used with file open/save as)
*/
void check_untitled_remove (gpointer data)
{
gchar *name = treeview_getname (data);
gint n = 0;
/* validate name */
if (!name) {
g_error ("check_untitled_remove() name NULL");
return;
}
/* check name "Untitled(n)", parse n */
if ((n = untitled_get_no (name)) >= 0)
untitled_remove (data, n); /* clear bit n in app->nultitled */
g_free(name);
}
/** given kinst_t instance, set treeview name for entry to Untitled(n)
* if inst is NULL, update app->nuntitled, or set name to inst->fname.
*/
gchar *treeview_initial_name (gpointer data, kinst_t *inst)
{
mainwin_t *app = data;
gchar *name;
if (inst->filename == NULL) {
gint n = untitled_get_next (app);
if (n)
name = g_strdup_printf ("Untitled(%d)", n);
else
name = g_strdup ("Untitled");
}
else
name = g_strdup (inst->fname);
return name;
}
kinst_t *treeview_append (mainwin_t *app, const gchar *filename)
{
GtkTreeStore *treestore;
GtkTreeIter toplevel;
GtkTreeSelection *selection;
kinst_t *inst = buf_new_inst (app, filename); /* new instance split filename */
gchar *name = NULL, *title = NULL;
// const gchar prefix[] = "Untitled";
// gboolean isuntitled = FALSE;
// gint i;
if (!inst) { /* validate instance allocated */
/* handle error */
g_print ("treeview_append() error : !inst\n");
return NULL;
}
name = treeview_initial_name (app, inst); /* get treeview name */
if (!name) {
/* handle error */
g_print ("treeview_append() error: !name\n");
return NULL;
}
/* TEMP - initial name in buffer & set modified FALSE */
// gtk_text_buffer_insert_at_cursor (GTK_TEXT_BUFFER(inst->buf),
// name, -1);
// gtk_text_buffer_set_modified (GTK_TEXT_BUFFER(inst->buf), FALSE);
treestore = GTK_TREE_STORE(gtk_tree_view_get_model (
GTK_TREE_VIEW(app->treeview)));
if (!treestore) {
/* handle no model error */
g_print ("treeview_append() error: !treestore\n");
g_free (name);
return NULL;
}
/* quick strncmp on name and prefix to determine whether
* to prepend or append (to add new "Untitled(n)" at top)
*/
// for (i = 0; name[i] && prefix[i]; i++)
// if (name[i] != prefix[i])
// break;
// isuntitled = (i + 1 == (gint)sizeof prefix) ? TRUE : FALSE;
/* appeand name and pointer to inst as entry in treeview */
// if (isuntitled)
// gtk_tree_store_prepend (treestore, &toplevel, NULL);
// else
gtk_tree_store_append (treestore, &toplevel, NULL);
gtk_tree_store_set (treestore, &toplevel, COLNAME, name, -1);
gtk_tree_store_set (treestore, &toplevel, COLINST, inst, -1);
/* initialize selection to current, set focus on new tree element */
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW(app->treeview));
gtk_tree_selection_select_iter (selection, &toplevel);
/* check/set modified state in window title text on selection change */
if (gtk_text_buffer_get_modified (GTK_TEXT_BUFFER(inst->buf)))
title = g_strdup_printf ("%s - %s [modified]", APPNAME, name);
else
title = g_strdup_printf ("%s - %s", APPNAME, name);
/* set window title */
gtk_window_set_title (GTK_WINDOW (app->window), title);
app->nfiles++; /* update file count */
g_free (title); /* free title */
g_free (name); /* free name */
return inst;
/* temp test */
// g_print ("model contains %d rows (nfiles: %d)\n",
// doctree_get_count (app->treeview), app->nfiles);
}
/** create tree model and initialize treestore with filenames provided in
* argv or Untitled, set app->treemodel. argv must end with sentinel NULL.
*/
GtkTreeModel *treemodel_init (mainwin_t *app, gchar **argv)
{
GtkTreeStore *treestore;
/* create model with string and pointer to inst.
* TODO - consider just using pointer to inst and change render to
* render name from inst->fname (but fname is NULL when Untitled)
*/
treestore = gtk_tree_store_new (NUMCOL, G_TYPE_STRING, G_TYPE_POINTER);
if (!treestore) {
g_print ("treemodel_init() error: !treestore\n");
}
app->treemodel = GTK_TREE_MODEL(treestore);
gtk_tree_view_set_model(GTK_TREE_VIEW(app->treeview), app->treemodel);
if (argv) {
while (*argv) {
treeview_append (app, *argv);
argv++;
}
}
else
treeview_append (app, NULL);
g_object_unref(app->treemodel);
return GTK_TREE_MODEL (treestore);
}
/** create treeview and initialize treemodel */
GtkWidget *create_view_and_model (mainwin_t *app, gchar **argv)
{
GtkTreeViewColumn *col;
GtkCellRenderer *renderer;
GtkWidget *view;
GtkTreeSelection *selection; /* tree selection for buffer inst */
view = gtk_tree_view_new();
app->treeview = view; /* set pointer to treeview */
/* TODO - consider (app->treeview)[n] to allow multiple views for
* showing files grouped by directory, or sorted based on opening or
* ascending/descending alphabetical sort.
*/
col = gtk_tree_view_column_new();
gtk_tree_view_column_set_title(col, "Documents");
/* set column header off (or on) FALSE/TRUE */
// gtk_tree_view_set_headers_visible (GTK_TREE_VIEW(view), FALSE);
gtk_tree_view_set_headers_visible (GTK_TREE_VIEW(view), TRUE);
/* if header is shown, below makes header clickable to sort ascending
* or descending -- but I don't like the header shown, so we will have
* to do sort in context menu.
*/
gtk_tree_view_column_set_sort_column_id (col, COLNAME);
gtk_tree_view_column_set_sort_indicator (col, TRUE);
gtk_tree_view_append_column(GTK_TREE_VIEW(view), col);
renderer = gtk_cell_renderer_text_new();
gtk_tree_view_column_pack_start(col, renderer, TRUE);
gtk_tree_view_column_add_attribute(col, renderer,
"text", COLNAME);
treemodel_init (app, argv); /* initialize treemodel, set app->treemodel */
/* set "has-tooltip" TRUE and then connect "query-tooltip" signal
* for handling by the gtk_tree_view_set_tooltip_row() function.
*/
// g_object_set (view, "has-tooltip", TRUE, NULL);
/* selection to set callback */
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW(view));
/* TreeView/model callbacks */
g_signal_connect (selection, "changed",
G_CALLBACK (treeview_changed), app);
return view;
}
/* simple test for_each to dump kinst_t information */
void doctree_for_each (GtkWidget *widget, mainwin_t *app)
{
GtkTreeModel *model = NULL;
GtkTreeIter iter;
gboolean valid;
gint nrows = 0;
model = gtk_tree_view_get_model(GTK_TREE_VIEW(app->treeview));
if (!model) {
g_print ("error: tree_view_get_model - failed.\n");
return;
}
valid = gtk_tree_model_get_iter_first (model, &iter);
while (valid) {
gchar *str = NULL;
kinst_t *inst = NULL;
gtk_tree_model_get (model, &iter,
COLNAME, &str, COLINST, &inst, -1);
// g_print ("name: %-12s filename: %s\n", str, inst->filename);
g_print ("name: %-16s filename: %-22s fname: %-16s path: %-10s ext: %s\n",
str, inst->filename, inst->fname, inst->fpath, inst->fext);
g_free (str);
nrows++;
valid = gtk_tree_model_iter_next (model, &iter);
}
g_print ("total: %d rows\n", nrows);
if (widget) {}
}
gint doctree_get_count (GtkWidget *treeview)
{
GtkTreeModel *model = NULL;
GtkTreeIter iter;
gboolean valid;
gint nrows = 0;
model = gtk_tree_view_get_model(GTK_TREE_VIEW(treeview));
if (!model) {
g_print ("error: tree_view_get_model - failed.\n");
return 0;
}
valid = gtk_tree_model_get_iter_first (model, &iter);
while (valid) {
nrows++;
valid = gtk_tree_model_iter_next (model, &iter);
}
return nrows;
}
/** returns buffer instance of selected document, which includes sourceview
* buffer, associated filename (or NULL if "Untitled"), and current line and
* column for insert mark in buffer so view can be restored.
*/
kinst_t *inst_get_selected (gpointer data)
{
GtkTreeSelection *selection; /* tree selection for buffer inst */
GtkTreeIter iter;
GtkTreeModel *model;
mainwin_t *app = data;
kinst_t *inst = NULL;
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW(app->treeview));
if (gtk_tree_selection_get_selected (selection, &model, &iter)) {
gtk_tree_model_get (model, &iter, COLINST, &inst, -1);
}
if (!inst)
g_warning ("inst_get_selected() - no selection found.");
return inst;
}
/** remove file identified by tree iter, clear buff if only file */
gboolean doctree_remove_iter (gpointer data, GtkTreeIter *iter)
{
mainwin_t *app = data;
gchar *name = NULL;
gint n;
/* get COLNAME from tree */
gtk_tree_model_get (app->treemodel, iter, COLNAME, &name, -1);
if (!name) {
g_warning ("doctree_remove_iter() - gtk_tree_model_get failed.");
return FALSE;
}
n = untitled_get_no (name); /* parse "Untitled(n)" for n */
g_free (name);
if (n >= 0)
untitled_remove (app, n); /* clear bit in app->nuntitled */
if (app->nfiles > 1) { /* only remove entry if > 1 file */
gtk_tree_store_remove (GTK_TREE_STORE(app->treemodel), iter);
app->nfiles--; /* decrement file count */
return TRUE;
}
else { /* no selection change, handle single file, clear buffer */
gchar *uname, *title;
kinst_t *inst;
/* get kinst_t inst from tree */
/* TODO set to app->einst[app->focused]->inst (shifted) */
gtk_tree_model_get (app->treemodel, iter, COLINST, &inst, -1);
/* if (newinst->fname) set COLNAME to newinst->fname and
* COLINST to newinst ELSE set to uname and clear!
*/
inst_reset_state (inst); /* set values 0/NULL */
uname = treeview_initial_name (app, inst); /* get Untitled(n) name */
gtk_tree_store_set (GTK_TREE_STORE(app->treemodel), iter,
COLNAME, uname, -1); /* set COLNAME in tree */
/* TODO move to buffer_clear() funciton */
gtk_text_buffer_set_text (GTK_TEXT_BUFFER(inst->buf), "", -1);
gtk_text_buffer_set_modified (GTK_TEXT_BUFFER(inst->buf), FALSE);
title = g_strdup_printf ("%s - %s", APPNAME, uname);
gtk_window_set_title (GTK_WINDOW (app->window), title);
g_free (title);
g_free (uname);
}
return FALSE;
}
/** gboolean treeview_remove_selected(), remove file from views and treeview.
* remove selected file from all views, closing views, if multiple files
* remain, set focus on previous file in tree, otherwise, reset current inst
* and clear inst->buf, highlight file in treeview, free all memory
* associated with removed inst filename, inst slice and allocated iter
* returned from tree_get_iter_from_view(). returns true on success, false
* otherwise.
*/
/* FIXME - make sure focus is set on textview instance to be removed, not
* simply focused on the name of the file in the tree. look at
* adding a grab_focus on the text view prior to calling removl.
*/
gboolean treeview_remove_selected (gpointer data)
{
mainwin_t *app = data;
kinst_t *inst = app->einst[app->focused]->inst,
*colinst = NULL,
*lastinst = NULL;;
GtkTreeModel *model = app->treemodel;
GtkTreeIter *victim = NULL,
iter,
*last = NULL;
GtkWidget *view = NULL;
gchar *name = NULL;
gint n = app->nview;
gboolean valid = FALSE, found = FALSE, first = TRUE;
/* save iter-to-selected here, then loop closing all instances */
victim = tree_get_iter_from_view (data);
/* get COLNAME from tree */
gtk_tree_model_get (app->treemodel, victim, COLINST, &colinst,
COLNAME, &name, -1);
if (inst != colinst) { /* validate inst matches colinst */
g_warning ("treeview_remove_current() - inst != colinst");
return FALSE;
}
if (!name) { /* validate name filled */
g_warning ("treeview_remove_current() - gtk_tree_model_get failed.");
return FALSE;
}
/* Remove all view (einst[x]) here, that will simplify the removal from
* the tree. Use the logic while (app->nview > 1 & n--) to handle the
* VIEWs removal, and then simply remove the files. VIEW removal is the
* same in both cases! Git rid of view, then remove 1-file.
*/
/* TODO move to separate function, 'view_close_all_with_inst()' */
while (app->nview > 1 && n--) {
einst_t *einst = app->einst[n];
if (einst->inst && einst->inst == inst) {
GtkWidget *ewin = einst->ebox; /* edit window bounding vbox */
gint focused = app->focused; /* currently focused textview */
gtk_widget_destroy (ewin); /* destroy widget */
/* shift existing higher einst down */
for (gint i = focused; (i + 1) < app->nview; i++)
einst_move (app->einst[i], app->einst[i+1]);
/* if focused and focused > 0 */
if (n == focused && focused > 0) {
/* if focus in highest number view */
if (focused < app->nview - 1) {
app->focused--; /* move focus to previous view */
} /* note: set focus on textview after loop complete */
}
app->nview--; /* decrement number of views shown */
}
}
/* update nuntitled bitfield */
n = untitled_get_no (name); /* check/parse "Untitled(n)" for n */
if (n >= 0)
untitled_remove (app, n); /* clear bit in app->nuntitled */
g_free (name);
/* free allocated memory associated with inst (except buf)
* set freed values 0/NULL, in prep for reuse or deletion
*/
inst_reset_state (inst);
/* TODO move all removal logic to treeview_remove_inst(), (possibly) */
if (app->nfiles == 1) { /* only single file, change buffer */
gchar *uname, *title; /* untitled name, window title */
uname = treeview_initial_name (app, inst); /* get Untitled(n) */
gtk_tree_store_set (GTK_TREE_STORE(app->treemodel), victim,
COLNAME, uname, -1); /* set COLNAME in tree */
/* TODO move to buffer_clear() funciton */
gtk_text_buffer_set_text (GTK_TEXT_BUFFER(inst->buf), "", -1);
gtk_text_buffer_set_modified (GTK_TEXT_BUFFER(inst->buf), FALSE);
inst->modified = FALSE;
/* set window title */
title = g_strdup_printf ("%s - %s", APPNAME, uname);
gtk_window_set_title (GTK_WINDOW (app->window), title);
found = TRUE; /* set found flag indicating successful removal */
g_free (title);
g_free (uname);
}
else if (app->nfiles > 1) { /* additional files remain, remove victim */
view = app->einst[app->focused]->view; /* get current view widget */
/* get first iter, iterate model to locate current inst, set selection */
valid = gtk_tree_model_get_iter_first (model, &iter);
while (valid) { /* loop over each entry in model */
/* get inst from iter, colinst set from victim above */
gtk_tree_model_get (model, &iter, COLINST, &inst, -1);
/* compare inst pointer to colinst of victim */
if (inst == colinst) {
/* add gboolean first = TRUE; and check if (first) get next
* and set next to current
*/
if (first) {
kinst_t *nextinst = NULL;
if (gtk_tree_model_iter_next (model, &iter)) {
/* get next instance from tree */
gtk_tree_model_get (model, &iter, COLINST, &nextinst, -1);
/* set current to next */
app->einst[app->focused]->inst = nextinst;
/* set buffer in active textview */
gtk_text_view_set_buffer (GTK_TEXT_VIEW(view),
GTK_TEXT_BUFFER(nextinst->buf));
/* set treeview selection on current buf COLNAME */
tree_set_selection (view, data);
}
found = TRUE; /* set found colinst flag */
goto removefirst;
}
else if (last) {
/* if last iter set, current to last */
app->einst[app->focused]->inst = lastinst;
/* set buffer in active textview */
gtk_text_view_set_buffer (GTK_TEXT_VIEW(view),
GTK_TEXT_BUFFER(lastinst->buf));
/* set treeview selection on current buf COLNAME */
tree_set_selection (view, data);
}
else /* found colinst in tree, but last iter invalid */
g_warning ("treeview_remove_selected() invalid last.\n");
found = TRUE; /* set found colinst flag */
break;
}
lastinst = inst; /* set last instance to current */
last = &iter; /* set last iter to current */
first = FALSE; /* unset flag for first file */
/* get next tree iter */
valid = gtk_tree_model_iter_next (model, &iter);
}
removefirst:;
/* remove victim from tree and free allocated slice */
gtk_tree_store_remove (GTK_TREE_STORE(app->treemodel), victim);
/* TEST (results in assertion 'handler_id > 0' failed) */
// buf_delete_inst (colinst); /* to unregister "changed" & "mark_set" */
g_slice_free (kinst_t, colinst); /* free colinst slice */
app->nfiles--; /* decrement file count */
}
if (!found) /* validate inst found or warn */
g_warning ("tree_set_selection inst not found.");
g_slice_free (GtkTreeIter, victim); /* free allocated iter */
return found;
}
/** check_unsaved buffers by iterating tree and checking inst->modified.
* takes pointer to mainwin_t instance and returns number of unsaved
* files on success (-1 on error).
*/
gint treeview_count_unsaved (gpointer data)
{
mainwin_t *app = data;
GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(app->treeview));
GtkTreeIter iter;
gboolean valid;
gint unsaved = 0;
if (!model) {
g_error ("error: check_unsaved() tree_view_get_model failed.");
return -1;
}
valid = gtk_tree_model_get_iter_first (model, &iter);
while (valid) {
kinst_t *inst = NULL;
gtk_tree_model_get (model, &iter, COLINST, &inst, -1);
if (inst->modified)
unsaved++;
valid = gtk_tree_model_iter_next (model, &iter);
}
return unsaved;
}
/** doctree callbacks */
/* GtkTreeSelection "changed" signal callback */
void treeview_changed (GtkWidget *widget, gpointer data)
{
GtkTreeIter iter;
GtkTreeModel *model;
mainwin_t *app = data;
GtkWidget *view = app->einst[app->focused]->view;
gchar *value, *title;
kinst_t *inst;
if (gtk_tree_selection_get_selected (GTK_TREE_SELECTION(widget),
&model, &iter)) {
gtk_tree_model_get (model, &iter, COLNAME, &value, COLINST, &inst, -1);
/* check/set modified state in window title text on selection change */
if (gtk_text_buffer_get_modified (GTK_TEXT_BUFFER(inst->buf)))
title = g_strdup_printf ("%s - %s [modified]", APPNAME, value);
else if (inst->readonly)
title = g_strdup_printf ("%s - %s [read-only]", APPNAME, value);
else
title = g_strdup_printf ("%s - %s", APPNAME, value);
/* set window title */
gtk_window_set_title (GTK_WINDOW (app->window), title);
/* set buffer in active textview */
gtk_text_view_set_buffer (GTK_TEXT_VIEW(view),
GTK_TEXT_BUFFER(inst->buf));
/* set editable based on readonly flag, set menu item sensitivity */
gtk_text_view_set_editable (GTK_TEXT_VIEW(view),
inst->readonly ? FALSE : TRUE);
gtk_widget_set_sensitive (app->rdonlyMi, inst->readonly ? TRUE : FALSE);
/* scroll to current line */
if (inst->line > 0)
/* scroll to insert mark */
gtk_text_view_scroll_to_mark (GTK_TEXT_VIEW(view),
gtk_text_buffer_get_insert (GTK_TEXT_BUFFER(inst->buf)),
0.0, TRUE, 0.0, 0.25);
/* set app->einst[app->focused]->inst to point to buffer inst */
app->einst[app->focused]->inst = inst;
/* update statusbar */
status_set_default (data);
g_free (value); /* values with type G_TYPE_STRING or G_TYPE_BOXED */
/* have to be freed, G_TYPE_POINTER does not */
g_free (title);
/* grab focus for textview -- cursor still not visible */
// gtk_widget_grab_focus (view);
}
}