Skip to content

Commit e41f505

Browse files
committed
Merge branch 'release/1.2.0'
2 parents a993fb7 + 1c53cfd commit e41f505

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

68 files changed

+2780
-625
lines changed

.github/workflows/ci.yml

+75-48
Original file line numberDiff line numberDiff line change
@@ -7,59 +7,86 @@ jobs:
77
name: "Build"
88
runs-on: ubuntu-latest
99
steps:
10-
- uses: actions/checkout@v3
11-
- name: Build library ZIP
12-
run: |
13-
mkdir BlenderServoAnimation
14-
cp -R src examples library.properties README.md LICENSE BlenderServoAnimation
15-
zip -r blender_servo_animation_arduino_library BlenderServoAnimation
16-
- name: Archive library ZIP
17-
uses: actions/upload-artifact@v3
18-
with:
19-
name: blender_servo_animation_arduino_library.zip
20-
path: |
21-
blender_servo_animation_arduino_library.zip
10+
- uses: actions/checkout@v3
11+
- name: Create library ZIP
12+
run: |
13+
mkdir BlenderServoAnimation
14+
cp -R src examples library.properties README.md LICENSE BlenderServoAnimation
15+
zip -r blender_servo_animation_arduino_library BlenderServoAnimation
16+
- name: Archive library ZIP
17+
uses: actions/upload-artifact@v3
18+
with:
19+
name: blender_servo_animation_arduino_library.zip
20+
path: |
21+
blender_servo_animation_arduino_library.zip
22+
build-examples:
23+
name: "Build examples"
24+
runs-on: ubuntu-latest
25+
strategy:
26+
fail-fast: false
27+
matrix:
28+
example:
29+
[
30+
AdafruitPCA9685,
31+
SerialLiveMode,
32+
WebSocketLiveMode,
33+
MultiplePCA9685,
34+
Show,
35+
StandardServoLib,
36+
SwitchModeButton,
37+
]
38+
steps:
39+
- uses: actions/checkout@v3
40+
- uses: actions/cache@v3
41+
with:
42+
path: |
43+
~/.cache/pip
44+
~/.platformio/.cache
45+
key: ${{ runner.os }}-pio
46+
- uses: actions/setup-python@v4
47+
with:
48+
python-version: "3.10"
49+
- name: Install PlatformIO Core
50+
run: pip install --upgrade platformio
51+
- name: Build PlatformIO examples
52+
run: |
53+
cd examples/${{ matrix.example }}
54+
pio run
2255
lint:
2356
name: "Lint"
2457
runs-on: ubuntu-22.04
2558
steps:
26-
- uses: actions/checkout@v3
27-
- name: Install dependencies
28-
run: |
29-
sudo apt-get update
30-
sudo apt-get install --no-install-recommends -y clang-format-13
31-
- name: Lint with clang-format
32-
run: |
33-
find src test -regex '.*\.\(cpp\|h\)' -exec clang-format-13 --dry-run --Werror {} \;
34-
find examples -regex '.*\.ino' -exec clang-format-13 --dry-run --Werror {} \;
35-
- name: Lint with arduino-lint
36-
uses: arduino/arduino-lint-action@v1
37-
with:
38-
library-manager: update
59+
- uses: actions/checkout@v3
60+
- name: Lint with clang-format
61+
uses: jidicula/[email protected]
62+
with:
63+
exclude-regex: 'examples\/.+\/.+\.h'
64+
- name: Lint with arduino-lint
65+
uses: arduino/arduino-lint-action@v1
66+
with:
67+
library-manager: update
3968
test:
4069
name: "Test"
4170
runs-on: ubuntu-latest
42-
env:
43-
LD_LIBRARY_PATH: /home/runner/.platformio/packages/tool-simavr/lib
4471
steps:
45-
- uses: actions/checkout@v3
46-
- name: Cache PlatformIO directory
47-
uses: actions/cache@v3
48-
with:
49-
path: /home/runner/.platformio
50-
key: platformio-dir
51-
- name: Set up Python 3.10
52-
uses: actions/setup-python@v3
53-
with:
54-
python-version: "3.10"
55-
cache: 'pip'
56-
cache-dependency-path: requirements-dev.txt
57-
- name: Install dependencies
58-
run: |
59-
sudo apt-get update
60-
sudo apt-get install --no-install-recommends -y libelf-dev
61-
python -m pip install --upgrade pip
62-
pip install -r requirements-dev.txt
63-
- name: Run tests with platformio
64-
run: |
65-
pio test --without-uploading
72+
- uses: actions/checkout@v3
73+
- name: Cache PlatformIO directory
74+
uses: actions/cache@v3
75+
with:
76+
path: /home/runner/.platformio
77+
key: platformio-dir
78+
- name: Set up Python 3.10
79+
uses: actions/setup-python@v4
80+
with:
81+
python-version: "3.10"
82+
cache: "pip"
83+
cache-dependency-path: requirements-dev.txt
84+
- name: Install dependencies
85+
run: |
86+
sudo apt-get update
87+
sudo apt-get install --no-install-recommends -y libelf-dev
88+
python -m pip install --upgrade pip
89+
pip install -r requirements-dev.txt
90+
- name: Run tests with platformIO
91+
run: |
92+
pio test -e native

.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -3,3 +3,5 @@
33
.vscode/c_cpp_properties.json
44
.vscode/launch.json
55
.vscode/ipch
6+
examples/*/.vscode
7+
examples/*/.gitignore

README.md

+127-13
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@
22

33
This library helps to control servos based on an exported Blender animation. It is specifically designed to work with the [Blender Servo Animation Add-on](https://github.com/timhendriks93/blender-servo-animation).
44

5+
[![Continuous Integration](https://github.com/timhendriks93/blender-servo-animation-arduino/actions/workflows/ci.yml/badge.svg)](https://github.com/timhendriks93/blender-servo-animation-arduino/actions/workflows/ci.yml)
6+
57
## Installation
68

79
Please refer to the official [Arduino documentation](https://docs.arduino.cc/software/ide-v1/tutorials/installing-libraries) to see how you can install this library.
@@ -27,6 +29,9 @@ BlenderServoAnimation::Servo(...);
2729

2830
// Blender animation object
2931
BlenderServoAnimation::Animation(...);
32+
33+
// Blender show object
34+
BlenderServoAnimation::Show();
3035
```
3136
3237
When not using the standard servo library, you can use the namespace and therefore skip the namespace prefix:
@@ -39,6 +44,9 @@ Servo(...);
3944
4045
// Blender animation object
4146
Animation(...);
47+
48+
// Blender show object
49+
Show();
4250
```
4351

4452
## Defining Servos
@@ -61,7 +69,9 @@ Servo(id, callback, threshold);
6169
| id | byte | Unique servo ID as specified via the Add-on |
6270
| positions | const int[] | Exported positions per frame |
6371
| callback | void (byte, int) | Function to trigger when a servo is moved |
64-
| threshold | byte | Max allowed position diff (default=20) |
72+
| threshold | byte | Max allowed position diff (default=0 / no threshold handling) |
73+
74+
> Note: the threshold is also used to define the speed for moving a servo to its neutral position when stopping an animation.
6575
6676
### Callback Function
6777
@@ -125,20 +135,21 @@ This is usually done inside the `setup` function after the servo objects have be
125135
Alternatively, we can also create an array of servos and call the `addServos` method instead:
126136

127137
```ino
138+
Animation myBlenderAnimation(30, 1000);
139+
128140
Servo myBlenderServos[] = {
129141
Servo(0, BoneA, move),
130142
Servo(1, BoneB, move),
131143
Servo(2, BoneC, move),
132-
...
133144
}
134145

135-
Animation myBlenderAnimation(30, 1000);
136-
137146
void setup() {
138-
myBlenderAnimation.addServos(myBlenderServos);
147+
myBlenderAnimation.addServos(myBlenderServos, 3);
139148
}
140149
```
141150
151+
> Note: the `addServos` function expects the amount of servos in the array to be passed via the second argument.
152+
142153
### Updating the Animation State
143154
144155
The animation needs to be triggered regularly in order to update its state and check if any servos have to be moved. We therefore need to call the `run` method during each `loop`:
@@ -174,12 +185,7 @@ myBlenderAnimation.live(stream);
174185

175186
> Note: the default mode can not be triggered as it is only handled internally.
176187
177-
When calling the `stop` method, it is possible to pass a delay in milliseconds. This delay will be used when the servos are moved to their neutral position during the stop mode. To get to the neutral position, the current position will either be increased or decreased by 1. The delay therefore controls how fast or smooth this movement will take place. The default value for this parameter is `20`.
178-
179-
```ino
180-
myBlenderAnimation.stop(); // Default reset speed applied
181-
myBlenderAnimation.stop(30); // Servos will reset slower
182-
```
188+
When calling the `stop` method, the threshold values of the animation's servos are considered to control how fast or smooth they are moving towards their neutral position. Keep in mind that the servos will not have a threshold value by default which results in the stop mode to immediately trigger the neutral position of the servos. A slower and safer movement can be achieved by setting the threshold values as low as possible with the actual animation still able to run properly.
183189

184190
To use the `live` method, we have to pass a stream instance which will be used for reading serial commands. For example, we can pass `Serial` if we want to use the standard USB connection of an Arduino compatible board:
185191

@@ -190,6 +196,8 @@ void setup() {
190196
}
191197
```
192198

199+
This library also comes with a `LiveStream` class which allows for a more generic way to listen to live commands. For example, it can be used as part of the web socket based live mode for which you can find a dedicated example [here](examples/WebSocketLiveMode).
200+
193201
To get the current animation mode, we can simply call the `getMode` method. This will return a `byte` representing one of the mode constants mentioned in the table above. We can then compare the return value to those constants to act according to the current mode:
194202

195203
```ino
@@ -210,12 +218,118 @@ On top of manually checking the animation mode, we can also register a callback
210218

211219
```ino
212220
void modeChanged(byte prevMode, byte newMode) {
213-
// Do something (e.g. using another switch)
221+
// Do something (e.g. using a switch statement)
214222
}
215223

216224
void setup() {
217225
myBlenderAnimation.onModeChange(modeChanged);
218226
}
219227
```
220228
221-
The [SwitchModeButton example](examples/SwitchModeButton) shows how to combine all mode methods to control an animation based on a single button. Make sure to also check out the other [examples](examples) to get started quickly.
229+
The [SwitchModeButton example](examples/SwitchModeButton) shows how to combine all mode methods to control an animation based on a single button.
230+
231+
## Defining a Show
232+
233+
A show object allows you to combine multiple animations and control their play back in an easy way. You can also think of a show as a playlist of animations. Since the show object does not expect any arguments, the initialization is very simple:
234+
235+
```ino
236+
Show myBlenderShow;
237+
```
238+
239+
### Registering Animations
240+
241+
After defining some animations as shown above, we have to register them to the show object by calling the `addAnimation` method:
242+
243+
```ino
244+
myBlenderShow.addAnimation(myBlenderAnimation);
245+
```
246+
247+
This is usually done inside the `setup` function after the animation and servo objects have been defined globally (outside of any function like `setup` or `loop`).
248+
249+
Alternatively, we can also create an array of animations and call the `addAnimations` method instead:
250+
251+
```ino
252+
Show myBlenderShow;
253+
254+
Animation animations[3] = {
255+
{FPS, FRAMES_A},
256+
{FPS, FRAMES_B},
257+
{FPS, FRAMES_C},
258+
};
259+
260+
void setup() {
261+
myBlenderShow.addAnimations(animations, 3);
262+
}
263+
```
264+
265+
> Note: the `addAnimations` function expects the amount of servos in the array to be passed via the second argument.
266+
267+
### Updating the Show State
268+
269+
Just like single animations, we have to regularly trigger the show instance in order to update its state and internally handle the servo movement of the current animation. We therefore need to call the `run` method during each `loop`:
270+
271+
```ino
272+
void loop() {
273+
myBlenderShow.run();
274+
}
275+
```
276+
277+
### Show Modes
278+
279+
The show modes are similar to the previously mentioned animation modes. In addition to those, there are various playback modes to handle a multitude of animations.
280+
281+
Just like with animation modes, a show will be in the default mode at first. The following table is focusing on the differences and additions of the show modes compared to the animation modes:
282+
283+
| Constant | Method | Description |
284+
|----------|--------|-------------|
285+
| MODE_PLAY | play() | Start or resume playing the show once |
286+
| MODE_PLAY_SINGLE | playSingle(index) | Start or resume playing a single animation once |
287+
| MODE_PLAY_RANDOM | playRandom() | Start or resume randomly playing animations of the show |
288+
289+
The modes can be changed or triggered by calling the respective methods on the show object:
290+
291+
```ino
292+
myBlenderShow.play();
293+
myBlenderShow.playSingle(index);
294+
myBlenderShow.playRandom();
295+
myBlenderShow.pause();
296+
myBlenderShow.loop();
297+
myBlenderShow.stop();
298+
myBlenderShow.live(stream);
299+
```
300+
301+
> Note: the default mode can not be triggered as it is only handled internally.
302+
303+
To get the current show mode, we can again call a `getMode` method. This will return a `byte` representing one of the mode constants mentioned in the table above. We can then compare the return value to those constants to act according to the current mode:
304+
305+
```ino
306+
byte currentMode = myBlenderShow.getMode();
307+
308+
switch (currentMode) {
309+
case Show::MODE_DEFAULT:
310+
// Do something
311+
break;
312+
case Show::MODE_PLAY:
313+
// Do something else
314+
break;
315+
...
316+
}
317+
```
318+
319+
> Note: The actual byte values of the show modes differ from the animation modes. For example, `Show::MODE_PAUSE != Animation::MODE_PAUSE`.
320+
321+
As with animations, we can also register an `onModeChange` callback function:
322+
323+
```ino
324+
void modeChanged(byte prevMode, byte newMode) {
325+
// Do something (e.g. using a switch statement)
326+
}
327+
328+
void setup() {
329+
myBlenderShow.onModeChange(modeChanged);
330+
}
331+
```
332+
333+
There is also a specific [Show example](examples/Show) to illustrate a simple setup based on 2 different animations.
334+
335+
Make sure to also check out the other [examples](examples) to get started more quickly.

examples/AdafruitPCA9685/AdafruitPCA9685.ino

-6
Original file line numberDiff line numberDiff line change
@@ -13,12 +13,6 @@
1313
// Using the namespace to have short class references (Animation and Servo)
1414
using namespace BlenderServoAnimation;
1515

16-
// Frames per second - see original Blender animation / ik.h
17-
#define FPS 30
18-
19-
// Total animation frames - see original Blender animation / ik.h
20-
#define FRAMES 100
21-
2216
// PWM driver instance to set PWM output
2317
Adafruit_PWMServoDriver pwm = Adafruit_PWMServoDriver();
2418

examples/AdafruitPCA9685/README.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
# Adafruit PCA9685
2+
3+
Using a PCA9685 PWM Servo Driver to animate 2 servos. The animation is based on the IK example of the Blender Servo Animation add-on which resembles a simple neck mechanism.
4+
5+
![Arduino Nano with PCA9685](../../images/arduino-nano-with-PCA9685.png)

examples/AdafruitPCA9685/ik.h

+6-2
Original file line numberDiff line numberDiff line change
@@ -6,19 +6,23 @@
66
Seconds: 3
77
Bones: 2
88
Armature: Armature
9+
Scene: Scene
910
File: ik.blend
1011
*/
1112

1213
#include <Arduino.h>
1314

15+
const byte FPS = 30;
16+
const int FRAMES = 100;
17+
1418
// Servo ID: 0
15-
const int NeckLeft[100] PROGMEM = {
19+
const int NeckLeft[FRAMES] PROGMEM = {
1620
375, 376, 377, 380, 384, 387, 391, 396, 400, 403, 406, 408, 410, 410, 409, 406, 402, 396, 390, 382, 373, 364, 355, 346, 338, 330, 323, 318, 314, 311, 309, 307, 305, 305, 305, 305, 306, 307, 309, 311, 314, 317, 320, 323, 327, 331, 335, 339, 343, 347,
1721
351, 356, 360, 364, 369, 374, 380, 386, 392, 398, 404, 410, 415, 420, 425, 429, 432, 435, 436, 437, 437, 436, 435, 434, 432, 430, 428, 426, 423, 421, 418, 415, 412, 409, 406, 403, 400, 397, 394, 391, 388, 386, 384, 381, 380, 378, 377, 376, 375, 375,
1822
};
1923

2024
// Servo ID: 1
21-
const int NeckRight[100] PROGMEM = {
25+
const int NeckRight[FRAMES] PROGMEM = {
2226
375, 376, 379, 383, 388, 394, 401, 409, 417, 426, 434, 443, 450, 457, 463, 468, 471, 472, 471, 469, 466, 462, 457, 452, 447, 441, 437, 432, 428, 424, 420, 416, 411, 407, 402, 398, 394, 389, 384, 380, 375, 370, 366, 361, 356, 352, 347, 342, 337, 333,
2327
328, 324, 319, 315, 312, 309, 308, 307, 306, 306, 306, 307, 308, 309, 310, 311, 312, 313, 313, 313, 313, 314, 315, 316, 318, 320, 322, 324, 327, 329, 332, 335, 338, 341, 344, 347, 350, 353, 356, 359, 362, 364, 366, 369, 370, 372, 373, 374, 375, 375,
2428
};

0 commit comments

Comments
 (0)