334 lines
14 KiB
Python
334 lines
14 KiB
Python
from Robot.UR.URRobot import URRobot
|
|
from time import sleep
|
|
import math
|
|
import threading
|
|
from Robot.UR.URModbusServer import ModbusError
|
|
|
|
class URSentry:
|
|
def __init__(self, host):
|
|
self.robot = URRobot(host)
|
|
self.sentry_pose = [0.785, -2.094, 0.96, -0.436, -1.571, 1.326]
|
|
#self.forward_pose = [1.571, -1.949, 1.974, -2.548, -1.571, 1.326]
|
|
self.imposing_pose = [1.571, -1.41, 1.411, -2.859, -1.604, 1.326]
|
|
self.looking_down_pose = [1.571, -2.3, 2.323, -2.452, -1.604, 1.326]
|
|
self.middle_point_pose = [(self.imposing_pose[i] + self.looking_down_pose[i]) / 2 for i in range(6)]
|
|
self.robot_speed = [0, 0, 0, 0, 0, 0]
|
|
self.detections = []
|
|
|
|
# Flags
|
|
self.ESTOP = False
|
|
|
|
self.await_stop = True
|
|
self.await_stop_ticks = 0
|
|
|
|
self.send_to_zero_on_stop = False
|
|
|
|
self.has_detected_once = False
|
|
|
|
self.none_input_ticks = 0
|
|
self.is_on_sentry_mode = True
|
|
self.has_initialized = False
|
|
|
|
# Smooth stopping
|
|
self.smooth_stop_delayed_call = None
|
|
|
|
self.modbus_healthy = self.Modbus_check()
|
|
|
|
def Modbus_check(self):
|
|
try:
|
|
self.robot.get_joint_angles()
|
|
self.robot.get_joint_speeds()
|
|
except KeyboardInterrupt as ki:
|
|
raise ki
|
|
except Exception as e:
|
|
print(e)
|
|
print("########################################################")
|
|
print("#### MODBUS HEALTHCHECK FAILED ####")
|
|
print("########################################################")
|
|
return False
|
|
print("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~")
|
|
print("~~~~ MODBUS HEALTHCHECK PASSED ~~~~")
|
|
print("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~")
|
|
return True
|
|
|
|
def initialize_pose(self):
|
|
self.await_stop = True
|
|
self.sentry_position()
|
|
self.has_initialized = True
|
|
self.is_on_sentry_mode = True
|
|
|
|
def get_joint_angles(self) -> "list[float]":
|
|
"""
|
|
Get the current joint angles of the robot in degrees
|
|
"""
|
|
return self.robot.get_joint_angles()
|
|
|
|
def sentry_position(self, a=0.5, v=1.5):
|
|
"""
|
|
Return the robot to the sentry position
|
|
"""
|
|
self.robot.movej(self.sentry_pose, a=0.5, v=1.5)
|
|
|
|
def forward_position(self, a=0.5, v=1.5):
|
|
self.robot.movej(self.imposing_pose, a, v)
|
|
|
|
def update_detections(self, detections):
|
|
self.detections = detections
|
|
|
|
def forward_position_to_base_angle_degrees(self, base_angle, a=0.5, v=1.5):
|
|
pose = self.middle_point_pose.copy()
|
|
pose[0] = math.radians(base_angle)
|
|
self.robot.movej(pose, a, v)
|
|
|
|
# def set_base_speed(self, speed):
|
|
# self.robot_speed[0] = speed
|
|
|
|
# def set_neck_speed(self, speed):
|
|
# self.robot_speed[4] = speed
|
|
|
|
def smooth_stop(self):
|
|
self.robot_speed = [0, 0, 0, 0, 0, 0]
|
|
self.robot.speedj([0, 0, 0, 0, 0, 0], 1.5)
|
|
print("xxxxxxxxx Smooth stopping xxxxxxxxx")
|
|
|
|
def lerp(self, a, b, t):
|
|
return a + t * (b - a)
|
|
|
|
|
|
def control_robot(self, joystick_pos: list[float] | None):
|
|
"""
|
|
Control the robot based on a joystick input.
|
|
"""
|
|
#print("Joystick pos: ", joystick_pos)
|
|
# Check for flags that would block control due to things happening
|
|
|
|
# Initialize robot if it hasn't already
|
|
if not self.modbus_healthy:
|
|
self.modbus_healthy = self.Modbus_check()
|
|
return
|
|
|
|
try:
|
|
|
|
if not self.has_initialized:
|
|
self.initialize_pose()
|
|
return
|
|
|
|
# If we are awaiting a stop, we do not control the robot until it remains at a standstill for a certain amount of ticks
|
|
ticks_to_wait = 6
|
|
if self.await_stop:
|
|
joint_speeds = self.robot.get_joint_speeds()
|
|
print("Awaiting stop -------------------- speeds: ", joint_speeds)
|
|
if all([speed == 0 for speed in joint_speeds]):
|
|
self.await_stop_ticks += 1
|
|
if self.await_stop_ticks >= ticks_to_wait:
|
|
self.await_stop = False
|
|
self.await_stop_ticks = 0
|
|
return
|
|
|
|
# If we are awaiting a stop to send to zero, we do so right after it passes the await_stop check
|
|
if self.send_to_zero_on_stop:
|
|
print("Sending to zero ~~~~~~~~~~~~~~~~~~~~~~~~")
|
|
self.await_stop = True
|
|
self.send_to_zero_on_stop = False
|
|
self.has_detected_once = False
|
|
self.forward_position_to_base_angle_degrees(0)
|
|
return
|
|
|
|
# If the joystick is None, we return to sentry mode after a certain amount of ticks
|
|
# We only do this if we have detected something at least once, as the camera gets buggy after fast movements
|
|
if joystick_pos is None:
|
|
if not self.is_on_sentry_mode:
|
|
ticks_for_return_to_sentry = 600 if self.has_detected_once else 3000
|
|
self.none_input_ticks += 1
|
|
print("None input ticks: ", self.none_input_ticks, " / ", ticks_for_return_to_sentry)
|
|
if self.none_input_ticks > ticks_for_return_to_sentry:
|
|
self.none_input_ticks = 0
|
|
self.is_on_sentry_mode = True
|
|
self.await_stop = True
|
|
self.sentry_position()
|
|
else:
|
|
self.move_robot_with_joystick(0, 0)
|
|
return
|
|
|
|
# If we have detected something, we reset the none input ticks
|
|
|
|
# If all flags are clear, we can control the robot
|
|
if self.is_on_sentry_mode:
|
|
self.awake_from_sentry_mode(joystick_pos[0], joystick_pos[1])
|
|
self.has_detected_once = False
|
|
else:
|
|
# is on forward mode
|
|
self.move_robot_with_joystick(joystick_pos[0], joystick_pos[1])
|
|
self.has_detected_once = True
|
|
self.none_input_ticks = 0
|
|
except ModbusError as me:
|
|
print("[Modbus error]: ", me)
|
|
self.modbus_healthy = False
|
|
try:
|
|
self.smooth_stop()
|
|
except:
|
|
pass
|
|
return
|
|
|
|
|
|
def awake_from_sentry_mode(self, joystick_pos_x: float, joystick_pos_y: float):
|
|
if joystick_pos_x == 0 and joystick_pos_y == 0:
|
|
return
|
|
|
|
theta_rad = math.atan2(joystick_pos_x, -joystick_pos_y)
|
|
theta_deg = math.degrees(theta_rad) - 45
|
|
print("Theta: ", theta_deg)
|
|
|
|
self.await_stop = True
|
|
self.forward_position_to_base_angle_degrees(-theta_deg, 1.5, 0.8)
|
|
self.is_on_sentry_mode = False
|
|
|
|
|
|
def move_robot_with_joystick(self, joystick_pos_x: float = 0, joystick_pos_y: float = 0):
|
|
"""
|
|
Move the robot based on a joystick input,
|
|
where the the center is (0, 0) and the bottom right corner is (1, 1).
|
|
|
|
Horizontally, the speed of the base is adjusted on how far left or right the detection is.
|
|
We take into account the maximum rotation limit, trying to slow down when nearing +-360 degrees,
|
|
and trying to never reach that point. If needed, the robot should attempt to do a full rotation the other way to keep following
|
|
the target.
|
|
This can be changed so we first change the horizontal "neck" (wrist_2) until it reaches a limit, and then rotate the base,
|
|
which would result in a more natural movement
|
|
|
|
Vertically, the speed of the vertical "neck" (wrist_1) is adjusted based on how far up or down the detection is.
|
|
We limit it's rotation to +- 45 degrees, as we do not need more to follow the target.
|
|
This can be changed in the future to acomplish more natural movements, by also adjusting the speed of the shoulder joint.
|
|
|
|
Movement is done entirely through speed control, which should result in a more fluid movement compared to pose control.
|
|
|
|
If flag await_stop is set True, we check if the current speed is 0 wait until all joints are stopped before moving again.
|
|
|
|
"""
|
|
# Reverse the x axis
|
|
joystick_pos_x = -joystick_pos_x
|
|
|
|
# Limit the y axis between +-0.5 to prevent speeding down too fast
|
|
joystick_pos_y = max(-0.5, min(0.5, joystick_pos_y))
|
|
|
|
# We omit all processing until the robot has stopped moving for a certain amount of calls ('ticks')
|
|
|
|
# print("Base Speed: ", self.robot_speed[0])
|
|
|
|
|
|
# Cancel smooth stopping due to no input
|
|
if self.smooth_stop_delayed_call is not None:
|
|
self.smooth_stop_delayed_call.cancel()
|
|
self.smooth_stop_delayed_call = None
|
|
|
|
# # Center detections, so that the center of the image is (0, 0)
|
|
# # We also invert the y axis, so that the upper part of the image is positive
|
|
# # Thus: lower left corner is (-500, -500) and upper right corner is (500, 500)
|
|
# joystick_pos_x -= 500
|
|
# joystick_pos_y -= 500
|
|
# joystick_pos_y = -joystick_pos_y
|
|
|
|
# Deadzones where we still consider the target in the middle, to avoid jittering
|
|
horizontal_dead_zone_radius = 0.05
|
|
vertical_dead_zone_radius = 0.01
|
|
|
|
# Speeds for the base and neck joints
|
|
base_max_speed = 1.5
|
|
base_min_speed = 0.0
|
|
vertical_speed = 2.0
|
|
|
|
base_acceleration = 1.5
|
|
|
|
# Maximum and minimum angles
|
|
base_max_angle = 315
|
|
base_danger_angle = 340
|
|
|
|
movement_happened = False
|
|
current_pose = None
|
|
#### Horizontal movement ####
|
|
if abs(joystick_pos_x) < horizontal_dead_zone_radius:
|
|
# We are in the deadzone, so we stop moving it
|
|
if abs(joystick_pos_y) < vertical_dead_zone_radius:
|
|
# Both deadzones, so we check if the robot is already stopped. If it is, return and do nothing
|
|
if all([speed == 0 for speed in self.robot_speed]):
|
|
return
|
|
self.robot_speed[0] = 0
|
|
|
|
else:
|
|
movement_happened = True
|
|
current_pose = self.get_joint_angles()
|
|
base_angle = round(math.degrees(current_pose[0]), 3)
|
|
# print("Base angle: ", base_angle)
|
|
# Check if we are past the maximum angle
|
|
if abs(base_angle) > base_max_angle:
|
|
# We are past the maximum angle, so we undo a rotation by sending to 0
|
|
self.send_to_zero_on_stop = True
|
|
self.await_stop = True
|
|
self.smooth_stop()
|
|
# self.robot_speed[0] = 0
|
|
# self.robot.speedj(self.robot_speed, base_acceleration, 1)
|
|
return
|
|
|
|
# We are not in the deadzone, so we need to move the base
|
|
# Calculate the speed based on the distance from the center
|
|
direction = math.copysign(1, joystick_pos_x)
|
|
base_speed = (
|
|
self.lerp(base_min_speed, base_max_speed, abs(joystick_pos_x))
|
|
* direction
|
|
)
|
|
self.robot_speed[0] = base_speed
|
|
|
|
#### Vertical movement ####
|
|
if abs(joystick_pos_y) < vertical_dead_zone_radius:
|
|
# We are in the deadzone, so we stop moving it
|
|
self.robot_speed = [self.robot_speed[0], 0, 0, 0, 0, 0]
|
|
else:
|
|
if current_pose is None:
|
|
current_pose = self.get_joint_angles()
|
|
movement_happened = True
|
|
direction = math.copysign(1, joystick_pos_y) # 1 if looking down, -1 if looking up
|
|
joint_limit = self.looking_down_pose if direction > 0 else self.imposing_pose
|
|
#print("Joint limit: ", joint_limit, "Direction: ", direction)
|
|
for i in range(1, 4):
|
|
joint_difference = self.looking_down_pose[i] - self.imposing_pose[i]
|
|
joint_direction = math.copysign(1, self.looking_down_pose[i] - self.imposing_pose[i]) * direction
|
|
difference_from_limit = (joint_limit[i] - current_pose[i])
|
|
# difference_from_limit = difference_from_limit if difference_from_limit * joint_direction > 0 else 0
|
|
# self.robot_speed[i] = round(difference_from_limit * vertical_speed * abs(joystick_pos_y), 5)
|
|
self.robot_speed[i] = round(joint_difference * vertical_speed * joystick_pos_y, 5) if difference_from_limit * joint_direction > 0 else 0
|
|
print("Joint: ", i, " limit:", joint_limit[i], " current:" , current_pose[i] , " reachedlimit: ", not (difference_from_limit * joint_direction > 0), " Speed: ", self.robot_speed[i])
|
|
print("")
|
|
#self.set_neck_speed(neck_speed)
|
|
|
|
|
|
if movement_happened:
|
|
# Schedule smooth stop if no input is given for a second
|
|
self.smooth_stop_delayed_call = threading.Timer(
|
|
0.4, self.move_robot_with_joystick, args=[]
|
|
)
|
|
self.smooth_stop_delayed_call.start()
|
|
|
|
#print("Base speed: ", self.robot_speed[0], " Joystick: ", joystick_pos_x)
|
|
self.robot.speedj(self.robot_speed, base_acceleration, 1)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
host = "172.22.114.160"
|
|
sentry = URSentry(host)
|
|
sentry.sentry_position()
|
|
# sleep(3)
|
|
#sentry.robot.movej(sentry.looking_down_pose, a=0.5, v=1.5)
|
|
#sleep(3)
|
|
#sentry.forward_position()
|
|
# sleep(3)
|
|
# sentry.robot.speedj([0.8, 0, 0, 0, 0, 0], 0.5, 3)
|
|
# sleep(2)
|
|
# sentry.smooth_stop()
|
|
# sleep(2)
|
|
# sentry.forward_position_to_base_angle_degrees(45)
|
|
|
|
#print(sentry.robot.get_joint_angles())
|
|
|
|
# while True:
|
|
# print(sentry.robot.get_joint_speeds()[0])
|