from panda3d.core import * from . import ObjectGlobals as OG class ActionMgr: def __init__(self): self.undoList = [] self.redoList = [] def reset(self): while len(self.undoList) > 0: action = self.undoList.pop() action.destroy() while len(self.redoList) > 0: action = self.redoList.pop() action.destroy() def push(self, action): self.undoList.append(action) if len(self.redoList) > 0: self.redoList.pop() def undo(self): if len(self.undoList) < 1: print('No more undo') else: action = self.undoList.pop() self.redoList.append(action) action.undo() def redo(self): if len(self.redoList) < 1: print('No more redo') else: action = self.redoList.pop() self.undoList.append(action) action.redo() class ActionBase(Functor): """ Base class for user actions """ def __init__(self, function, *args, **kargs): self.function = function if function is None: def nullFunc(): pass function = nullFunc Functor.__init__(self, function, *args, **kargs) self.result = None def _do__call__(self, *args, **kargs): self.saveStatus() self.result = Functor._do__call__(self, *args, **kargs) self.postCall() return self.result # needed this line to override _do__call__ __call__ = _do__call__ def redo(self): self.result = self._do__call__() return self.result def saveStatus(self): # save object status for undo here pass def postCall(self): # implement post process here pass def undo(self): print("undo method is not defined for this action") class ActionAddNewObj(ActionBase): """ Action class for adding new object """ def __init__(self, editor, *args, **kargs): self.editor = editor function = self.editor.objectMgr.addNewObject ActionBase.__init__(self, function, *args, **kargs) self.uid = None def postCall(self): obj = self.editor.objectMgr.findObjectByNodePath(self.result) if obj: self.uid = obj[OG.OBJ_UID] def redo(self): if self.uid is None: print("Can't redo this add") else: self.result = self._do__call__(uid=self.uid) return self.result def undo(self): if self.result is None: print("Can't undo this add") else: print("Undo: addNewObject") if self.uid: obj = self.editor.objectMgr.findObjectById(self.uid) else: obj = self.editor.objectMgr.findObjectByNodePath(self.result) if obj: self.uid = obj[OG.OBJ_UID] self.editor.ui.sceneGraphUI.delete(self.uid) base.direct.deselect(obj[OG.OBJ_NP]) base.direct.removeNodePath(obj[OG.OBJ_NP]) self.result = None else: print("Can't undo this add") class ActionDeleteObj(ActionBase): """ Action class for deleting object """ def __init__(self, editor, *args, **kargs): self.editor = editor function = base.direct.removeAllSelected ActionBase.__init__(self, function, *args, **kargs) self.selectedUIDs = [] self.hierarchy = {} self.objInfos = {} self.objTransforms = {} def saveStatus(self): selectedNPs = base.direct.selected.getSelectedAsList() def saveObjStatus(np, isRecursive=True): obj = self.editor.objectMgr.findObjectByNodePath(np) if obj: uid = obj[OG.OBJ_UID] if not isRecursive: self.selectedUIDs.append(uid) objNP = obj[OG.OBJ_NP] self.objInfos[uid] = obj self.objTransforms[uid] = objNP.getMat() parentNP = objNP.getParent() if parentNP == render: self.hierarchy[uid] = None else: parentObj = self.editor.objectMgr.findObjectByNodePath(parentNP) if parentObj: self.hierarchy[uid] = parentObj[OG.OBJ_UID] for child in np.getChildren(): if child.hasTag('OBJRoot'): saveObjStatus(child) for np in selectedNPs: saveObjStatus(np, False) def undo(self): if len(self.hierarchy) == 0 or\ len(self.objInfos) == 0: print("Can't undo this deletion") else: print("Undo: deleteObject") def restoreObject(uid, parentNP): obj = self.objInfos[uid] objDef = obj[OG.OBJ_DEF] objModel = obj[OG.OBJ_MODEL] objProp = obj[OG.OBJ_PROP] objRGBA = obj[OG.OBJ_RGBA] objNP = self.editor.objectMgr.addNewObject(objDef.name, uid, obj[OG.OBJ_MODEL], parentNP) self.editor.objectMgr.updateObjectColor(objRGBA[0], objRGBA[1], objRGBA[2], objRGBA[3], objNP) self.editor.objectMgr.updateObjectProperties(objNP, objProp) objNP.setMat(self.objTransforms[uid]) while len(self.hierarchy) > 0: for uid in self.hierarchy: if self.hierarchy[uid] is None: parentNP = None restoreObject(uid, parentNP) del self.hierarchy[uid] else: parentObj = self.editor.objectMgr.findObjectById(self.hierarchy[uid]) if parentObj: parentNP = parentObj[OG.OBJ_NP] restoreObject(uid, parentNP) del self.hierarchy[uid] base.direct.deselectAllCB() for uid in self.selectedUIDs: obj = self.editor.objectMgr.findObjectById(uid) if obj: self.editor.select(obj[OG.OBJ_NP], fMultiSelect=1, fUndo=0) self.selecteUIDs = [] self.hierarchy = {} self.objInfos = {} class ActionDeleteObjById(ActionBase): """ Action class for deleting object """ def __init__(self, editor, uid): self.editor = editor function = self.editor.objectMgr.removeObjectById self.uid = uid ActionBase.__init__(self, function, self.uid) self.hierarchy = {} self.objInfos = {} self.objTransforms = {} def saveStatus(self): def saveObjStatus(uid_np, isUID=False): if isUID: obj = self.editor.objectMgr.findObjectById(uid_np) else: obj = self.editor.objectMgr.findObjectByNodePath(uid_np) if obj: uid = obj[OG.OBJ_UID] objNP = obj[OG.OBJ_NP] self.objInfos[uid] = obj self.objTransforms[uid] = objNP.getMat() parentNP = objNP.getParent() if parentNP == render: self.hierarchy[uid] = None else: parentObj = self.editor.objectMgr.findObjectByNodePath(parentNP) if parentObj: self.hierarchy[uid] = parentObj[OG.OBJ_UID] for child in objNP.getChildren(): if child.hasTag('OBJRoot'): saveObjStatus(child) saveObjStatus(self.uid, True) def undo(self): if len(self.hierarchy) == 0 or\ len(self.objInfos) == 0: print("Can't undo this deletion") else: print("Undo: deleteObjectById") def restoreObject(uid, parentNP): obj = self.objInfos[uid] objDef = obj[OG.OBJ_DEF] objModel = obj[OG.OBJ_MODEL] objProp = obj[OG.OBJ_PROP] objRGBA = obj[OG.OBJ_RGBA] objNP = self.editor.objectMgr.addNewObject(objDef.name, uid, obj[OG.OBJ_MODEL], parentNP) self.editor.objectMgr.updateObjectColor(objRGBA[0], objRGBA[1], objRGBA[2], objRGBA[3], objNP) self.editor.objectMgr.updateObjectProperties(objNP, objProp) objNP.setMat(self.objTransforms[uid]) while len(self.hierarchy) > 0: for uid in self.hierarchy: if self.hierarchy[uid] is None: parentNP = None restoreObject(uid, parentNP) del self.hierarchy[uid] else: parentObj = self.editor.objectMgr.findObjectById(self.hierarchy[uid]) if parentObj: parentNP = parentObj[OG.OBJ_NP] restoreObject(uid, parentNP) del self.hierarchy[uid] self.hierarchy = {} self.objInfos = {} class ActionChangeHierarchy(ActionBase): """ Action class for changing Scene Graph Hierarchy """ def __init__(self, editor, oldGrandParentId, oldParentId, newParentId, childName, *args, **kargs): self.editor = editor self.oldGrandParentId = oldGrandParentId self.oldParentId = oldParentId self.newParentId = newParentId self.childName = childName function = self.editor.ui.sceneGraphUI.parent ActionBase.__init__(self, function, self.oldParentId, self.newParentId, self.childName, **kargs) def undo(self): self.editor.ui.sceneGraphUI.parent(self.oldParentId, self.oldGrandParentId, self.childName) class ActionSelectObj(ActionBase): """ Action class for adding new object """ def __init__(self, editor, *args, **kargs): self.editor = editor function = base.direct.selectCB ActionBase.__init__(self, function, *args, **kargs) self.selectedUIDs = [] def saveStatus(self): selectedNPs = base.direct.selected.getSelectedAsList() for np in selectedNPs: obj = self.editor.objectMgr.findObjectByNodePath(np) if obj: uid = obj[OG.OBJ_UID] self.selectedUIDs.append(uid) def undo(self): print("Undo : selectObject") base.direct.deselectAllCB() for uid in self.selectedUIDs: obj = self.editor.objectMgr.findObjectById(uid) if obj: self.editor.select(obj[OG.OBJ_NP], fMultiSelect=1, fUndo=0) self.selectedUIDs = [] class ActionTransformObj(ActionBase): """ Action class for object transformation """ def __init__(self, editor, *args, **kargs): self.editor = editor function = self.editor.objectMgr.setObjectTransform ActionBase.__init__(self, function, *args, **kargs) self.uid = args[0] #self.xformMat = Mat4(args[1]) self.origMat = None def saveStatus(self): obj = self.editor.objectMgr.findObjectById(self.uid) if obj: self.origMat = Mat4(self.editor.objectMgr.objectsLastXform[obj[OG.OBJ_UID]]) #self.origMat = Mat4(obj[OG.OBJ_NP].getMat()) def _do__call__(self, *args, **kargs): self.result = ActionBase._do__call__(self, *args, **kargs) obj = self.editor.objectMgr.findObjectById(self.uid) if obj: self.editor.objectMgr.objectsLastXform[self.uid] = Mat4(obj[OG.OBJ_NP].getMat()) return self.result def undo(self): if self.origMat is None: print("Can't undo this transform") else: print("Undo: transformObject") obj = self.editor.objectMgr.findObjectById(self.uid) if obj: obj[OG.OBJ_NP].setMat(self.origMat) self.editor.objectMgr.objectsLastXform[self.uid] = Mat4(self.origMat) del self.origMat self.origMat = None class ActionDeselectAll(ActionBase): """ Action class for adding new object """ def __init__(self, editor, *args, **kargs): self.editor = editor function = base.direct.deselectAllCB ActionBase.__init__(self, function, *args, **kargs) self.selectedUIDs = [] def saveStatus(self): selectedNPs = base.direct.selected.getSelectedAsList() for np in selectedNPs: obj = self.editor.objectMgr.findObjectByNodePath(np) if obj: uid = obj[OG.OBJ_UID] self.selectedUIDs.append(uid) def undo(self): print("Undo : deselectAll") base.direct.deselectAllCB() for uid in self.selectedUIDs: obj = self.editor.objectMgr.findObjectById(uid) if obj: self.editor.select(obj[OG.OBJ_NP], fMultiSelect=1, fUndo=0) self.selectedUIDs = [] class ActionUpdateObjectProp(ActionBase): """ Action class for updating object property """ def __init__(self, editor, fSelectObject, obj, propName, val, oldVal, function, undoFunc, *args, **kargs): self.editor = editor self.fSelectObject = fSelectObject self.obj = obj self.propName = propName self.newVal = val self.oldVal = oldVal self.undoFunc = undoFunc ActionBase.__init__(self, function, *args, **kargs) def saveStatus(self): self.obj[OG.OBJ_PROP][self.propName] = self.newVal def redo(self): self.result = self._do__call__()#uid=self.uid, xformMat=self.xformMat) if self.editor and self.fSelectObject: base.direct.select(self.obj[OG.OBJ_NP], fUndo=0) return self.result def undo(self): print("Undo : updateObjectProp") if self.oldVal: self.obj[OG.OBJ_PROP][self.propName] = self.oldVal if self.undoFunc: self.undoFunc() if self.editor and self.fSelectObject: base.direct.select(self.obj[OG.OBJ_NP], fUndo=0)