150 lines
7.5 KiB
Python
150 lines
7.5 KiB
Python
|
"""ParentMgr module: contains the ParentMgr class"""
|
||
|
|
||
|
from direct.directnotify import DirectNotifyGlobal
|
||
|
from direct.showbase.PythonUtil import isDefaultValue
|
||
|
|
||
|
|
||
|
class ParentMgr:
|
||
|
# This is now used on the AI as well.
|
||
|
"""ParentMgr holds a table of nodes that avatars may be parented to
|
||
|
in a distributed manner. All clients within a particular zone maintain
|
||
|
identical tables of these nodes, and the nodes are referenced by 'tokens'
|
||
|
which the clients can pass to each other to communicate distributed
|
||
|
reparenting information.
|
||
|
|
||
|
The functionality of ParentMgr used to be implemented with a simple
|
||
|
token->node dictionary. As distributed 'parent' objects were manifested,
|
||
|
they would add themselves to the dictionary. Problems occured when
|
||
|
distributed avatars were manifested before the objects to which they
|
||
|
were parented to.
|
||
|
|
||
|
Since the order of object manifestation depends on the order of the
|
||
|
classes in the DC file, we could maintain an ordering of DC definitions
|
||
|
that ensures that the necessary objects are manifested before avatars.
|
||
|
However, it's easy enough to keep a list of pending reparents and thus
|
||
|
support the general case without requiring any strict ordering in the DC.
|
||
|
"""
|
||
|
|
||
|
notify = DirectNotifyGlobal.directNotify.newCategory('ParentMgr')
|
||
|
|
||
|
def __init__(self):
|
||
|
self.token2nodepath = {}
|
||
|
# these are nodepaths that have requested to be parented to
|
||
|
# a node that has not yet registered as a parent
|
||
|
self.pendingParentToken2children = {}
|
||
|
# Multiple reparent requests may come in for a given child
|
||
|
# before that child can successfully be reparented. We need to
|
||
|
# make sure that each child is only scheduled to be parented to
|
||
|
# a single parent, at most.
|
||
|
# For efficient removal of pending children, we keep a dict
|
||
|
# of pending children to the token of the parent they're waiting for
|
||
|
self.pendingChild2parentToken = {}
|
||
|
|
||
|
def destroy(self):
|
||
|
del self.token2nodepath
|
||
|
del self.pendingParentToken2children
|
||
|
del self.pendingChild2parentToken
|
||
|
|
||
|
def privRemoveReparentRequest(self, child):
|
||
|
""" this internal function removes any currently-pending reparent
|
||
|
request for the given child nodepath """
|
||
|
if child in self.pendingChild2parentToken:
|
||
|
self.notify.debug("cancelling pending reparent of %s to '%s'" %
|
||
|
(repr(child),
|
||
|
self.pendingChild2parentToken[child]))
|
||
|
parentToken = self.pendingChild2parentToken[child]
|
||
|
del self.pendingChild2parentToken[child]
|
||
|
self.pendingParentToken2children[parentToken].remove(child)
|
||
|
|
||
|
def requestReparent(self, child, parentToken):
|
||
|
if parentToken in self.token2nodepath:
|
||
|
# this parent has registered
|
||
|
# this child may already be waiting on a different parent;
|
||
|
# make sure they aren't any more
|
||
|
self.privRemoveReparentRequest(child)
|
||
|
self.notify.debug("performing wrtReparent of %s to '%s'" %
|
||
|
(repr(child), parentToken))
|
||
|
child.wrtReparentTo(self.token2nodepath[parentToken])
|
||
|
else:
|
||
|
if isDefaultValue(parentToken):
|
||
|
self.notify.error('child %s requested reparent to default-value token: %s' % (repr(child), parentToken))
|
||
|
self.notify.debug(
|
||
|
"child %s requested reparent to parent '%s' that is not (yet) registered" %
|
||
|
(repr(child), parentToken))
|
||
|
# cancel any pending reparent on behalf of this child
|
||
|
self.privRemoveReparentRequest(child)
|
||
|
# make note of this pending parent request
|
||
|
self.pendingChild2parentToken[child] = parentToken
|
||
|
self.pendingParentToken2children.setdefault(parentToken, [])
|
||
|
self.pendingParentToken2children[parentToken].append(child)
|
||
|
# there is no longer any valid place for the child in the
|
||
|
# scenegraph; put it under hidden
|
||
|
child.reparentTo(hidden)
|
||
|
|
||
|
def registerParent(self, token, parent):
|
||
|
if token in self.token2nodepath:
|
||
|
self.notify.error(
|
||
|
"registerParent: token '%s' already registered, referencing %s" %
|
||
|
(token, repr(self.token2nodepath[token])))
|
||
|
|
||
|
if isDefaultValue(token):
|
||
|
self.notify.error('parent token (for %s) cannot be a default value (%s)' % (repr(parent), token))
|
||
|
|
||
|
if type(token) is int:
|
||
|
if token > 0xFFFFFFFF:
|
||
|
self.notify.error('parent token %s (for %s) is out of uint32 range' % (token, repr(parent)))
|
||
|
|
||
|
self.notify.debug("registering %s as '%s'" % (repr(parent), token))
|
||
|
self.token2nodepath[token] = parent
|
||
|
|
||
|
# if we have any pending children, add them
|
||
|
if token in self.pendingParentToken2children:
|
||
|
children = self.pendingParentToken2children[token]
|
||
|
del self.pendingParentToken2children[token]
|
||
|
for child in children:
|
||
|
# NOTE: We do a plain-old reparentTo here (non-wrt)
|
||
|
# under the assumption that the node has been
|
||
|
# positioned as if it is already under the new parent.
|
||
|
#
|
||
|
# The only case that I can think of where the parent
|
||
|
# node would not have been registered at the time of
|
||
|
# the reparent request is when we're entering a new
|
||
|
# zone and manifesting remote toons along with
|
||
|
# other distributed objects, and a remote toon is
|
||
|
# requesting to be parented to geometry owned by a
|
||
|
# distributed object that has not yet been manifested.
|
||
|
#
|
||
|
# (The situation in the factory is a little different;
|
||
|
# the distributed toons of your companions are never
|
||
|
# disabled, since the toons are in the factory's uberzone.
|
||
|
# They send out requests to be parented to nodes that
|
||
|
# may be distributed objects, which may not be generated
|
||
|
# on your client)
|
||
|
#
|
||
|
# It is therefore important for that remote toon to
|
||
|
# have his position set as a required field, relative
|
||
|
# to the parent node, after the reparent request.
|
||
|
# If the node has already been registered, the toon will
|
||
|
# be in the correct position. Otherwise, the toon will
|
||
|
# have the correct position but the wrong parent node,
|
||
|
# until this code runs and corrects the toon's parent
|
||
|
# node. Since we don't start rendering until all objects
|
||
|
# in a new zone have been generated, all of that action
|
||
|
# will happen in a single frame, and the net result will
|
||
|
# be that the toon will be in the right place when
|
||
|
# rendering starts.
|
||
|
self.notify.debug("performing reparent of %s to '%s'" %
|
||
|
(repr(child), token))
|
||
|
child.reparentTo(self.token2nodepath[token])
|
||
|
# remove this child from the child->parent table
|
||
|
assert self.pendingChild2parentToken[child] == token
|
||
|
del self.pendingChild2parentToken[child]
|
||
|
|
||
|
def unregisterParent(self, token):
|
||
|
if token not in self.token2nodepath:
|
||
|
self.notify.warning("unregisterParent: unknown parent token '%s'" %
|
||
|
token)
|
||
|
return
|
||
|
self.notify.debug("unregistering parent '%s'" % (token))
|
||
|
del self.token2nodepath[token]
|