Skip to content

Commit 3d3c3e4

Browse files
committed
update readme
1 parent 950ea2a commit 3d3c3e4

File tree

1 file changed

+46
-124
lines changed

1 file changed

+46
-124
lines changed

README.md

+46-124
Original file line numberDiff line numberDiff line change
@@ -4,15 +4,17 @@ This library helps to control servos based on an exported Blender animation. It
44

55
[![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)
66

7-
## Installation
7+
## First Steps
8+
9+
### Installation
810

911
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.
1012

11-
## Quick Start
13+
### Quick Start
1214

1315
Take a look at the [StandardServoLib](examples/StandardServoLib) example to get started quickly. It represents the most simple setup and is based on the standard Servo Arduino library with which you might already be familiar.
1416

15-
## Usage
17+
### Usage
1618

1719
To start using this library, add the following include statement to your script or sketch:
1820

@@ -30,7 +32,7 @@ Start simply by creating a new animation instance at the outer scope of your ske
3032
BlenderServoAnimation animation;
3133
```
3234

33-
### Define the Position Change Callback
35+
### Register a Position Change Callback
3436

3537
To specify what should happen when a servo needs to be moved to a new position, we have to define a callback function. It receives 2 arguments - the servo ID as `byte` and the new position as `int`:
3638

@@ -52,77 +54,87 @@ animation.onPositionChange(move);
5254

5355
Before we can play back and control an animation, we first have to add at least one scene to our animation. There are 2 ways to do this depending on the type of animation data you want to provide:
5456

55-
1. test
57+
```ino
58+
animation.addScene(data, size, fps, frames);
59+
animation.addScene(stream, fps, frames);
60+
```
5661

5762
| Parameter | Type | Description |
5863
|-----------|------|-------------|
64+
| data | const byte[] | Exported position data |
65+
| stream | Stream | Stream instance to read positions from |
66+
| size | int | Size of the position data |
5967
| fps | byte | Frames per second as specified in Blender |
6068
| frames | int | Total amount of frames as specified in Blender |
6169

62-
In general, the provided values should align with the Blender animation you would like to export or work on. Given that the Blender animation consists of 1000 frames and plays back at a rate of 30 frames per second, the animation object should be created as follows:
63-
64-
```ino
65-
Animation myBlenderAnimation(30, 1000);
66-
```
70+
> Note: make sure the `fps` and `frames` values align exactly with the respective settings of the animation in Blender.
6771
6872
### Updating the Animation State
6973

7074
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`:
7175

7276
```ino
7377
void loop() {
74-
myBlenderAnimation.run();
78+
animation.run();
7579
}
7680
```
7781

82+
## Controlling an Animation
83+
7884
### Animation Modes
7985

8086
At first, an animation will be in the default mode. In this mode, the animation is simply not doing anything and waits until the mode has changed.
8187

8288
| Constant | Method | Description |
8389
|----------|--------|-------------|
84-
| MODE_DEFAULT | | Not playing / waiting |
85-
| MODE_PLAY | play() | Start or resume playing the animation once |
86-
| MODE_PAUSE | pause() | Pausing the animation at the current frame |
87-
| MODE_STOP | stop() | Slowly moving the servos to their neutral position |
88-
| MODE_LOOP | loop() | Start or resume playing the animation in a loop |
89-
| MODE_LIVE | live(stream) | Reading serial commands to move the servos in real-time |
90+
| MODE_DEFAULT | n/a | Not playing / waiting |
91+
| MODE_PLAY | play() | Start or resume playing all scenes once |
92+
| MODE_PLAY_SINGLE | playSingle(index) | Start or resume playing a single scene once |
93+
| MODE_PLAY_RANDOM | playRandom() | Start or resume randomly playing scenes |
94+
| MODE_PAUSE | pause() | Pausing the scene at the current frame |
95+
| MODE_STOP | stop() | Slowly moving the servos to their starting position |
96+
| MODE_LOOP | loop() | Start or resume playing all scenes in a loop |
97+
| MODE_LIVE | live()/live(stream) | Reading serial commands to move the servos in real-time |
9098

9199
The modes can be changed or triggered by calling the above methods on the animation object:
92100

93101
```ino
94-
myBlenderAnimation.play();
95-
myBlenderAnimation.pause();
96-
myBlenderAnimation.loop();
97-
myBlenderAnimation.stop();
98-
myBlenderAnimation.live(stream);
102+
animation.play();
103+
animation.playSingle(index);
104+
animation.playRandom();
105+
animation.pause();
106+
animation.loop();
107+
animation.stop();
108+
animation.live(stream);
99109
```
100110

101111
> Note: the default mode can not be triggered as it is only handled internally.
102112
103113
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.
104114

115+
### Live Mode
116+
105117
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:
106118

107119
```ino
108120
void setup() {
109121
Serial.begin(115200);
110-
myBlenderAnimation.live(Serial);
122+
animation.live(Serial);
111123
}
112124
```
113125

114-
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).
126+
### React to Mode Changes
115127

116128
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:
117129

118130
```ino
119-
byte currentMode = myBlenderAnimation.getMode();
131+
byte currentMode = animation.getMode();
120132

121133
switch (currentMode) {
122-
case Animation::MODE_DEFAULT:
134+
case BlenderServoAnimation::MODE_DEFAULT:
123135
// Do something
124136
break;
125-
case Animation::MODE_PLAY:
137+
case BlenderServoAnimation::MODE_PLAY:
126138
// Do something else
127139
break;
128140
...
@@ -137,114 +149,24 @@ void modeChanged(byte prevMode, byte newMode) {
137149
}
138150

139151
void setup() {
140-
myBlenderAnimation.onModeChange(modeChanged);
141-
}
142-
```
143-
144-
The [SwitchModeButton example](examples/SwitchModeButton) shows how to combine all mode methods to control an animation based on a single button.
145-
146-
## Defining a Show
147-
148-
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:
149-
150-
```ino
151-
Show myBlenderShow;
152-
```
153-
154-
### Registering Animations
155-
156-
After defining some animations as shown above, we have to register them to the show object by calling the `addAnimation` method:
157-
158-
```ino
159-
myBlenderShow.addAnimation(myBlenderAnimation);
160-
```
161-
162-
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`).
163-
164-
Alternatively, we can also create an array of animations and call the `addAnimations` method instead:
165-
166-
```ino
167-
Show myBlenderShow;
168-
169-
Animation animations[3] = {
170-
{FPS, FRAMES_A},
171-
{FPS, FRAMES_B},
172-
{FPS, FRAMES_C},
173-
};
174-
175-
void setup() {
176-
myBlenderShow.addAnimations(animations, 3);
177-
}
178-
```
179-
180-
> Note: the `addAnimations` function expects the amount of servos in the array to be passed via the second argument.
181-
182-
### Updating the Show State
183-
184-
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`:
185-
186-
```ino
187-
void loop() {
188-
myBlenderShow.run();
152+
animation.onModeChange(modeChanged);
189153
}
190154
```
191155
192-
### Show Modes
193-
194-
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.
195-
196-
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:
197-
198-
| Constant | Method | Description |
199-
|----------|--------|-------------|
200-
| MODE_PLAY | play() | Start or resume playing the show once |
201-
| MODE_PLAY_SINGLE | playSingle(index) | Start or resume playing a single animation once |
202-
| MODE_PLAY_RANDOM | playRandom() | Start or resume randomly playing animations of the show |
203-
204-
The modes can be changed or triggered by calling the respective methods on the show object:
205-
206-
```ino
207-
myBlenderShow.play();
208-
myBlenderShow.playSingle(index);
209-
myBlenderShow.playRandom();
210-
myBlenderShow.pause();
211-
myBlenderShow.loop();
212-
myBlenderShow.stop();
213-
myBlenderShow.live(stream);
214-
```
215-
216-
> Note: the default mode can not be triggered as it is only handled internally.
156+
### React to Scene Changes
217157
218-
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:
158+
To execute logic whenever there is a transition into a different scene, another callback function can be registered. The function will receive the index of both the previous and the new scene as `byte` values. To register the function, we can call the `onSceneChange` method:
219159
220160
```ino
221-
byte currentMode = myBlenderShow.getMode();
222-
223-
switch (currentMode) {
224-
case Show::MODE_DEFAULT:
225-
// Do something
226-
break;
227-
case Show::MODE_PLAY:
228-
// Do something else
229-
break;
230-
...
231-
}
232-
```
233-
234-
> Note: The actual byte values of the show modes differ from the animation modes. For example, `Show::MODE_PAUSE != Animation::MODE_PAUSE`.
235-
236-
As with animations, we can also register an `onModeChange` callback function:
237-
238-
```ino
239-
void modeChanged(byte prevMode, byte newMode) {
240-
// Do something (e.g. using a switch statement)
161+
void sceneChanged(byte prevSceneIndex, byte newSceneIndex) {
162+
// Do something (e.g. opening another animation file on an SD card)
241163
}
242164
243165
void setup() {
244-
myBlenderShow.onModeChange(modeChanged);
166+
animation.onSceneChange(sceneChanged);
245167
}
246168
```
247169

248-
There is also a specific [Show example](examples/Show) to illustrate a simple setup based on 2 different animations.
170+
## Examples
249171

250172
Make sure to also check out the other [examples](examples) to get started more quickly.

0 commit comments

Comments
 (0)