"""ParentMgr module: contains the ParentMgr class"""

from direct.directnotify import DirectNotifyGlobal
from direct.showbase.PythonUtil import isDefaultValue
import types

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 types.IntType:
            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]