O widget GtkTreeView é a visão na aproximação Modelo/Vista/Controlador. Ela cuida de mostrar os dados guardados em um modelo (GtkTreeStore ou GtkListStore) para o usuário. Você pode ter vários GtkTreeViews para um único modelo, e mudanças no modelo serão exibidos automaticamente em todos eles.
Para usar o GtkTreeView, você precisa criar o widget em sÃ, então criar GtkTreeViewColumns para as colunas a exibir e GtkCellRenderers para dizer para as colunas como exibir uma célula na coluna.
Exemplo 10.3. TreeView e Renderer
$view = new GtkTreeView($model); $renderer = new GtkCellRendererText(); $column = new GtkTreeViewColumn("Folder", $renderer, "text", 1); $view->append_column($column); |
Este código cria um novo GtkTreeView a anexa-o ao modelo. Então cria um cell renderer de texto e uma coluna e adiciona estes para a visão. "Folder" é o nome da coluna, exibida ao topo da coluna.
Existem vários GtkCellRenderers na biblioteca Gtk+, e você pode escrever o seu próprio, se estes não forem o suficiente.
As seleções do usuário em um GtkTreeView (por visão) são seguidas usando o objeto GtkTreeSelection. Se o seu código necessita saber quando o usuário muda a seleção, conecte com o sinal "changed" do GtkTreeSelection. Você também pode fazer o objeto da seleção chamar uma função para cada ramo selecionado, ou mudar atráves de programação a seleção. Veja a documentação da API para detalhes.
Exemplo 10.4. O sinal "changed" da seleção
$selection = $view->get_selection(); $selection->connect("changed", "display_selected_folder"); |
Este código conecta o sinal "changed" a função (display_selected_folder()) que exibe o conteúdo da pasta selecionada.
Você precisa dizer ao GtkTreeView explicitamente que é permitido ao usuário rearanjar a arvore de pastas usando drag-and-drop. Felizmente, após dizer isto uma vez, o widget se encarrega de cuidar do resto.
Exemplo 10.5. Reordenando com Drag-and-Drop
$view->set_reorderable(true); |
Isto é tudo o que você precisa para ativar o drag-and-drop no widget.
A mesma coisa pode ser feita manualmente se você escolher não ativar o drag-and-drop removendo o ramo filho da arvore e inserindo-o de volta como filho de outro ramo.
Exemplo 10.6. Reordenamento manual
$folder = $model->get_value($old_iter, 0); $model->remove($old_iter); $new_iter = $model->insert_before($new_parent, null); $model->set($new_iter, 0, $folder); $model->set($new_iter, 1, $folder['name']); |
Este código move um ramo de old_iter para ser o último filho do ramo new_parent.
Quando este tutorial foi escrito/portado havia um bug no PHP-Gtk2 que a ordem dos parãmetros para insert_before() e insert_after() foram trocadas. O código acima não irá funcionar (nem o código do exemplo abaixo) sem trocar a ordem dos parâmetros. Na versão php-gtk-2.0.0 alpha, a ordem ainda é (sibling, parent) e deveria ser (parent, sibling). Isto já foi modificado no CVS.
Exemplo 10.7. Código fonte de exemplo
<?php // This is an example for demonstrating use of the GtkTreeView widget. // The code in this example is not particularly good: it is written to // concentrate on widget usage demonstration, not for maintainability. $view = null; $choose_parent_view = null; $dialog = null; function move($old_iter = null, $new_parent, $model) { if ($old_iter) { $folder = $model->get_value($old_iter, 0); $model->remove($old_iter); $new_iter = $model->insert_before($new_parent, null); $model->set($new_iter, 0, $folder); $model->set($new_iter, 1, $folder['name']); } } function dialog_ok($args) { global $dialog, $choose_parent_view, $view; $dialog->hide(); list($model, $parent_iter) = $choose_parent_view->get_selection()->get_selected(); list($model, $old_iter) = $view->get_selection()->get_selected(); if ($parent_iter && $old_iter) { move($old_iter, $parent_iter, $model); } } function dialog_cancel($args) { global $dialog; $dialog->hide(); } function choose_parent($args) { global $dialog; $dialog->show(); } function move_to_bottom($args) { global $view; list ($model, $old_iter) = $view->get_selection()->get_selected(); if ($old_iter) { move($old_iter, null, $model); } } function quit($args) { Gtk::main_quit(); } function make_view($model) { $view = new GtkTreeView($model); $view->set_reorderable(true); $renderer = new GtkCellRendererText(); $column = new GtkTreeViewColumn("Folder", $renderer, "text", 1); $view->append_column($column); $view->show(); $scrolled = new GtkScrolledWindow(); $scrolled->add($view); $scrolled->show(); return array($view, $scrolled); } function make_buttons($list) { $buttonbox = new GtkHBox(); foreach ($list as $label => $func) { $button = new GtkButton(); $button->set_label($label); $button->connect("clicked", $func); $button->show(); $buttonbox->pack_start($button, false, false); } $buttonbox->show(); return $buttonbox; } $model = new GtkTreeStore(Gtk::TYPE_PHP_VALUE, Gtk::TYPE_STRING); for ($i=0; $i < 100; $i++) { $folder = array('name' => 'folder ' . $i, 'files' => array('foo', 'bar')); $iter = $model->insert_before(null, null); $model->set($iter, 0, $folder); $model->set($iter, 1, $folder['name']); } list($view, $scrolled) = make_view($model); $view->set_reorderable(true); $buttons = array( "Quit" => "quit", "Choose parent" => "choose_parent", "Move to bottom" => "move_to_bottom" ); $buttonbox = make_buttons($buttons); $vbox = new GtkVBox(); $vbox->pack_start($buttonbox, false, false); $vbox->pack_start($scrolled, true, true); $vbox->show(); $win = new GtkWindow(Gtk::WINDOW_TOPLEVEL); $win->connect("delete_event", "quit"); $win->add($vbox); $win->show(); $win->resize(300, 500); list($choose_parent_view, $scrolled) = make_view($model); $buttons = array( "OK" => "dialog_ok", "Cancel" => "dialog_cancel" ); $buttonbox = make_buttons($buttons); $vbox = new GtkVBox(); $vbox->pack_start($scrolled, true, true); $vbox->pack_start($buttonbox, false, false); $vbox->show(); $dialog = new GtkWindow(Gtk::WINDOW_TOPLEVEL); $dialog->set_default_size(200, 400); $dialog->add($vbox); Gtk::main(); ?> |