9 #define YUILogComponent "gtk" 10 #include <yui/Libyui_config.h> 14 #include "YSelectionWidget.h" 15 #include "YGSelectionStore.h" 16 #include "ygtktreeview.h" 30 YGTreeView (YWidget *ywidget, YWidget *parent,
const std::string &label,
bool tree)
31 :
YGScrolledWidget (ywidget, parent, label, YD_VERT, YGTK_TYPE_TREE_VIEW, NULL),
34 gtk_tree_view_set_headers_visible (getView(), FALSE);
39 gtk_tree_selection_set_mode (getSelection(), GTK_SELECTION_BROWSE);
41 connect (getSelection(),
"changed", G_CALLBACK (selection_changed_cb),
this);
42 connect (getWidget(),
"row-activated", G_CALLBACK (activated_cb),
this);
43 connect (getWidget(),
"right-click", G_CALLBACK (right_click_cb),
this);
46 markColumn = -1; m_count = NULL;
48 g_signal_connect (getWidget(),
"map", G_CALLBACK (block_init_cb),
this);
52 {
if (m_blockTimeout) g_source_remove (m_blockTimeout); }
54 inline GtkTreeView *getView()
55 {
return GTK_TREE_VIEW (getWidget()); }
56 inline GtkTreeSelection *getSelection()
57 {
return gtk_tree_view_get_selection (getView()); }
59 void addTextColumn (
int iconCol,
int textCol)
60 { addTextColumn (
"", YAlignUnchanged, iconCol, textCol); }
62 void addTextColumn (
const std::string &header, YAlignmentType align,
int icon_col,
int text_col)
66 case YAlignBegin: xalign = 0.0;
break;
67 case YAlignCenter: xalign = 0.5;
break;
68 case YAlignEnd: xalign = 1.0;
break;
69 case YAlignUnchanged:
break;
72 GtkTreeViewColumn *column = gtk_tree_view_column_new();
73 gtk_tree_view_column_set_title (column, header.c_str());
75 GtkCellRenderer *renderer;
76 renderer = gtk_cell_renderer_pixbuf_new();
77 gtk_tree_view_column_pack_start (column, renderer, FALSE);
78 gtk_tree_view_column_set_attributes (column, renderer,
"pixbuf", icon_col, NULL);
80 renderer = gtk_cell_renderer_text_new();
81 gtk_tree_view_column_pack_start (column, renderer, TRUE);
82 gtk_tree_view_column_set_attributes (column, renderer,
"text", text_col, NULL);
84 g_object_set (renderer,
"xalign", xalign, NULL);
86 gtk_tree_view_column_set_resizable (column, TRUE);
87 gtk_tree_view_append_column (getView(), column);
88 if (gtk_tree_view_get_search_column (getView()) == -1)
89 gtk_tree_view_set_search_column (getView(), text_col);
92 void addCheckColumn (
int check_col)
94 GtkCellRenderer *renderer = gtk_cell_renderer_toggle_new();
95 g_object_set_data (G_OBJECT (renderer),
"column", GINT_TO_POINTER (check_col));
96 GtkTreeViewColumn *column = gtk_tree_view_column_new_with_attributes (
97 NULL, renderer,
"active", check_col, NULL);
98 gtk_tree_view_column_set_cell_data_func (column, renderer, inconsistent_mark_cb,
this, NULL);
99 g_signal_connect (G_OBJECT (renderer),
"toggled",
100 G_CALLBACK (toggled_cb),
this);
102 gtk_tree_view_column_set_resizable (column, TRUE);
103 gtk_tree_view_append_column (getView(), column);
104 if (markColumn == -1)
105 markColumn = check_col;
109 { gtk_tree_view_set_model (getView(), getModel()); }
111 void addCountWidget (YWidget *yparent)
113 bool mainWidget = !yparent || !strcmp (yparent->widgetClass(),
"YVBox") || !strcmp (yparent->widgetClass(),
"YReplacePoint");
115 m_count = gtk_label_new (
"0");
116 GtkWidget *hbox = YGTK_HBOX_NEW(4);
117 gtk_box_set_homogeneous (GTK_BOX (hbox), FALSE);
119 GtkWidget *label = gtk_label_new (_(
"Total selected:"));
121 gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, TRUE, 0);
122 gtk_box_pack_start (GTK_BOX (hbox), m_count, FALSE, TRUE, 0);
123 gtk_box_pack_start (GTK_BOX (YGWidget::getWidget()), hbox, FALSE, TRUE, 0);
124 gtk_widget_show_all (hbox);
130 if (!m_count)
return;
133 static gboolean
foreach (
134 GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer _pThis)
138 gtk_tree_model_get (model, iter, pThis->markColumn, &mark, -1);
140 int count = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (model),
"count"));
141 g_object_set_data (G_OBJECT (model),
"count", GINT_TO_POINTER (count+1));
147 GtkTreeModel *model = getModel();
148 g_object_set_data (G_OBJECT (model),
"count", 0);
149 gtk_tree_model_foreach (model, inner::foreach,
this);
151 int count = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (model),
"count"));
152 gchar *str = g_strdup_printf (
"%d", count);
153 gtk_label_set_text (GTK_LABEL (m_count), str);
157 void focusItem (YItem *item,
bool select)
160 getTreeIter (item, &iter);
164 GtkTreePath *path = gtk_tree_model_get_path (getModel(), &iter);
165 gtk_tree_view_expand_to_path (getView(), path);
167 if (gtk_tree_selection_get_mode (getSelection()) != GTK_SELECTION_MULTIPLE)
168 gtk_tree_view_scroll_to_cell (getView(), path, NULL, TRUE, 0.5, 0);
169 gtk_tree_path_free (path);
171 gtk_tree_selection_select_iter (getSelection(), &iter);
174 gtk_tree_selection_unselect_iter (getSelection(), &iter);
177 void unfocusAllItems()
180 gtk_tree_selection_unselect_all (getSelection());
186 static gboolean foreach_unmark (
187 GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer _pThis)
190 pThis->setRowMark (iter, pThis->markColumn, FALSE);
195 gtk_tree_model_foreach (getModel(), inner::foreach_unmark,
this);
198 YItem *getFocusItem()
201 if (gtk_tree_selection_get_selected (getSelection(), NULL, &iter))
202 return getYItem (&iter);
206 virtual bool _immediateMode() {
return true; }
207 virtual bool _shrinkable() {
return false; }
208 virtual bool _recursiveSelection() {
return false; }
210 void setMark (GtkTreeIter *iter, YItem *yitem, gint column,
bool state,
bool recursive)
212 setRowMark (iter, column, state);
213 yitem->setSelected (state);
216 for (YItemConstIterator it = yitem->childrenBegin();
217 it != yitem->childrenEnd(); it++) {
219 getTreeIter (*it, &_iter);
220 setMark (&_iter, *it, column, state,
true);
224 void toggleMark (GtkTreePath *path, gint column)
227 if (!gtk_tree_model_get_iter (getModel(), &iter, path))
230 gtk_tree_model_get (getModel(), &iter, column, &state, -1);
233 YItem *yitem = getYItem (&iter);
234 setMark (&iter, yitem, column, state, _recursiveSelection());
236 emitEvent (YEvent::ValueChanged);
241 virtual unsigned int getMinSize (YUIDimension dim)
244 return YGUtils::getCharsHeight (getWidget(), _shrinkable() ? 2 : 5);
249 static gboolean block_selected_timeout_cb (gpointer data)
252 pThis->m_blockTimeout = 0;
258 if (m_blockTimeout) g_source_remove (m_blockTimeout);
259 m_blockTimeout = g_timeout_add_full (G_PRIORITY_LOW, 50, block_selected_timeout_cb,
this, NULL);
262 static void block_init_cb (GtkWidget *widget,
YGTreeView *pThis)
263 { pThis->blockSelected(); }
267 static bool all_marked (GtkTreeModel *model, GtkTreeIter *iter,
int mark_col)
270 GtkTreeIter child_iter;
271 if (gtk_tree_model_iter_children (model, &child_iter, iter))
273 gtk_tree_model_get (model, &child_iter, mark_col, &marked, -1);
274 if (!marked)
return false;
275 all_marked (model, &child_iter, mark_col);
276 }
while (gtk_tree_model_iter_next (model, &child_iter));
280 static void inconsistent_mark_cb (GtkTreeViewColumn *column,
281 GtkCellRenderer *cell, GtkTreeModel *model, GtkTreeIter *iter, gpointer data)
285 gtk_tree_model_get (model, iter, pThis->markColumn, &marked, -1);
286 gboolean consistent = !marked || all_marked (model, iter, pThis->markColumn);
287 g_object_set (G_OBJECT (cell),
"inconsistent", !consistent, NULL);
290 static void selection_changed_cb (GtkTreeSelection *selection,
YGTreeView *pThis)
293 static gboolean foreach_sync_select (
294 GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer _pThis)
297 GtkTreeSelection *selection = pThis->getSelection();
298 bool sel = gtk_tree_selection_iter_is_selected (selection, iter);
299 pThis->getYItem (iter)->setSelected (sel);
304 if (pThis->m_blockTimeout)
return;
305 if (pThis->markColumn == -1)
306 gtk_tree_model_foreach (pThis->getModel(), inner::foreach_sync_select, pThis);
307 if (pThis->_immediateMode())
308 pThis->emitEvent (YEvent::SelectionChanged, IF_NOT_PENDING_EVENT);
311 static void activated_cb (GtkTreeView *tree_view, GtkTreePath *path,
314 if (pThis->markColumn >= 0)
315 pThis->toggleMark (path, pThis->markColumn);
318 if (gtk_tree_view_row_expanded (tree_view, path))
319 gtk_tree_view_collapse_row (tree_view, path);
321 gtk_tree_view_expand_row (tree_view, path, FALSE);
323 pThis->emitEvent (YEvent::Activated);
327 static void toggled_cb (GtkCellRendererToggle *renderer, gchar *path_str,
330 GtkTreePath *path = gtk_tree_path_new_from_string (path_str);
331 gint column = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (renderer),
"column"));
332 pThis->toggleMark (path, column);
333 gtk_tree_path_free (path);
336 if (gtk_tree_path_get_depth (path) >= 2)
337 gtk_widget_queue_draw (pThis->getWidget());
342 { pThis->emitEvent (YEvent::ContextMenuActivated); }
346 #include "YGDialog.h" 347 #include <gdk/gdkkeysyms.h> 353 YGTable (YWidget *parent, YTableHeader *headers,
bool multiSelection)
354 : YTable (NULL, headers, multiSelection),
355 YGTreeView (
this, parent, std::string(),
false)
357 gtk_tree_view_set_headers_visible (getView(), TRUE);
358 gtk_tree_view_set_rules_hint (getView(), columns() > 1);
359 ygtk_tree_view_set_empty_text (YGTK_TREE_VIEW (getView()), _(
"No entries."));
361 gtk_tree_selection_set_mode (getSelection(), GTK_SELECTION_MULTIPLE);
363 GType types [columns()*2];
364 for (
int i = 0; i < columns(); i++) {
366 types[t+0] = GDK_TYPE_PIXBUF;
367 types[t+1] = G_TYPE_STRING;
368 addTextColumn (header(i), alignment (i), t, t+1);
370 createStore (columns()*2, types);
376 YAlignmentType lastAlign = alignment (columns()-1);
377 if (lastAlign == YAlignCenter || lastAlign == YAlignEnd)
378 gtk_tree_view_append_column (getView(), gtk_tree_view_column_new());
380 g_signal_connect (getWidget(),
"key-press-event", G_CALLBACK (key_press_event_cb),
this);
383 void setSortable (
bool sortable)
386 GList *columns = gtk_tree_view_get_columns (getView());
387 for (GList *i = columns; i; i = i->next, n++) {
388 GtkTreeViewColumn *column = (GtkTreeViewColumn *) i->data;
389 if (n >= YGTable::columns())
393 gtk_tree_sortable_set_sort_func (
394 GTK_TREE_SORTABLE (getModel()), index, tree_sort_cb,
395 GINT_TO_POINTER (index), NULL);
396 gtk_tree_view_column_set_sort_column_id (column, index);
399 gtk_tree_view_column_set_sort_column_id (column, -1);
401 g_list_free (columns);
404 void setCell (GtkTreeIter *iter,
int column,
const YTableCell *cell)
407 std::string label (cell->label());
409 label = YUI::app()->glyph (YUIGlyph_CheckMark);
411 int index = column * 2;
412 setRowText (iter, index, cell->iconName(), index+1, label,
this);
417 virtual bool _immediateMode() {
return immediateMode(); }
421 virtual void setKeepSorting (
bool keepSorting)
423 YTable::setKeepSorting (keepSorting);
424 setSortable (!keepSorting);
426 GtkTreeViewColumn *column = gtk_tree_view_get_column (getView(), 0);
428 gtk_tree_view_column_clicked (column);
432 virtual void cellChanged (
const YTableCell *cell)
435 getTreeIter (cell->parent(), &iter);
436 setCell (&iter, cell->column(), cell);
441 void doAddItem (YItem *_item)
443 YTableItem *item = dynamic_cast <YTableItem *> (_item);
446 addRow (item, &iter);
448 for (YTableCellIterator it = item->cellsBegin();
449 it != item->cellsEnd(); it++)
452 yuiWarning() <<
"Item contains too many columns, current column is (starting from 0) " << i
453 <<
" but only " << columns() <<
" columns are configured. Skipping..." << std::endl;
456 setCell (&iter, i++, *it);
457 if (item->selected())
458 focusItem (item,
true);
461 yuiError() <<
"Can only add YTableItems to a YTable.\n";
464 void doSelectItem (YItem *item,
bool select)
465 { focusItem (item, select); }
467 void doDeselectAllItems()
468 { unfocusAllItems(); }
472 static void activateButton (YWidget *button)
474 YWidgetEvent *
event =
new YWidgetEvent (button, YEvent::Activated);
475 YGUI::ui()->sendEvent (event);
480 if (pThis->notifyContextMenu())
481 return YGTreeView::right_click_cb (view, outreach, pThis);
486 static void key_activate_cb (GtkMenuItem *item, YWidget *button)
487 { activateButton (button); }
488 static void appendItem (GtkWidget *menu,
const gchar *stock,
int key)
490 YWidget *button = YGDialog::currentDialog()->getFunctionWidget (key);
493 item = gtk_menu_item_new_with_mnemonic (stock);
494 gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
495 g_signal_connect (G_OBJECT (item),
"activate",
496 G_CALLBACK (key_activate_cb), button);
501 GtkWidget *menu = gtk_menu_new();
502 YGDialog *dialog = YGDialog::currentDialog();
503 if (dialog->getClassWidgets (
"YTable").size() == 1) {
506 if (dialog->getFunctionWidget(3))
507 inner::appendItem (menu,
"list-add", 3);
510 if (dialog->getFunctionWidget(4))
511 inner::appendItem (menu,
"edit-cut", 4);
512 if (dialog->getFunctionWidget(5))
513 inner::appendItem (menu,
"list-remove", 5);
517 menu = ygtk_tree_view_append_show_columns_item (YGTK_TREE_VIEW (view), menu);
518 ygtk_tree_view_popup_menu (view, menu);
521 static gboolean key_press_event_cb (GtkWidget *widget, GdkEventKey *event,
YGTable *pThis)
523 if (event->keyval == GDK_KEY_Delete) {
524 YWidget *button = YGDialog::currentDialog()->getFunctionWidget (5);
526 activateButton (button);
528 gtk_widget_error_bell (widget);
534 static gint tree_sort_cb (
535 GtkTreeModel *model, GtkTreeIter *a, GtkTreeIter *b, gpointer _index)
537 int index = GPOINTER_TO_INT (_index);
538 gchar *str_a, *str_b;
539 gtk_tree_model_get (model, a, index, &str_a, -1);
540 gtk_tree_model_get (model, b, index, &str_b, -1);
541 if (!str_a) str_a = g_strdup (
"");
542 if (!str_b) str_b = g_strdup (
"");
543 int ret = strcmp (str_a, str_b);
544 g_free (str_a); g_free (str_b);
548 YGLABEL_WIDGET_IMPL (YTable)
549 YGSELECTION_WIDGET_IMPL (YTable)
552 YTable *YGWidgetFactory::createTable (YWidget *parent, YTableHeader *headers,
555 return new YGTable (parent, headers, multiSelection);
558 #include "YSelectionBox.h" 564 : YSelectionBox (NULL, label),
567 GType types [2] = { GDK_TYPE_PIXBUF, G_TYPE_STRING };
568 addTextColumn (0, 1);
569 createStore (2, types);
575 virtual bool _shrinkable() {
return shrinkable(); }
579 void doAddItem (YItem *item)
582 addRow (item, &iter);
583 setRowText (&iter, 0, item->iconName(), 1, item->label(),
this);
584 if (item->selected())
585 focusItem (item,
true);
588 void doSelectItem (YItem *item,
bool select)
589 { focusItem (item, select); }
591 void doDeselectAllItems()
592 { unfocusAllItems(); }
594 YGLABEL_WIDGET_IMPL (YSelectionBox)
595 YGSELECTION_WIDGET_IMPL (YSelectionBox)
598 YSelectionBox *YGWidgetFactory::createSelectionBox (YWidget *parent,
const std::string &label)
601 #include "YMultiSelectionBox.h" 607 : YMultiSelectionBox (NULL, label),
610 GType types [3] = { G_TYPE_BOOLEAN, GDK_TYPE_PIXBUF, G_TYPE_STRING };
612 addTextColumn (1, 2);
613 createStore (3, types);
615 addCountWidget (parent);
620 virtual bool _shrinkable() {
return shrinkable(); }
624 void doAddItem (YItem *item)
627 addRow (item, &iter);
628 setRowMark (&iter, 0, item->selected());
629 setRowText (&iter, 1, item->iconName(), 2, item->label(),
this);
633 void doSelectItem (YItem *item,
bool select)
636 getTreeIter (item, &iter);
637 setRowMark (&iter, 0, select);
641 void doDeselectAllItems()
642 { unmarkAll(); syncCount(); }
646 virtual YItem *currentItem()
647 {
return getFocusItem(); }
649 virtual void setCurrentItem (YItem *item)
650 { focusItem (item,
true); }
652 YGLABEL_WIDGET_IMPL (YMultiSelectionBox)
653 YGSELECTION_WIDGET_IMPL (YMultiSelectionBox)
656 YMultiSelectionBox *YGWidgetFactory::createMultiSelectionBox (YWidget *parent,
const std::string &label)
660 #include "YTreeItem.h" 665 YGTree (YWidget *parent,
const std::string &label,
bool multiselection,
bool recursiveSelection)
666 : YTree (NULL, label, multiselection, recursiveSelection),
669 if (multiselection) {
670 GType types [3] = { GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_BOOLEAN };
672 addTextColumn (0, 1);
673 createStore (3, types);
674 addCountWidget (parent);
678 GType types [2] = { GDK_TYPE_PIXBUF, G_TYPE_STRING };
679 addTextColumn (0, 1);
680 createStore (2, types);
684 g_signal_connect (getWidget(),
"row-collapsed", G_CALLBACK (row_collapsed_cb),
this);
685 g_signal_connect (getWidget(),
"row-expanded", G_CALLBACK (row_expanded_cb),
this);
688 virtual bool _recursiveSelection() {
return recursiveSelection(); }
690 void addNode (YItem *item, GtkTreeIter *parent)
693 addRow (item, &iter, parent);
694 setRowText (&iter, 0, item->iconName(), 1, item->label(),
this);
695 if (item->selected()) {
696 if (hasMultiSelection())
697 setRowMark (&iter, 2, item->selected());
699 focusItem (item,
true);
701 if (((YTreeItem *) item)->isOpen())
703 for (YItemConstIterator it = item->childrenBegin();
704 it != item->childrenEnd(); it++)
705 addNode (*it, &iter);
709 void expand (GtkTreeIter *iter)
711 GtkTreePath *path = gtk_tree_model_get_path (getModel(), iter);
712 gtk_tree_view_expand_row (getView(), path, FALSE);
713 gtk_tree_path_free (path);
716 bool isReallyOpen (YTreeItem *item)
718 for (YTreeItem *i = item; i; i = i->parent())
727 virtual void rebuildTree()
732 for (YItemConstIterator it = YTree::itemsBegin(); it != YTree::itemsEnd(); it++)
735 int depth = getTreeDepth();
736 gtk_tree_view_set_show_expanders (getView(), depth > 1);
737 gtk_tree_view_set_enable_tree_lines (getView(), depth > 3);
742 static gboolean foreach_sync_open (
743 GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer _pThis)
746 YTreeItem *item = (YTreeItem *) pThis->getYItem (iter);
748 gtk_tree_view_expand_row (pThis->getView(), path, FALSE);
753 g_signal_handlers_block_by_func (getWidget(), (gpointer) row_expanded_cb,
this);
754 gtk_tree_model_foreach (getModel(), inner::foreach_sync_open,
this);
755 g_signal_handlers_unblock_by_func (getWidget(), (gpointer) row_expanded_cb,
this);
760 virtual YTreeItem *currentItem()
761 {
return (YTreeItem *) getFocusItem(); }
763 void _markItem (YItem *item,
bool select,
bool recursive) {
765 getTreeIter (item, &iter);
766 setRowMark (&iter, 2, select);
769 YTreeItem *_item = (YTreeItem *) item;
770 for (YItemConstIterator it = _item->childrenBegin();
771 it != _item->childrenEnd(); it++)
772 _markItem (*it, select,
true);
778 void doAddItem (YItem *item) {}
780 void doSelectItem (YItem *item,
bool select)
782 if (hasMultiSelection()) {
783 _markItem (item, select, recursiveSelection());
787 focusItem (item, select);
790 void doDeselectAllItems()
792 if (hasMultiSelection()) {
802 void reportRowOpen (GtkTreeIter *iter,
bool open)
804 YTreeItem *item = static_cast <YTreeItem *> (getYItem (iter));
805 item->setOpen (open);
808 static void row_collapsed_cb (GtkTreeView *view, GtkTreeIter *iter,
809 GtkTreePath *path,
YGTree *pThis)
810 { pThis->reportRowOpen (iter,
false); }
812 static void row_expanded_cb (GtkTreeView *view, GtkTreeIter *iter,
813 GtkTreePath *path,
YGTree *pThis)
814 { pThis->reportRowOpen (iter,
true); }
820 YTreeItem *item = static_cast <YTreeItem *> (pThis->getYItem (iter));
821 for (YItemConstIterator it = item->childrenBegin();
822 it != item->childrenEnd(); it++) {
823 const YTreeItem *child = static_cast <YTreeItem *> (*it);
824 if (child->isOpen()) {
826 if (pThis->getIter (child, &iter))
827 pThis->expand (&iter);
837 YGUI::ui()->sendEvent(
new YWidgetEvent(
this,YEvent::Activated ) );
841 YGLABEL_WIDGET_IMPL (YTree)
842 YGSELECTION_WIDGET_IMPL (YTree)
845 YTree *YGWidgetFactory::createTree (YWidget *parent,
const std::string &label,
bool multiselection,
bool recursiveSelection)
846 {
return new YGTree (parent, label, multiselection, recursiveSelection); }