> I have a model like
>
> class Task(models.Model):
>   name = Models.CharField(max_length = 100)
>   parent = models.ForeignKey('Task', null = True)
>
> Using this model say I have got a table like,
>
> Task
> -------
> ID  Name Parent_id
> 1   Foo     null
> 2  Bar       1
> 3 Baz        1
> 4 Bax        2
>
> I want to display this information hierchically in the template,
> something like
>
> <ul>
> <li> Foo </li>
> <ul>
>   <li> Bar</li>
>    <ul><li> Bax </li></ul>
>   <li> Baz </li>
> </ul>
> </ul>
>
> Essentially I want to convert a relational data to monarchical data.
> Does any body have snippets/recipe to do something similar? I am using
> mySQL


Shabda, does the piece of code below help? Your problem is very  
similar to what I just have tried to do. In my case, for a threaded  
forum display.
I've created a class ThreadList, which creates ThreadListItems one by  
one. You then create a ThreadList object on a QuerySet, and step  
through it in your template.
Let me know if it's useful, and suggestions for improvements if you  
use it (one thing I'm thinking of, is allowing the user to specify  
the "<li><ul>" and "</ul></li>" tags; could be <div>s, for example).  
And I could certainly use a better solution for iterating through the  
query_set (this is where my last question came from: http:// 
groups.google.com/group/django-users/browse_thread/thread/ 
15c236dedb08b4c4/cf6464805af429ff?hl=en#cf6464805af429ff

   Evert


Class definitions all the way at the bottom.

Somewhere in your views.py:

   responseDict['thread'] = ThreadList(Task.objects.all())   # filter  
as you like; you can use Paginator objects as well, as long as the  
result a QuerySet (actually, a list might also work)


Somewhere in your template:

{% if thread %}
<ul class="thread">
{% for t in thread %}
{{ t.step }}
<li>
{{ t.name }}
</li>
{% endfor %}
{{ thread.final }}
</ul>
{% endif %}

The if-clause isn't really necessary, just to prevent an empty <ul></ 
ul>.
The t.step method steps you 'inwards' or 'outwards', depending on the  
level.
The thread.final is necessary to trace backwards enough, so that you  
close all <ul> levels


The class definitions:

class ThreadListItem(object):
     def __init__(self, item, depth):
         self._item = item
         self.depth = depth
     def step(self):
         if self.depth > 0:
             return mark_safe(u'<li><ul>' * self.depth)
         else:
             return mark_safe(u'</ul></li>' * abs(self.depth))
     def item(self):
         return self._item
     def __unicode__(self):
         return self.thread_in() + unicode(self.item()) +  
self.thread_out()


class ThreadList(object):
     def __init__(self, query_set, depth='depth', parent='parent'):
         # Note: I'm using len() instead of .count(), so the query_set
         # gets evaluated. Without evaluation, the query_set indexing
         # in next() goes awry
         # 'quick note added in proof/email': the 'depth' key could  
be used if your model has a
         # specific thread-depth parameter. Otherwise, preferably set  
to None.
         self.query_set = query_set
         self.max = len(self.query_set)
         self.depth = depth
         self.parent = parent
         self.reset()
     def __iter__(self):
         return self
     def next(self):
         self.i += 1
         if self.i >= self.max:
             raise StopIteration
         else:
             item = self.query_set[self.i]
             if self.depth and hasattr(item, self.depth):
                 depth = int(item.__getattribute__(self.depth))
             elif self.parent and hasattr(item, self.parent):
                 depth = 0
                 p = item.__getattribute__(self.parent)
                 while p:
                     depth += 1
                     p = p.__getattribute__(self.parent)
             d = depth - self.current_level
             self.current_level = depth
             return ThreadListItem(item, d)
     def final(self):
         return mark_safe(u'</ul></li>' * self.current_level)
     def reset(self):
         self.i = -1
         self.current_level = 0





--~--~---------~--~----~------------~-------~--~----~
You received this message because you are subscribed to the Google Groups 
"Django users" group.
To post to this group, send email to django-users@googlegroups.com
To unsubscribe from this group, send email to [EMAIL PROTECTED]
For more options, visit this group at 
http://groups.google.com/group/django-users?hl=en
-~----------~----~----~----~------~----~------~--~---

Reply via email to