Loading...
Logo
Processing Forum
Is it possible to fix a body so that it may be translated and not rotated?

Replies(22)

So I see that their is a field for fixedRotation in the Body class, however since it is final I have tried to declare a Body using:

    BodyDef manDef;
    Body crazyMan;
    manDef = new BodyDef();
    manDef.fixedRotation = true;
    crazyMan = physics.getWorld().createBody(manDef);
    PolygonDef pDef = new PolygonDef();
    pDef.setAsBox(mWidth, mHeight, new Vec2(x,y), 0.0);
    crazyMan.createShape(pDef); 

I am getting an assertion error, can anyone shed some light on this?
" I am getting an assertion error"
Can you be more precise?
Here is the full log, below I will post the full code for all classes:





Cannot run Java in 32 bit mode. Continuing in 64 bit mode.
processing.app.debug.RunnerException: AssertionError
at processing.app.Sketch.placeException(Sketch.java:1543)
at processing.app.debug.Runner.findException(Runner.java:583)
at processing.app.debug.Runner.reportException(Runner.java:558)
at processing.app.debug.Runner.exception(Runner.java:498)
at processing.app.debug.EventThread.exceptionEvent(EventThread.java:367)
at processing.app.debug.EventThread.handleEvent(EventThread.java:255)
at processing.app.debug.EventThread.run(EventThread.java:89)
Exception in thread "Animation Thread" java.lang.AssertionError
at org.jbox2d.collision.Shape.createProxy(Shape.java:308)
at org.jbox2d.dynamics.Body.createShape(Body.java:354)
at CrazyMan2$Man.<init>(CrazyMan2.java:193)
at CrazyMan2.setup(CrazyMan2.java:50)
at processing.core.PApplet.handleDraw(PApplet.java:1583)
at processing.core.PApplet.run(PApplet.java:1503)
at java.lang.Thread.run(Thread.java:680)
Below is the full program code, however since everything works when I take away the code from the Man class to create a custom Body through BodyDefs and attach a Shape via ShapeDef, and just use createRect(), I suspect that it is either in CrazyMan or Man, or possibly in the way that the Physics class is initialized behind the scenes or the World class. To make the body non-rotational I looked at the createRect() code in the source for BoxWrap2d and tried to mimic it, and just add the fixedRotation field.


The CrazyMan2 class: 

class Man{
  BodyDef manDef;
  PolygonDef manShapeDef;
  Body crazyMan;
  ForceUtils futil = new ForceUtils();
  Vec2 pointLeft;
  Vec2 pointRight;
  Vec2 pointDown;
  Vec2 pointUp;
  Boolean isJumping;

  
  Man(float mWidth, float mHeight, float x, float y, Physics physics){
    physics.setDensity(1.0);
    physics.setFriction(0.5);
    manDef = new BodyDef();
    manDef.fixedRotation = true;
    crazyMan = physics.getWorld().createBody(manDef);
    PolygonDef pDef = new PolygonDef();
    pDef.setAsBox(mWidth, mHeight, new Vec2(x,y), 0.0);
    crazyMan.createShape(pDef); 
    //crazyMan = physics.createRect(x1, y1, x2, y2);
    pointLeft = new Vec2(-width,0);
    pointRight = new Vec2(width,0);
    pointUp = new Vec2(0, height);
    pointDown = new Vec2(0,-height);
  }
  
  float getMass(){
    return crazyMan.m_mass;
  }
  
  void moveLeft(){
    futil.push(crazyMan, pointLeft, 3000.0);  
  }

  void moveRight(){
    futil.push(crazyMan, pointRight, 3000.0);  
  }

  void moveDown(){
    futil.push(crazyMan, pointDown, 3000.0);  
  }

  void moveUp(){
    futil.push(crazyMan, pointUp, 3000.0);  
  }
}

The Man class:

class Man{
  BodyDef manDef;
  PolygonDef manShapeDef;
  Body crazyMan;
  ForceUtils futil = new ForceUtils();
  Vec2 pointLeft;
  Vec2 pointRight;
  Vec2 pointDown;
  Vec2 pointUp;
  Boolean isJumping;

  
  Man(float mWidth, float mHeight, float x, float y, Physics physics){
    physics.setDensity(1.0);
    physics.setFriction(0.5);
    manDef = new BodyDef();
    manDef.fixedRotation = true;
    crazyMan = physics.getWorld().createBody(manDef);
    PolygonDef pDef = new PolygonDef();
    pDef.setAsBox(mWidth, mHeight, new Vec2(x,y), 0.0);
    crazyMan.createShape(pDef); 
    //crazyMan = physics.createRect(x1, y1, x2, y2);
    pointLeft = new Vec2(-width,0);
    pointRight = new Vec2(width,0);
    pointUp = new Vec2(0, height);
    pointDown = new Vec2(0,-height);
  }
  
  float getMass(){
    return crazyMan.m_mass;
  }
  
  void moveLeft(){
    futil.push(crazyMan, pointLeft, 3000.0);  
  }

  void moveRight(){
    futil.push(crazyMan, pointRight, 3000.0);  
  }

  void moveDown(){
    futil.push(crazyMan, pointDown, 3000.0);  
  }

  void moveUp(){
    futil.push(crazyMan, pointUp, 3000.0);  
  }
}

The Joints class:

class Joints{
  private LinkedList<Joint> jointList;
  Boolean hasFirstEnd;
  Body firstEnd;
  Physics physics;
  
  
  Joints(Physics p){
    physics = p;
    hasFirstEnd = false;
    jointList = new LinkedList<Joint>();
  }
  
  void createNew(Body attachment){
    if(attachment == null) return;
    if(hasFirstEnd){
      physics.setRestitution(1.0);
      jointList.add(JointUtils.createDistanceJoint(attachment, firstEnd));
      hasFirstEnd = false;
    } else {
      hasFirstEnd = true;
      firstEnd = attachment;
    }
  }
}

The Circles class:

class Circles{
  
  private LinkedList<Body> circleList;
  Physics physics;
  
  Circles(Physics p){
    circleList = new LinkedList<Body>();
    this.physics = p;
  } 
  
  void createNew(int x, int y, float r){
    physics.setDensity(0.0);
    circleList.add(physics.createCircle((float) x,(float) y,r));
  }
    void createNewDynamic(int x, int y, float r){
    physics.setDensity(1.0);
    circleList.add(physics.createCircle((float) x,(float) y,r));
  }
}
I cannot test your code: you pasted Man code instead of CrazyMan2 one, and I don't have setup()/draw() code.
So I looked at the source, but SVN progressed a lot since 2008 (it might be interesting the grab the latest jars, they can be found in the SVN repository). And the Shape class moved from org.jbox2d.collision org.jbox2d.collision.shapes (maybe you had a jar with this organization, when you had a problem with Shape?).

So I just used a decompiler (JD-GUI) to look at the actual source of the jar we have. Indeed, at line 308 there is an assert() line, checking broadPhase.inRange(aabb) is true, whatever it means (it is in createProxy and I am not sure what a proxy is in this context).
So the only advice I can give is to check the values you use, in case they go out of the range of the current world.
Here is the code for CrazyMan2, the class I should've posted, and much thanks for all your help:

import org.jbox2d.util.nonconvex.*;
import org.jbox2d.dynamics.contacts.*;
import org.jbox2d.testbed.*;
import org.jbox2d.collision.*;
import org.jbox2d.common.*;
import org.jbox2d.dynamics.joints.*;
import org.jbox2d.p5.*;
import org.jbox2d.dynamics.*;

Physics physics;
Man someguy;
char creatorMode;
Circles circles;
Joints joints;
World world;

void setup(){
  size(300, 200);
  frameRate(60);
  smooth();
  InitScene();
  someguy = new Man(width/10.0, height/10.0, width/2.0, height/2.0, physics);
  circles = new Circles(physics);
  joints = new Joints(physics);
}

void draw(){
  background(255);
}

void InitScene()
{
  // Set up the engine with the sketch's dimensions
  physics = new Physics(this, width, height);
  world = physics.getWorld();
}

void keyPressed()
{
  switch(key) {
    case 'a':
      someguy.moveLeft();
      return;
    
    case 'd':
      someguy.moveRight();
      return;
    
    case 'w':
      someguy.moveUp();
      return;
    case 'c':
      creatorMode = 'c';
      return;
    case 'x':
      creatorMode = 'x';
      return;
    case 'v':
      creatorMode = 'j';
      return;    
  }
  // Can be used to reset the sketch, for example
  physics.destroy();
  physics = null;
  InitScene();
  //createObjects();
}

void mouseReleased(){
  if(creatorMode == 'c'){
    circles.createNew(mouseX, mouseY, 10.0);
  }
  else if(creatorMode == 'x'){
    circles.createNewDynamic(mouseX, mouseY, 10.0);
  }
  else if(creatorMode == 'j'){
    joints.createNew(getBodyAtPoint(mouseX, mouseY));
  }
}

Body getBodyAtPoint(float x, float y)
{
  // Create a small box at mouse point
  Vec2 v = physics.screenToWorld(x, y);
  AABB aabb = new AABB(new Vec2(v.x - 0.001, v.y - 0.001), new Vec2(v.x + 0.001, v.y + 0.001));
  // Look at the shapes intersecting this box (max.: 10)
  org.jbox2d.collision.Shape[] shapes = world.query(aabb, 10);
  if (shapes == null)
    return null;  // No body there...
  for (int is = 0; is < shapes.length; is++)
  {
    org.jbox2d.collision.Shape s = shapes[is];
// Ensure it is really at this point
if (s.testPoint(s.m_body.getXForm(), v))
 return s.m_body; // Return the first body found
    }
  return null;
}

I think the problem is that you use the PolygonDef before adding vertices to it, so it is empty.
can you explain what you mean? wouldn't the setAsBox() method populate the polygonDef?
It appears that a proxy is a dynamic tree that is created to arrange data in a binary tree, (now I recall reading about it in the manual). It is used to check for the presence of an object inside of an AABB. So it makes sense that upon object creation it must be added to a proxy. I have played with the arguments to the constructor of Man, and using   

someguy = new Man(width/2.0, height/2.0, width/2.0, height/20.0, physics);

I don't get an error, however now the body is static, and I can't figure out why
Yes, sorry, I have read the code too fast, the setAsBox you use calls another version which add vertices.
It looks like I have made a slight mistake, the code I use to set up my box works, however it is static and tricky to create (the dimensions seem to be off). I realized that I wanted the code from the createRect() function, however in my delirium I accidentally used createBox(); Right now I am working on porting the createRect() code to make a custom function for the man class that will set the fixedRotation field. This is proving slightly tricky due to the Screen vs World coord translations using some other methods in the Physics class, but I am sure I will conquer it. I realized that before the density was not getting explicitly set, which is why I mention in the above post that the body is static, I took it for granted that the Physics class would set it, but this is not the case of Jbox2d and is only a helper of BoxWrap2d so in making a custom object using Jbox2d it isn't done for you.
Here is the new modified version of createRect, the main modifications were in changing internal members of the Physics class to be compatible outside of the class. From here I am looking to do some contact listening and change the way the dynamic circles are created so that they are static initially and can be set to dynamic instead of having them be created as dynamic (which makes it difficult to add a joint and keep the circle in the right place since a dynamic one will fall). Thanks phi.lho for the help. I am thinking also about simply using the Jbox2d library as the current boxwrap2d is quite old I believe, or possibly recompiling the library and updating some of the wrapper code to support stuff included in the newer versions of box2d like fixtures.

update: I have tried setting the circles using the setMassFromShapes() method, but nothing is happening, the objects are remaining dynamic. Undoubtedly this will also require some digging into the library to understand.

  private Body createManRect(float x0, float y0, float x1, float y1) {
    float cxs = (x0 + x1) * .5f;
    float cys = (y0 + y1) * .5f;
    float wxs = Math.abs(x1-x0);
    float wys = Math.abs(y1-y0);
    //System.out.println("Screen: ("+cxs + ","+cys+")");
    Vec2 center = physics.screenToWorld(cxs, cys);
    //System.out.println("World: "+center);
    float halfWidthWorld = .5f*physics.screenToWorld(wxs);
    float halfHeightWorld = .5f*physics.screenToWorld(wys);
    //System.out.println("Half Width world: "+halfWidthWorld);
    PolygonDef pd = new PolygonDef();
    pd.setAsBox(halfWidthWorld, halfHeightWorld);
    pd.density = 1.0;
    pd.restitution = 0.2;
    pd.friction = 0.5;
    pd.isSensor= false;
    
    BodyDef bd = new BodyDef();
    bd.fixedRotation = true;
    bd.isBullet = false;

    Body b = physics.getWorld().createBody(bd);
    b.createShape(pd);
    b.setMassFromShapes();

    b.setXForm(center, 0.0f);

    return b;
  }
You might also want to consider using Fisica http://www.ricardmarxer.com/fisica

It is a wrapper around jBox2D.  It hides quite a few details about the jBox2D, but you can also access to the underlying objects to use the jBox2D API directly when needed.

For the rotation problem you can simply use:

myBody.setRotatable(false);

before adding the body to the world.  See the doc here:

Good luck with your project
ricard
I was looking over the Fisica docs and wanted to know if it is possible to implement a custom renderer? I am debating whether to use Fisica or code the next iteration of CrazyMan, CrazyMan3 in jbox2d without a wrapper. I believe that  implementing a custom renderer can be done by extending FWorld and overriding the void draw() method?
Thanks for the info Ricard, I am going to check out fisica.

Right now I am trying to make a Body that is static and then later assign it a density, I have tried using setMassFromShapes(), but have had no such luck, perhaps this is because even if there are shapes it is calculating the mass based on the density (which would be set to 0). Right now I am going to make a slight work-around, however my end goal definitely needs this feature, Ricard or Philippe, do either of you know if this change from static to dynamic is possible?
In Fisica you can do:

myBody.setStatic(true);

or:

myBody.setStatic(false);

In jbox2d I think you might have to recalculate the mass of the shape from the density, and probably you might have to remove the body from the world and add it again with the same properties of velocity, angular velocity, rotation and position.  But I'm not sure about that, you'll have to check.

cheers
Hey Ricardo,
I am looking at the Fisica doc for FBody and it shows that its superclass is Object, while this is true, is not Body (from JBox2d) also its superclass?
Hey!

Actually FBody does not inherit from Body.  FBody contains the underlying Body as a member variable.  This variable is named m_body and it is protected:

  protected Body m_body;

So you can inherit from any FBody and access it's underlying Body using m_body.  However beware that before adding the FBody to the FWorld m_body is null.  So handle with care.

However if there is something that you need and can't do with Fisica let me know.  Maybe it's something that can be easily added to the lib.
Fisica is excellent and includes many useful things, I was able to port my code pretty quickly. 

However one of the things that surprised me was that it doesn't extend the base. In particular last night I wanted to get a JointEdge (linked list of joints) from a FBody, as far as I can tell there is no getJointList() method like exists in Box2d, requiring you to manually define and track the joints you would like to destroy.

 I see that FWorld extends world so I believe that I could use an AABB from Jbox2d and register it the same way. If some additional coding needs to be done to do this, I will be happy to do so. 

Sometime this week I will start to dig in and read your source code through well for a better understanding.
 However one of the things that surprised me was that it doesn't extend the base.

Yep, I first tried inheriting from Body.  However that didn't allow me to make the API I wanted.  And I had to move them into a member variable.  For the details, I guess you'll understand when you see the source, in short it is because in jBox2d Body objects only exist in a World object, and I wanted an API that allows you to freely add the and remove them from a world.


In particular last night I wanted to get a JointEdge (linked list of joints) from a FBody, as far as I can tell there is no getJointList() method like exists in Box2d, requiring you to manually define and track the joints you would like to destroy.
This is a good feature request.  I will see if I can add it to the lib during this week.

Let me know if you see any missing methods that are in jBox2d.  You can drop me an email if you want.

ricard

Actually it was a quick thing to add.

I am now uploading the new version of Fisica
With the FBody.getJoints() and a new example that illustrates it here.

It's a bit different from JBox2d's because I didn't want to add a new class. But you can easily get the two bodies connected using:

myJoint.getBody1() and myJoint.getBody2();
Ricard, I am currently trying to figure out how to make my character jump. I have the 'w' key applying an upward force, but this lets the character fly. Ideally the character can jump not only when it is on the ground, but also when it is on platforms above the ground that are floating. Any suggestions?