Maxon Developers Maxon Developers
    • Documentation
      • Cinema 4D Python API
      • Cinema 4D C++ API
      • Cineware API
      • ZBrush Python API
      • ZBrush GoZ API
      • Code Examples on Github
    • Forum
    • Downloads
    • Support
      • Support Procedures
      • Registered Developer Program
      • Plugin IDs
      • Contact Us
    • Categories
      • Overview
      • News & Information
      • Cinema 4D SDK Support
      • Cineware SDK Support
      • ZBrush 4D SDK Support
      • Bugs
      • General Talk
    • Unread
    • Recent
    • Tags
    • Users
    • Login

    Can I reset the keyframe of an object through Python script?

    General Talk
    2
    2
    768
    Loading More Posts
    • Oldest to Newest
    • Newest to Oldest
    • Most Votes
    Reply
    • Reply as topic
    Log in to reply
    This topic has been deleted. Only users with topic management privileges can see it.
    • H
      happygrass_cn
      last edited by happygrass_cn

      Hi, I am making a DEMO for my research work of crowd simulation. The people may walk and stop during the animation. The position and drection(Velocity) of the crowd have been recorded in a text file. I have written a Python script successfully read the data file at each keyframe. Now, I use TP partilce to animate the crowd. And, I subsitute the particle with a Man model through XPresso Pshape tags. Shown as followed.
      b3afd9da-2ab6-40ad-be83-c18da2d7dbc6-image.png

      When the position is not changed the model's motion should stop(stay at present keyframe) untill he moved again. The motion of the model is pre-rendered like a re-cycle movie.

      The keyframe of the Model as follow:
      26a31c95-5f23-46fa-81e0-454749ede650-image.png

      What should I do to hold on the present keyframe? Sorry ablout my poor question description!

      My Python scripts as followed:

      # Boids for Py4D by smart-page.net
      
      import c4d
      import math
      
      # particles' params
      boids_number = 1000
      currentframe = None
      
      # used for read user data frame by frame
      frame_step = boids_number+1 
      frame_total = 0
      
      
      def main():
          global tp
          global doc
          global currentframe    
          
          currentframe = doc.GetTime().GetFrame(doc.GetFps())
          
          # particles born at 0 frame
          if currentframe == 0:   
              tp.FreeAllParticles()
              tp.AllocParticles(boids_number)
                   
                      
          # life time for particles
          lt = c4d.BaseTime(1000)
      
          # user data for paritlces
          filename = op[c4d.ID_USERDATA, 1]
         
          # open the user file. First, read a frame of data from the user file. Then, read lines one bye one to feed the particles.
          with open(filename, 'r') as fn:
              
              # read all lines of the user data
              lines = fn.readlines()
              
              # compute how many frames of data in the file
              frame_total = int(len(lines) / frame_step)
              
              # 
              frame = 1
              i=0
      
              #read a frame of data according to the scene keyframe
              for frame in range(frame_total): 
                  if frame == currentframe:
                      t_lines = lines[frame * frame_step:frame * frame_step + frame_step - 1]  
                      
                      #pase lines of the readed data 
                      for line in t_lines: 
                          if line == t_lines[0]:  # filter the first line of each frame in the text file, because is just flag words
                              print(line)                       
      
                          else:
                              #split position(x,y,z) and direction (dx,dy,dz)
                              x, y, z, dx, dy, dz = line.split()                      
                              pos = c4d.Vector(float(x), float(y), float(z) )
                              vol = c4d.Vector(float(dx), float(dy), float(dz))
                                                     
                              temp=(pos-tp.Position(i)).GetLength()  
                              
                              # some codes wanted here
                              if temp==0.0:
                                  # the motion of the man should stop.
                                  # should I  edit the keyframe of the walking Man model?
                                  # when temp==0, then the walking Man stay at present keyframe but not go ahead along with the scene keyframe. 
                                                     
                                                   
                              
                             
                              # align to velocity direction
                              vel = vol.GetNormalized()
                              side = c4d.Vector(c4d.Vector(0, 1, 0).Cross(vel)).GetNormalized()
                              up = vel.Cross(side)
                              m = c4d.Matrix(c4d.Vector(0), side, up, vel)
                              tp.SetAlignment(i, m)
                             
                              # set position
                              tp.SetPosition(i,pos)
                              
                              # set life time for particle i
                              tp.SetLife(i, lt)  
                               
                              i=i+1
      
          c4d.EventAdd()
      
      if __name__=='__main__':
          main()
      code_text
      

      My user data format as followed.
      c6d1c4a1-6356-4120-a174-5dc56e19e6b2-image.png

      1 Reply Last reply Reply Quote 0
      • M
        m_adam
        last edited by m_adam

        Hi, @happygrass_cn first of all welcome in the plugincafe community.

        While you have deleted your topic (probably because you solved it). I had a solution that explores a few areas in Cinema 4D that are not so well documented so I restored it.

        So my solution uses a cloner with 2 clones (one animated and the other one similar but without animation).
        I also define a thinking particle channel data( a boolean "isMoving").
        A python effector will read this attribute and according to this attribute will affect the displayed cloned object to be either the animated one or not.
        Of course, this does not provide interpolation between animation so it can look a bit odd.

        Finally in Python SetPData does not support boolean, so "IsMoving" is not a boolean (DTYPE_BOOL) but an integer(DTYPE_LONG) that I thread as a boolean.

        The Python generator does the next things:

        • Thinking Particles Creation.
        • Custom Data Channel Creation/Assignation.
        • Movement of particles.
        import c4d
        
        def GetParticleSystemAndRootGroup():
            # Retrieves the Particles System of the current document
            pSys = doc.GetParticleSystem()
            if pSys is None:
                raise RuntimeError("op is none, please select one object.")
        
            # Retrieves the Root group (where all particles belongs as default)
            rootGrp = pSys.GetRootGroup()
            if rootGrp is None:
                raise RuntimeError("Failed to retrieve root group of tp master system.")
        
            return pSys, rootGrp
        
        def GetIsMovingChannelId():
            # Retrieves the Particles System and the root group of the current document
            pSys, rootGrp = GetParticleSystemAndRootGroup()
        
            for x in xrange(pSys.NumDataChannels()):
                if pSys.DataChannelName(x) == "isMoving(Integer)":
                    return x
        
            return False
        
        
        def CreateParticle(count):
            # Retrieves the Particles System and the root group of the current document
            pSys, rootGrp = GetParticleSystemAndRootGroup()
        
            if pSys.NumParticles() >= count:
                return
        
            # Allows each particles to get a custom colors
            rootGrp[c4d.PGROUP_USE_COLOR] = False
        
            # Creates 90 Particles
            particlesIds = pSys.AllocParticles(count)
            if not particlesIds:
                raise RuntimeError("Failed to create X TP particles.")
        
            # Check if the "isMoving"" Channel data already exist and if not create it
            # This Boolean "isMoving" will store the moving state of a particle
            if GetIsMovingChannelId() is False:
                pSys.AddDataChannel(c4d.DTYPE_LONG, "isMoving")
        
            isMovingId = GetIsMovingChannelId()
            if isMovingId is False:
                raise RuntimeError("Failed to retrieve IsMoving particle Data")
        
            # Assigns position and colors for each particles
            for particleId in particlesIds:
                # Checks if particles ID is ok
                if particleId == c4d.NOTOK:
                    continue
        
                # Defines a lifetime of 1000 frame for each particles
                pSys.SetLife(particleId, c4d.BaseTime(1000))
        
                # Calculates a position
                sin, cos = c4d.utils.SinCos(particleId)
                pos = c4d.Vector(sin * 300.0, cos * 300.0, particleId * 30.0)
                # Assigns position
                pSys.SetPosition(particleId, pos)
        
                # Calculates a color
                hsv = c4d.Vector(float(particleId) * 1.0 / count, 1.0, 1.0)
                rgb = c4d.utils.HSVToRGB(hsv)
        
                # Assigns color
                pSys.SetColor(particleId, rgb)
        
                # Assigns a "freeze" data to indicate either the particle is stoped or not
                pSys.SetPData(particleId, isMovingId, False)
        
            return particlesIds
        
        def MoveParticle(frame):
            # Retrieves the Particles System and the root group of the current document
            pSys, rootGrp = GetParticleSystemAndRootGroup()
        
            isMovingId = GetIsMovingChannelId()
            if isMovingId is False:
                raise RuntimeError("Failed to retrieve IsMoving particle Data")
        
            for particleId in xrange(pSys.NumParticles()):
                # If particle don't belong to the root group
                if pSys.Group(particleId) != rootGrp:
                    continue
        
                sin, cos = c4d.utils.SinCos(particleId + (frame/40.00))
                pos = c4d.Vector(sin * 300.0, cos * 300.0, particleId * 30.0)
                # Assigns position
                pSys.SetPosition(particleId, pos)
        
                # Assigns a "freeze" data to indicate either the particle is stoped or not
                pSys.SetPData(particleId, isMovingId, True)
        
        
        def FreezeParticle():
            # Retrieves the Particles System and the root group of the current document
            pSys, rootGrp = GetParticleSystemAndRootGroup()
        
            isMovingId = GetIsMovingChannelId()
            if isMovingId is False:
                raise RuntimeError("Failed to retrieve IsMoving particle Data")
        
            for particleId in xrange(pSys.NumParticles()):
                # If particle don't belong to the root group
                if pSys.Group(particleId) != rootGrp:
                    continue
        
                # Assigns a "freeze" data to indicate either the particle is stoped or not
                pSys.SetPData(particleId, isMovingId, 0)
        
        def main():
            currentFrame = doc.GetTime().GetFrame(doc.GetFps())
            if currentFrame == 0:
                CreateParticle(20)
        
            # Don't move particles from frame 150 to 250
            elif 150 <= currentFrame <= 250:
                if currentFrame == 150:
                    FreezeParticle()
            else:
                MoveParticle(currentFrame)
        

        The Python Effector set in Full Control does the next things:

        • Read the data for each particle of the Custom Data Channel
        • Defines the cloned used according to this value
        import c4d
        
        def GetParticleSystemAndRootGroup():
            # Retrieves the Particles System of the current document
            pSys = doc.GetParticleSystem()
            if pSys is None:
                raise RuntimeError("op is none, please select one object.")
        
            # Retrieves the Root group (where all particles belongs as default)
            rootGrp = pSys.GetRootGroup()
            if rootGrp is None:
                raise RuntimeError("Failed to retrieve root group of tp master system.")
        
            return pSys, rootGrp
        
        def GetIsMovingChannelId():
            # Retrieves the Particles System and the root group of the current document
            pSys, rootGrp = GetParticleSystemAndRootGroup()
        
            for x in xrange(pSys.NumDataChannels()):
                if pSys.DataChannelName(x) == "isMoving(Integer)":
                    return x
        
            return False
        
        def main() :
            moData = c4d.modules.mograph.GeGetMoData(op)
            if moData is None:
                return False
        
            cnt = moData.GetCount()
            mdClone = moData.GetArray(c4d.MODATA_CLONE)
        
            hasField = op[c4d.FIELDS].HasContent()
        
            # Retrieves the Particles System and the root group of the current document
            pSys, rootGrp = GetParticleSystemAndRootGroup()
        
            # Retrieve the isMovingData channel ID
            isMovingId = GetIsMovingChannelId()
            if isMovingId is False:
                raise RuntimeError("Failed to retrieve IsMoving particle Data")
        
            # Maybe there is more particles allocated than what's is used in the cloner (aka mutlipe particle group)
            # So we loop over them
            cloneId = 0
            for particleId in xrange(pSys.NumParticles()):
        
                # If particle don't belong to the root group
                if pSys.Group(particleId) != rootGrp:
                    continue
        
                if cloneId > cnt:
                    raise RuntimeError("There is more cloneID than allowed")
        
                # Retrieve the internal IsMoving Data and check if the particle is moving,
                isMoving = pSys.GetPData(particleId, isMovingId)
                if isMoving == 1:
                    # Defines the float value representing the cloned 0 = First child, 1 = last child
                    mdClone[cloneId] = 0.0
                else:
                    mdClone[cloneId] = 1.0
        
                cloneId +=1
        
            moData.SetArray(c4d.MODATA_CLONE, mdClone, hasField)
            return True
        

        And the attached scene with everything together.
        generatorTp.c4d

        Cheers,
        Maxime.

        MAXON SDK Specialist

        Development Blog, MAXON Registered Developer

        1 Reply Last reply Reply Quote 0
        • First post
          Last post