Я пытаюсь отобразить mocap-файл Biovision Hierarchy с помощью pygame.На данный момент в ортографической проекции перспектива будет делением z.Мне кажется, что кости импортированы, но я не могу выбрать правильные оси даже в неподвижной арматуре (XY и XZ не работают).
Самое забавное случается, когда я пытаюсь применить анимацию.Похоже на искривление андроида от Blade Runner.
Покажите мне, как это исправить.Я использую 3dsMax версию CMU mocap.
Вот мой код:
import numpy as np
class Cell(list):
def getname(self):
if hasattr(self, "name"):
return self.name
return ""
def setname(self, name):
self.name=name
def __repr__(self):
if hasattr(self, "name"):
if self:
return self.name+" "+list.__repr__(self)
return self.name
return list.__repr__(self)
class Node(object):
def __init__(self, name, parent, offset, channels):
self.name=name
self.parent=parent
self.children=[]
self.offset=offset
self.channels=channels
def add_child(self, child):
self.children.append(child)
def get_bones(self):
bones=[]
if self.parent is not None:
bones.append((self.parent.abspos(),self.abspos()))
for child in self.children:
bones.extend(child.get_bones())
return bones
def abspos(self):
if self.parent is None:
return np.dot(self.get_matrix(),np.hstack((self.offset, 1)))[:3]#also need rotation
return self.parent.abspos()+np.dot(self.get_matrix(),np.hstack((self.offset, 1)))[:3]#also need rotation
def __repr__(self):
return "<"+self.name+" ["+", ".join([repr(child) for child in self.children])+"]>"
def feed_transform(self, transform):
obj=transform
matrix=np.zeros((4,4))
matrix[0,0]=1
matrix[1,1]=1
matrix[2,2]=1
matrix[3,3]=1
for channel in self.channels:
if channel=="Xposition":
mod=np.zeros((4,4))
mod[0,0]=1
mod[1,1]=1
mod[2,2]=1
mod[3,3]=1
mod[3,0]=obj[0]
obj=obj[1:]
matrix=np.dot(matrix, mod)
elif channel=="Yposition":
mod=np.zeros((4,4))
mod[0,0]=1
mod[1,1]=1
mod[2,2]=1
mod[3,3]=1
mod[3,1]=obj[0]
obj=obj[1:]
matrix=np.dot(matrix, mod)
elif channel=="Zposition":
mod=np.zeros((4,4))
mod[0,0]=1
mod[1,1]=1
mod[2,2]=1
mod[3,3]=1
mod[3,2]=obj[0]
obj=obj[1:]
matrix=np.dot(matrix, mod)
elif channel=="Xrotation":
mod=np.zeros((4,4))
mod[0,0]=1
mod[1,1]=np.cos(obj[0])
mod[2,2]=np.cos(obj[0])
mod[1,2]=np.sin(obj[0])
mod[2,1]=-np.sin(obj[0])
mod[3,3]=1
obj=obj[1:]
matrix=np.dot(matrix, mod)
elif channel=="Yrotation":
mod=np.zeros((4,4))
mod[1,1]=1
mod[0,0]=np.cos(obj[0])
mod[2,2]=np.cos(obj[0])
mod[2,0]=np.sin(obj[0])
mod[0,2]=-np.sin(obj[0])
mod[3,3]=1
obj=obj[1:]
matrix=np.dot(matrix, mod)
elif channel=="Zrotation":
mod=np.zeros((4,4))
mod[2,2]=1
mod[0,0]=np.cos(obj[0])
mod[1,1]=np.cos(obj[0])
mod[0,1]=np.sin(obj[0])
mod[1,0]=-np.sin(obj[0])
mod[3,3]=1
obj=obj[1:]
matrix=np.dot(matrix, mod)
self.matrix=matrix
for child in self.children:
obj=child.feed_transform(obj)
return obj
def get_matrix(self):
if self.parent is None:
return self.matrix
return np.dot(self.matrix,self.parent.get_matrix())
def parse(obj, parent=None):
if obj is None:
return None
tp, nm=obj.getname().split(" ")
offset=None
channels=None
joints=[]
if tp not in ("ROOT", "JOINT"):
print("warning", tp)
for line in obj:
if line.getname().startswith("OFFSET"):
offset=np.array([float(x) for x in line.getname().split(" ")[1:4]])
elif line.getname().startswith("CHANNELS"):
channels=tuple(line.getname().split(" ")[2:])
elif line.getname().startswith("JOINT"):
joints.append(line)
res=Node(nm, parent, offset, channels)
for joint in joints:
res.add_child(parse(joint, parent=res))
return res
tree=Cell()
depth=0
with open("/media/stuntj/ESD-ISO/cmu/01/01_01.bvh","r") as f:
for line in f:
ls=line.strip(" \t\r\n")
obj=tree
for i in range(depth):
obj=obj[-1]
if not ls:
pass
elif ls=="{":
depth+=1
elif ls=="}":
depth-=1
else:
obj.append(Cell())
obj[-1].setname(ls)
#print(tree)
hier=False
moti=False
parsed=[]
frames=[]
fps=30
for line in tree:
if line.getname() == "HIERARCHY":
hier=True
moti=False
elif line.getname() == "MOTION":
hier=False
moti=True
elif hier:
tp, nm=line.getname().split(" ")
if tp == "ROOT":
parsed.append(parse(line))
elif moti:
if line.getname().startswith("Frame Time:"):
fps=int(1.0/float(line.getname()[11:].strip(" \t\r\n"))+.5)
elif line.getname()[0:1] in "-.0123456789":
frames.append([float(x) for x in line.getname().split(" ")])
#print(parsed)
#print(frames)
#print(fps)
#print(parsed[0].get_bones())import pygame, sys
import pygame, sys
from pygame.locals import *
pygame.init()
DISPLAYSURF = pygame.display.set_mode((800, 800))
pygame.display.set_caption('Yet Another BVH Lexer')
fpsClock = pygame.time.Clock()
WHITE = (255, 255, 255)
BLACK = (0, 0, 0)
for frame in frames:
obj=frame
for ps in parsed:
obj=ps.feed_transform(obj)
DISPLAYSURF.fill(WHITE)
for bone in ps.get_bones():
pygame.draw.line(DISPLAYSURF, BLACK, (bone[0][0]+400,bone[0][2]+400), (bone[1][0]*8+400,bone[1][2]*-8+400),1)
pygame.display.update()
if obj:
print("Malformed data")
fpsClock.tick(fps)
pygame.quit()
sys.exit()