Skip to content

Commit b06d76c

Browse files
author
Emiliano Borghi
authored
Dashboard (#24)
* Add services * Fix import in load_calib.py * Minor changes to ca_node make file * Dashboard package -- first approach * Add launchfile for Dashboard * Edit dockerfiles * Add header to create_dashboard script * Add empty line at the end * Add libcreate as run dependency * Fix compilation of ca_node * Move some remainder maps to ca_mapping * Add missing plugin * Fix linter
1 parent 4898bb0 commit b06d76c

24 files changed

+367
-39
lines changed

ca_dashboard/CMakeLists.txt

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
cmake_minimum_required(VERSION 2.8.3)
2+
project(ca_dashboard)
3+
4+
find_package(catkin REQUIRED COMPONENTS
5+
ca_msgs
6+
ca_node
7+
diagnostic_msgs
8+
rospy
9+
rqt_gui
10+
rqt_gui_py
11+
rqt_robot_dashboard
12+
)
13+
14+
catkin_python_setup()
15+
16+
catkin_package()
17+
18+
install(PROGRAMS scripts/create_dashboard
19+
DESTINATION ${CATKIN_PACKAGE_BIN_DESTINATION})
20+
21+
install(DIRECTORY plugins
22+
DESTINATION ${CATKIN_PACKAGE_SHARE_DESTINATION})

ca_dashboard/launch/dashboard.launch

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
<launch>
2+
<node pkg="ca_dashboard" type="create_dashboard" name="create_dashboard" output="screen"/>
3+
</launch>

ca_dashboard/package.xml

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
<?xml version="1.0"?>
2+
<package format="2">
3+
<name>ca_dashboard</name>
4+
<version>2.3.1</version>
5+
<description>
6+
The Create dashboard is a RQT-based plug-in for visualising data from the Create and giving easy access
7+
to basic functionalities.
8+
</description>
9+
<maintainer email="[email protected]">Marcus Liebhardt</maintainer>
10+
<license>BSD</license>
11+
12+
<url type="website">http://ros.org/wiki/create_dashboard</url>
13+
<url type="repository">https://github.com/turtlebot/turtlebot_create_desktop</url>
14+
<url type="bugtracker">https://github.com/turtlebot/turtlebot_create_desktop/issues</url>
15+
<author email="[email protected]">Ze'ev Klapow</author>
16+
<author email="[email protected]">Marcus Liebhardt</author>
17+
18+
<buildtool_depend>catkin</buildtool_depend>
19+
20+
<build_depend>ca_msgs</build_depend>
21+
<build_depend>ca_node</build_depend>
22+
<build_depend>diagnostic_msgs</build_depend>
23+
<build_depend>rospy</build_depend>
24+
<build_depend>rqt_gui</build_depend>
25+
<build_depend>rqt_gui_py</build_depend>
26+
<build_depend>rqt_robot_dashboard</build_depend>
27+
28+
<exec_depend>ca_msgs</exec_depend>
29+
<exec_depend>ca_node</exec_depend>
30+
<exec_depend>diagnostic_msgs</exec_depend>
31+
<exec_depend>rospy</exec_depend>
32+
<exec_depend>rqt_gui</exec_depend>
33+
<exec_depend>rqt_gui_py</exec_depend>
34+
<exec_depend>rqt_robot_dashboard</exec_depend>
35+
<exec_depend>rqt_virtual_joy</exec_depend>
36+
37+
<export>
38+
<rqt_gui plugin="${prefix}/plugins/plugin.xml"/>
39+
</export>
40+
</package>

ca_dashboard/plugins/plugin.xml

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
<library path="src">
2+
<class name="CreateDashboard" type="ca_dashboard.dashboard.CreateDashboard" base_class_type="rqt_gui_py::Plugin">
3+
<description>
4+
Dashboard for the iRobot Create
5+
</description>
6+
<qtgui>
7+
<label>Create Dashboard</label>
8+
<statustip>Test robot_dashboard.</statustip>
9+
</qtgui>
10+
</class>
11+
</library>

ca_dashboard/scripts/create_dashboard

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
#!/bin/bash
2+
rqt -s ca_dashboard

ca_dashboard/setup.py

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
#!/usr/bin/env python
2+
3+
from distutils.core import setup
4+
from catkin_pkg.python_setup import generate_distutils_setup
5+
6+
d = generate_distutils_setup(packages=['ca_dashboard'],
7+
package_dir={'': 'src'})
8+
9+
setup(**d)

ca_dashboard/src/ca_dashboard/__init__.py

Whitespace-only changes.
+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
import roslib;roslib.load_manifest('ca_dashboard')
2+
import rospy
3+
4+
from rqt_robot_dashboard.widgets import BatteryDashWidget
5+
6+
def non_zero(value):
7+
if value < 0.00001 and value > -0.00001:
8+
return 0.00001
9+
return value
10+
11+
class TurtlebotBattery(BatteryDashWidget):
12+
def __init__(self, name='battery'):
13+
icons = []
14+
charge_icons = []
15+
icons.append(['ic-battery-0.svg'])
16+
icons.append(['ic-battery-20.svg'])
17+
icons.append(['ic-battery-40.svg'])
18+
icons.append(['ic-battery-60-green.svg'])
19+
icons.append(['ic-battery-80-green.svg'])
20+
icons.append(['ic-battery-100-green.svg'])
21+
charge_icons.append(['ic-battery-charge-0.svg'])
22+
charge_icons.append(['ic-battery-charge-20.svg'])
23+
charge_icons.append(['ic-battery-charge-40.svg'])
24+
charge_icons.append(['ic-battery-charge-60-green.svg'])
25+
charge_icons.append(['ic-battery-charge-80-green.svg'])
26+
charge_icons.append(['ic-battery-charge-100-green.svg'])
27+
super(TurtlebotBattery, self).__init__(name=name, icons=icons, charge_icons=charge_icons)
28+
29+
self._power_consumption = 0.0
30+
self._pct = 0.0
31+
self._cap = 2.7
32+
self._char_cap =2.7
33+
self._time_remaining = 0.0
34+
35+
def set_power_state(self, msg):
36+
last_pct = self._pct
37+
last_plugged_in = self._charging
38+
last_time_remaining = self._time_remaining
39+
self._char_cap = 0.8*self._char_cap +0.2*float(msg['Charge (Ah)'])
40+
#make sure that battery percentage is not greater than 100%
41+
if self._char_cap < float(msg['Capacity (Ah)']):
42+
self._cap = float(msg['Capacity (Ah)'])
43+
else:
44+
self._cap = self._char_cap
45+
46+
self._power_consumption = float(msg['Current (A)'])*float(msg['Voltage (V)'])
47+
#determine if we're charging or discharging
48+
if float(msg['Current (A)'])<0:
49+
tmp = (float(msg['Charge (Ah)'])/non_zero(float(msg['Current (A)'])))*60.0
50+
else:
51+
tmp = ((float(msg['Charge (Ah)'])-self._cap)/non_zero(float(msg['Current (A)'])))*60.0
52+
53+
self._time_remaining = 0.8*self._time_remaining + 0.2*tmp
54+
55+
self._pct = float(msg['Charge (Ah)'])/self._cap
56+
57+
if self._pct == 1 and float(msg['Current (A)']) == 0:
58+
self._charging = True
59+
else:
60+
self._charging = (float(msg['Current (A)'])>0)
61+
62+
self.update_perc(self._pct*100)
63+
self.update_time(self._pct*100)
64+
+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
from rqt_robot_dashboard.widgets import ButtonDashWidget
2+
3+
class BreakerControl(ButtonDashWidget):
4+
def __init__(self, context, name, index, cb = None, icon = None):
5+
super(BreakerControl, self).__init__(context, name, cb, icon)
6+
7+
self.clicked.connect(self.toggle)
8+
self._power_control = rospy.ServiceProxy('turtlebot_node/set_digital_outputs', ca_node.srv.SetDigitalOutputs)
+168
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,168 @@
1+
import roslib;roslib.load_manifest('ca_dashboard')
2+
import rospy
3+
4+
import diagnostic_msgs
5+
import ca_msgs.srv
6+
import ca_msgs.msg
7+
8+
from rqt_robot_dashboard.dashboard import Dashboard
9+
from rqt_robot_dashboard.widgets import MonitorDashWidget, ConsoleDashWidget, MenuDashWidget, BatteryDashWidget, IconToolButton, NavViewDashWidget
10+
from QtWidgets import QMessageBox, QAction
11+
from python_qt_binding.QtCore import QSize
12+
13+
from .battery import TurtlebotBattery
14+
15+
import rospkg
16+
import os.path
17+
18+
rp = rospkg.RosPack()
19+
20+
image_path = image_path = os.path.join(rp.get_path('ca_dashboard'), 'images')
21+
22+
class BreakerButton(IconToolButton):
23+
def __init__(self, name, onclick):
24+
self._on_icon = ['bg-green.svg', 'ic-breaker.svg']
25+
self._off_icon = ['bg-red.svg', 'ic-breaker.svg']
26+
27+
icons = [self._off_icon, self._on_icon]
28+
29+
super(BreakerButton, self).__init__(name, icons=icons)
30+
31+
self.setFixedSize(self._icons[0].actualSize(QSize(50,30)))
32+
33+
self.clicked.connect(onclick)
34+
35+
class CreateDashboard(Dashboard):
36+
def setup(self, context):
37+
self.message = None
38+
39+
self._dashboard_message = None
40+
self._last_dashboard_message_time = 0.0
41+
42+
self._raw_byte = None
43+
self.digital_outs = [0,0,0]
44+
45+
# These were moved out of get_widgets because they are sometimes not defined
46+
# before being used by dashboard_callback. Could be done more cleanly than this
47+
# though.
48+
self.lap_bat = BatteryDashWidget("Laptop")
49+
self.create_bat = TurtlebotBattery("Create")
50+
self.breakers = [BreakerButton('breaker0', lambda: self.toggle_breaker(0)),
51+
BreakerButton('breaker1', lambda: self.toggle_breaker(1)),
52+
BreakerButton('breaker2', lambda: self.toggle_breaker(2))]
53+
54+
# This is what gets dashboard_callback going eagerly
55+
self._dashboard_agg_sub = rospy.Subscriber('diagnostics_agg', diagnostic_msgs.msg.DiagnosticArray, self.dashboard_callback)
56+
self._power_control = rospy.ServiceProxy('turtlebot_node/set_digital_outputs', ca_msgs.srv.SetDigitalOutputs)
57+
58+
def get_widgets(self):
59+
self.mode = MenuDashWidget('Mode')
60+
61+
self.mode.add_action('Full', self.on_full_mode)
62+
self.mode.add_action('Passive', self.on_passive_mode)
63+
self.mode.add_action('Safe', self.on_safe_mode)
64+
65+
return [[MonitorDashWidget(self.context), ConsoleDashWidget(self.context), self.mode],
66+
self.breakers,
67+
[self.lap_bat, self.create_bat],
68+
[NavViewDashWidget(self.context)]]
69+
70+
def dashboard_callback(self, msg):
71+
self._dashboard_message = msg
72+
self._last_dashboard_message_time = rospy.get_time()
73+
74+
battery_status = {} # Used to store TurtlebotBattery status info
75+
breaker_status = {}
76+
laptop_battery_status = {}
77+
op_mode = None
78+
for status in msg.status:
79+
print("Status callback %s"%status.name)
80+
if status.name == "/Power System/Battery":
81+
for value in status.values:
82+
print value
83+
battery_status[value.key]=value.value
84+
# Create battery actually doesn't check public or check
85+
# charging state. Could maybe done by looking at +-
86+
# of current value
87+
# Below is kobuki battery code.
88+
# Best place to fix this is in create driver to match the kobuki
89+
# diagnostics
90+
#elif value.key == "Charging State":
91+
# if value.value == "Trickle Charging" or value.value == "Full Charging":
92+
# self.create_bat.set_charging(True)
93+
# else:
94+
# self.create_bat.set_charging(False)
95+
elif status.name == "/Mode/Operating Mode":
96+
op_mode=status.message
97+
elif status.name == "/Digital IO/Digital Outputs":
98+
#print "got digital IO"
99+
for value in status.values:
100+
breaker_status[value.key]=value.value
101+
elif status.name == "/Power System/Laptop Battery":
102+
for value in status.values:
103+
laptop_battery_status[value.key]=value.value
104+
105+
# If battery diagnostics were found, calculate percentages and stuff
106+
if (laptop_battery_status):
107+
percentage = float(laptop_battery_status['Charge (Ah)'])/float(laptop_battery_status['Capacity (Ah)'])
108+
self.lap_bat.update_perc(percentage*100)
109+
self.lap_bat.update_time(percentage*100)
110+
charging_state = True if float(laptop_battery_status['Current (A)']) > 0.0 else False
111+
self.lap_bat.set_charging(charging_state)
112+
113+
if (battery_status):
114+
self.create_bat.set_power_state(battery_status)
115+
116+
if (breaker_status):
117+
self._raw_byte = int(breaker_status['Raw Byte'])
118+
self._update_breakers()
119+
120+
def toggle_breaker(self, index):
121+
try:
122+
self.digital_outs[index] = not self.digital_outs[index]
123+
power_cmd = ca_msgs.srv.SetDigitalOutputsRequest(self.digital_outs[0], self.digital_outs[1], self.digital_outs[2])
124+
#print power_cmd
125+
self._power_control(power_cmd)
126+
127+
return True
128+
except rospy.ServiceException, e:
129+
self.message = QMessageBox()
130+
self.message.setText("Service call failed with error: %s"%(e))
131+
self.message.exec_()
132+
return False
133+
134+
return False
135+
136+
def _update_breakers(self):
137+
tmp = self._raw_byte
138+
for i in range(0,3):
139+
self.digital_outs[i]=tmp%2
140+
self.breakers[i].update_state(self.digital_outs[i])
141+
tmp = tmp >> 1
142+
143+
def on_passive_mode(self):
144+
passive = rospy.ServiceProxy("/turtlebot_node/set_operation_mode",ca_msgs.srv.SetTurtlebotMode )
145+
try:
146+
passive(ca_msgs.msg.TurtlebotSensorState.OI_MODE_PASSIVE)
147+
except rospy.ServiceException, e:
148+
self.message = QMessageBox()
149+
self.message.setText("Failed to put the turtlebot in passive mode: service call failed with error: %s"%(e))
150+
self.message.exec_()
151+
152+
def on_safe_mode(self):
153+
safe = rospy.ServiceProxy("/turtlebot_node/set_operation_mode",ca_msgs.srv.SetTurtlebotMode)
154+
try:
155+
safe(ca_msgs.msg.TurtlebotSensorState.OI_MODE_SAFE)
156+
except rospy.ServiceException, e:
157+
self.message = QMessageBox()
158+
self.message.setText("Failed to put the turtlebot in safe mode: service call failed with error: %s"%(e))
159+
self.message.exec_()
160+
161+
def on_full_mode(self):
162+
full = rospy.ServiceProxy("/turtlebot_node/set_operation_mode", ca_msgs.srv.SetTurtlebotMode)
163+
try:
164+
full(ca_msgs.msg.TurtlebotSensorState.OI_MODE_FULL)
165+
except rospy.ServiceException, e:
166+
self.message = QMessageBox()
167+
self.message.setText("Failed to put the turtlebot in full mode: service call failed with error: %s"%(e))
168+
self.message.exec_()

ca_msgs/CMakeLists.txt

+4-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,10 @@ add_message_files(
1111
DIRECTORY msg
1212
)
1313

14-
# Generate added messages and services with any dependencies listed here
14+
add_service_files(
15+
DIRECTORY srv
16+
)
17+
1518
generate_messages(
1619
DEPENDENCIES
1720
geometry_msgs

ca_msgs/srv/SetDigitalOutputs.srv

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
uint8 digital_out_0
2+
uint8 digital_out_1
3+
uint8 digital_out_2
4+
---
5+
bool done

ca_msgs/srv/SetTurtlebotMode.srv

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
ca_msgs/Mode mode #This sets the operating mode to one of the OI_MODE states from the TurtlebotSensorState.msg
2+
---
3+
bool valid_mode # will return false if a unvalid mode was requested

ca_node/CMakeLists.txt

+3-8
Original file line numberDiff line numberDiff line change
@@ -8,23 +8,22 @@ find_package(catkin REQUIRED COMPONENTS
88
ecl_threads
99
nodelet
1010
pluginlib
11+
roscpp
1112
)
1213

1314
find_package(Boost REQUIRED system thread)
1415

1516
catkin_package(
1617
LIBRARIES ${PROJECT_NAME}let ca_driver
1718
CATKIN_DEPENDS
18-
ca_description
19-
ca_msgs
19+
ca_driver
2020
ecl_threads
2121
geometry_msgs
2222
nodelet
2323
pluginlib
24+
roscpp
2425
rospy
25-
smach
2626
smach_ros
27-
smach_viewer
2827
std_msgs
2928
)
3029

@@ -41,11 +40,7 @@ install(DIRECTORY launch plugins
4140
install(FILES plugins/nodelet_plugins.xml
4241
DESTINATION ${CATKIN_PACKAGE_SHARE_DESTINATION}/plugins)
4342

44-
install(DIRECTORY scripts
45-
DESTINATION ${CATKIN_PACKAGE_BIN_DESTINATION})
46-
4743
install(DIRECTORY scripts
4844
DESTINATION ${CATKIN_PACKAGE_BIN_DESTINATION}
4945
USE_SOURCE_PERMISSIONS
50-
PATTERN ".svn" EXCLUDE
5146
)

0 commit comments

Comments
 (0)