Skip to content

Commit

Permalink
Added variable holders (namespace) to massspring.py so the variables …
Browse files Browse the repository at this point in the history
…can be changed from outside the package. Also redirected main README to README in massspring.
  • Loading branch information
pooya-shams committed Mar 22, 2020
1 parent 75e50c5 commit aedeb81
Show file tree
Hide file tree
Showing 8 changed files with 44 additions and 153 deletions.
127 changes: 1 addition & 126 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,134 +7,9 @@ Inspired by Saeed Sarkarati
## Brief

using this library you can create great simulations of most of physical environments which can be calculated presizely. you can create many types of objects from which "mass" and "spring" are main ones.
more on [project homepage on github](https://github.com/pooya-shams/massspring) and [README](https://github.com/pooya-shams/massspring/blob/master/massspring/README.md)
more information is available on [project homepage on github](https://github.com/pooya-shams/massspring) and [README](https://github.com/pooya-shams/massspring/blob/master/massspring/README.md)

## license

licensed under MIT License.
you can freely use and modify this package, but please let me know if you are doing something interesting.

## usage

you can import the library by using the following code (it is recommended to import as m or ms)

```python
import massspring as m
```

### mass

mass is the only type of **object** you can create right now.
you can create a mass object by using the mass class. you can learn more about the arguments and their usage in the [mass class documention](./massspring.py#mass)

```python
m1 = m.mass(x=0, y=0, z=0, vx=0, vy=0, vz=0, m=10, r=10, q=0, moveable=False, solid=True, bound=True, gravitateable=False, resistable=False, electrical=False, conductive=False, color=(0, 255, 0), visible=True)
```

### force

there are four types of forces (technically force subclass) you can create:
1.gravity
2.electricity
3.spring
4.collision

#### gravity

the gravity class provides a gravity force between two masses according to the **"Newoton's law of universal gravitation"** which you can see bellow:
![gravity formula](https://wikimedia.org/api/rest_v1/media/math/render/svg/8c6ee5510ba3c7d6664775c0e76b53e72468303a)
more information is available at [Newton's law of universal gravitation](https://en.wikipedia.org/wiki/Newton%27s_law_of_universal_gravitation) wikipedia page.
you can create a gravity force between two masses using the code bellow:

```python
g1 = m.gravity(m1=m1, m2=m2)
```

(m1 and m2 are the two mass objects between which you want to set gravity force.)

#### electricity

the electricity class provides an electricity force between two masses according to the **"Coulomb's law"** (the vector version) which you can see bellow:
![Coulomb's law](https://wikimedia.org/api/rest_v1/media/math/render/svg/6a815b94d95194b85a8fe53c0b1b5935764ea458)
more information is available at [Coulomb's law](https://en.wikipedia.org/wiki/Coulomb%27s_law) wikipedia page.
you can create an electricity force between two masses using the code bellow:

```python
e1 = m.electricity(m1=m1, m2=m2)
```

(m1 and m2 are the two mass objects between which you want to set electricity force.)

#### spring

spring can also be known as an object. However, spring class is a subclass of force class. So technically spring is a force. the force will be calculated using the **"Hooke's law"**:
![Hooke's law: F=-kx](https://wikimedia.org/api/rest_v1/media/math/render/svg/66774554c5cee5c22c57b0950c06ded0d36da720)
more information is available on [Hooke's law](https://en.wikipedia.org/wiki/Hooke%27s_law) wikipedia page.
you can create a spring object by using the spring class. you can learn more about the arguments and their usage in the [spring class documention](./massspring.py#spring).

```python
s1 = m.spring(k=1500000, l=0, m1=m1, m2=m2, color=(255, 0, 0), visible=True)
```

#### collision

uses the conservation of momentum and kinetic energy and the equations presented at the [Elastic Collision](https://en.wikipedia.org/wiki/Elastic_collision) wikipedia page.

```python
c1 = m.collision(m1=m1, m2=m2)
```

### mainloop

after creating all of the objects and forces you want, you should call the mainloop function which is an infinite while loop and ends when the user closes the window or presses the ESCape button.
arguments are:
1.speed: the more this number is the faster your program runs. it will decrease the number of frames in which the screen will be updated(increases the space between them).
2.FPS: sets the fps parameter for the pygame.time.Clock().tick(FPS) function. the less this number is the less your cpu will be under pressure.
3.frame: it's a function which will be called every frame of the program.
4.\*args: the arguments you want to give to the frame function.

```python
m.mainloop(speed=2, FPS=0, frame=None, *args)
```

## examples

You can see a simple example of using massspring library to create a pendulum.
It has an acceleration of -g in the Y ordinate.
!["pendulum"](https://github.com/pooya-shams/massspring/tree/master/images/massspring%20(pendulum).gif)

available at [example 1](./example%201%20(pendulum).py)

Here is anothere example of using this library.</br>
As you can see there is a double pendulum shown in the image. However, it is show in both x-y and z-y coordinates (x-y at left and z-y at right). you might feel they are completely different objects but they are just showing one object from two different points of veiw.

![double pendulum](https://github.com/pooya-shams/massspring/tree/master/images/massspring%20(double%20pendulum).gif)

available at [example 2](./example%202%20(double%20pendulum%203d).py).

3.[example 3]("./example%203%20(3d%20pyramid).py")
4.[example 4](./example%204%20(rope).py)
5.[example 5](./example%205%20(collision).py)

## Requirements

python >= 3.6
pygame >= 1.9.2
Note: if you want to hide the pygame support message, place this code before importing massspring.

```python
import os
os.environ["PYGAME_HIDE_SUPPORT_PROMPT"] = '1'
```

## TODO

(feel free to complete these tasks)

- [ ] Update the Elastic collision.
- [ ] Divide the collision to Elastic collision and Inelastic collision.
- [ ] Create non-sphere/non-mass objects.
- [ ] Collision between mass and spring (mass: sphere with mass, spring: line without mass).
- [ ] Collision between non-sphere objects, sphere objects and line objects(spring).
- [ ] Add statistic screen.
- [ ] Create the c/c++ project.
2 changes: 1 addition & 1 deletion massspring/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -105,7 +105,7 @@ It has an acceleration of -g in the Y ordinate.

available at [example 1](./example%201%20(pendulum).py)

Here is anothere example of using this library.</br>
Here is anothere example of using this library.
As you can see there is a double pendulum shown in the image. However, it is show in both x-y and z-y coordinates (x-y at left and z-y at right). you might feel they are completely different objects but they are just showing one object from two different points of veiw.

![double pendulum](./images/massspring%20(double%20pendulum).gif)
Expand Down
2 changes: 1 addition & 1 deletion massspring/example 1 (pendulum).py
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import massspring as m

m.ay0 = -m.ge
m.acceleration.y = -m.ge

m1 = m.mass(x=0, y=0, z=0, vx=0, vy=0, vz=0, m=10, r=10, q=0, moveable=False,
solid=True, bound=True, gravitateable=False, resistable=False,
Expand Down
6 changes: 3 additions & 3 deletions massspring/example 2 (double pendulum 3d).py
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
import massspring as m

m.ax0 = 0
m.ay0 = -m.ge
m.az0 = -m.ge
m.acceleration.x = 0
m.acceleration.y = -m.ge
m.acceleration.z = -m.ge

m1 = m.mass(x=0, y=0, z=0, vx=0, vy=0, vz=0, m=20, r=10, q=0, moveable=False,
solid=True, bound=True, gravitateable=False, resistable=False,
Expand Down
6 changes: 3 additions & 3 deletions massspring/example 3 (3d pyramid).py
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
import massspring as m

m.ax0 = -m.ge
m.ay0 = -m.ge
m.az0 = -m.ge
m.acceleration.x = -m.ge
m.acceleration.y = -m.ge
m.acceleration.z = -m.ge

n = 5

Expand Down
6 changes: 3 additions & 3 deletions massspring/example 4 (rope).py
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
import massspring as m

m.ax0 = 0
m.ay0 = -m.ge
m.az0 = 0
m.acceleration.x = 0
m.acceleration.y = -m.ge
m.acceleration.z = 0

NM = 20
d = 10
Expand Down
44 changes: 30 additions & 14 deletions massspring/massspring.py
Original file line number Diff line number Diff line change
Expand Up @@ -33,8 +33,6 @@
Cd is drag coefficient -> 0.47 for sphere; non-united
da is ro -> density of air; measured in kg*m^-3
where kg is kilograms, m is meters.
ax0, ay0 and az0 are acceleration in 3 directions; measured in m*s^-2
where m is meters, s is seconds.
ge is earth's gravity acceleration; measured in m*s^-2.
"""
dt = .001
Expand All @@ -43,9 +41,6 @@
k = 1 / (4 * pi * e0)
Cd = .47
da = 1.2
ax0 = 0
ay0 = 0
az0 = 0
ge = 9.8

# programming variables
Expand All @@ -66,9 +61,6 @@
WINW = 600 # window width (_)
WINH = 600 # window height (|)
WIND = 600 # window depth (.)
x0 = WINW / 2 # the graphical x of the 0 point
y0 = WINH / 2 # the graphical y of the 0 point
z0 = WIND / 2 # the graphical z of the 0 point

# colors [= ( R , G , B )]
WHITE = (255, 255, 255)
Expand Down Expand Up @@ -130,7 +122,11 @@ def hypot3d(x, y, z):


def sign(x):
return int(copysign(1, x))
if x < 0:
return -1
elif x > 0:
return 1
return 0


def warn_same_pos(m1, m2):
Expand Down Expand Up @@ -173,6 +169,26 @@ class NonElectricalConductive(Exception):
class SamePosition(Warning):
pass

# variable holders (namespaces)


class acceleration:
"""
x, y, z are default acceleration in 3 directions; measured in m*s^-2
where m is meters, s is seconds.
"""
x = 0
y = 0
z = 0


class position:
""" x, y, z are default 0 point position in pygame surface """
x = WINW // 2
y = WINH // 2
z = WIND // 2


# objects(physical meaning) classes
# mass

Expand Down Expand Up @@ -345,9 +361,9 @@ def move(self):
# attention! self.ax(), self.ay() and self.az() are
# calculated by vx/dt, vy/dt and vz/dt
ax, ay, az = self.fx / self.m, self.fy / self.m, self.fz / self.m
ax += ax0
ay += ay0
az += az0
ax += acceleration.x
ay += acceleration.y
az += acceleration.z
self.vx += ax * dt
self.vy += ay * dt
self.vz += az * dt
Expand All @@ -357,11 +373,11 @@ def move(self):

def show_pos_xy(self):
""" returns the position of the mass on the screen (just x and y) """
return int(x0 + self.x), int(y0 - self.y)
return int(position.x + self.x), int(position.y - self.y)

def show_pos_zy(self):
""" returns the position of the mass on the screen (just z and y) """
return int(z0 + self.z), int(y0 - self.y)
return int(position.z + self.z), int(position.y - self.y)

def show_color(self):
"""
Expand Down
4 changes: 2 additions & 2 deletions setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,15 +6,15 @@
setup(
name="massspring",
packages=["massspring"],
version='0.0.5',
version='0.0.6',
license='MIT',
description='A better mass-spring real world simulator with new types of forces(gravity, electricity, spring, collision, ...)',
long_description=long_description,
long_description_content_type="text/markdown",
author='Pooya Shams kolahi',
author_email='[email protected]',
url='https://github.com/pooya-shams/massspring',
download_url='https://github.com/pooya-shams/massspring/archive/v0.0.5.tar.gz',
download_url='https://github.com/pooya-shams/massspring/archive/v0.0.6.tar.gz',
keywords=["physics",
"physics-3d",
"physics-simulation",
Expand Down

0 comments on commit aedeb81

Please sign in to comment.