Following Microsoft's purchase of Github, this project has migrated to gitlab.
Simulation Software For Oliver's Machine
This Processing application simulates Oliver's machine.
NEW : A web version is now available so you can run it in your browser. Thank you, processing.js.
To run the stand alone version, load it into the Processing 3 IDE, then click the "play" arrow under Processing's main menu.
A window will appear with the simulation running.
A key stroke or mouse click will pause/restart the simulation. If image saving is enabled, a copy of the current display will be saved as a png file
Simulation parameters that can be set by the user are:
- Small disk sizes
- Initial position of radius arms (point on small disk connected to pen arm - positve is counter-clockwise)
- distance between the small disks (center to center)
- big wheel size
- wheel speeds
- pen holder arm lengths
- pen color
- simulation speed
- directory in which to save image
- enable/disable image saving on pause
These are all set in the Config¨ tab of the IDE.
There should be no need to make changes in any other tab.
Some screencast demos are available:
The real machine comprises 3 rotating disks and 2 arms joined at one end to hold the pen and the other ends are connected to points on the smaller rotating disks. The 3rd disk is much larger and carries the canvas on which the pen writes. The disks and arms are arranged so that the pen sits on the canvas.
As the 3 disks turn, the pen moves on the canvas, leaving a curved track.
Changing the disk rotation speeds, and/or the small disk radii and/or the arm lengths will change the pattern produced.
Conceptually, the 2 pen arms are radii of circles which intersect at the pen. The center of each of these circles is the other end point of the pen arm. Thus, the center of each circle rotates around one of the small disks.
An illustration is provided in this diagram.
It is straightforward to compute coordinates of the center of each of the intersecting circles based on the rotation speed of the disks. Once the center points are known, it is again straightforward to compute the intersection point.
By making the simulation take short steps, we can approximate continuous drawing by making a line from the current point to the previous one.
However, the canvas is rotating about its center as well. That means that each point will rotate about the canvas disk center. So, as this rotation takes place, the program has to compute the complex trajectory of each previously drawn point. This would be a nightmare, if we didn't use a trick.
Our programming environment provides a Cartesian coordinate system for all our drawing. The origin of this coordinate system is the upper left corner of the window, with positive Y downwards and positive X to the right.
Our trick will be to create another coordinate system with the origin at the center of the canvas disk. Now, in this coordinate system the computation of a point's trajectory is straightforward. Also, the conversion between the 2 coordinate systems is straightforward as well.
Once we understand what is going on, the code is very straightforward.
The Object paradigm was employed and the execution is performed by the collaboration of instances of classes. These classes are:
- App: the application itself, that is called at the top level by the basic processing code, i.e. setup() and draw()
- Circle: implementation of all that deals with circles, including their creation and computation of intersection points
- Point: management of our points
- Config: a place to put user config data
In addition to the classes, the top level processing code is held in the file MachineRunner.pde.
So how does it work?
- Processing starts up and executes the code in the MachineRunner.pde file. An instance of the Config class is created, and the setup function creates an instance of the App class. draw() simply calls the App.draw() method. Mouse clicks and key presses are also handed in the MachineRunner.pde file.
- At instantiation, the App creates the disks (instances of Circle) and the end points of the pen arms (instances of Point) as well as an ArrayList of Point instances to record each point. It alwso draws the 3 disks.
- Processing then repeatedly calls app.draw(). This method first checks to see if simulation is paused or not. Then, if not paused, and depending on the debugging parameters, it will draw the disks, update the radius arm end points, and compute the pen point. Finally, depending on the debugging parameters it draws the arms and/or the base line, and or the points. At the very end it increments the iteration counter.
- A subtle part of the code is the drawing of the points in the method App.drawPenPoints. After setting the stroke and fill, we push the current coordinate system onto the matrix stack, then translate our coordinate system to the center of the canvas disk. If there are already points in the ArrayList, we take them one by one, rotate them about the origin (in the new coordinate system), then take the next point, rotate it, and draw a line from the previous point to it. Incredibly, this works! We then make the current point the last point and iterate through all the previously computed points. Then, we pop the coordinate matrix stack to get back to the original coordinate system and display the current pen point, which is still based on the original coordinates. At the very end of the method, we add a translated copy of the pen point to the ArrayList. We are good to go!
All the remaining functionality is easily understood by examining the code.