Previous
Bedtime Songs Bot
Create your own version of the famous arcade claw machine game using a robotic arm, an arcade claw grabber, and some fun items to pick from. Fine-tune every intricate detail, from the precision of each grab, to the claw’s strength, and even the aesthetics of your control interface.
In this tutorial, you will:
To build your own claw game machine, you need the following hardware:
To build your own claw game machine, you need the following software:
You will also need the following tools:
First, assemble the hardware components.
To support the arm and create a surface to hold the prizes, create a flat surface that also has supporting beams so you can securely mount the arm. You will need to build the table to a size that allows for the arm to reach anywhere on the table surface. For the xArm6, a table that is 4’x4’ works.
Cut one 2"x4" in half, creating two 48 inch sides.
Then take two 2x4s and cut them to four 46.5 inch lengths.
Next, place the 48 inch sides parallel to each other on a flat surface, and place the four other sides perpendicular to and between the 48 inch sides.
Attach the 48 inch sides to two of the shorter lengths with deck screws, forming a 4 foot by 4 foot square.
Then find the center of the 48 inch sides and mark this on both sides.
Measure 2.5 inches from that center mark in either direction, and mark those points as well.
These will be the points where we attach the remaining lengths, which will provide a support structure for the robotic arm.
The provided photo does not have the center lengths in the correct location - they are not yet centered or spaced.
Attach these lengths with deck screws.
Use the fiberboard as a tabletop and cut a prize exit hole into it. Center the exit, make it 10 inches wide, 8 inches long and cut the opening with a jigsaw. We used two 2 foot by 4 foot fiberboards, but ideally you can use one 4 foot by 4 foot fiberboard.
Glue and clamp the tabletop and let it dry overnight.
To finish the table construction, attach the supporting legs and mount the arm to the center of the table.
Cut four 2x4s to 78 inches to serve as legs for the cabinet.
Measure 30 inches from the top of each and mark this with the sliding square rule.
For each leg, line the 30 inch mark up with the top surface of the table, and drill two holes ⅜ inch holes through the leg and table surface.
Push two hex bolts through the drilled holes, add washers and nuts, and tighten.
For extra stability, cut two more 2x4s to 48 inch lengths, and mount these with deck screws on all four sides of the bottom of the cabinet, bridging the legs. You can also cut an additional 2x4 and mount it as a fifth leg, going from the center arm supports to the floor.
Finally, mount the xArm6 to the top of the table using the lag screws. Be sure that the lag screws sink into the 2x4 posts below, and that you are mounting the arm so that it is straight, with the X axis facing the player. You’ll need at least two people to ensure a smooth installation of the arm.
Next, configure your newly-built robot.
Add a new machine in the Viam app.
On the machine’s page, follow the setup instructions to install viam-server
on the computer you’re using for your project.
Wait until your machine has successfully connected to the Viam app.
Machines are organized into parts, where each part represents a computer (a single-board computer, desktop, laptop, or other computer) running viam-server
, the hardware components attached to it, and any services or other resources running on it.
Every machine has a main part which is automatically created when you create the machine.
Since you just created a new machine, your machine’s main part is already defined.
Multi-part machines also have one or more sub-parts representing additional computers running viam-server
.
If you have two computers within the same machine, you can use one as the main part and connect the other to it as a sub-part.
This is the approach this tutorial follows: you’ll run the motion planning service on a laptop and connect that laptop as a sub-part to your machine.
Technically you could configure all the components within one part, but motion planning is more performant when running on a computer like a macOS or Linux laptop running viam-server
.
Use the parts dropdown menu in the top banner of your machine’s page on the Viam app to add a new sub-part called planning
:
Follow the instructions on the Setup tab to install viam-server
on your development machine and connect to your robot’s sub-part.
For more information about parts, see Machine Architecture: Parts.
Now you are ready to configure the individual components in the Viam app. Navigate to the Config tab of your machine’s page and select your main part from the parts dropdown.
Click the Components subtab. Click the Create component button in the lower-left corner.
Add your board with type board
and model viam:raspberry-pi:rpi
if you are using a Raspberry Pi 4, Raspberry Pi 3 or Raspberry Pi Zero 2 W.
If you are using a Raspberry Pi 5, use the pi5
model.
Name your board myBoard
and click Create.
You can name your board whatever you want as long as you use the same name to refer to it in your code.
We named it myBoard
for simplicity.
This is the only component in the main machine.
Click Save config in the lower-left corner of the screen.
On the Raw JSON
tab, replace the configuration with the following JSON configuration for your board:
{
"components": [
{
"model": "pi",
"attributes": {},
"depends_on": [],
"name": "myBoard",
"type": "board"
}
]
}
Click Save config in the lower-left corner of the screen.
Use the parts dropdown menu to navigate to the planning
sub-part.
Click the Components subtab. Click the Create component button in the lower-left corner.
Add your arm with type arm
, and model xArm6
.
Name it myArm
and click Create.
Configure the arm component with the arm’s IP address in the host
field.
Click the {} (Switch to Advanced) button in the top right of the component panel to edit the component’s attributes directly with JSON.
Our arm’s address was 10.1.1.26
, but you should use the IP address for your arm.
For more information on xArm6 configuration, see Configure an xArm6 Arm.
Click Save config in the lower-left corner of the screen.
On the Raw JSON
tab, replace the configuration with the following JSON configuration for your arm:
{
"components": [
{
"model": "xArm6",
"type": "arm",
"namespace": "rdk",
"attributes": {
"acceleration_degs_per_sec_per_sec": 0,
"host": "10.1.1.26",
"speed_degs_per_sec": 0
},
"depends_on": [],
"frame": {
"parent": "world",
"translation": {
"x": 0,
"y": 0,
"z": 0
},
"orientation": {
"type": "ov_degrees",
"value": {
"th": 0,
"x": 0,
"y": 0,
"z": 1
}
}
},
"name": "myArm"
}
]
}
Click Save config in the bottom left corner of the screen.
Click Create component and add your gripper.
Choose type gripper
and model fake
.
Name it gripper
and click Create.
Set up a fake
model.
You will only use this as a placeholder for the size of the gripper to use with Viam’s frame system later.
Measure the claw’s height and width, and enter these values for the fake
model.
Ours was 120mm for the width and 180mm for the height.
Click Save config in the lower-left corner of the screen.
On the Raw JSON
tab, replace the configuration with the following JSON configuration for your arm and gripper:
{
"components": [
{
"model": "xArm6",
"type": "arm",
"namespace": "rdk",
"attributes": {
"acceleration_degs_per_sec_per_sec": 0,
"host": "10.1.1.26",
"speed_degs_per_sec": 0
},
"depends_on": [],
"frame": {
"parent": "world",
"translation": {
"x": 0,
"y": 0,
"z": 0
},
"orientation": {
"type": "ov_degrees",
"value": {
"th": 0,
"x": 0,
"y": 0,
"z": 1
}
}
},
"name": "myArm"
},
{
"frame": {
"translation": {
"y": 0,
"z": 0,
"x": 0
},
"orientation": {
"type": "ov_degrees",
"value": {
"z": 1,
"th": 0,
"x": 0,
"y": 0
}
},
"geometry": {
"r": 120,
"translation": {
"z": 180,
"x": 0,
"y": 0
}
},
"parent": "myArm"
},
"name": "gripper",
"model": "fake",
"type": "gripper",
"namespace": "rdk",
"attributes": {},
"depends_on": []
}
]
}
Click Save config in the bottom left corner of the screen.
Download the STL for the claw mount, and use a 3D printer to print the mount for in between the claw and the xArm6.
Using two M20 screws, attach the printed mount to the end of the arm and tighten.
Using hook-and-loop cable ties, run the claw’s cable along each segment of the arm to the arm base, making sure the cord is secure but with some slack to allow for movement.
The arcade claw is actuated when a solenoid is powered, acting as a magnet to pull the claw shut. For this project, we use a relay, which allows us to programmatically control when power flows to the claw’s solenoid.
COM
terminal on the relay.NO
terminal on the relay and the negative terminal on the barrel jack adapter.
This creates a normally open circuit, which means the circuit is normally not complete and the claw is therefore normally not powered.In order to control the claw through Viam, you will now wire the relay to the Raspberry Pi.
DC+
terminal on the relay to pin 2 (5v
) on the Pi.DC-
terminal on the relay to pin 6 (ground
) on the Pi.IN
terminal on the relay to pin 8 (GPIO
) on the Pi.Once viam-server
has started, you can test closing and opening the claw.
The Viam board
component gives us an interface for this.
Go to the Control tab for your machine, open the MyBoard
card, enter 8
next to Set under the GPIO interface, choose high
and click Set Pin State.
The relay will trigger the claw circuit to be closed when the GPIO pin state is set to high and your claw will close.
Now select low
and click Set Pin State again: the claw will open.
The claw game machine will use the motion service to plan its movements. To make sure the arm doesn’t hit the walls of the enclosure or the prize drop hole, you need to create representations of obstacles around the arm that the motion service can use when planning.
Obstacles are geometries located at a pose relative to some frame. When solving a motion plan with movable frames that contain inherent geometries, for example parts of the arm, the solved path is constrained such that none of those inherent geometries intersect with the obstacles.
You can pass information about the robot’s environment, including obstacles, to the Viam platform through a data structure named WorldState.
These obstacle representations are defined in a JSON. You can find the file we used in our claw game repository.
Represented in that file are obstacles for the prize drop hole, and each of the four walls based on measurements we took from our enclosure.
If the dimensions of your enclosure differ from ours, adjust your obstacles.json
file to match your enclosure.
The obstacles for our arm are configured in reference to the “world” frame which is defined as a , which is a special frame that represents the starting point for all other frames in the robot’s world.
The list of obstacles are defined in a WorldState
object, which is passed as an argument in each move() call.
If the arm is not mounted exactly perpendicular to the x/y axis of the enclosure, you can adjust the theta (th) of the arm within the arm component configuration by a number of degrees to compensate. Obstacles can then be configured as if the arm were straight in the enclosure. See the frame system documentation for more information.
By moving the arm through the Control tab in the Viam app, you can determine the arm’s ideal home pose, which is the position the arm starts each game and the one it returns to after making a grab.
You can also determine the desired distance between the lateral plane and the pick up level, which is how you determine how far to drop the grabber.
If your enclosure is sufficiently different from ours, these values may be different: it’s best to test your own measurements by moving your arm around your enclosure from the Control tab.
For our enclosure, the hole pose and dimensions are as follows:
hole_origin = Pose(x=470, y=125, z=0, o_x=0, o_y=0, o_z=1, theta=15)
hole_dims = Vector3(x=250, y=400, z=300)
Additionally, the home position origin and dimensions where the arm needs to be in order to drop the prize are as follows:
home_pose = Pose(x=390.0, y=105.0, z=500.0, o_x=0, o_y=0, o_z=-1, theta=0)
home_pose_in_frame = PoseInFrame(reference_frame="world", pose=home_pose)
The floor level where the claw drops has different numbers for x
and y
since you can pick items from anywhere on the plane, but the Z axis is always the same since you always drop to the same level.
We tested between 240
and 280
for this level, but you can adjust it to your liking.
Use git
to clone the Claw Game project repository:
git clone https://github.com/viam-labs/claw-game
The claw game repository includes the Python test script CLI-test.py
, which connects to your robot, creates an orientation constraint so the last arm joint is always facing down, and provides functions to:
This script provides an interface to run a single move command, or to run move commands in sequences. You can adjust the code to your liking.
In the following, you can read about what the different sections of the code do.
First, the code imports the required packages:
import asyncio
import argparse
import json
from viam.robot.client import RobotClient
from viam.rpc.dial import Credentials, DialOptions
from viam.components.board import Board
from viam.components.arm import Arm
from viam.services.motion import MotionClient
from viam.proto.common import Pose, PoseInFrame, Vector3, Geometry, \
GeometriesInFrame, RectangularPrism, WorldState
from viam.proto.service.motion import Constraints, LinearConstraint, \
OrientationConstraint
Then it creates an argument parser, defining required and optional arguments to create a user-friendly command line interface:
parser = argparse.ArgumentParser()
parser.add_argument('--command', type=str, required=True)
parser.add_argument('--sequence', type=str, required=False)
parser.add_argument('--location', type=str, required=True)
parser.add_argument('--password', type=str, required=True)
args = parser.parse_args()
Next, it creates some constants to define how much the arm should move at each call, where the home position should be and where it should drop to grab the prizes. You can tweak these numbers as needed.
# The amount to move in mm for each command forward, backward, left, right
move_increment = 50
# Define home position to return to
home_plane = 500.0
home_pose = Pose(x=390.0, y=105.0, z=home_plane, o_x=0, o_y=0, o_z=-1, theta=0)
# Define plane to grab on
grab_plane = 240.0
Then we define the constraints - in this case we are using an orientation constraint. The orientation constraint places a restriction on the orientation change during a motion, as the arm in a claw game should always face down so the gripper is always in a position where is can descend and grab a prize:
constraints = Constraints(orientation_constraint=[OrientationConstraint()])
Next the code imports the world_state
representing the robot’s physical environment:
def get_world_state():
with open('obstacles.json', 'r') as f:
geometries = json.load(f)
world_state_obstacles = []
for geometry in geometries:
center = geometry['translation']
orientation = geometry['orientation']['value']
center = Pose(
x=center['x'],
y=center['y'],
z=center['z'],
o_x=orientation['x'],
o_y=orientation['y'],
o_z=orientation['z'],
theta=orientation['th'],
)
dims = Vector3(x=geometry['x'], y=geometry['y'], z=geometry['z'])
world_state_obstacles.append(
Geometry(center=center,
box=RectangularPrism(dims_mm=dims),
label=geometry['label']))
obstacles_in_frame = GeometriesInFrame(reference_frame="world",
geometries=world_state_obstacles)
return WorldState(obstacles=[obstacles_in_frame])
world_state = get_world_state()
Next, the code defines a grab function to use GPIO to open and close the gripper by setting the Raspberry Pi pin state to true
or false
:
async def grab(board, doGrab):
# Note that the pin supplied is a placeholder. Please change this to a
# valid pin you are using.
pin = await board.gpio_pin_by_name('8')
if doGrab:
# opens the gripper/release
await pin.set(True)
else:
# closes the gripper/grab
await pin.set(False)
Lastly, the code defines the functions move_absolute()
, home()
, move_to_offset()
and move_z()
, which construct new pose requests to send to the motion service.
async def move_absolute(arm, motion_service, pose):
destination = PoseInFrame(reference_frame="world", pose=pose)
await motion_service.move(component_name=arm,
destination=destination,
world_state=world_state,
constraints=constraints)
async def home(arm, motion_service):
# Makes sure to first move the arm up in z axis
await move_z(arm, motion_service, 500)
# Generate a sample "home" pose around the drop hole and demonstrate motion
home_pose_in_frame = PoseInFrame(reference_frame="world", pose=home_pose)
await motion_service.move(component_name=arm,
destination=home_pose_in_frame,
world_state=world_state,
constraints=constraints)
async def move_to_offset(arm, motion_service, offset):
# Get current position of the arm
current_position = await motion_service.get_pose(
component_name=arm,
destination_frame="",
supplemental_transforms=None)
print('current position: ', current_position)
# Calculate new pose to move the arm to
pose = Pose(
x=current_position.pose.x + offset.x,
y=current_position.pose.y + offset.y,
z=current_position.pose.z + offset.z,
o_x=0,
o_y=0,
o_z=-1, # negative z means claw will point down
theta=0
)
print('moving to position: ', pose)
# Move arm
destination = PoseInFrame(reference_frame="world", pose=pose)
await motion_service.move(component_name=arm,
destination=destination,
world_state=world_state,
constraints=constraints)
async def move_z(arm, motion_service, z):
# Get current position of the arm
current_position = await motion_service.get_pose(
component_name=arm,
destination_frame="",
supplemental_transforms=None)
print('current_position: ', current_position)
# Construct new pose to get to desired z position
pose = Pose(
x=current_position.pose.x,
y=current_position.pose.y,
z=z,
o_x=0,
o_y=0,
o_z=-1, # negative z means claw will point down
theta=0
)
print('moving to position: ', pose)
# Move arm
destination = PoseInFrame(reference_frame="world", pose=pose)
await motion_service.move(component_name=arm,
destination=destination,
world_state=world_state,
constraints=constraints)
The main()
function initializes different resources and then handles command line arguments to move the arm in a sequence for testing and debugging:
async def main():
robot = await connect()
print('Resources:')
print(robot.resource_names)
# Pose using motion service, grabbing the service from local computer
motion_service = MotionClient.from_robot(robot, "planning:builtin")
# myBoard
my_board = Board.from_robot(robot, "myBoard")
# my Subpart name, arm
my_arm_resource = Arm.get_resource_name("planning:myArm")
my_arm_resource.name = "myArm"
print("arm resource", my_arm_resource)
commands = [args.command]
if args.command == "sequence":
commands = args.sequence.split(",")
for command in commands:
if command == "drop":
print("will drop")
# Moves the arm's z position to grab plane
await move_z(my_arm_resource, motion_service, grab_plane)
if command == "up":
print("will go up")
# Moves the arm's z position to home plane
await move_z(my_arm_resource, motion_service, home_plane)
if command == "home":
print("will return home")
# Goes to home position
await home(my_arm_resource, motion_service)
if command == "left":
print("will move left")
# Moves the arm's y position to left
await move_to_offset(my_arm_resource,
motion_service,
Vector3(x=0, y=-move_increment, z=0))
if command == "right":
print("will move right")
# Moves the arm's y position to right
await move_to_offset(my_arm_resource,
motion_service,
Vector3(x=0, y=move_increment, z=0))
if command == "forward":
print("will move forward")
# Moves the arm's x position to forward
await move_to_offset(my_arm_resource,
motion_service,
Vector3(x=move_increment, y=0, z=0))
if command == "backward":
print("will move backward")
# Moves the arm's x position to backwards
await move_to_offset(my_arm_resource,
motion_service,
Vector3(x=-move_increment, y=0, z=0))
if command == "grab":
print("will grab")
# Closes the gripper
await grab(my_board, True)
if command == "release":
print("will release")
# Opens the gripper
await grab(my_board, False)
if command == "sleep":
print("will sleep one second")
await asyncio.sleep(1)
if command == "test":
print("""will move to the test position, drop, grab, return home
and release""")
await move_absolute(
my_arm_resource,
motion_service,
Pose(x=0.0, y=380, z=home_plane, o_x=0, o_y=0, o_z=-1, theta=0)
)
await move_z(my_arm_resource, motion_service, grab_plane)
await grab(my_board, True)
await home(my_arm_resource, motion_service)
await grab(my_board, False)
# Don't forget to close the robot when you're done!
await robot.close()
if __name__ == '__main__':
asyncio.run(main())
Use CLI-test.py
to run these commands from the command line, for example:
python3 CLI-test.py --password mypass --location mylocation --command grab
Or, you can run sequences of these commands together, for example:
python3 CLI-test.py --password mypass --location mylocation --command sequence \
--sequence grab,sleep,release,sleep,grab,sleep,release
Now that the arm is set up and you have a CLI script you can use for testing - try testing the motion and claw grab with different items. Consider that the size, weight, and shape of the item being grabbed will affect the claw’s ability to successfully grab it. You can also try increasing the voltage of the claw’s power supply if you feel like it is struggling to pick up objects, especially if the objects are heavy (but don’t exceed the limits of the relay).
We tested with different Viam swag items such as t-shirts and hats, but found that foam balls worked the best for us.
In order to house the prizes, you’ll need to enclose three of the sides (front, left, and right sides) of the upper cabinet with plexiglass.
Now you can fill the enclosure with prizes! Fill the enclosure generously, but make sure that the prizes don’t overflow from the prize exit hole or the backstop.
We ended up using over 600 foam balls.
Now that you’ve built out and tested all of the main functionality of your claw game, it’s time to design a custom interface to control it. This tutorial shows you how to create the interfaces for a touch screen tablet (in our project, we mounted an iPad to the facade of our claw machine prototype). The interface is hosted on a separate macOS or Linux computer locally and then accessed on the tablet using your computer’s local address.
Within the project code repository, the static
folder contains the frontend code, including styling and HTML.
To use the Viam TypeScript SDK you must install the dependencies in your main project folder.
Make sure you have the latest version of Node.JS installed.
Once you have installed Node, you can now fetch all dependencies, including the Viam TypeScript SDK, by running the following command in your project directory:
npm install
Determine your machine’s location secret and host address by navigating to the Code sample tab in the Viam app. Use the host address of your main machine part, as it will reference all parts of your machine.
To show your machine’s API key in the sample code, toggle Include API key.
We strongly recommend that you add your API key and machine address as an environment variable. Anyone with these secrets can access your machine, and the computer running your machine.
Then, run the following command to start the custom TypeScript interface, inserting your machine’s API key as the argument for VIAM_API_KEY
,your machine’s API key ID as the argument for VIAM_API_KEY_ID
, and your main part host address for VIAM_LOCATION
:
export VIAM_LOCATION=<mylocation>;VIAM_API_KEY=<myapikey>;VIAM_API_KEY_ID=<myapikeyid>;npm run start-simple
Providing these values in this fashion ensures that your security keys are not accessible to others when publishing your code.
Visit localhost:8000
in a browser.
Press the buttons to execute the control commands defined in main.ts and watch your robot arm move around using the simple user interface.
The TypeScript app reads in the obstacles defined in the same obstacles.json
JSON file that you used with the Python testing script, and creates a world state.
The TypeScript interface code includes functions that define movements in each direction on a plane relative to the base of the game.
This includes forward, back, left, right, and home functions.
If you look through the code you will notice each of these functions follow the same convention by using the Motion Client method move()
and passing in all of the obstacles defined globally as well as the WorldState
in each function:
async function right(motionClient: MotionClient, armClient: ArmClient) {
if (ignoreInterrupts && await armClient.isMoving()) { return }
// Get current position of the arm
console.log('im trying to print the current position')
let currentPosition = await motionClient.getPose(myArm, 'world', [])
console.log('current position:' + JSON.stringify(currentPosition))
// Define the new right position by adding to the y value incrementally so that your robot arm will move
let rightPose: Pose = {
x: currentPosition.pose!.x,
y: currentPosition.pose!.y + moveDistance,
z: currentPosition.pose!.z,
theta: 0,
oX: 0,
oY: 0,
oZ: -1
};
let rightPoseInFrame: SDK.PoseInFrame ={
referenceFrame: "world",
pose: rightPose
}
// Call the move method with the new rightPose values to move your arm and pass in your machine component name, WorldState, and constraints
await motionClient.move(rightPoseInFrame, myArm, myWorldState, constraints)
}
For each function that moves the arm in the Python script, the front end has corresponding buttons that trigger these actions in your browser environment. Each of these button functions is initialized the same way for each corresponding function.
For example, the following code handles movement to the right:
// Creating a button function that corresponds to the HTML element that will show on your webpage
function rightbutton() {
return <HTMLButtonElement>document.getElementById('right-button');
}
// Since we are designing for a touch screen, we are using a touchstart event for our button command
rightbutton().ontouchstart = async () => {
rightHandler()
};
// This function is where the move function is called as well as adding some
// timeout and color changing media queries
async function rightHandler() {
if (rightbutton().classList.contains('error')) return;
try {
await right(motionClient, armClient);
if (rightbutton().classList.contains('custom-box-shadow-active')) {await rightHandler()};
} catch (error) {
console.log(error);
rightbutton().classList.add('error');
rightbutton()?.querySelector('svg')?.classList.add('icon');
setTimeout( () => { rightbutton().classList.remove('error'); }, 3000 )
}
}
As with the Python test script, the TypeScript code also controls our arcade claw with GPIO on a Raspberry Pi.
You control this in TypeScript by setting the pin state to high
or low
on the board component by using the setGPIO()
method in the BoardClient
.
For example, here is the grab()
function that calls the setGPIO()
method.
// Global variable: GPIO pin used for claw relay on the board
const grabberPin = '8'
// Print out the pin state and set the GPIO state with a boolean value
async function grab(boardClient: BoardClient) {
try {
console.log(await boardClient.getGPIO(grabberPin));
console.log('i`m grabbin');
await boardClient.setGPIO(grabberPin, true);
}
}
In this tutorial, you learned how to:
For some next steps, you could:
Use the advanced interface included in the project repository to leverage the motion service for larger, more complex arm movement within the enclosure.
Add a camera and use the vision service to add color detection, or use an ML model to determine grab success rate and create a score counter.
Design a hard mode where the prizes are shuffled around with the arm every few attempts.
Add a camera and extend the interface to allow folks from anywhere in the world to play the claw game and win.
Viam already allows you to securely control machines from anywhere.
Or you can head over to our Tutorials page and try one of our other tutorials to continue building other robots.
Was this page helpful?
Glad to hear it! If you have any other feedback please let us know:
We're sorry about that. To help us improve, please tell us what we can do better:
Thank you!