#!/usr/bin/env python3
# Filename: A16b_simple_airtrack_forces_P3.py
"""
Air Track with point forces...
Self-contained pygame-based example of pybox2d.
This file is based on the simple_01.py example in the pybox2d example directory.
"""
import pygame
from pygame.locals import *
# import Box2D # The main library
from Box2D import *
# This maps Box2D.b2Vec2 to vec2, and so on. (confusing, so disabled it)
#from Box2D.b2 import *
#=====================================================================
# Classes
#=====================================================================
class fwQueryCallback(b2QueryCallback):
# Checks for objects at particular locations (p) like under the cursor.
def __init__(self, p):
super(fwQueryCallback, self).__init__()
self.point = p
self.fixture = None
def ReportFixture(self, fixture):
body = fixture.body
if body.type == b2_dynamicBody:
inside=fixture.TestPoint(self.point)
if inside:
self.fixture=fixture
# We found the object, so stop the query
return False
# Continue the query
return True
class Environment:
def __init__(self, screensize_tuple, world):
self.world = world
self.viewZoom = 10.0
self.viewCenter = b2Vec2(0, 0)
self.viewOffset = b2Vec2(0, 0)
self.screenSize = b2Vec2(*screensize_tuple)
self.rMouseDown = False
self.mouseJoint = None
# Needed for the mousejoint
self.groundbody = self.world.CreateBody()
self.flipX = False
self.flipY = True
# pass viewZoom to init in DrawToScreen class. Call the DrawToScreen function by use of the "renderer" name.
self.renderer = DrawToScreen(self.viewZoom)
self.pointSize = 2.5
self.colors={
'mouse_point' : b2Color(1,0,0),
'bomb_center' : b2Color(0,0,1.0),
'joint_line' : b2Color(0.8,0.8,0.8),
'contact_add' : b2Color(0.3, 0.95, 0.3),
'contact_persist' : b2Color(0.3, 0.3, 0.95),
'contact_normal' : b2Color(0.4, 0.9, 0.4),
'force_point' : b2Color(0,1,0)
}
def MouseDown(self, p):
"""
Indicates that there was a left click at point p (world coordinates)
"""
# If there is already a mouse joint just get out of here.
if self.mouseJoint != None:
return
# Create a mouse joint on the selected body (assuming it's dynamic)
# Make a small box.
aabb = b2AABB(lowerBound=p-(0.001, 0.001), upperBound=p+(0.001, 0.001))
# Query the world for overlapping shapes.
query = fwQueryCallback(p)
self.world.QueryAABB(query, aabb)
if query.fixture:
body = query.fixture.body
# A body was selected, create the mouse joint
self.mouseJoint = self.world.CreateMouseJoint(
bodyA=self.groundbody,
bodyB=body,
target=p,
maxForce=1000.0*body.mass)
body.awake = True
def MouseUp(self, p):
"""
Left mouse button up.
"""
if self.mouseJoint:
self.world.DestroyJoint(self.mouseJoint)
self.mouseJoint = None
def MouseMove(self, p):
"""
Mouse moved to point p, in world coordinates.
"""
self.mouseWorld = p
if self.mouseJoint:
self.mouseJoint.target = p
def Keyboard_Event(self, key, down=True):
global apply_jet_1_TF, apply_jet_2_TF
if down:
if key == K_z: # Zoom in
self.viewZoom = min(1.1 * self.viewZoom, 50.0)
elif key == K_x: # Zoom out
self.viewZoom = max(0.9 * self.viewZoom, 0.02)
elif key == K_UP:
self.viewCenter -= (0, 20)
elif key == K_DOWN:
self.viewCenter += (0, 20)
elif key == K_RIGHT:
self.viewCenter -= (20, 0)
elif key == K_LEFT:
self.viewCenter += (20, 0)
elif key == K_HOME:
self.viewZoom = 10.0
self.viewCenter = b2Vec2(0, 0)
self.viewOffset = b2Vec2(0, 0)
elif key == K_f:
apply_jet_1_TF = True
print("applying force 1")
elif key == K_g:
apply_jet_2_TF = True
print("applying force 2")
else:
pass
# If up
else:
if key == K_f:
apply_jet_1_TF = False
elif key == K_g:
apply_jet_2_TF = False
else:
pass
def CheckEvents(self):
"""
Check for pygame events (mainly keyboard/mouse events).
Passes the events onto the GUI also.
"""
for event in pygame.event.get():
if event.type == QUIT or (event.type == KEYDOWN and event.key == K_ESCAPE):
print("early bailout")
return False
elif event.type == KEYDOWN:
self.Keyboard_Event(event.key, down=True)
elif event.type == KEYUP:
self.Keyboard_Event(event.key, down=False)
elif event.type == MOUSEBUTTONDOWN:
p = self.ConvertScreenToWorld(*event.pos)
if event.button == 1: # left
self.MouseDown( p )
elif event.button == 2: #middle
pass
elif event.button == 3: #right
self.rMouseDown = True
elif event.button == 4:
self.viewZoom *= 1.1
elif event.button == 5:
self.viewZoom /= 1.1
elif event.type == MOUSEBUTTONUP:
p = self.ConvertScreenToWorld(*event.pos)
if event.button == 3: #right
self.rMouseDown = False
else:
self.MouseUp(p)
elif event.type == MOUSEMOTION:
p = self.ConvertScreenToWorld(*event.pos)
self.MouseMove(p)
if self.rMouseDown:
#print(f"mouse motion: position=({event.pos[0]:4d},{event.pos[1]:4d}), change=({event.rel[0]:3d},{event.rel[1]:3d})")
self.viewCenter -= (event.rel[0]/1.0, -event.rel[1]/1.0) #... it was /5.0
return True
def ConvertScreenToWorld(self, x, y):
# self.viewOffset = self.viewCenter - self.screenSize/2
self.viewOffset = self.viewCenter
return b2Vec2((x + self.viewOffset.x) / self.viewZoom,
((self.screenSize.y - y + self.viewOffset.y) / self.viewZoom))
def ConvertWorldtoScreen(self, point):
"""
Convert from world to screen coordinates.
In the class instance, we store a zoom factor, an offset indicating where
the view extents start at, and the screen size (in pixels).
"""
# The zoom factor works to define and scale the relationship between pixels (screen) and meters (world).
self.viewOffset = self.viewCenter
x = (point.x * self.viewZoom) - self.viewOffset.x
if self.flipX:
x = self.screenSize.x - x
y = (point.y * self.viewZoom) - self.viewOffset.y
if self.flipY:
y = self.screenSize.y - y
return (round(x), round(y)) # return tuple of integers
def DrawMouseJoint(self):
if self.mouseJoint:
p1_screen = self.ConvertWorldtoScreen(self.mouseJoint.anchorB) # point on the object converted to screen coordinates
p2_screen = self.ConvertWorldtoScreen(self.mouseJoint.target) # current mouse position converted to screen coordinates
self.renderer.DrawPoint(p1_screen, self.pointSize, self.colors['mouse_point'])
self.renderer.DrawPoint(p2_screen, self.pointSize, self.colors['mouse_point'])
self.renderer.DrawSegment(p1_screen, p2_screen, self.colors['joint_line'])
def DrawForcePoint(self, forcePoint_2d_m):
forcePoint_screen = self.ConvertWorldtoScreen( forcePoint_2d_m)
self.renderer.DrawPoint( forcePoint_screen, self.pointSize, self.colors['force_point'])
class DrawToScreen:
def __init__(self, viewZoom):
self.viewZoom = viewZoom
self.surface = screen
def DrawPoint(self, p, size, color):
# Draw a single point at point p given a pixel size and color.
self.DrawCircle(p, size/self.viewZoom, color, drawwidth=0)
def DrawSegment(self, p1, p2, color):
# Draw the line segment from p1-p2 with the specified color.
pygame.draw.aaline(self.surface, color.bytes, p1, p2)
def DrawCircle(self, center, radius, color, drawwidth=1):
# Draw a wireframe circle.
radius *= self.viewZoom
if radius < 1:
radius = 1
else:
radius = round(radius)
pygame.draw.circle(self.surface, color.bytes, center, radius, drawwidth)
#=================================================================================
# Some functions
#=================================================================================
def AddCar(theWorld, bodies, x=0, vx=0, width=1, height=1, rest=1.00, angDamp=0.0):
# Create a dynamic body.
dynamic_body = theWorld.CreateDynamicBody(position=b2Vec2(x, 0), angle=0, linearVelocity=b2Vec2(vx, 0), angularDamping=angDamp)
# Add to the bodies list
bodies.append( dynamic_body)
# And add a box fixture onto the body.
dynamic_body.CreatePolygonFixture(box=(width, height), density=1.0, friction=0.00, restitution=rest)
# Note the area used in the mass calc will be area = (2 * w) * (2 * h)
# Then mass = density * area
print("Mass data:", dynamic_body.mass)
#=================================================================================
# Main program
#=================================================================================
# --- constants ---
TARGET_FPS = 120
TIME_STEP = 1.0/TARGET_FPS
screenXY = SCREEN_WIDTH, SCREEN_HEIGHT = 1140, 480
# --- pygame setup ---
pygame.init()
screen = pygame.display.set_mode((SCREEN_WIDTH, SCREEN_HEIGHT), 0, 32)
pygame.display.set_caption('Simple pybox2d example')
clock = pygame.time.Clock()
# --- pybox2d world setup ---
# Create the world
world = b2World(gravity=(-0.0, -10.0), doSleep=True)
# Initialize the environment object.
env = Environment(screenXY, world)
# list of bodies
bodies = []
# And a static body to be the base of the air track.
ground_body = world.CreateStaticBody(
position=(0,0),
shapes=b2PolygonShape(box=(150,0.5))
)
bodies.append( ground_body)
# left bumper
air_track_bumper1 = world.CreateStaticBody(
position=(0,0),
shapes=b2PolygonShape(box=(0.5,5))
)
bodies.append( air_track_bumper1)
# right bumper
air_track_bumper2 = world.CreateStaticBody(
position=(100,0),
shapes=b2PolygonShape(box=(0.5,5))
)
bodies.append( air_track_bumper2)
# Add cars to the track.
x = 15
for j in range(5):
x += 5
vx = j + 2
AddCar(world, bodies, x, vx, height=1.0, width=2.0, rest=0.8, angDamp=0.7)
colors = {
b2_staticBody : (255,255,255,255),
b2_dynamicBody : (127,127,127,255),
}
# Initialize the jet toggles.
apply_jet_1_TF = False
apply_jet_2_TF = False
# --- main game loop ---
running = True
while running:
running = env.CheckEvents()
# Apply forces
if apply_jet_1_TF:
force_point_1_2d_m = bodies[5].GetWorldPoint(b2Vec2(2,0))
bodies[5].ApplyForce(force=b2Vec2(0.0, 100.0), point=force_point_1_2d_m, wake=True)
if apply_jet_2_TF:
force_point_2_2d_m = bodies[5].GetWorldPoint(b2Vec2(-2,0))
bodies[5].ApplyForce(force=b2Vec2(0.0, 100.0), point=force_point_2_2d_m, wake=True)
# Make Box2D simulate the physics of our world for one step.
# Instruct the world to perform a single step of simulation. It is
# generally best to keep the time step and iterations fixed.
# See the manual (Section "Simulating the World") for further discussion
# on these parameters and their implications.
world.Step(TIME_STEP, 10, 10)
screen.fill((0,0,0,0))
# Draw the world
for body in bodies: # or: world.bodies
# The body gives us the position and angle of its shapes
for fixture in body.fixtures:
# The fixture holds information like density and friction,
# and also the shape.
shape = fixture.shape
# This assumes that this is a polygon shape. (not good normally!)
# We take the body's transform and multiply it with each
# vertex, and then convert from world to screen.
# The "*" operators are overloaded from the class (operator overloading).
vertices_screen = []
for vertex_object in shape.vertices:
vertex_world = body.transform * vertex_object # Overload operation
vertex_screen = env.ConvertWorldtoScreen( vertex_world) # This returns a tuple
vertices_screen.append( vertex_screen) # Append to the list.
pygame.draw.polygon(screen, colors[body.type], vertices_screen)
# Draw mouse joint.
env.DrawMouseJoint()
# Draw force points.
if apply_jet_1_TF:
env.DrawForcePoint( force_point_1_2d_m)
if apply_jet_2_TF:
env.DrawForcePoint( force_point_2_2d_m)
# Flip the screen and try to keep at the target FPS
pygame.display.flip()
clock.tick(TARGET_FPS)
pygame.quit()
print('Done!')