generated from cyberbotics/wrestling
-
Notifications
You must be signed in to change notification settings - Fork 2
/
participant.py
106 lines (90 loc) · 3.88 KB
/
participant.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
# Copyright 1996-2023 Cyberbotics Ltd.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
'''
Controller example for the Robot Wrestling Tournament.
Demonstrates how to use a sensor, here an accelerometer to detect a fall.
Depending on the fall direction, the robot will play a different motion, which is implemented by a simple Finite State Machine.
Beats Charlie by getting up after being knocked down.
'''
import sys
from controller import Robot
sys.path.append('..')
from utils.accelerometer import Accelerometer
from utils.finite_state_machine import FiniteStateMachine
from utils.motion_library import MotionLibrary
from utils.current_motion_manager import CurrentMotionManager
class David (Robot):
def __init__(self):
Robot.__init__(self)
# retrieves the WorldInfo.basicTimeTime (ms) from the world file
self.time_step = int(self.getBasicTimeStep())
# the Finite State Machine (FSM) is a way of representing a robot's behavior as a sequence of states
self.fsm = FiniteStateMachine(
states=['DEFAULT', 'BLOCKING_MOTION', 'FRONT_FALL', 'BACK_FALL'],
initial_state='DEFAULT',
actions={
'BLOCKING_MOTION': self.pending,
'DEFAULT': self.walk,
'FRONT_FALL': self.front_fall,
'BACK_FALL': self.back_fall
}
)
self.accelerometer = Accelerometer(self, self.time_step)
self.leds = {
'right': self.getDevice('Face/Led/Right'),
'left': self.getDevice('Face/Led/Left')
}
# Shoulder roll motors for getting up from a side fall
self.RShoulderRoll = self.getDevice('RShoulderRoll')
self.LShoulderRoll = self.getDevice('LShoulderRoll')
# load motion files
self.current_motion = CurrentMotionManager()
self.library = MotionLibrary()
def run(self):
self.leds['right'].set(0x0000ff)
self.leds['left'].set(0x0000ff)
self.current_motion.set(self.library.get('Stand'))
self.fsm.transition_to('BLOCKING_MOTION')
while self.step(self.time_step) != -1:
self.detect_fall()
self.fsm.execute_action()
def detect_fall(self):
'''Detect a fall and update the FSM state.'''
[acc_x, acc_y, _] = self.accelerometer.get_new_average()
if acc_x < -7:
self.fsm.transition_to('FRONT_FALL')
elif acc_x > 7:
self.fsm.transition_to('BACK_FALL')
if acc_y < -7:
# Fell to its right, pushing itself on its back
self.RShoulderRoll.setPosition(-1.2)
elif acc_y > 7:
# Fell to its left, pushing itself on its back
self.LShoulderRoll.setPosition(1.2)
def pending(self):
# waits for the current motion to finish before doing anything else
if self.current_motion.is_over():
self.fsm.transition_to('DEFAULT')
def walk(self):
if self.current_motion.get() != self.library.get('ForwardLoop'):
self.current_motion.set(self.library.get('ForwardLoop'))
def front_fall(self):
self.current_motion.set(self.library.get('GetUpFront'))
self.fsm.transition_to('BLOCKING_MOTION')
def back_fall(self):
self.current_motion.set(self.library.get('GetUpBack'))
self.fsm.transition_to('BLOCKING_MOTION')
# create the Robot instance and run main loop
wrestler = David()
wrestler.run()