-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathreactive_assembly_v2.py
561 lines (444 loc) · 25.7 KB
/
reactive_assembly_v2.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
#!/usr/bin/env python3
import pdb
import sys
import time
import pickle
import numpy as np
from copy import deepcopy
from threading import Thread
import adapy
import rospy
from std_msgs.msg import Float64MultiArray
from moveit_ros_planning_interface._moveit_roscpp_initializer import roscpp_init
from PyQt5.QtWidgets import *
from PyQt5.QtGui import *
from PyQt5.QtCore import *
import common
# set to False if operating real robot
IS_SIM = False
# directory path for each machine
directory_syspath = "/home/icaros/ros_ws/src/ada_manipulation_demos"
# urdf files path
urdf_filepath = "package://ada_manipulation_demos/urdfs"
# ------------------------------------------------------- MAIN ------------------------------------------------------- #
class AssemblyController(QMainWindow):
def __init__(self):
super(AssemblyController, self).__init__()
# initialize robot
self.ada = adapy.Ada(IS_SIM)
# ------------------------------------------ Create sim environment ---------------------------------------------- #
# objects in airplane assembly
storageURDFUri = urdf_filepath + "/storage.urdf"
storagePose = [0., -0.3, -0.77, 0, 0, 0, 0]
wingURDFUri = urdf_filepath + "/abstract_main_wing.urdf"
wingPose = [0.75, -0.3, 0., 0.5, 0.5, 0.5, 0.5]
tailURDFUri = urdf_filepath + "/abstract_tail_wing.urdf"
tailPose = [-0.7, -0.25, 0.088, 0.5, 0.5, 0.5, 0.5]
container1URDFUri = urdf_filepath + "/container_1.urdf"
container1_1Pose = [0.4, -0.4, 0., 0., 0., 0., 0.]
container1_2Pose = [-0.4, -0.4, 0., 0., 0., 0., 0.]
container1_3Pose = [0.55, -0.3, 0., 0., 0., 0., 0.]
container1_4Pose = [-0.55, -0.3, 0., 0., 0., 0., 0.]
container2URDFUri = urdf_filepath + "/container_2.urdf"
container2_1Pose = [0.4, -0.1, 0, 0., 0., 0., 0.]
container2_2Pose = [-0.4, -0.1, 0., 0., 0., 0., 0.]
container3URDFUri = urdf_filepath + "/container_3.urdf"
container3_1Pose = [0.6, 0., 0., 0., 0., 0., 0.]
container3_2Pose = [-0.6, 0., 0., 0., 0, 0, 0]
# grasp TSR and offsets
tailGraspPose = [[1., 0., 0.], [0., 1., 0.], [0., 0., 1]]
tailGraspOffset = [0., 0.175, 0.]
container1GraspPose = [[0., 1., 0., 0.], [1., 0., 0., -0.05], [0., 0., -1, 0.8], [0., 0., 0., 1.]]
container1GraspOffset = [0., 0., -0.07]
container2GraspPose = [[0., 1., 0., 0.], [1., 0., 0., -0.1], [0., 0., -1, 0.1], [0., 0., 0., 1.]]
container2GraspOffset = [0., -0.115, 0.]
container3GraspPose = [[-1., 0., 0., 0.], [0., 1., 0., 0.], [0., 0., -1., 0.1], [0., 0., 0., 1.]]
container3GraspOffset = [0., 0., 0.]
# hard-coded grasps
self.graspConfig, self.deliveryRotation, self.deliveryHandRotation, self.lowerDistance, self.liftOffset, self.waitTime = {}, {}, {}, {}, {}, {}
self.graspConfig["long bolts"] = [-2.06624655, 4.37198852, 2.3886246, -2.84061763, 4.90123373, -6.59571791]
self.graspConfig["long bolts back"] = [-2.06807695, 4.2959132, 2.41701177, -2.789237, 11.03463327, -6.56036019]
#[-2.09854350, 4.27745083, 2.12565941, -2.91795263, -1.19278937, -0.39783092]
self.deliveryRotation["long bolts"] = -1.85
self.deliveryHandRotation["long bolts"] = -1.65
self.lowerDistance["long bolts"] = -0.135
self.liftOffset["long bolts"] = [0., 0., 0.15]
self.waitTime["long bolts"] = 4
self.graspConfig["short bolts"] = [ -7.07682948, 4.45124074, 2.65111774, -2.60687012, 17.12044212, -11.50343272]
#[-0.72561783, 4.31588712, 2.28856202, -2.71514972, -1.42200445, 1.01089267]
self.graspConfig["short bolts back"] = [ -7.07814016, 4.39303122, 2.67651871, -2.55499361, 16.98924735, -11.45468195]
self.deliveryRotation["short bolts"] = 1.85
self.deliveryHandRotation["short bolts"] = 1.6
self.lowerDistance["short bolts"] = -0.135
self.liftOffset["short bolts"] = [0., 0., 0.15]
self.waitTime["short bolts"] = 4
# self.graspConfig["propeller nut"] = [0.49700125, 1.86043184, 3.78425230, 2.63384048, 1.44808279, 1.67817618]
self.graspConfig["propeller nut"] = [-2.03877631, 4.09967790, 1.60438025, -0.19636232, 0.71718155, 2.21799853]
self.deliveryRotation["propeller nut"] = -2.1
self.deliveryHandRotation["propeller nut"] = -2.25
self.lowerDistance["propeller nut"] = -0.135
self.liftOffset["propeller nut"] = [0.2, 0., 0.15]
self.waitTime["propeller nut"] = 4
#self.outDistance["propeller nut"] = 0.2
self.graspConfig["tail screw"] = [-0.46015322, 4.47079882, 2.68192519, -2.584758426, -1.74260217, 1.457295330]
self.deliveryRotation["tail screw"] = 1.5
self.deliveryHandRotation["tail screw"] = 1.35
self.lowerDistance["tail screw"] = -0.135
self.liftOffset["tail screw"] = [0., 0., 0.15]
self.waitTime["tail screw"] = 4
self.graspConfig["propeller blades"] = [-2.65224753, 4.04064601, 1.51335391, -0.18831809, 0.71782515, 1.8779379 ]
#[-2.34985128, 4.05563485, 1.56017775, -2.94642629, -0.75563986, -0.58171179]# grasping from bottom of the container
# [-2.65224753 4.04064601 1.51335391 -0.18831809 0.71782515 1.8779379 ] a little to the right
#self.graspConfig["propeller blades"] = [-2.4191907, 3.9942575, 1.29241768, 3.05926906, -0.50726387, -0.52933128]
self.deliveryRotation["propeller blades"] = -1.5
self.deliveryHandRotation["propeller blades"] = -1.5
self.lowerDistance["propeller blades"] = -0.14
self.liftOffset["propeller blades"] = [0.1, 0., 0.15]
self.waitTime["propeller blades"] = 4
#self.outDistance["propeller blades"] = 0.1
# self.graspConfig["tool"] = [-0.32843145, 4.02576609, 1.48440087, -2.87877031, -0.79457283, 1.40310179]
self.graspConfig["tool"] = [-0.39286100, 4.03060775, 1.54378641, -2.93910479, -0.79306859, 1.08912509]
self.deliveryRotation["tool"] = 1.5
self.deliveryHandRotation["tool"] = 1.65
self.lowerDistance["tool"] = -0.14
self.liftOffset["tool"] = [0., 0., 0.15]
self.waitTime["tool"] = 4
#self.outDistance["tool"] = -0.1
self.graspConfig["propeller hub"] = [-15.45968209, 4.28690186, 2.20877388, -0.26991138, 1.17136384, -0.10107539]
#[3.00773842, 4.21352853, 1.98663177, -0.17330897, 1.01156224, -0.46210507]
self.deliveryRotation["propeller hub"] = -1.2
self.deliveryHandRotation["propeller hub"] = -1.35
self.lowerDistance["propeller hub"] = -0.14
self.liftOffset["propeller hub"] = [0., 0., 0.2]
self.waitTime["propeller hub"] = 4
self.graspConfig["tail wing"] = [3.129024, 1.87404028, 3.40826295, 0.53502216, -1.86749865, -0.99044654]
self.deliveryRotation["tail wing"] = 1.15
self.deliveryHandRotation["tail wing"] = -0.65
self.liftOffset["tail wing"] = [0., 0., 0.15]
self.waitTime["tail wing"] = 2
self.graspConfig["main wing"] = [-2.86840265, 3.89315136, 1.47980743, -3.07256298, 0.95719655, 2.37149834]
self.deliveryRotation["main wing"] = -0.75
self.deliveryHandRotation["main wing"] = 0.0
self.waitTime["main wing"] = 2
self.liftOffset["main wing"] = [0., 0.10, 0.10]
# initialize sim environment
self.world = self.ada.get_world()
viewer = self.ada.start_viewer("airplane_assembly_demo", "map")
# add parts to sim environment
storageInWorld = self.world.add_body_from_urdf(storageURDFUri, storagePose)
container1_1 = self.world.add_body_from_urdf(container1URDFUri, container1_1Pose)
container1_2 = self.world.add_body_from_urdf(container1URDFUri, container1_2Pose)
container1_3 = self.world.add_body_from_urdf(container1URDFUri, container1_3Pose)
container1_4 = self.world.add_body_from_urdf(container1URDFUri, container1_4Pose)
container2_1 = self.world.add_body_from_urdf(container2URDFUri, container2_1Pose)
container2_2 = self.world.add_body_from_urdf(container2URDFUri, container2_2Pose)
container3_1 = self.world.add_body_from_urdf(container3URDFUri, container3_1Pose)
# container3_2 = self.world.add_body_from_urdf(container3URDFUri, container3_2Pose)
tailWing = self.world.add_body_from_urdf(tailURDFUri, tailPose)
mainWing = self.world.add_body_from_urdf(wingURDFUri, wingPose)
# dict of all objects
self.objects = {"propeller hub": [container3_1, container3_1Pose, container3GraspPose, container3GraspOffset, container3URDFUri],
"long bolts": [container1_1, container1_1Pose, container1GraspPose, container1GraspOffset, container1URDFUri],
"short bolts": [container1_2, container1_2Pose, container1GraspPose, container1GraspOffset, container1URDFUri],
"tail wing": [tailWing, tailPose, tailGraspPose, tailGraspOffset, tailURDFUri],
"propeller nut": [container1_3, container1_3Pose, container1GraspPose, container1GraspOffset, container1URDFUri],
"tail screw": [container1_4, container1_4Pose, container1GraspPose, container1GraspOffset, container1URDFUri],
"propeller blades": [container2_1, container2_1Pose, container2GraspPose, container2GraspOffset, container2URDFUri],
"tool": [container2_2, container2_2Pose, container2GraspPose, container2GraspOffset, container2URDFUri],
"main wing": [mainWing, wingPose, 0,0, wingURDFUri],
"airplane body": []}
# ------------------------------------------------ Get robot config ---------------------------------------------- #
collision = self.ada.get_self_collision_constraint()
self.arm_skeleton = self.ada.get_arm_skeleton()
self.arm_state_space = self.ada.get_arm_state_space()
self.hand = self.ada.get_hand()
self.hand_node = self.hand.get_end_effector_body_node()
viewer.add_frame(self.hand_node)
# ------------------------------- Start executor for real robot (not needed for sim) ----------------------------- #
if not IS_SIM:
self.ada.start_trajectory_controllers()
# move to home position
self.armHome = [-1.57, 3.14, 1.23, -2.19, 1.8, 1.2]
trajectory = self.ada.plan_to_configuration(self.armHome, self.ada.get_world_collision_constraint())
self.ada.execute_trajectory(trajectory)
# open hand
self.hand.execute_preshape([0.15, 0.15])
# ------------------------------------------------- Assembly Info ------------------------------------------------ #
# objects yet to be delivered
self.remaining_objects = list(self.objects.keys())
# subscribe to action recognition
sub_act = rospy.Subscriber("/april_tag_detection", Float64MultiArray, self.callback, queue_size=1)
# initialize user sequence
self.time_step = 0
self.user_sequence = []
self.anticipated_action_name = []
self.suggested_objects = []
# ------------------------------------------------ GUI details --------------------------------------------------- #
# window title and size
self.setWindowTitle("Robot Commander")
self.setGeometry(0, 0, 1280, 720)
# prompt
query = QLabel(self)
query.setText("Which part(s) do you want?")
query.setFont(QFont('Arial', 28))
query.adjustSize()
query.move(95, 135)
# task info
assembly_image = QLabel(self)
pixmap = QPixmap(directory_syspath + "/media/actual_task.jpg")
pixmap = pixmap.scaledToWidth(1125)
assembly_image.setPixmap(pixmap)
assembly_image.adjustSize()
assembly_image.move(660, 145)
# inputs
options = deepcopy(self.remaining_objects)
# print the options
option_x, option_y = 210, 200
buttons = []
for opt in options:
opt_button = QPushButton(self)
opt_button.setText(opt)
opt_button.setFont(QFont('Arial', 20))
opt_button.setGeometry(option_x, option_y, 225, 50)
opt_button.setCheckable(True)
opt_button.setStyleSheet("QPushButton::checked {background-color : lightpink;}")
buttons.append(opt_button)
option_y += 50
self.option_buttons = buttons
# button for performing selected actions
option_x = 85
option_y += 60
self.selected_button = QPushButton(self)
self.selected_button.setText("Give me the selected parts.")
self.selected_button.setFont(QFont('Arial', 20))
self.selected_button.setGeometry(option_x, option_y, 500, 50)
self.selected_button.setStyleSheet("background-color : lightpink")
self.selected_button.setCheckable(True)
self.selected_button.clicked.connect(self.deliver_part)
# print current time step
self.step_label = QLabel(self)
self.step_label.setText("Current time step: " + str(self.time_step))
self.step_label.setFont(QFont('Arial', 36))
self.step_label.adjustSize()
self.step_label.move(715, 65)
# update timer
self.time_to_respond = 10
self.timer = QTimer()
self.timer.timeout.connect(self.update_application)
self.time_left = deepcopy(self.time_to_respond)
self.countdown = QLabel(self)
self.countdown.setText(str(self.time_left))
self.countdown.setFont(QFont('Arial', 36))
self.countdown.setStyleSheet("background-color: khaki")
self.countdown.adjustSize()
self.countdown.move(1720, 65)
self.countdown_timer = QTimer()
self.countdown_timer.timeout.connect(self.timer_update)
self.timer.start(self.time_to_respond*1000)
self.countdown_timer.start(1000)
def timer_update(self):
self.time_left -=1
self.countdown.setText(" " + str(self.time_left) + " ")
if self.time_left == 0:
self.time_left = deepcopy(self.time_to_respond)
self.countdown.setText(str(self.time_left))
def update_application(self):
# update time stamp
self.step_label.setText("Current time step: " + str(self.time_step))
# update suggested options
for opt_button in self.option_buttons:
# opt_button.setChecked(False)
if opt_button.text() not in self.remaining_objects:
opt_button.setChecked(False)
opt_button.setCheckable(False)
opt_button.setStyleSheet("QPushButton {color : lightgrey;}")
else:
opt_button.setStyleSheet("QPushButton::checked {background-color : lightpink;}")
# update action buttons
self.selected_button.setChecked(False)
def callback(self, data):
# current recognised action sequence
detected_sequence = [int(a) for a in data.data]
# current recognised parts
detected_parts = data.layout.dim[0].label.split(",")
# update action sequence
self.user_sequence = detected_sequence
self.time_step = len(self.user_sequence)
# update remaining parts
self.remaining_objects = [rem_obj for rem_obj in list(self.objects.keys()) if rem_obj not in detected_parts]
# self.remaining_objects = [rem_obj for rem_obj in self.remaining_objects if rem_obj not in detected_parts]
def deliver_part(self):
# check which objects were selected by the user
if self.selected_button.isChecked():
objects_to_deliver = []
for option in self.option_buttons:
if option.isChecked():
objects_to_deliver.append(option.text())
else:
objects_to_deliver = []
# loop over all objects to be delivered
for chosen_obj in objects_to_deliver:
# instruct the user to retreive the parts that cannot be delivered by the robot
if chosen_obj in ["airplane body", "none"]:
print("Cannot provide this part.")
msg = QMessageBox()
msg.setText("Get the parts you need while the robot waits.")
msg.setFont(QFont('Arial', 20))
msg.setWindowTitle("Robot Message")
QTimer.singleShot(10000, msg.close)
msg.exec_()
else:
# deliver parts requested by the user whenever possible
print("Providing the required part.")
# ---------------------------------------- Collision detection --------------------------------------- #
# collision_free_constraint = self.ada.set_up_collision_detection(ada.get_arm_state_space(), self.ada.get_arm_skeleton(),
# [obj])
# full_collision_constraint = self.ada.get_full_collision_constraint(ada.get_arm_state_space(),
# self.ada.get_arm_skeleton(),
# collision_free_constraint)
# collision = self.ada.get_self_collision_constraint()
# ---------------------------------------- Plan path to object --------------------------------------- #
obj = self.objects[chosen_obj][0]
# use pre-computed grasp configuration if available
if chosen_obj in self.graspConfig.keys():
print("Running hard-coded...")
grasp_configuration = self.graspConfig[chosen_obj]
else:
print("Creating new TSR.")
objPose = self.objects[chosen_obj][1]
objGraspPose = self.objects[chosen_obj][2]
# grasp TSR for object
objTSR = common.createTSR(objPose, objGraspPose)
# marker = viewer.add_tsr_marker(objTSR)
# perform IK to compute grasp configuration
ik_sampleable = adapy.create_ik(self.arm_skeleton, self.arm_state_space, objTSR, self.hand_node)
ik_generator = ik_sampleable.create_sample_generator()
configurations = []
samples, max_samples = 0, 10
while samples < max_samples and ik_generator.can_sample():
samples += 1
goal_state = ik_generator.sample(self.arm_state_space)
if len(goal_state) == 0:
continue
configurations.append(goal_state)
print("Found new configuration.")
grasp_configuration = configurations[0]
# plan path to grasp configuration
# waypoints = [(0.0, self.armHome),(1.0, grasp_configuration)]
# trajectory = self.ada.compute_joint_space_path(waypoints)
# remove object to grasp from collision check
self.world.remove_skeleton(obj)
# plan path to grasp configuration with world collision check
trajectory = self.ada.plan_to_configuration(grasp_configuration, self.ada.get_world_collision_constraint())
# add object urdf back to world and update obj pointer
self.objects[chosen_obj][0] = self.world.add_body_from_urdf(self.objects[chosen_obj][4], self.objects[chosen_obj][1])
obj = self.objects[chosen_obj][0]
if not trajectory:
print("Failed to find a solution!")
else:
# execute the planned trajectory
self.ada.execute_trajectory(trajectory)
# ---------------------------------- Move closer to object for grasping ------------------------------ #
# lower gripper
if chosen_obj == 'main wing':
traj = self.ada.plan_to_offset("j2n6s200_hand_base", [0.04, 0., 0.])
else:
traj = self.ada.plan_to_offset("j2n6s200_hand_base", [0., 0., -0.045])
self.ada.execute_trajectory(traj)
# grasp the object
self.hand.execute_preshape([1.3, 1.3])
time.sleep(1)
self.hand.grab(obj)
# lift up grasped object
# if chosen_obj == 'main wing':
# traj = self.ada.plan_to_offset("j2n6s200_hand_base", [0., 0.10, 0.10])
# self.ada.execute_trajectory(traj)
# traj = self.ada.plan_to_offset("j2n6s200_hand_base", [-0.1, 0.0, 0.0])
# self.ada.execute_trajectory(traj)
# else:
traj = self.ada.plan_to_offset("j2n6s200_hand_base", self.liftOffset[chosen_obj])
self.ada.execute_trajectory(traj)
if chosen_obj == "main wing":
traj = self.ada.plan_to_offset("j2n6s200_hand_base", [-0.1, 0.0, 0.0])
self.ada.execute_trajectory(traj)
#--------------------------------- Move grasped object to workbench ---------------------------------- #
# move outwards if propeller nut, propeller blades, tool
# if chosen_obj in ["propeller nut", "propeller blades", "tool"]:
# traj = self.ada.plan_to_offset("j2n6s200_hand_base", [self.outDistance[chosen_obj], 0. ,0.])
# self.ada.execute_trajectory(traj)
current_position = self.arm_skeleton.get_positions()
new_position = current_position.copy()
new_position[0] += self.deliveryRotation[chosen_obj]
new_position[5] += self.deliveryHandRotation[chosen_obj]
waypoints = [(0.0, current_position), (1.0, new_position)]
traj = self.ada.compute_joint_space_path(waypoints)
self.ada.execute_trajectory(traj)
# ------------------------ Lower grasped object using Jacobian pseudo-inverse ------------------------ #
if chosen_obj in ["main wing", "tail wing"]:
# ungrasp the main wing
time.sleep(self.waitTime[chosen_obj])
self.hand.ungrab()
self.hand.execute_preshape([0.15, 0.15])
self.world.remove_skeleton(obj)
else:
# hold the grasped object and wait for user to grab one propeller blade
if chosen_obj == "tool":
# hold tool box until user picks up tool and put tool back
while("tool" in self.remaining_objects):
time.sleep(0.1)
print("picked up tool")
time.sleep(0.5)
while ("tool" not in self.remaining_objects):
time.sleep(1)
print("tool is out")
print("tool put back")
else:
time.sleep(self.waitTime[chosen_obj])
# ---------------------- Move the container back if not main wing and tail wing ----------------------- #
# move grasped object back to parts
# turn the grasp obj back
current_position = self.arm_skeleton.get_positions()
new_position = current_position.copy()
new_position[0] -= self.deliveryRotation[chosen_obj]
new_position[5] -= self.deliveryHandRotation[chosen_obj]
waypoints = [(0.0, current_position),(1.0, new_position)]
trajectory = self.ada.compute_joint_space_path(waypoints)
if not trajectory:
print("Failed to find a solution!")
else:
# execute the planned trajectory
self.ada.execute_trajectory(trajectory)
# move inwards if propeller nut, propeller blades, tool
traj = self.ada.plan_to_offset("j2n6s200_hand_base", [-self.liftOffset[chosen_obj][0], 0. ,self.lowerDistance[chosen_obj]])
self.ada.execute_trajectory(traj)
# ungrab
self.hand.ungrab()
self.hand.execute_preshape([0.15, 0.15])
self.world.remove_skeleton(obj)
time.sleep(1)
# raise gripper
traj = self.ada.plan_to_offset("j2n6s200_hand_base", [0., 0., 0.04])
if traj:
self.ada.execute_trajectory(traj)
# ------------------- Move robot back to home ------------------- #
waypoints = [(0.0, self.ada.get_arm_positions()), (1.0, self.armHome)]
traj = self.ada.compute_joint_space_path(waypoints)
self.ada.execute_trajectory(traj)
# unselect part button after delivery
for opt_button in self.option_buttons:
if opt_button.text() == chosen_obj:
opt_button.setChecked(False)
print("Finished executing actions.")
# MAIN
# initialise ros node
rospy.init_node("reactive_assembly")
roscpp_init('reactive_assembly', [])
app = QApplication(sys.argv)
win = AssemblyController()
win.showMaximized()
app.exec_()
try:
rospy.spin()
except KeyboardInterrupt:
print ("Shutting down")
input("Press Enter to Quit...")