[TransWarp] peak.web and forms
Phillip J. Eby
pje at telecommunity.com
Tue Aug 5 14:47:09 EDT 2003
At 08:15 PM 8/5/03 +0200, Roché Compaan wrote:
>* Phillip J. Eby <pje at telecommunity.com> [2003-08-05 16:32]:
> > Now that I've explained all that,
>
>Thank you very much, I really appreciate it. There are a lot classes in
>the web mix at the moment and to get one's head around - an updated web
>example will really help ground the concepts and make more feedback
>possible.
It's coming. I'm just trying to get everything finished, first.
> > Of course, I'm not sure that this really saves anything. It may be that I
> > should just work out how to easily invoke a template from inside a method
> > on an object.
>
>At the moment they can only be invoked from a TraversalContext, right?
Well, it's more correct to say that a DOMlet needs a traversal context in
order to run. renderTo() now takes a context and a state. (Hm, I should
update the interface, which I think still says 'data' is a
traversable. Now it's a context.)
>And a WebTraversable has access to the traversal context so one should
>be able to instruct a template to render itself from there as well?
No, traversables don't have access to a context, they always take it as a
parameter. This is important for traversables to be reusable in different
contexts. Consider a traversable that wraps a "Contact" object... it
can't/shouldn't know if the Contact was accessed by traversing from an
"Order", or by direct access from a specialist.
Before last weekend, traversables did know this, and things were too
complicated and variable. Now, there are three objects involved in any
traversal:
1. The traversal context, which knows how you got to where you are, what
interaction this is part of, and also holds the next two items...
2. A traversable, which knows how to traverse the..
3. Current domain object being traversed. If the domain object is
traversable, then #2 above is the same object as this.
So, the traversal context holds everything you need to do
anything. context.contextFor('name') returns the context that results from
traversing to 'name'. It does this by asking the traversable (#2 above) to
traverseTo('name',interaction), and then it wraps a new context around the
result, using self.subcontext('name',result).
> > One thing that talking about this scenario has brought up,
> > is that there's currently no way to get at the traversal context from
> > inside a method like this, so that makes things harder.
>
>Inside methods of which other objects should a template be accessible?
>It sounds convenient to have access to the context all over the place
>but on the other hand if one only has access to the traversal context
>from within a web traversable or a decorator won't this enforce a modular
>design? Anyway, it feels like I might be missing the point again,
>or lacking context ;-)
Well, the problem is equivalent, because it might be that the method we're
talking about is on the traversable, or it might be on the domain object,
or the domain object might be traversable. So that issue isn't really
important. :)
> > Otherwise, you
> > could just say something like:
> >
> > return context.subcontext('addForm', self.addForm).render()
> >
> > or:
> > return context.contextFor('addForm').render()
>
>What is the difference between just "context" and
>"contextFor('addForm')" in the line above.
If you're actually asking here about the difference between
'subcontext(name,ob)' and 'contextFor(name)', it's that 'contextFor(name)'
does the work of figuring out for you, what 'ob' should be in the call to
'subcontext()'. :)
> I know the interface says
>contextFor returns a new traversal context but why would you want one?
Because contexts want to be immutable. That way, if I pass a context
through to a method I call, I don't have to worry about that method
modifying my context. It's a bad idea to divorce allocation from usage in
this fashion. If I always create a new object when I need new data, then
the management burden doesn't propagate all over the code. Think of it as
making the code that wants something, pay the price for getting it.
Another convenient side effect, is that because contexts chain to one
another, they get the caching benefit of their use of bindings. For
example, a given context will compute its absolute or traversed URL at most
once, and any child contexts spun off from it will get the benefit of this
in how long it takes for them to compute their own URLs. E.g.:
Traversal ('/')
/
TraversalContext ('/foo')
/
TraversalContext('/foo/bar')
If I start with the root traversal object, and do 'contextFor("foo")', and
then take that object and get its 'contextFor("bar")', I will have a
component tree as shown above. Asking for a 'contextFor("..")' will go
back up the tree and reuse the previous traversal context, which also
contributes to caching effects.
More information about the PEAK
mailing list