918 lines
39 KiB
Python
918 lines
39 KiB
Python
|
"""
|
||
|
Defines Graph Editor
|
||
|
"""
|
||
|
import wx
|
||
|
import math
|
||
|
from .PaletteTreeCtrl import *
|
||
|
from . import ObjectGlobals as OG
|
||
|
from . import AnimGlobals as AG
|
||
|
from wx.lib.embeddedimage import PyEmbeddedImage
|
||
|
|
||
|
property = [
|
||
|
"translateX",
|
||
|
"translateY",
|
||
|
"translateZ"
|
||
|
]
|
||
|
|
||
|
#----------------------------------------------------------------------
|
||
|
ZoomIn = PyEmbeddedImage(
|
||
|
"iVBORw0KGgoAAAANSUhEUgAAABwAAAAcCAIAAAD9b0jDAAAAA3NCSVQICAjb4U/gAAACgklE"
|
||
|
"QVRIid3VW0vjQBQA4DlJMziZlCKCSB/9Bf4a3wRtNU6SHyXK6pay2nVbQbasskXFO15SqmwL"
|
||
|
"LtYnn7Zg20zSyT50wZikXsCHxXk9Z74ZZuacgebtLXrvIb27+PFQjDEADIoqiiLL8htQAMAY"
|
||
|
"Hx4dPTw8SHEuIaRWqzWbzUQi8SoUAIaGhgqFgn15SSn1Iwmqqh4fH5c2NpLJpO8/xhORzEeR"
|
||
|
"EJLL5VqtViaT8TxPCBFMoJTu7e1tb28zxpLJpOu6L6D9PS6vrHQ6nWw267puVNzZ2alUKqZh"
|
||
|
"UE0LivFoX/y0vOxynpmZ4ZxHxZ+Vyu7urmmaqqqGxBhUkiSM8eLiIgDMTE87EVHTtB9bW4cH"
|
||
|
"B5ZpEkKiYhjt3/XCwoKiKFNTU47jRPf4vVw+PTmxLAtjHCs+QQEAAJaWljRNm5yc7Ha7wQtF"
|
||
|
"CGGMy+WybduWZSmK4nlerIiCTwoAOOfX19cTExMSgPDDT0iSJNu2x8fHU6nUM+ITVAhBCLFM"
|
||
|
"83MuV63VKCGhVM65ruv1RqNYLFJKX4UihFzXTafTc7Oz+XzerlZVVQ1GhRCyLJuGcXV1VSqV"
|
||
|
"QtGBKEKo6zhjY2Pzur66tnZxcRGa2ev1AMA0zXqj8W3wfmPK1HGc0dFRNj//dX397Ows6iKE"
|
||
|
"DMZ+39wUCgVKabTXxNe+4zgjIyMGY8VS6eT0NOr6vs8Yu7u7+7K6SggJuQMbCud8eHjYYGxz"
|
||
|
"c3N/f19RlJDb6/V0Xb+/v8/n829ofZzzVCqlz82dn5//abUk6UmyEEIIkc1k2u32r3o9uCq8"
|
||
|
"+PHJsuz7vud5IfTffABZljnnwejLnb9/grEiQih2vVd9J36kup4fH+w3/S/QvwejQg8ibHgo"
|
||
|
"AAAAAElFTkSuQmCC")
|
||
|
|
||
|
#----------------------------------------------------------------------
|
||
|
ZoomOut = PyEmbeddedImage(
|
||
|
"iVBORw0KGgoAAAANSUhEUgAAABwAAAAcCAIAAAD9b0jDAAAAA3NCSVQICAjb4U/gAAACmUlE"
|
||
|
"QVRIieXVz0sbQRQH8NmdH8smh3jwEIIlB+tNEW1NVps0WxPBv0BQXKQx4N2/xR2xTYQ9iIHc"
|
||
|
"/A+shyqevHiQVBGLeqgoxM1kNjHTQ8DayWb14KX0Hfe9+fDd4bGr/Ly4AK9d6quL/y6qKIqm"
|
||
|
"aQHThBBVfT7HnwlFUTjn3/f3MUKKoviMqurh4WG9XkcIvRQVQoRCodPT0+1yWdO0bhchxD2v"
|
||
|
"WCxyzjHGL0IBAA8PD9biIgDAcRxN06Q39Tzvk2kahmHbNmMswP3rmBCi0WhYlkUI2dzcxBhL"
|
||
|
"ruu6mUwmnU6v2bbrur1c+dY77sLCQjgcLpVKGKFuN5VKmaZpU1qr1Qghz6OP7vz8fF9f39di"
|
||
|
"Efq5H6amcrkcpfTu7q7b9d8PIQRjbG5urr+//8vGBoSw200mErOzs3R9/fb2VnLh6uqqrwsA"
|
||
|
"aLVao6OjV9fX3/b2RkZGJLfZbMbj8UgkUqlU3g4NhcNhIURQ0k612+12u/1ufPxHtcoY6157"
|
||
|
"z/OGh4fv7++vLi8hhI/Pg9ZY1/Xz83PHcVZWViKRSLPZfNpFCHmeRynNZrNjY2ONRuOx1TOp"
|
||
|
"rutnZ2eO41iWNTg4KIkYY875mm2/n5iYnp5+KvZMqut6tVrd2tpaWlqKx+OMMUmsM0YpnZqc"
|
||
|
"NE3TdV3puA+q6/rJycl2ufw5n38zMCClwBjXXdem9GMmk06lukUfNBQKHR8fVyqVwvJyLBaT"
|
||
|
"REJIrVbr3KNhGL6ijGKMj46OdnZ2CoVCNBqVRAjhr5ubUqmUm5kxksleIpA+fYyxg4ODfD4f"
|
||
|
"jUY559IohHB3dzebzSYTiQARAKA8/fEJIVRVhRC2Wi3faSEEIUTahKCknbAAgF4iAEBV1WdF"
|
||
|
"Ge1kCZgO7vZEX6X+c/Q3Qy8zmmCLWqIAAAAASUVORK5CYII=")
|
||
|
|
||
|
#----------------------------------------------------------------------
|
||
|
OneTangent = PyEmbeddedImage(
|
||
|
"iVBORw0KGgoAAAANSUhEUgAAABwAAAAcCAIAAAD9b0jDAAAAA3NCSVQICAjb4U/gAAACKUlE"
|
||
|
"QVRIia2Wz2saURDH5/3Yt6aVsE3SomArGO0phxCSU6A2EFgSklP+qPwrXkOQkBCoaG8eQrBQ"
|
||
|
"KRYVU1tabUij3V1337weAtZI1FX3e9qF2c/OzJv9zpKbRgMGRAhRSsF8oo9uOCeUMk0LDEo1"
|
||
|
"rVUuF46Pv2SzhDEgJBhoLZer5/OViwu32yWUjnlsvHj/SrluPJ222m3GORMC5ugsGTwoyjkT"
|
||
|
"4nuxeFetvj06ko4zG/RRjeh5rmVFNjeZrt8UCnxhIQAoAIBSnm0n9vb+1Gq/y2Wm60FAAUAp"
|
||
|
"UGr18LCRz9vtNuH8iZipoQAopW4Yb3Z2vmazgDjtJIyMlo7zIpUyksnq2RkTIhgoAHiWFdve"
|
||
|
"RsTmx4IuNP/ZToiUvV7CNNt3natb61b57cKEMIXIOeu+N0uof5IaEl8f7+R3K1SG8sKc0FqF"
|
||
|
"oPTjCZOhEiDF0Vyk8da3+occC4UCgAKAAhBe73X6Xa/T+VEs8klcv0eKCtD1kgcHP6+v7xsN"
|
||
|
"OtZzp5hqhUh1PbG/Xz0/l5ZFGAsACgDous+j0cjWVuX0lI428qmdWNr2q/X10NJS/fJyVHNn"
|
||
|
"sXfPtuO7u39brV+lkgiHmRBDKZOhbepThDHpOJ8zmZ7jLMZiq6YJhPSXxYyLSEkZMgx9ebl8"
|
||
|
"clLKZO6bTTbgkFN7ZV+e47xcW4tubIQjkWcrKyjl/zpmK/9BlDH0vIfZUogBZAoAKCWhVCk1"
|
||
|
"tHrnggLAk/9I/wC0eekA1bLbIQAAAABJRU5ErkJggg==")
|
||
|
|
||
|
#----------------------------------------------------------------------
|
||
|
TwoTangents = PyEmbeddedImage(
|
||
|
"iVBORw0KGgoAAAANSUhEUgAAABwAAAAcCAIAAAD9b0jDAAAAA3NCSVQICAjb4U/gAAACL0lE"
|
||
|
"QVRIie3UzU8TQRQA8DczO1vatRgh1YhWD/WD+kG91FgOXLhx4+CpiYnE9L8g/gVwJHhp4lUP"
|
||
|
"pPHgQRMOGiEaDabV1LRAtdjUAG23m3W/Znc8YEo/trRNOGjCO+7M++3MezODdopFOO7Axy6e"
|
||
|
"oC0oAqAEY4QGyCcE4faVtaII7dQUy7b7dDEheq3GdB0R4o76KH2+kZ158uzxyzcO56iXS0Rx"
|
||
|
"N5tdnZ9fW1w0FaXZPUQxRl/Le7+q8mruh2paQsem2kLweCq5XGVrq7K5adTruGk+ahx+AeOS"
|
||
|
"rLz4ktctdtYvPYje0i3Gu22cUr1a3VheHg6FhoPBC9EoBwDO21EOQDH2igIALL39NOrz3r8z"
|
||
|
"rppWp3jQmQ8LC5enp4NTU0zTmGG0/PJwKgBzHEU3VcN6eHfie7X++ltBEmknSoaG0snkSDg8"
|
||
|
"Njmp12ptIrieU4dzAEjEIu8KPz8Wy14qNI9SScqvrHCA67Ozlqq6F8f1K3McL6WPYpFUOre9"
|
||
|
"L3uEv50VfL7S+vpuJjMxN8cMo1HEvlAAMG37nF+KR28+fZ/e/62JBAuiWC8U8qlUJJFAhHDH"
|
||
|
"6ZZ71LnRLXY1MDJzI5Rc+6wwR5PlTDIZjsd9gYBjuTSwEajneypR4dV2qXxq1F/dG5fLodg9"
|
||
|
"U9OOThGOHgYAg7FrF8fKKqjng/4rl+xeYl+ozeEMsNsiwmCftnnXQg6EcgABICxyAGAcut2x"
|
||
|
"wdADt+uFdYv/9OU/Qf9d9A9dhOgRU8MmewAAAABJRU5ErkJggg==")
|
||
|
|
||
|
#----------------------------------------------------------------------
|
||
|
|
||
|
class GraphEditorWindow(wx.Window):
|
||
|
"""
|
||
|
This is the main graph editor window.
|
||
|
"""
|
||
|
def __init__(self, parent, windowSize, property, xRange, yRange, curFrame, object):
|
||
|
wx.Window.__init__(self, parent, size = windowSize, style = wx.SUNKEN_BORDER)
|
||
|
|
||
|
self._mainDialog = wx.GetTopLevelParent(self)
|
||
|
self.w,self.h = self.GetClientSize()
|
||
|
|
||
|
self.zoom = float(2)
|
||
|
self._mouseIn = False
|
||
|
self._selectRec = False
|
||
|
self._selectHandler = False
|
||
|
self._OneTangent = True
|
||
|
|
||
|
self.object = object
|
||
|
self.curFrame = curFrame
|
||
|
self.property = property
|
||
|
|
||
|
self.zeroPos = (float(0), self.h/float(2))
|
||
|
self.zero = 0
|
||
|
self.unitWidth = self.w/float(xRange)
|
||
|
self.unitHeight = self.h/float(yRange)
|
||
|
|
||
|
self.generateInfo()
|
||
|
self.InitBuffer()
|
||
|
|
||
|
self.Bind(wx.EVT_PAINT, self.OnPaint)
|
||
|
self.Bind(wx.EVT_SIZE, self.OnSize)
|
||
|
self.Bind(wx.EVT_LEFT_DOWN, self.OnLeftDown)
|
||
|
self.Bind(wx.EVT_LEFT_UP, self.OnLeftUp)
|
||
|
self.Bind(wx.EVT_MIDDLE_DOWN, self.OnMiddleDown)
|
||
|
self.Bind(wx.EVT_MIDDLE_UP, self.OnMiddleUp)
|
||
|
self.Bind(wx.EVT_MOTION, self.OnMotion)
|
||
|
|
||
|
def refresh(self):
|
||
|
self._mouseIn = False
|
||
|
self._selectRec = False
|
||
|
self._selectHandler = False
|
||
|
self.generateInfo()
|
||
|
|
||
|
def generateInfo(self):
|
||
|
#generate the information for animation curve generation
|
||
|
self.X = []
|
||
|
self.Y = []
|
||
|
self.Z = []
|
||
|
|
||
|
if self._mainDialog.editor.animMgr.keyFramesInfo != {}:
|
||
|
self.keyFramesInfo = self._mainDialog.editor.animMgr.keyFramesInfo
|
||
|
for key in self.keyFramesInfo:
|
||
|
if key == (self.object[OG.OBJ_UID], 'X'):
|
||
|
for i in range(len(self.keyFramesInfo[key])):
|
||
|
item = self.keyFramesInfo[key][i]
|
||
|
handler = self.generateHandler(item)
|
||
|
self.X.append([key, i, handler[0], handler[1], handler[2], handler[3], handler[4]])
|
||
|
if key == (self.object[OG.OBJ_UID], 'Y'):
|
||
|
for i in range(len(self.keyFramesInfo[key])):
|
||
|
item = self.keyFramesInfo[key][i]
|
||
|
handler = self.generateHandler(item)
|
||
|
self.Y.append([key, i, handler[0], handler[1], handler[2], handler[3], handler[4]])
|
||
|
if key == (self.object[OG.OBJ_UID], 'Z'):
|
||
|
for i in range(len(self.keyFramesInfo[key])):
|
||
|
item = self.keyFramesInfo[key][i]
|
||
|
handler = self.generateHandler(item)
|
||
|
self.Z.append([key, i, handler[0], handler[1], handler[2], handler[3], handler[4]])
|
||
|
|
||
|
def generateHandler(self, item):
|
||
|
#generate the position for the control handler
|
||
|
x1 = self.zeroPos[0] + float(item[AG.FRAME])*self.unitWidth
|
||
|
y1 = self.zeroPos[1] - float(item[AG.VALUE])*self.unitHeight
|
||
|
|
||
|
t1x = item[AG.INSLOPE][0]*self.unitWidth
|
||
|
t1y = item[AG.INSLOPE][1]*self.unitHeight
|
||
|
|
||
|
t2x = item[AG.OUTSLOPE][0]*self.unitWidth
|
||
|
t2y = item[AG.OUTSLOPE][1]*self.unitHeight
|
||
|
|
||
|
tanA = t1y/t1x
|
||
|
temp1 = float(1)/(tanA**2+1)
|
||
|
if t1x <0 :
|
||
|
cosA = -math.sqrt(abs(temp1))
|
||
|
if t1x >=0:
|
||
|
cosA = math.sqrt(abs(temp1))
|
||
|
temp2 = (tanA**2)*temp1
|
||
|
if t1y <0 :
|
||
|
sinA = -math.sqrt(abs(temp2))
|
||
|
if t1y >=0:
|
||
|
sinA = math.sqrt(abs(temp2))
|
||
|
|
||
|
x2 = x1-float(self.unitWidth*self.zoom)*cosA
|
||
|
y2 = y1+float(self.unitWidth*self.zoom)*sinA
|
||
|
|
||
|
tanA = t2y/t2x
|
||
|
temp1 = float(1)/(tanA**2+1)
|
||
|
if t2x <0 :
|
||
|
cosA = -math.sqrt(abs(temp1))
|
||
|
if t2x >=0:
|
||
|
cosA = math.sqrt(abs(temp1))
|
||
|
temp2 = (tanA**2)*temp1
|
||
|
if t2y <0 :
|
||
|
sinA = -math.sqrt(abs(temp2))
|
||
|
if t2y >=0:
|
||
|
sinA = math.sqrt(abs(temp2))
|
||
|
|
||
|
x3 = x1+float(self.unitWidth*self.zoom)*cosA
|
||
|
y3 = y1-float(self.unitWidth*self.zoom)*sinA
|
||
|
|
||
|
return [[(x1,y1),0],[(x2,y2),0],[(x3,y3),0],[t1x,t1y],[t2x,t2y]]
|
||
|
|
||
|
def InitBuffer(self):
|
||
|
self.buffer = wx.EmptyBitmap(self.w, self.h)
|
||
|
dc = wx.BufferedDC(wx.ClientDC(self), self.buffer)
|
||
|
self.DrawXCoord(dc)
|
||
|
self.DrawYCoord(dc)
|
||
|
self.DrawFrame(dc)
|
||
|
self.DrawCurve(dc)
|
||
|
self.DrawSelectRec(dc)
|
||
|
|
||
|
def SetGraphEditorData(self, property, curFrame = 1):
|
||
|
self.curFrame = curFrame
|
||
|
self.property = property
|
||
|
|
||
|
self.InitBuffer()
|
||
|
|
||
|
def OnPaint(self, evt):
|
||
|
dc = wx.BufferedPaintDC(self, self.buffer)
|
||
|
|
||
|
def DrawXCoord(self,dc):
|
||
|
dc.SetBackground(wx.Brush(wx.Colour(200, 200, 200)))
|
||
|
dc.Clear()
|
||
|
|
||
|
dc.SetPen(wx.BLACK_PEN)
|
||
|
dc.SetBrush(wx.BLACK_BRUSH)
|
||
|
dc.SetFont(wx.Font(8, wx.SWISS, wx.NORMAL, wx.NORMAL))
|
||
|
|
||
|
dc.DrawLine(self.zeroPos[0], float(0), self.zeroPos[0], self.h)
|
||
|
st = str(self.zero)
|
||
|
self.tw,self.th = dc.GetTextExtent(st)
|
||
|
dc.DrawText(st, self.zeroPos[0]+1.0, self.h-self.th-0.5)
|
||
|
|
||
|
dc.SetPen(wx.Pen(wx.Colour(150, 150, 150)))
|
||
|
dc.SetBrush(wx.Brush(wx.Colour(150, 150, 150)))
|
||
|
|
||
|
if self.unitWidth >= 25:
|
||
|
posPos = self.zeroPos[0]+self.unitWidth
|
||
|
posNum = self.zero + 1
|
||
|
while posPos <= self.w:
|
||
|
dc.DrawLine(posPos, float(0), posPos, self.h)
|
||
|
st = str(posNum)
|
||
|
self.drawXNumber(dc, st, posPos)
|
||
|
posPos += self.unitWidth
|
||
|
posNum += 1
|
||
|
|
||
|
negPos = self.zeroPos[0]-self.unitWidth
|
||
|
negNum = self.zero - 1
|
||
|
while negPos >= float(0):
|
||
|
dc.DrawLine(negPos, float(0), negPos, self.h)
|
||
|
st = str(negNum)
|
||
|
self.drawXNumber(dc, st, negPos)
|
||
|
negPos -= self.unitWidth
|
||
|
posNum -= 1
|
||
|
|
||
|
elif self.unitWidth >= 10 and self.unitWidth <= 25:
|
||
|
posPos = self.zeroPos[0]+self.unitWidth*float(2)
|
||
|
posNum = self.zero + 2
|
||
|
while posPos <= self.w:
|
||
|
dc.DrawLine(posPos, float(0), posPos, self.h)
|
||
|
st = str(posNum)
|
||
|
self.drawXNumber(dc, st, posPos)
|
||
|
posPos += self.unitWidth*float(2)
|
||
|
posNum += 2
|
||
|
|
||
|
negPos = self.zeroPos[0]-self.unitWidth*float(2)
|
||
|
negNum = self.zero - 2
|
||
|
while negPos >= float(0):
|
||
|
dc.DrawLine(negPos, float(0), negPos, self.h)
|
||
|
st = str(negNum)
|
||
|
self.drawXNumber(dc, st, negPos)
|
||
|
negPos -= self.unitWidth*float(2)
|
||
|
posNum -= 2
|
||
|
|
||
|
elif self.unitWidth >= 2 and self.unitWidth <= 10:
|
||
|
posPos = self.zeroPos[0]+self.unitWidth*float(5)
|
||
|
posNum = self.zero + 5
|
||
|
while posPos <= self.w:
|
||
|
dc.DrawLine(posPos, float(0), posPos, self.h)
|
||
|
st = str(posNum)
|
||
|
self.drawXNumber(dc, st, posPos)
|
||
|
posPos += self.unitWidth*float(5)
|
||
|
posNum += 5
|
||
|
|
||
|
negPos = self.zeroPos[0]-self.unitWidth*float(5)
|
||
|
negNum = self.zero - 5
|
||
|
while negPos >= float(0):
|
||
|
dc.DrawLine(negPos, float(0), negPos, self.h)
|
||
|
st = str(negNum)
|
||
|
self.drawXNumber(dc, st, negPos)
|
||
|
negPos -= self.unitWidth*float(5)
|
||
|
posNum -= 5
|
||
|
|
||
|
def DrawYCoord(self,dc):
|
||
|
dc.SetPen(wx.BLACK_PEN)
|
||
|
dc.SetBrush(wx.BLACK_BRUSH)
|
||
|
dc.SetFont(wx.Font(8, wx.SWISS, wx.NORMAL, wx.NORMAL))
|
||
|
|
||
|
dc.DrawLine(float(0), self.zeroPos[1], self.w, self.zeroPos[1])
|
||
|
st = str(self.zero)
|
||
|
dc.DrawText(st, 5.0, self.zeroPos[1]-1.0)
|
||
|
|
||
|
dc.SetPen(wx.Pen(wx.Colour(150, 150, 150)))
|
||
|
dc.SetBrush(wx.Brush(wx.Colour(150, 150, 150)))
|
||
|
dc.SetLogicalFunction(wx.AND)
|
||
|
|
||
|
posPos = self.zeroPos[1]-self.unitHeight*float(5)
|
||
|
posNum = self.zero + 5
|
||
|
while posPos >= float(0):
|
||
|
dc.DrawLine(float(0), posPos, self.w, posPos)
|
||
|
st = str(posNum)
|
||
|
self.drawYNumber(dc, st, posPos)
|
||
|
posPos -= self.unitHeight*float(5)
|
||
|
posNum += 5
|
||
|
|
||
|
negPos = self.zeroPos[1]+self.unitHeight*float(5)
|
||
|
negNum = self.zero - 5
|
||
|
while negPos <= self.h:
|
||
|
dc.DrawLine(float(0), negPos, self.w, negPos)
|
||
|
st = str(negNum)
|
||
|
self.drawYNumber(dc, st, negPos)
|
||
|
negPos += self.unitHeight*float(5)
|
||
|
negNum -= 5
|
||
|
|
||
|
def drawXNumber(self, dc, st, pos):
|
||
|
oldPen, oldBrush, oldMode = dc.GetPen(), dc.GetBrush(), dc.GetLogicalFunction()
|
||
|
|
||
|
dc.SetPen(wx.BLACK_PEN)
|
||
|
dc.SetBrush(wx.BLACK_BRUSH)
|
||
|
dc.DrawText(st, pos+1.0, self.h-self.th-0.5)
|
||
|
|
||
|
dc.SetPen(oldPen)
|
||
|
dc.SetBrush(oldBrush)
|
||
|
dc.SetLogicalFunction(oldMode)
|
||
|
|
||
|
def drawYNumber(self, dc, st, pos):
|
||
|
oldPen, oldBrush, oldMode = dc.GetPen(), dc.GetBrush(), dc.GetLogicalFunction()
|
||
|
|
||
|
dc.SetPen(wx.BLACK_PEN)
|
||
|
dc.SetBrush(wx.BLACK_BRUSH)
|
||
|
dc.DrawText(st, 5.0, pos-1.0)
|
||
|
|
||
|
dc.SetPen(oldPen)
|
||
|
dc.SetBrush(oldBrush)
|
||
|
dc.SetLogicalFunction(oldMode)
|
||
|
|
||
|
def DrawFrame(self, dc):
|
||
|
if self._mainDialog.editor.mode == self._mainDialog.editor.ANIM_MODE:
|
||
|
curFramePos = self.zeroPos[0]+self.curFrame*self.unitWidth
|
||
|
dc.SetPen(wx.Pen("red"))
|
||
|
dc.SetBrush(wx.Brush("red"))
|
||
|
dc.DrawLine(curFramePos, float(0), curFramePos, self.h)
|
||
|
else:
|
||
|
pass
|
||
|
|
||
|
def drawX(self, dc):
|
||
|
dc.SetPen(wx.Pen("red"))
|
||
|
dc.SetBrush(wx.Brush("red"))
|
||
|
self.drawSingleCurve(self.X, dc)
|
||
|
self.drawKeys(self.X, dc)
|
||
|
self.drawHandler(self.X, dc)
|
||
|
|
||
|
def drawY(self, dc):
|
||
|
dc.SetPen(wx.Pen("green"))
|
||
|
dc.SetBrush(wx.Brush("green"))
|
||
|
self.drawSingleCurve(self.Y, dc)
|
||
|
self.drawKeys(self.Y, dc)
|
||
|
self.drawHandler(self.Y, dc)
|
||
|
|
||
|
def drawZ(self, dc):
|
||
|
dc.SetPen(wx.Pen("blue"))
|
||
|
dc.SetBrush(wx.Brush("blue"))
|
||
|
self.drawSingleCurve(self.Z, dc)
|
||
|
self.drawKeys(self.Z, dc)
|
||
|
self.drawHandler(self.Z, dc)
|
||
|
|
||
|
def DrawCurve(self, dc):
|
||
|
if self.property == self._mainDialog.namestr:
|
||
|
self.drawX(dc)
|
||
|
self.drawY(dc)
|
||
|
self.drawZ(dc)
|
||
|
return
|
||
|
if self.property == property[AG.X]:
|
||
|
self.drawX(dc)
|
||
|
return
|
||
|
if self.property == property[AG.Y]:
|
||
|
self.drawY(dc)
|
||
|
return
|
||
|
if self.property == property[AG.Z]:
|
||
|
self.drawZ(dc)
|
||
|
return
|
||
|
|
||
|
def drawSingleCurve(self, list, dc):
|
||
|
if len(list) == 1:
|
||
|
dc.DrawPoint(list[0][AG.KEYFRAME][AG.LOCAL_VALUE][0], list[0][AG.KEYFRAME][AG.LOCAL_VALUE][1])
|
||
|
return
|
||
|
|
||
|
if len(list) == 2:
|
||
|
dc.DrawLine(list[0][AG.KEYFRAME][AG.LOCAL_VALUE][0], list[0][AG.KEYFRAME][AG.LOCAL_VALUE][1], list[1][AG.KEYFRAME][AG.LOCAL_VALUE][0], list[1][AG.KEYFRAME][AG.LOCAL_VALUE][1])
|
||
|
return
|
||
|
|
||
|
if len(list)>=3 :
|
||
|
for i in range(len(list)-1):
|
||
|
x1 = list[i][AG.KEYFRAME][AG.LOCAL_VALUE][0]
|
||
|
y1 = list[i][AG.KEYFRAME][AG.LOCAL_VALUE][1]
|
||
|
|
||
|
x4 = list[i+1][AG.KEYFRAME][AG.LOCAL_VALUE][0]
|
||
|
y4 = list[i+1][AG.KEYFRAME][AG.LOCAL_VALUE][1]
|
||
|
|
||
|
t1x = list[i][AG.OUT_SLOPE][0]
|
||
|
t1y = list[i][AG.OUT_SLOPE][1]
|
||
|
|
||
|
t2x = list[i+1][AG.IN_SLOPE][0]
|
||
|
t2y = list[i+1][AG.IN_SLOPE][1]
|
||
|
|
||
|
x2 = x1 + (x4 - x1) / float(3);
|
||
|
scale1 = (x2 - x1) / t1x;
|
||
|
y2 = y1 - t1y * scale1;
|
||
|
|
||
|
|
||
|
x3 = x4 - (x4 - x1) / float(3);
|
||
|
scale2 = (x4 - x3) / t2x;
|
||
|
y3 = y4 + t2y * scale2;
|
||
|
|
||
|
ax = - float(1) * x1 + float(3) * x2 - float(3) * x3 + float(1) * x4;
|
||
|
bx = float(3) * x1 - float(6) * x2 + float(3) * x3 + float(0) * x4;
|
||
|
cx = - float(3) * x1 + float(3) * x2 + float(0) * x3 + float(0) * x4;
|
||
|
dx = float(1) * x1 + float(0) * x2 - float(0) * x3 + float(0) * x4;
|
||
|
|
||
|
ay = - float(1) * y1 + float(3) * y2 - float(3) * y3 + float(1) * y4;
|
||
|
by = float(3) * y1 - float(6) * y2 + float(3) * y3 + float(0) * y4;
|
||
|
cy = - float(3) * y1 + float(3) * y2 + float(0) * y3 + float(0) * y4;
|
||
|
dy = float(1) * y1 + float(0) * y2 - float(0) * y3 + float(0) * y4;
|
||
|
|
||
|
preX = x1
|
||
|
preY = y1
|
||
|
t = 0.001
|
||
|
|
||
|
while t<=float(1):
|
||
|
x = ax * t*t*t + bx * t*t + cx * t + dx;
|
||
|
y = ay * t*t*t + by * t*t + cy * t + dy;
|
||
|
|
||
|
curX = x
|
||
|
curY = y
|
||
|
|
||
|
dc.DrawLine(preX, preY, curX, curY)
|
||
|
|
||
|
preX = curX
|
||
|
preY = curY
|
||
|
|
||
|
t += 0.001
|
||
|
|
||
|
def drawKeys(self, list, dc):
|
||
|
for i in range(len(list)):
|
||
|
pointX = list[i][AG.KEYFRAME][AG.LOCAL_VALUE][0]
|
||
|
pointY = list[i][AG.KEYFRAME][AG.LOCAL_VALUE][1]
|
||
|
|
||
|
if list[i][AG.KEYFRAME][AG.SELECT] == 0:
|
||
|
dc.SetPen(wx.Pen("black",3))
|
||
|
dc.SetBrush(wx.Brush("black"))
|
||
|
dc.DrawCircle(pointX, pointY, 2)
|
||
|
|
||
|
if list[i][AG.KEYFRAME][AG.SELECT] == 1:
|
||
|
dc.SetPen(wx.Pen("cyan",3))
|
||
|
dc.SetBrush(wx.Brush("cyan"))
|
||
|
dc.DrawCircle(pointX, pointY, 2)
|
||
|
|
||
|
def drawHandler(self, list, dc):
|
||
|
for i in range(len(list)):
|
||
|
if list[i][AG.KEYFRAME][AG.SELECT] == 1:
|
||
|
X1 = list[i][AG.KEYFRAME][AG.LOCAL_VALUE][0]
|
||
|
Y1 = list[i][AG.KEYFRAME][AG.LOCAL_VALUE][1]
|
||
|
if self._OneTangent == True:
|
||
|
for j in range(3,5):
|
||
|
X = list[i][j][AG.LOCAL_VALUE][0]
|
||
|
Y = list[i][j][AG.LOCAL_VALUE][1]
|
||
|
if list[i][j][AG.SELECT] == 1:
|
||
|
dc.SetPen(wx.Pen("cyan",3))
|
||
|
dc.SetBrush(wx.Brush("cyan"))
|
||
|
dc.DrawCircle(X, Y, 2)
|
||
|
|
||
|
dc.SetPen(wx.Pen("cyan",1))
|
||
|
dc.DrawLine(X1, Y1, X, Y)
|
||
|
|
||
|
if list[i][j][AG.SELECT] == 0:
|
||
|
dc.SetPen(wx.Pen("brown",3))
|
||
|
dc.SetBrush(wx.Brush("brown"))
|
||
|
dc.DrawCircle(X, Y, 2)
|
||
|
|
||
|
dc.SetPen(wx.Pen("brown",1))
|
||
|
dc.DrawLine(X1, Y1, X, Y)
|
||
|
|
||
|
if self._OneTangent == False:
|
||
|
if list[i][AG.IN_TANGENT][AG.SELECT] == 1:
|
||
|
X = list[i][AG.IN_TANGENT][AG.LOCAL_VALUE][0]
|
||
|
Y = list[i][AG.IN_TANGENT][AG.LOCAL_VALUE][1]
|
||
|
dc.SetPen(wx.Pen("cyan",3))
|
||
|
dc.SetBrush(wx.Brush("cyan"))
|
||
|
dc.DrawCircle(X, Y, 2)
|
||
|
|
||
|
dc.SetPen(wx.Pen("cyan",1))
|
||
|
dc.DrawLine(X1, Y1, X, Y)
|
||
|
|
||
|
if list[i][AG.IN_TANGENT][AG.SELECT] == 0:
|
||
|
X = list[i][AG.IN_TANGENT][AG.LOCAL_VALUE][0]
|
||
|
Y = list[i][AG.IN_TANGENT][AG.LOCAL_VALUE][1]
|
||
|
dc.SetPen(wx.Pen("navy",3))
|
||
|
dc.SetBrush(wx.Brush("navy"))
|
||
|
dc.DrawCircle(X, Y, 2)
|
||
|
|
||
|
dc.SetPen(wx.Pen("navy",1))
|
||
|
dc.DrawLine(X1, Y1, X, Y)
|
||
|
|
||
|
if list[i][AG.OUT_TANGENT][AG.SELECT] == 1:
|
||
|
X = list[i][AG.OUT_TANGENT][AG.LOCAL_VALUE][0]
|
||
|
Y = list[i][AG.OUT_TANGENT][AG.LOCAL_VALUE][1]
|
||
|
dc.SetPen(wx.Pen("cyan",3))
|
||
|
dc.SetBrush(wx.Brush("cyan"))
|
||
|
dc.DrawCircle(X, Y, 2)
|
||
|
|
||
|
dc.SetPen(wx.Pen("cyan",1))
|
||
|
dc.DrawLine(X1, Y1, X, Y)
|
||
|
|
||
|
if list[i][AG.OUT_TANGENT][AG.SELECT] == 0:
|
||
|
X = list[i][AG.OUT_TANGENT][AG.LOCAL_VALUE][0]
|
||
|
Y = list[i][AG.OUT_TANGENT][AG.LOCAL_VALUE][1]
|
||
|
dc.SetPen(wx.Pen("brown",3))
|
||
|
dc.SetBrush(wx.Brush("brown"))
|
||
|
dc.DrawCircle(X, Y, 2)
|
||
|
|
||
|
dc.SetPen(wx.Pen("brown",1))
|
||
|
dc.DrawLine(X1, Y1, X, Y)
|
||
|
|
||
|
def DrawSelectRec(self, dc):
|
||
|
if self._selectRec == True:
|
||
|
dc.SetPen(wx.Pen("navy", 1))
|
||
|
dc.SetBrush(wx.Brush("navy"))
|
||
|
## dc.SetLogicalFunction(wx.AND)
|
||
|
dc.DrawLine(self.pos[0], self.pos[1], self.pos[0], self.newPos[1])
|
||
|
dc.DrawLine(self.pos[0], self.pos[1], self.newPos[0], self.pos[1])
|
||
|
dc.DrawLine(self.newPos[0], self.newPos[1], self.pos[0], self.newPos[1])
|
||
|
dc.DrawLine(self.newPos[0], self.newPos[1], self.newPos[0], self.pos[1])
|
||
|
|
||
|
def OnSize(self,evt):
|
||
|
self.InitBuffer()
|
||
|
|
||
|
def OnLeftDown(self,evt):
|
||
|
point = (evt.GetX(), evt.GetY())
|
||
|
|
||
|
if point[1]>= float(0) and point[1]<= float(self.h):
|
||
|
if point[0]>= float(0) and point[0]<= float(self.w):
|
||
|
self._mouseIn = True
|
||
|
|
||
|
if self._mouseIn:
|
||
|
self.CaptureMouse()
|
||
|
self.pos = point
|
||
|
|
||
|
def OnLeftUp(self,evt):
|
||
|
if self.GetCapture():
|
||
|
self.ReleaseMouse()
|
||
|
self._mouseIn = False
|
||
|
self._selectRec = False
|
||
|
self.setSelection()
|
||
|
self.SetGraphEditorData(self.property, self.curFrame)
|
||
|
|
||
|
def OnMiddleDown(self,evt):
|
||
|
point = (evt.GetX(), evt.GetY())
|
||
|
|
||
|
if point[1]>= float(0) and point[1]<= float(self.h):
|
||
|
if point[0]>= float(0) and point[0]<= float(self.w):
|
||
|
self._mouseIn = True
|
||
|
|
||
|
if self._mouseIn:
|
||
|
self.CaptureMouse()
|
||
|
self.midPos = point
|
||
|
|
||
|
def OnMiddleUp(self, evt):
|
||
|
if self.GetCapture():
|
||
|
self.ReleaseMouse()
|
||
|
|
||
|
def OnMotion(self,evt):
|
||
|
self._mouseIn = False
|
||
|
if evt.Dragging() and evt.LeftIsDown():
|
||
|
self.newPos = (evt.GetX(), evt.GetY())
|
||
|
if self.newPos[1]>= float(0) and self.newPos[1]<= float(self.h):
|
||
|
if self.newPos[0]>= float(0) and self.newPos[0]<= float(self.w):
|
||
|
self._mouseIn = True
|
||
|
|
||
|
if self._mouseIn:
|
||
|
if self.newPos == self.pos:
|
||
|
evt.Skip()
|
||
|
self._mouseIn = False
|
||
|
else:
|
||
|
self._selectRec = True
|
||
|
self.SetGraphEditorData(self.property, self.curFrame)
|
||
|
|
||
|
if evt.Dragging() and evt.MiddleIsDown():
|
||
|
self.newMidPos = (evt.GetX(), evt.GetY())
|
||
|
if self.newMidPos[1]>= float(0) and self.newMidPos[1]<= float(self.h):
|
||
|
if self.newMidPos[0]>= float(0) and self.newMidPos[0]<= float(self.w):
|
||
|
self._mouseIn = True
|
||
|
|
||
|
if self._mouseIn:
|
||
|
if self.newMidPos == self.midPos:
|
||
|
evt.Skip()
|
||
|
self._mouseIn = False
|
||
|
else:
|
||
|
self.recalculateSlope()
|
||
|
self.onAnimation()
|
||
|
self.midPos = self.newMidPos
|
||
|
|
||
|
evt.Skip()
|
||
|
self._mouseIn = False
|
||
|
self._selectRec = False
|
||
|
|
||
|
def setExistKey(self, list):
|
||
|
flag = False
|
||
|
for i in range(len(list)):
|
||
|
if list[i][AG.KEYFRAME][AG.SELECT] == 1:
|
||
|
inside = self.inside(self.pos, self.newPos, (list[i][AG.KEYFRAME][AG.LOCAL_VALUE][0], list[i][AG.KEYFRAME][AG.LOCAL_VALUE][1]))
|
||
|
if inside == True:
|
||
|
list[i][AG.KEYFRAME][AG.SELECT] = 0
|
||
|
if inside == False:
|
||
|
find = False
|
||
|
for j in range(3,5):
|
||
|
inside = self.inside(self.pos, self.newPos, (list[i][j][AG.LOCAL_VALUE][0], list[i][j][AG.LOCAL_VALUE][1]))
|
||
|
if inside == False:
|
||
|
list[i][j][AG.SELECT] = 0
|
||
|
if inside == True:
|
||
|
list[i][j][AG.SELECT] = 1
|
||
|
find = True
|
||
|
flag = True
|
||
|
if find == False:
|
||
|
list[i][AG.KEYFRAME][AG.SELECT] == 0
|
||
|
|
||
|
return flag
|
||
|
|
||
|
def setNewKey(self, list):
|
||
|
for i in range(len(list)):
|
||
|
inside = self.inside(self.pos, self.newPos, (list[i][2][0][0], list[i][2][0][1]))
|
||
|
if inside == True:
|
||
|
list[i][AG.KEYFRAME][AG.SELECT] = 1
|
||
|
if inside == False:
|
||
|
list[i][AG.KEYFRAME][AG.SELECT] = 0
|
||
|
|
||
|
def setSelection(self):
|
||
|
if self.property == self._mainDialog.namestr:
|
||
|
self.setSelectionBase(self.X)
|
||
|
self.setSelectionBase(self.Y)
|
||
|
self.setSelectionBase(self.Z)
|
||
|
return
|
||
|
if self.property == property[AG.X]:
|
||
|
self.setSelectionBase(self.X)
|
||
|
return
|
||
|
if self.property == property[AG.Y]:
|
||
|
self.setSelectionBase(self.Y)
|
||
|
return
|
||
|
if self.property == property[AG.Z]:
|
||
|
self.setSelectionBase(self.Z)
|
||
|
return
|
||
|
|
||
|
def setSelectionBase(self, list):
|
||
|
self.setExistKey(list)
|
||
|
if self.setExistKey(list) == True:
|
||
|
return
|
||
|
else:
|
||
|
self.setNewKey(list)
|
||
|
|
||
|
def inside(self, point0, point1, point):
|
||
|
if point0[0]<=point1[0] and point0[1]<=point1[1]:
|
||
|
if point0[0]<point[0] and point[0]<point1[0] and point0[1]<point[1] and point[1]<point1[1]:
|
||
|
return True
|
||
|
else:
|
||
|
return False
|
||
|
elif point1[0]<=point0[0] and point0[1]<=point1[1]:
|
||
|
if point1[0]<point[0] and point[0]<point0[0] and point0[1]<point[1] and point[1]<point1[1]:
|
||
|
return True
|
||
|
else:
|
||
|
return False
|
||
|
elif point0[0]<=point1[0] and point1[1]<=point0[1]:
|
||
|
if point0[0]<point[0] and point[0]<point1[0] and point1[1]<point[1] and point[1]<point0[1]:
|
||
|
return True
|
||
|
else:
|
||
|
return False
|
||
|
elif point1[0]<=point0[0] and point1[1]<=point0[1]:
|
||
|
if point1[0]<point[0] and point[0]<point0[0] and point1[1]<point[1] and point[1]<point0[1]:
|
||
|
return True
|
||
|
else:
|
||
|
return False
|
||
|
else:
|
||
|
return False
|
||
|
|
||
|
def recalculateSlope(self):
|
||
|
if self.property == self._mainDialog.namestr:
|
||
|
self.recalculateSlopeBase(self.X)
|
||
|
self.recalculateSlopeBase(self.Y)
|
||
|
self.recalculateSlopeBase(self.Z)
|
||
|
return
|
||
|
if self.property == property[AG.X]:
|
||
|
self.recalculateSlopeBase(self.X)
|
||
|
return
|
||
|
if self.property == property[AG.Y]:
|
||
|
self.recalculateSlopeBase(self.Y)
|
||
|
return
|
||
|
if self.property == property[AG.Z]:
|
||
|
self.recalculateSlopeBase(self.Z)
|
||
|
return
|
||
|
|
||
|
def recalculateSlopeBase(self, list):
|
||
|
#recalculate the tangent slope
|
||
|
moveX = self.newMidPos[0]-self.midPos[0]
|
||
|
moveY = self.newMidPos[1]-self.midPos[1]
|
||
|
|
||
|
for i in range(len(list)):
|
||
|
if list[i][AG.KEYFRAME][AG.SELECT] == 1:
|
||
|
if list[i][AG.IN_TANGENT][AG.SELECT] == 1:
|
||
|
newPointX = list[i][AG.IN_TANGENT][AG.LOCAL_VALUE][0] + moveX
|
||
|
newPointY = list[i][AG.IN_TANGENT][AG.LOCAL_VALUE][1] + moveY
|
||
|
|
||
|
newSlope = [list[i][AG.KEYFRAME][AG.LOCAL_VALUE][0] - newPointX , newPointY - list[i][AG.KEYFRAME][AG.LOCAL_VALUE][1]]
|
||
|
|
||
|
temp0 = self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.INSLOPE][0]
|
||
|
temp1 = self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.INSLOPE][1]
|
||
|
self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.INSLOPE][0] = newSlope[0]/self.unitWidth
|
||
|
self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.INSLOPE][1] = newSlope[1]/self.unitHeight
|
||
|
handler = self.generateHandler(self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]])
|
||
|
if handler[1][0][0]>= list[i][AG.KEYFRAME][AG.LOCAL_VALUE][0]:
|
||
|
self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.INSLOPE][0] = temp0
|
||
|
self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.INSLOPE][1] = temp1
|
||
|
return
|
||
|
if handler[1][0][0] < list[i][AG.KEYFRAME][AG.LOCAL_VALUE][0]:
|
||
|
if self._OneTangent == False:
|
||
|
list[i][AG.IN_TANGENT][0] = handler[1][0]
|
||
|
list[i][AG.IN_SLOPE][0] = handler[3][0]
|
||
|
list[i][AG.IN_SLOPE][1] = handler[3][1]
|
||
|
|
||
|
if self._OneTangent == True:
|
||
|
self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.OUTSLOPE][0] = newSlope[0]/self.unitWidth
|
||
|
self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.OUTSLOPE][1] = newSlope[1]/self.unitHeight
|
||
|
handler = self.generateHandler(self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]])
|
||
|
list[i][AG.IN_TANGENT][0] = handler[1][0]
|
||
|
list[i][AG.OUT_TANGENT][0] = handler[2][0]
|
||
|
list[i][AG.IN_SLOPE][0] = handler[3][0]
|
||
|
list[i][AG.IN_SLOPE][1] = handler[3][1]
|
||
|
list[i][AG.OUT_SLOPE][0] = handler[4][0]
|
||
|
list[i][AG.OUT_SLOPE][1] = handler[4][1]
|
||
|
|
||
|
self.SetGraphEditorData(self.property, self.curFrame)
|
||
|
|
||
|
if list[i][AG.OUT_TANGENT][AG.SELECT] == 1:
|
||
|
newPointX = list[i][AG.OUT_TANGENT][AG.LOCAL_VALUE][0] + moveX
|
||
|
newPointY = list[i][AG.OUT_TANGENT][AG.LOCAL_VALUE][1] + moveY
|
||
|
|
||
|
newSlope = [newPointX - list[i][AG.KEYFRAME][AG.LOCAL_VALUE][0] , list[i][AG.KEYFRAME][AG.LOCAL_VALUE][1] - newPointY]
|
||
|
|
||
|
temp0 = self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.OUTSLOPE][0]
|
||
|
temp1 = self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.OUTSLOPE][1]
|
||
|
self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.OUTSLOPE][0] = newSlope[0]/self.unitWidth
|
||
|
self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.OUTSLOPE][1] = newSlope[1]/self.unitHeight
|
||
|
|
||
|
handler = self.generateHandler(self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]])
|
||
|
if handler[2][0][0] <= list[i][AG.KEYFRAME][AG.LOCAL_VALUE][0]:
|
||
|
self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.OUTSLOPE][0] = temp0
|
||
|
self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.OUTSLOPE][1] = temp1
|
||
|
return
|
||
|
if handler[2][0][0] > list[i][AG.KEYFRAME][AG.LOCAL_VALUE][0]:
|
||
|
if self._OneTangent == False:
|
||
|
list[i][AG.OUT_TANGENT][0] = handler[2][0]
|
||
|
list[i][AG.OUT_SLOPE][0] = handler[4][0]
|
||
|
list[i][AG.OUT_SLOPE][1] = handler[4][1]
|
||
|
|
||
|
if self._OneTangent == True:
|
||
|
self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.INSLOPE][0] = newSlope[0]/self.unitWidth
|
||
|
self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]][AG.INSLOPE][1] = newSlope[1]/self.unitHeight
|
||
|
handler = self.generateHandler(self._mainDialog.editor.animMgr.keyFramesInfo[list[i][AG.KEY]][list[i][AG.I]])
|
||
|
list[i][AG.IN_TANGENT][0] = handler[1][0]
|
||
|
list[i][AG.OUT_TANGENT][0] = handler[2][0]
|
||
|
list[i][AG.IN_SLOPE][0] = handler[3][0]
|
||
|
list[i][AG.IN_SLOPE][1] = handler[3][1]
|
||
|
list[i][AG.OUT_SLOPE][0] = handler[4][0]
|
||
|
list[i][AG.OUT_SLOPE][1] = handler[4][1]
|
||
|
|
||
|
self.SetGraphEditorData(self.property, self.curFrame)
|
||
|
|
||
|
def selectHandler(self):
|
||
|
self._selectHandler = False
|
||
|
|
||
|
def onAnimation(self):
|
||
|
if self._mainDialog.editor.mode == self._mainDialog.editor.ANIM_MODE:
|
||
|
self._mainDialog.editor.ui.animUI.OnAnimation(self._mainDialog.editor.ui.animUI.curFrame)
|
||
|
else:
|
||
|
pass
|
||
|
|
||
|
|
||
|
class GraphEditorUI(wx.Dialog):
|
||
|
"""
|
||
|
This is the graph editor main class implementation.
|
||
|
"""
|
||
|
def __init__(self, parent, editor, object):
|
||
|
wx.Dialog.__init__(self, parent, id=wx.ID_ANY, title="Graph Editor",
|
||
|
pos=wx.DefaultPosition, size=(735, 535))
|
||
|
|
||
|
self.editor = editor
|
||
|
self.editor.GRAPH_EDITOR = True
|
||
|
self.object = object
|
||
|
self.xRange = 24+1
|
||
|
self.yRange = 50
|
||
|
if self.editor.mode == self.editor.ANIM_MODE:
|
||
|
self.curFrame = self.editor.ui.animUI.curFrame
|
||
|
self.curFrame = 1
|
||
|
|
||
|
self.mainPanel1 = wx.Panel(self, -1)
|
||
|
|
||
|
bmpZoomIn = ZoomIn.GetBitmap()
|
||
|
bmpZoomOut = ZoomOut.GetBitmap()
|
||
|
bmpOneTangent = OneTangent.GetBitmap()
|
||
|
bmpTwoTangents = TwoTangents.GetBitmap()
|
||
|
|
||
|
self.buttonZoomIn = wx.BitmapButton(self.mainPanel1, -1, bmpZoomIn, size = (30,30),style = wx.BU_AUTODRAW)
|
||
|
self.buttonZoomOut = wx.BitmapButton(self.mainPanel1, -1, bmpZoomOut, size = (30,30),style = wx.BU_AUTODRAW)
|
||
|
self.buttonOneTangent = wx.BitmapButton(self.mainPanel1, -1, bmpOneTangent, size = (30,30),style = wx.BU_AUTODRAW)
|
||
|
self.buttonTwoTangents = wx.BitmapButton(self.mainPanel1, -1, bmpTwoTangents, size = (30,30),style = wx.BU_AUTODRAW)
|
||
|
|
||
|
self.mainPanel2 = wx.Panel(self, -1)
|
||
|
|
||
|
self.tree = self.tree = wx.TreeCtrl(self.mainPanel2, id=-1, pos=wx.DefaultPosition,size=wx.Size(200, 450), style=wx.TR_MULTIPLE|wx.TR_DEFAULT_STYLE,validator=wx.DefaultValidator, name="treeCtrl")
|
||
|
self.namestr = "%s"%(object[OG.OBJ_DEF].name)
|
||
|
self.root = self.tree.AddRoot(self.namestr)
|
||
|
self.AddTreeNodes(self.root, property)
|
||
|
self.tree.Expand(self.root)
|
||
|
self.tree.SelectItem(self.root,select=True)
|
||
|
self.str = self.tree.GetItemText(self.root)
|
||
|
|
||
|
self.graphEditorWindow =GraphEditorWindow(self.mainPanel2, wx.Size(500, 450), str(object[OG.OBJ_DEF].name), self.xRange, self.yRange, self.curFrame, self.object)
|
||
|
|
||
|
self.SetProperties()
|
||
|
self.DoLayout()
|
||
|
|
||
|
self.Bind(wx.EVT_BUTTON, self.OnZoomIn, self.buttonZoomIn)
|
||
|
self.Bind(wx.EVT_BUTTON, self.OnZoomOut, self.buttonZoomOut)
|
||
|
self.Bind(wx.EVT_BUTTON, self.OnOneTangent, self.buttonOneTangent)
|
||
|
self.Bind(wx.EVT_BUTTON, self.OnTwoTangents, self.buttonTwoTangents)
|
||
|
|
||
|
self.Bind(wx.EVT_CLOSE, self.OnExit)
|
||
|
self.tree.Bind(wx.EVT_TREE_SEL_CHANGED, self.OnSelChanged)
|
||
|
|
||
|
def SetProperties(self):
|
||
|
pass
|
||
|
|
||
|
def DoLayout(self):
|
||
|
dialogSizer = wx.BoxSizer(wx.VERTICAL)
|
||
|
mainSizer1 = wx.FlexGridSizer(1, 4, 0, 0)
|
||
|
mainSizer2 = wx.FlexGridSizer(1, 2, 0, 0)
|
||
|
|
||
|
mainSizer1.Add(self.buttonOneTangent, 0, wx.ALIGN_CENTER_VERTICAL|wx.LEFT, 570)
|
||
|
mainSizer1.Add(self.buttonTwoTangents, 0, wx.ALIGN_CENTER_VERTICAL|wx.LEFT, 10)
|
||
|
mainSizer1.Add(self.buttonZoomIn, 0, wx.ALIGN_CENTER_VERTICAL|wx.LEFT, 10)
|
||
|
mainSizer1.Add(self.buttonZoomOut, 0, wx.ALIGN_CENTER_VERTICAL|wx.LEFT|wx.RIGHT, 10)
|
||
|
|
||
|
mainSizer2.Add(self.tree, 0, wx.ALIGN_CENTER_VERTICAL|wx.LEFT|wx.RIGHT, 10)
|
||
|
mainSizer2.Add(self.graphEditorWindow, 0, wx.ALIGN_CENTER_VERTICAL|wx.RIGHT, 10)
|
||
|
|
||
|
self.mainPanel1.SetSizerAndFit(mainSizer1)
|
||
|
self.mainPanel2.SetSizer(mainSizer2)
|
||
|
|
||
|
dialogSizer.Add(self.mainPanel2, 0, wx.ALIGN_CENTER_VERTICAL|wx.TOP, 10)
|
||
|
dialogSizer.Add(self.mainPanel1, 0, wx.ALIGN_CENTER_VERTICAL|wx.TOP, 5)
|
||
|
|
||
|
self.SetSizer(dialogSizer)
|
||
|
self.Layout()
|
||
|
|
||
|
self.dialogSizer = dialogSizer
|
||
|
|
||
|
def AddTreeNodes(self, parentItem, items):
|
||
|
for item in items:
|
||
|
if type(item) == str:
|
||
|
self.tree.AppendItem(parentItem, item)
|
||
|
|
||
|
def OnSelChanged(self, evt):
|
||
|
item = evt.GetItem()
|
||
|
if item:
|
||
|
self.str = self.tree.GetItemText(item)
|
||
|
self.graphEditorWindow.refresh()
|
||
|
self.graphEditorWindow.SetGraphEditorData(self.str, self.curFrame)
|
||
|
|
||
|
def OnZoomIn(self,evt):
|
||
|
self.graphEditorWindow.zoom = self.graphEditorWindow.zoom/float(1.2)
|
||
|
self.graphEditorWindow.unitWidth = self.graphEditorWindow.unitWidth*float(1.2)
|
||
|
self.graphEditorWindow.unitHeight = self.graphEditorWindow.unitHeight*float(1.2)
|
||
|
self.graphEditorWindow.generateInfo()
|
||
|
self.graphEditorWindow.SetGraphEditorData(self.str, self.curFrame)
|
||
|
|
||
|
def OnZoomOut(self,evt):
|
||
|
self.graphEditorWindow.zoom = self.graphEditorWindow.zoom*float(1.2)
|
||
|
self.graphEditorWindow.unitWidth = self.graphEditorWindow.unitWidth/float(1.2)
|
||
|
self.graphEditorWindow.unitHeight = self.graphEditorWindow.unitHeight/float(1.2)
|
||
|
self.graphEditorWindow.generateInfo()
|
||
|
self.graphEditorWindow.SetGraphEditorData(self.str, self.curFrame)
|
||
|
|
||
|
def OnOneTangent(self,evt):
|
||
|
self.graphEditorWindow._OneTangent = True
|
||
|
self.graphEditorWindow.SetGraphEditorData(self.str, self.curFrame)
|
||
|
|
||
|
def OnTwoTangents(self,evt):
|
||
|
self.graphEditorWindow._OneTangent = False
|
||
|
self.graphEditorWindow.SetGraphEditorData(self.str, self.curFrame)
|
||
|
|
||
|
def curFrameChange(self):
|
||
|
if self.editor.mode == self.editor.ANIM_MODE:
|
||
|
self.curFrame = self.editor.ui.animUI.curFrame
|
||
|
self.graphEditorWindow.SetGraphEditorData(self.str, self.curFrame)
|
||
|
else:
|
||
|
pass
|
||
|
|
||
|
def OnExit(self,evt):
|
||
|
self.Destroy()
|
||
|
self.editor.ui.graphEditorMenuItem.Check(False)
|
||
|
self.object = None
|
||
|
self.editor.GRAPH_EDITOR = False
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|