6.1. Analytical kinematics
For some applications it is useful to retrieve more than one solution for an inverse kinematics request, this can be achieved through analytic solvers. For a general 6-DoF robot (industrial arm with 6 revolute joints), up to eight possible solutions can be found. (More, if we include joint positions ± 360 degrees).
The resulting eight solutions have an order. That means that if you call IK on
two subsequent frames and compare the 8 configurations of the first frame with
the 8 configurations of the second frame at their respective indices, then these
configurations are “close” to one another. For this reason, for certain use
cases, e.g. for cartesian path planning, it makes sense to keep the order of
solutions. This can be achieved by setting the optional parameter keep_order
to True
. The configurations that are in collision or outside joint
boundaries are then not removed from the list of solutions, they are set to
None
.
We currently support the following robotic arms:
- Offset-Wrist manipulators:
UR3, UR3e
UR5, UR5e
UR10, UR10e
- Spherical-Wrist manipulators:
Staubli_TX260L
ABB_IRB4600_40_255
6.1.1. Links
6.1.2. Inverse kinematics
The inverse kinematics function calculates the joint states required for the end-effector to reach a certain target pose.
Here is an example of using AnalyticalInverseKinematics
:
from compas.geometry import Frame
from compas_fab.robots.ur5 import Robot
from compas_fab.backends.kinematics import AnalyticalInverseKinematics
ik = AnalyticalInverseKinematics()
robot = Robot()
frame_WCF = Frame((0.381, 0.093, 0.382), (0.371, -0.292, -0.882), (0.113, 0.956, -0.269))
for jp, jn in ik.inverse_kinematics(robot, frame_WCF, options={'solver': 'ur5'}): # knows that we need the IK for the UR5 robot
print(jp)
The above solutions, could however be in collision. In order to check for collisions,
we have to use a client. See below the example for using the PyBulletClient
from compas.geometry import Frame
from compas.robots import LocalPackageMeshLoader
import compas_fab
from compas_fab.backends.kinematics import AnalyticalInverseKinematics
from compas_fab.backends import PyBulletClient
urdf_filename = compas_fab.get('universal_robot/ur_description/urdf/ur5.urdf')
srdf_filename = compas_fab.get('universal_robot/ur5_moveit_config/config/ur5.srdf')
frame_WCF = Frame((0.381, 0.093, 0.382), (0.371, -0.292, -0.882), (0.113, 0.956, -0.269))
with PyBulletClient(connection_type='direct') as client:
# Load UR5
loader = LocalPackageMeshLoader(compas_fab.get('universal_robot'), 'ur_description')
robot = client.load_robot(urdf_filename, [loader])
client.load_semantics(robot, srdf_filename)
ik = AnalyticalInverseKinematics(client)
# set a new IK function
client.inverse_kinematics = ik.inverse_kinematics
options = {"solver": "ur5", "check_collision": True, "keep_order": True}
for solution in robot.iter_inverse_kinematics(frame_WCF, options=options):
print(solution)
Or, alternatively, use the AnalyticalPyBulletClient
:
import compas_fab
from compas.geometry import Frame
from compas.robots import LocalPackageMeshLoader
from compas_fab.backends.kinematics.client import AnalyticalPyBulletClient
urdf_filename = compas_fab.get('universal_robot/ur_description/urdf/ur5.urdf')
srdf_filename = compas_fab.get('universal_robot/ur5_moveit_config/config/ur5.srdf')
with AnalyticalPyBulletClient(connection_type='direct') as client:
loader = LocalPackageMeshLoader(compas_fab.get('universal_robot'), 'ur_description')
robot = client.load_robot(urdf_filename, [loader])
client.load_semantics(robot, srdf_filename)
frame_WCF = Frame((0.381, 0.093, 0.382), (0.371, -0.292, -0.882), (0.113, 0.956, -0.269))
options = {"solver": "ur5", "check_collision": True, "keep_order": True}
# 8 solutions, `None` are those in collision
for config in robot.iter_inverse_kinematics(frame_WCF, options=options):
print(config)
We can also use the AnalyticalPyBulletClient
to calculate a cartesian path:
import matplotlib.pyplot as plt
from compas.geometry import Frame
from compas.robots import LocalPackageMeshLoader
import compas_fab
from compas_fab.backends.kinematics.client import AnalyticalPyBulletClient
urdf_filename = compas_fab.get('universal_robot/ur_description/urdf/ur5.urdf')
srdf_filename = compas_fab.get('universal_robot/ur5_moveit_config/config/ur5.srdf')
frames_WCF = [Frame((0.407, 0.073, 0.320), (0.922, 0.000, 0.388), (0.113, 0.956, -0.269)),
Frame((0.404, 0.057, 0.324), (0.919, 0.000, 0.394), (0.090, 0.974, -0.210)),
Frame((0.390, 0.064, 0.315), (0.891, 0.000, 0.454), (0.116, 0.967, -0.228)),
Frame((0.388, 0.079, 0.309), (0.881, 0.000, 0.473), (0.149, 0.949, -0.278)),
Frame((0.376, 0.087, 0.299), (0.850, 0.000, 0.528), (0.184, 0.937, -0.296))]
with AnalyticalPyBulletClient(connection_type='direct') as client:
loader = LocalPackageMeshLoader(compas_fab.get('universal_robot'), 'ur_description')
robot = client.load_robot(urdf_filename, [loader])
client.load_semantics(robot, srdf_filename)
options = {"solver": "ur5", "check_collision": True}
start_configuration = list(robot.iter_inverse_kinematics(frames_WCF[0], options=options))[-1]
trajectory = robot.plan_cartesian_motion(frames_WCF, start_configuration=start_configuration, options=options)
assert(trajectory.fraction == 1.)
j = [c.joint_values for c in trajectory.points]
plt.plot(j)
plt.show()