Как настроить Gtk# TreeModelFilter, который фильтрует базовый TreeStore?

Я прочитал "Учебное пособие по GtkSharp TreeView", в котором автор описывает, как настроить и использовать TreeModelFilter для базового ListStore (в разделе учебного пособия "Фильтрация данных"). Техника, похоже, не работает для лежащего в основе иерархического TreeStore. Я хочу отфильтровать многоуровневый TreeStore и показать результаты в TreeView. Это дает мне очень тяжелое время. Есть ли уроки, образцы или предложения для этого?

Ниже приведен код. В основном это тот же код, что и в учебнике, за исключением изменений, связанных с созданием и заполнением TreeStore, а не ListStore. {TreeStore используется для сохранения "имен" и "адресов электронной почты" контактов, разделенных (и сохраненных как) дочерних элементов корней "друзья" и "родственники"}

// compilation requires references to:
// gtk-sharp, atk-sharp and glib-sharp


using System;
using Gtk;

public class TreeViewExample
{
    public static void Main()
    {
        Gtk.Application.Init();
        new TreeViewExample();
        Gtk.Application.Run();
    }

    Gtk.Entry filterEntry;
    Gtk.TreeModelFilter filter;

    public TreeViewExample()
    {
        // Create a Window
        Gtk.Window window = new Gtk.Window("TreeView Example");
        window.SetSizeRequest(500, 200);
        window.DeleteEvent += delegate { Application.Quit(); };

        // Create an Entry used to filter the tree
        filterEntry = new Gtk.Entry();

        // Fire off an event when the text in the Entry changes
        filterEntry.Changed += OnFilterEntryTextChanged;

        // Create a nice label describing the Entry
        Gtk.Label filterLabel = new Gtk.Label("Search:");

        // Put them both into a little box so they show up side by side
        Gtk.HBox filterBox = new Gtk.HBox();
        filterBox.PackStart(filterLabel, false, false, 5);
        filterBox.PackStart(filterEntry, true, true, 5);

        // Create our TreeView
        Gtk.TreeView tv = new Gtk.TreeView();

        // Create a box to hold the Entry and Tree
        Gtk.VBox box = new Gtk.VBox();

        // Add the widgets to the box
        box.PackStart(filterBox, false, false, 5);
        box.PackStart(tv, true, true, 5);
        window.Add(box);

        //setting up columns and renderers
        Gtk.TreeViewColumn nameColumn = new Gtk.TreeViewColumn { Title = "Name" };
        Gtk.CellRendererText nameCell = new Gtk.CellRendererText();
        nameColumn.PackStart(nameCell, true);
        Gtk.TreeViewColumn emailColumn = new Gtk.TreeViewColumn { Title = "Email" };
        Gtk.CellRendererText emailCell = new Gtk.CellRendererText();
        emailColumn.PackStart(emailCell, true);

        // Add the columns to the TreeView
        tv.AppendColumn(nameColumn);
        tv.AppendColumn(emailColumn);

        // Tell the Cell Renderers which items in the model to display
        nameColumn.AddAttribute(nameCell, "text", 0);
        emailColumn.AddAttribute(emailCell, "text", 1);

        // Create a model that will hold two strings 
        Gtk.TreeStore contacts = new Gtk.TreeStore(typeof(string), typeof(string));

        // Add some hierarchical data
        Gtk.TreeIter treeiter;

        //first root
        treeiter = contacts.AppendValues("FRIENDS");

        // 2 children of first root
        contacts.AppendValues(treeiter, "Ogre", "stinky@hotmale.com");
        contacts.AppendValues(treeiter, "Bee", "stingy@coolguy.com");

        // second root
        treeiter = contacts.AppendValues("RELATIVES");

        // 3 children of second root
        contacts.AppendValues(treeiter, "Mommy", "mother@family.com");
        contacts.AppendValues(treeiter, "Daddy", "father@family.com");
        contacts.AppendValues(treeiter, "tom", "cousin@family.com");

        filter = new Gtk.TreeModelFilter(contacts, null);
        // Specify the function that determines which rows to filter out and which ones to display
        filter.VisibleFunc = new Gtk.TreeModelFilterVisibleFunc(FilterTree);

        // Assign the filter as our treeview's model
        tv.Model = filter;

        // Show the window and everything on it
        window.ShowAll();
    }

    private void OnFilterEntryTextChanged(object o, System.EventArgs args)
    {
        // Since the filter text changed, tell the filter to re-determine which rows to display
        filter.Refilter();
    }

    private bool FilterTree(Gtk.TreeModel model, Gtk.TreeIter iter)
    {
        string contactname = model.GetValue(iter, 0).ToString();
        if (filterEntry.Text == "")
            return true;
        if (contactname.IndexOf(filterEntry.Text) > -1)
            return true;
        else
            return false;
    }
}

[Я использую mono 2.6.4 /monodevelop 2.4 / gtk-sharp 2.12 для Windows Vista.]

2 ответа

Решение

Кажется, что при фильтрации строк в древовидной модели строка видна, только если ВСЕ ее родители тоже видны. Поскольку ваша функция фильтра скрывает родительские узлы, она не будет отображать дочерние узлы, даже если текст соответствует. Я изменил ваш код, чтобы проиллюстрировать эту проблему:

Теперь один из родительских узлов начинается с "test". Если вы введете "test", вы увидите, что фильтрация работает правильно.

using System;
using Gtk;

public class TreeViewExample
{

public static void Main ()

{

    Gtk.Application.Init ();

    new TreeViewExample ();

    Gtk.Application.Run ();

}


Gtk.Entry filterEntry;
Gtk.TreeModelFilter filter;



public TreeViewExample ()
{

    // Create a Window

    Gtk.Window window = new Gtk.Window ("TreeView Example");

    window.SetSizeRequest (500,200);

    window.DeleteEvent+=delegate {Application.Quit();};



    // Create an Entry used to filter the tree

    filterEntry = new Gtk.Entry ();



    // Fire off an event when the text in the Entry changes

    filterEntry.Changed += OnFilterEntryTextChanged;



    // Create a nice label describing the Entry

    Gtk.Label filterLabel = new Gtk.Label ("Search:");



    // Put them both into a little box so they show up side by side

    Gtk.HBox filterBox = new Gtk.HBox ();

    filterBox.PackStart (filterLabel, false, false, 5);

    filterBox.PackStart (filterEntry, true, true, 5);



    // Create our TreeView

    Gtk.TreeView tv = new Gtk.TreeView ();



    // Create a box to hold the Entry and Tree

    Gtk.VBox box = new Gtk.VBox ();



    // Add the widgets to the box

    box.PackStart (filterBox, false, false, 5);

    box.PackStart (tv, true, true, 5);



    window.Add (box);





    //setting up columns and renderers





    Gtk.TreeViewColumn nameColumn = new Gtk.TreeViewColumn{Title="Name"}; 

    Gtk.CellRendererText nameCell = new Gtk.CellRendererText ();        

    nameColumn.PackStart (nameCell, true);









    Gtk.TreeViewColumn emailColumn = new Gtk.TreeViewColumn {Title="Email"}; 

    Gtk.CellRendererText emailCell = new Gtk.CellRendererText ();

    emailColumn.PackStart (emailCell, true);







    // Add the columns to the TreeView

    tv.AppendColumn (nameColumn);

    tv.AppendColumn (emailColumn);







    // Tell the Cell Renderers which items in the model to display

    nameColumn.AddAttribute (nameCell, "text", 0);

    emailColumn.AddAttribute (emailCell, "text", 1);







    // Create a model that will hold two strings 

    Gtk.TreeStore contacts = new Gtk.TreeStore (typeof (string), typeof (string));





    // Add some hierarchical data



    Gtk.TreeIter treeiter;





    //first root

    treeiter= contacts.AppendValues("testFRIENDS"); 



        // 2 children of first root

        contacts.AppendValues(treeiter, "testOgre", "stinky@hotmale.com");

        contacts.AppendValues(treeiter, "testBee", "stingy@coolguy.com");







    // second root

    treeiter= contacts.AppendValues("RELATIVES"); 



        // 3 children of second root

        contacts.AppendValues (treeiter,"Mommy","mother@family.com");

        contacts.AppendValues (treeiter,"Daddy", "father@family.com");

        contacts.AppendValues (treeiter,"tom", "cousin@family.com");









    filter = new Gtk.TreeModelFilter (contacts, null);



    // Specify the function that determines which rows to filter out and which ones to display

    filter.VisibleFunc = new Gtk.TreeModelFilterVisibleFunc (FilterTree);



    // Assign the filter as our treeview's model

    tv.Model = filter;



    // Show the window and everything on it

    window.ShowAll ();

}



private void OnFilterEntryTextChanged (object o, System.EventArgs args)

{

    // Since the filter text changed, tell the filter to re-determine which rows to display

    filter.Refilter ();

}



private bool FilterTree (Gtk.TreeModel model, Gtk.TreeIter iter)

{

    string contactname = model.GetValue (iter, 0).ToString ();



    if (filterEntry.Text == "")

        return true;



    if (contactname.IndexOf (filterEntry.Text) > -1)

        return true;

    else

        return false;

}

}

Самым простым решением с вашей текущей структурой было бы использование функции фильтра, всегда возвращающей значение ИСТИНА для узлов "контейнера" ("Друзья и родственники") на основе значения в скрытом столбце в модели. Это не будет выглядеть точно так, как вы хотите, но это будет работать.

GTK + Treeview Tutorial, хотя и не обновляющийся в течение некоторого времени, по-прежнему ОЧЕНЬ полезный ресурс для всех ваших потребностей TreeView. Код и примеры написаны на C, но большая часть все еще относится к GTK#.

Чтобы достичь правильной функциональности вашего кода, я предлагаю вам изменить его следующим образом:

1.Добавить новое поле private filterBool = false; в ваш класс

2. Модифицировать свой FilterTree метод до этого состояния:

private bool FilterTree (Gtk.TreeModel model, Gtk.TreeIter iter)
{
string contactname = model.GetValue (iter, 0).ToString ();

if (filterEntry.Text == "")
    return true;

if (contactname.IndexOf (filterEntry.Text) > -1)
    return true;

if (model.IterHasChild(iter)) 
{
    filerBool = false;
    investigateChildNodes(model, iter); //method checking if currently investigated
                               //node has any child fulfilling filter contitions
    return filerBool;
}
return false;
}

3. добавить отсутствующий метод

 private void investigateChildNodes(TreeModel model, TreeIter iter) 
    {       
        TreeIter childIter;
        model.IterChildren(out childIter, iter); 
        do
        {
            if (model.GetValue(childIter, 0).ToString().IndexOf(filterEntry.Text) > -1)
                filerBool = true;

            if (model.IterHasChild(childIter))
                investigateChildNodes(model, childIter);

        } while (model.IterNext(ref childIter));
    }

С этой модификацией каждый узел проверяется на возможные дочерние узлы, которые могут выполнять условия фильтрации. Если что-либо обнаружено, узел не отбрасывается.

Версия Тинки работает безотказно. Единственное, что мне не нравится, это приватная переменная. Этого можно избежать, используя возвращаемое значение в функции InvestigateChildrenNodes.

    private bool InvestigateChildNodes(TreeModel model, TreeIter iter)
    {
        TreeIter childIter;
        model.IterChildren(out childIter, iter);

        bool result = false;
        do
        {
            if (model.GetValue(childIter, 0).ToString().Contains(FilterEntry.Text))
            {
                result = true;
                break;
            }

            if (model.IterHasChild(childIter))
            {
                result = InvestigateChildNodes(model, childIter);
                if (result)
                    break;
            }

        } while (model.IterNext(ref childIter));
        return result;
    }
Другие вопросы по тегам