from direct.showbase.DirectObject import DirectObject
from direct.directtools.DirectGeometry import *

from panda3d.core import NodePath, LineSegs

class Mopath(DirectObject):

    nameIndex = 1

    def __init__(self, name = None, fluid = 1, objectToLoad = None, upVectorNodePath = None, reverseUpVector = False):
        if (name == None):
            name = 'mopath%d' % self.nameIndex
            self.nameIndex = self.nameIndex + 1
        self.name = name
        self.fluid = fluid
        self.tPoint = Point3(0)
        self.posPoint = Point3(0)
        self.hprPoint = Point3(0)
        self.tangentVec = Vec3(0)
        self.fFaceForward = 0
        self.faceForwardDelta = None
        self.faceForwardNode = None
        self.timeScale = 1
        self.upVectorNodePath = upVectorNodePath
        self.reverseUpVector = reverseUpVector
        self.reset()
        if isinstance( objectToLoad, NodePath ):
            self.loadNodePath( objectToLoad )
        elif isinstance( objectToLoad, str ):
            self.loadFile( objectToLoad )
        elif objectToLoad is not None:
            print "Mopath: Unable to load object '%s', objectToLoad must be a file name string or a NodePath" % objectToLoad

    def getMaxT(self):
        return self.maxT * self.timeScale

    def loadFile(self, filename, fReset = 1):
        nodePath = loader.loadModel(filename)
        if nodePath:
            self.loadNodePath(nodePath)
            nodePath.removeNode()
        else:
            print 'Mopath: no data in file: %s' % filename


    def loadNodePath(self, nodePath, fReset = 1):
        if fReset:
            self.reset()

        self.__extractCurves(nodePath)
        if (self.tNurbsCurve != []):
            self.maxT = self.tNurbsCurve[-1].getMaxT()
        elif (self.xyzNurbsCurve != None):
            self.maxT = self.xyzNurbsCurve.getMaxT()
        elif (self.hprNurbsCurve != None):
            self.maxT = self.hprNurbsCurve.getMaxT()
        else:
            print 'Mopath: no valid curves in nodePath: %s' % nodePath


    def reset(self):
        self.maxT = 0.0
        self.loop = 0
        self.xyzNurbsCurve = None
        self.hprNurbsCurve = None
        self.tNurbsCurve = []
        self.node = None

    def __extractCurves(self, nodePath):
        node = nodePath.node()
        if isinstance(node, ParametricCurve):
            if node.getCurveType() == PCTXYZ:
                self.xyzNurbsCurve = node
            elif node.getCurveType() == PCTHPR:
                self.hprNurbsCurve = node
            elif node.getCurveType() == PCTNONE:
                if (self.xyzNurbsCurve == None):
                    self.xyzNurbsCurve = node
                else:
                    print 'Mopath: got a PCT_NONE curve and an XYZ Curve in nodePath: %s' % nodePath
            elif (node.getCurveType() == PCTT):
                self.tNurbsCurve.append(node)
        else:
            # Iterate over children if any
            for child in nodePath.getChildren():
                self.__extractCurves(child)

    def calcTime(self, tIn):
        return self.__calcTime(tIn, self.tNurbsCurve)

    def __calcTime(self, tIn, tCurveList):
        if tCurveList:
            tCurveList[-1].getPoint(tIn, self.tPoint)
            return self.__calcTime(self.tPoint[0], tCurveList[:-1])
        else:
            return tIn

    def getFinalState(self):
        pos = Point3(0)
        if (self.xyzNurbsCurve != None):
            self.xyzNurbsCurve.getPoint(self.maxT, pos)
        hpr = Point3(0)
        if (self.hprNurbsCurve != None):
            self.hprNurbsCurve.getPoint(self.maxT, hpr)
        return (pos, hpr)

    def goTo(self, node, time):
        if (self.xyzNurbsCurve == None) and (self.hprNurbsCurve == None):
            print 'Mopath: Mopath has no curves'
            return
        time /= self.timeScale
        self.playbackTime = self.calcTime(CLAMP(time, 0.0, self.maxT))
        if (self.xyzNurbsCurve != None):
            self.xyzNurbsCurve.getPoint(self.playbackTime, self.posPoint)
            if self.fluid:
                node.setFluidPos(self.posPoint)
            else:
                node.setPos(self.posPoint)
        if (self.hprNurbsCurve != None):
            self.hprNurbsCurve.getPoint(self.playbackTime, self.hprPoint)
            node.setHpr(self.hprPoint)
        elif (self.fFaceForward and (self.xyzNurbsCurve != None)):
            if self.faceForwardDelta:
                # Look at a point a bit ahead in parametric time.
                t = min(self.playbackTime + self.faceForwardDelta, self.xyzNurbsCurve.getMaxT())
                lookPoint = Point3()
                self.xyzNurbsCurve.getPoint(t, lookPoint)
                if self.faceForwardNode:
                    self.faceForwardNode.setPos(lookPoint)
            else:
                self.xyzNurbsCurve.getTangent(self.playbackTime, self.tangentVec)
                lookPoint = self.posPoint + self.tangentVec

            # use the self.upVectorNodePath position if it exists to
            # create an up vector for lookAt
            if (self.upVectorNodePath is None):
                node.lookAt(lookPoint)
            else:
                if (self.reverseUpVector == False):
                     node.lookAt(lookPoint,
                                 self.upVectorNodePath.getPos() - self.posPoint)
                else:
                     node.lookAt(lookPoint,
                                 self.posPoint - self.upVectorNodePath.getPos())

    def play(self, node, time = 0.0, loop = 0):
        if (self.xyzNurbsCurve == None) and (self.hprNurbsCurve == None):
            print 'Mopath: Mopath has no curves'
            return
        self.node = node
        self.loop = loop
        self.stop()
        t = taskMgr.add(self.__playTask, self.name + '-play')
        t.currentTime = time
        t.lastTime = globalClock.getFrameTime()

    def stop(self):
        taskMgr.remove(self.name + '-play')

    def __playTask(self, task):
        time = globalClock.getFrameTime()
        dTime = time - task.lastTime
        task.lastTime = time
        if (self.loop):
            cTime = (task.currentTime + dTime) % self.getMaxT()
        else:
            cTime = task.currentTime + dTime
        if ((self.loop == 0) and (cTime > self.getMaxT())):
            self.stop()
            messenger.send(self.name + '-done')
            self.node = None
            return task.done
        self.goTo(self.node, cTime)
        task.currentTime = cTime
        return task.cont

    def draw(self, subdiv = 1000):
        """ Draws a quick and cheesy visualization of the Mopath using
        LineSegs.  Returns the NodePath representing the drawing. """

        ls = LineSegs('mopath')
        p = Point3()
        for ti in range(subdiv):
            t = float(ti) / float(subdiv) * self.maxT
            tp = self.calcTime(t)
            self.xyzNurbsCurve.getPoint(tp, p)
            ls.drawTo(p)
        
        return NodePath(ls.create())