Skip to content

Commit 3a92145

Browse files
authored
Merge pull request #18 from arduino/dev
0.4.0 RC1
2 parents 690e6d4 + 8ccf676 commit 3a92145

16 files changed

+353
-73
lines changed

README.md

+72
Original file line numberDiff line numberDiff line change
@@ -70,4 +70,76 @@ You can now use Arduino Lab for Micropython to run your examples remotely from t
7070

7171
<br>
7272

73+
## Default Demo
74+
75+
Use `mpremote` to copy following the files from the examples folder:
76+
- `main.py`, this file allows you to automatically start the demo
77+
- `demo.py`, demo launcher
78+
- `touch_move.py`, programming the robot movements via touch buttons demo
79+
- `line_follower.py`, black line follower demo
80+
- `hand_follower.py`, hand following demo, the robot stays always at 10cm from an obstacle placed in front of it.
81+
82+
When the robot is turned on, the demo launcher starts after Alvik boot.
83+
84+
Blue leds on top turn on.
85+
86+
By pressing up and down arrows, it is possible to select different demos identified by different colors (blue, green and red).
87+
88+
Each color allows to run a different demo as following:
89+
- `red` launches the touch-move demo
90+
- `green` launches the hand following demo
91+
- `blue` launches the line follower demo
92+
93+
To run a demo, press the `OK touch button`, after selecting the right demo color.
94+
95+
To run a different demo, turn the robot off and on again or reset the Arduino® Nano ESP32.
96+
97+
### 1. Touch mode example (RED)
98+
This example starts with the red leds on.
99+
100+
`directional touch buttons` (UP, DOWN, LEFT, RIGHT) program the desired movements.
101+
102+
Everytime a `directional touch button` is pressed, the leds blink in a purple color indicating that the command has been registered.
103+
- `UP touch button` will register a 10 cm forward movement
104+
- `DOWN touch button` will register a 10 cm backward movement
105+
- `RIGHT touch button` will register a 90° clockwise rotation movement
106+
- `LEFT touch button` will register a 90° counterclockwise rotation movement
107+
108+
To clear the commands queue, press the `CANCEL touch button`.
109+
The leds will blink in red.
110+
111+
To start the sequence, press the `OK touch button`.
112+
113+
Pressing the `CANCEL touch button` at any time stops the robot and resets the sequence.
114+
115+
<br>
116+
117+
### 2. Hand follower example (GREEN)
118+
This example starts with the green leds on.
119+
120+
Place an obstacle or your hand in front of the robot.
121+
122+
To start the robot press the `OK touch button`.
123+
124+
The robot will move to keep a 10 centimeters distance from the obstacle/hand.
125+
126+
It is possible to stop the robot at any time by pressing the `CANCEL touch button`.
127+
128+
<br>
129+
130+
### 3. Line Follower example (BLUE)
131+
This example starts with the blue leds on.
132+
133+
To run this example, a white board and black tape (2cm wide) is required.
134+
135+
Place the robot at the center of the line and press the `OK touch button`.
136+
137+
It is possible to stop the robot at any time by pressing the `CANCEL touch button`.
138+
139+
140+
141+
<br>
142+
<br>
143+
<br>
144+
73145
__Note: not open bin files with Arduino Lab for Micropython because they will be corrupted__

arduino_alvik/arduino_alvik.py

+77-22
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
import struct
44
from machine import I2C
55
import _thread
6-
from time import sleep_ms
6+
from time import sleep_ms, ticks_ms, ticks_diff
77

88
from ucPack import ucPack
99

@@ -15,7 +15,6 @@
1515

1616

1717
class ArduinoAlvik:
18-
1918
_update_thread_running = False
2019
_update_thread_id = None
2120
_touch_events_thread_running = False
@@ -31,10 +30,10 @@ def __init__(self):
3130
self.left_wheel = _ArduinoAlvikWheel(self._packeter, ord('L'))
3231
self.right_wheel = _ArduinoAlvikWheel(self._packeter, ord('R'))
3332
self._led_state = list((None,))
34-
self.left_led = _ArduinoAlvikRgbLed(self._packeter, 'left', self._led_state,
35-
rgb_mask=[0b00000100, 0b00001000, 0b00010000])
36-
self.right_led = _ArduinoAlvikRgbLed(self._packeter, 'right', self._led_state,
37-
rgb_mask=[0b00100000, 0b01000000, 0b10000000])
33+
self.left_led = self.DL1 = _ArduinoAlvikRgbLed(self._packeter, 'left', self._led_state,
34+
rgb_mask=[0b00000100, 0b00001000, 0b00010000])
35+
self.right_led = self.DL2 = _ArduinoAlvikRgbLed(self._packeter, 'right', self._led_state,
36+
rgb_mask=[0b00100000, 0b01000000, 0b10000000])
3837
self._battery_perc = None
3938
self._touch_byte = None
4039
self._behaviour = None
@@ -67,7 +66,8 @@ def __init__(self):
6766
self._bottom_tof = None
6867
self._linear_velocity = None
6968
self._angular_velocity = None
70-
self._last_ack = ''
69+
self._last_ack = None
70+
self._waiting_ack = None
7171
self._version = [None, None, None]
7272
self._touch_events = _ArduinoAlvikTouchEvents()
7373

@@ -131,7 +131,7 @@ def _idle(self, delay_=1, check_on_thread=False) -> None:
131131
LEDR.value(led_val)
132132
LEDG.value(1)
133133
led_val = (led_val + 1) % 2
134-
print("Alvik is on")
134+
print("********** Alvik is on **********")
135135
except KeyboardInterrupt:
136136
self.stop()
137137
sys.exit()
@@ -176,7 +176,7 @@ def begin(self) -> int:
176176
:return:
177177
"""
178178
if not self.is_on():
179-
print("\nTurn on your Arduino Alvik!\n")
179+
print("\n********** Please turn on your Arduino Alvik! **********\n")
180180
sleep_ms(1000)
181181
self._idle(1000)
182182
self._begin_update_thread()
@@ -199,6 +199,7 @@ def _wait_for_ack(self) -> None:
199199
Waits until receives 0x00 ack from robot
200200
:return:
201201
"""
202+
self._waiting_ack = 0x00
202203
while self._last_ack != 0x00:
203204
sleep_ms(20)
204205

@@ -229,24 +230,31 @@ def _stop_update_thread(cls):
229230
"""
230231
cls._update_thread_running = False
231232

232-
def _wait_for_target(self):
233-
while not self.is_target_reached():
234-
pass
233+
def _wait_for_target(self, idle_time):
234+
start = ticks_ms()
235+
while True:
236+
if ticks_diff(ticks_ms(), start) >= idle_time * 1000 and self.is_target_reached():
237+
break
238+
else:
239+
# print(self._last_ack)
240+
sleep_ms(100)
235241

236242
def is_target_reached(self) -> bool:
237243
"""
238244
Returns True if robot has sent an M or R acknowledgment.
239245
It also responds with an ack received message
240246
:return:
241247
"""
242-
if self._last_ack != ord('M') and self._last_ack != ord('R'):
243-
sleep_ms(50)
244-
return False
245-
else:
248+
if self._waiting_ack is None:
249+
return True
250+
if self._last_ack == self._waiting_ack:
246251
self._packeter.packetC1B(ord('X'), ord('K'))
247252
uart.write(self._packeter.msg[0:self._packeter.msg_size])
248-
sleep_ms(200)
253+
sleep_ms(100)
254+
self._last_ack = 0x00
255+
self._waiting_ack = None
249256
return True
257+
return False
250258

251259
def set_behaviour(self, behaviour: int):
252260
"""
@@ -269,8 +277,9 @@ def rotate(self, angle: float, unit: str = 'deg', blocking: bool = True):
269277
sleep_ms(200)
270278
self._packeter.packetC1F(ord('R'), angle)
271279
uart.write(self._packeter.msg[0:self._packeter.msg_size])
280+
self._waiting_ack = ord('R')
272281
if blocking:
273-
self._wait_for_target()
282+
self._wait_for_target(idle_time=(angle / MOTOR_CONTROL_DEG_S))
274283

275284
def move(self, distance: float, unit: str = 'cm', blocking: bool = True):
276285
"""
@@ -284,8 +293,9 @@ def move(self, distance: float, unit: str = 'cm', blocking: bool = True):
284293
sleep_ms(200)
285294
self._packeter.packetC1F(ord('G'), distance)
286295
uart.write(self._packeter.msg[0:self._packeter.msg_size])
296+
self._waiting_ack = ord('M')
287297
if blocking:
288-
self._wait_for_target()
298+
self._wait_for_target(idle_time=(distance / MOTOR_CONTROL_MM_S))
289299

290300
def stop(self):
291301
"""
@@ -610,7 +620,11 @@ def _parse_message(self) -> int:
610620
_, self._linear_velocity, self._angular_velocity = self._packeter.unpacketC2F()
611621
elif code == ord('x'):
612622
# robot ack
613-
_, self._last_ack = self._packeter.unpacketC1B()
623+
if self._waiting_ack is not None:
624+
_, self._last_ack = self._packeter.unpacketC1B()
625+
else:
626+
self._packeter.unpacketC1B()
627+
self._last_ack = 0x00
614628
elif code == ord('z'):
615629
# robot ack
616630
_, self._x, self._y, self._theta = self._packeter.unpacketC3F()
@@ -899,9 +913,9 @@ def hsv2label(h, s, v) -> str:
899913
label = 'GREEN'
900914
elif 170 <= h < 210:
901915
label = 'LIGHT BLUE'
902-
elif 210 <= h < 260:
916+
elif 210 <= h < 250:
903917
label = 'BLUE'
904-
elif 260 <= h < 280:
918+
elif 250 <= h < 280:
905919
label = 'VIOLET'
906920
else: # h<20 or h>=280 is more problematic
907921
if v < 0.5 and s < 0.45:
@@ -1322,3 +1336,44 @@ def register_callback(self, event_name: str, callback: callable, args: tuple = N
13221336
if event_name not in self.__class__.available_events:
13231337
return
13241338
super().register_callback(event_name, callback, args)
1339+
1340+
1341+
# UPDATE FIRMWARE METHOD #
1342+
1343+
def update_firmware(file_path: str):
1344+
"""
1345+
1346+
:param file_path: path of your FW bin
1347+
:return:
1348+
"""
1349+
1350+
from sys import exit
1351+
from stm32_flash import (
1352+
CHECK_STM32,
1353+
STM32_endCommunication,
1354+
STM32_startCommunication,
1355+
STM32_NACK,
1356+
STM32_eraseMEM,
1357+
STM32_writeMEM, )
1358+
1359+
if CHECK_STM32.value() is not 1:
1360+
print("Turn on your Alvik to continue...")
1361+
while CHECK_STM32.value() is not 1:
1362+
sleep_ms(500)
1363+
1364+
ans = STM32_startCommunication()
1365+
if ans == STM32_NACK:
1366+
print("Cannot establish connection with STM32")
1367+
exit(-1)
1368+
1369+
print('\nSTM32 FOUND')
1370+
1371+
print('\nERASING MEM')
1372+
STM32_eraseMEM(0xFFFF)
1373+
1374+
print("\nWRITING MEM")
1375+
STM32_writeMEM(file_path)
1376+
print("\nDONE")
1377+
print("\nLower Boot0 and reset STM32")
1378+
1379+
STM32_endCommunication()

arduino_alvik/constants.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11

22
# COLOR SENSOR
33
COLOR_FULL_SCALE = 4097
4-
WHITE_CAL = [444, 342, 345]
5-
BLACK_CAL = [153, 135, 123]
4+
WHITE_CAL = [450, 500, 510]
5+
BLACK_CAL = [160, 200, 190]

arduino_alvik/firmware_updater.py

-24
This file was deleted.

arduino_alvik/robot_definitions.py

+2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,8 @@
33
MOTOR_KI_DEFAULT = 450.0
44
MOTOR_KD_DEFAULT = 0.0
55
MOTOR_MAX_RPM = 70.0
6+
MOTOR_CONTROL_DEG_S = 100
7+
MOTOR_CONTROL_MM_S = 100
68
WHEEL_TRACK_MM = 89.0
79

810
# Wheels parameters

examples/demo.py

+53
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
from arduino_alvik import ArduinoAlvik
2+
from time import sleep_ms
3+
import sys
4+
5+
alvik = ArduinoAlvik()
6+
alvik.begin()
7+
8+
menu_status = 0
9+
10+
11+
def update_led_status(val):
12+
if val == 0:
13+
alvik.left_led.set_color(0, 0, 1)
14+
alvik.right_led.set_color(0, 0, 1)
15+
elif val == 1:
16+
alvik.left_led.set_color(0, 1, 0)
17+
alvik.right_led.set_color(0, 1, 0)
18+
elif val == -1:
19+
alvik.left_led.set_color(1, 0, 0)
20+
alvik.right_led.set_color(1, 0, 0)
21+
22+
23+
while True:
24+
25+
update_led_status(menu_status)
26+
27+
try:
28+
29+
if alvik.get_touch_ok():
30+
if menu_status == 0:
31+
import line_follower
32+
elif menu_status == 1:
33+
import hand_follower
34+
elif menu_status == -1:
35+
import touch_move
36+
37+
if alvik.get_touch_up() and menu_status < 1:
38+
menu_status += 1
39+
update_led_status(menu_status)
40+
while alvik.get_touch_up():
41+
sleep_ms(100)
42+
if alvik.get_touch_down() and menu_status > -1:
43+
menu_status -= 1
44+
update_led_status(menu_status)
45+
while alvik.get_touch_down():
46+
sleep_ms(100)
47+
48+
sleep_ms(100)
49+
50+
except KeyboardInterrupt as e:
51+
print('over')
52+
alvik.stop()
53+
sys.exit()

examples/flash_firmware.py

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
# from machine import reset
2+
from arduino_alvik import update_firmware
3+
4+
update_firmware('./firmware.bin')
5+
# reset()

0 commit comments

Comments
 (0)