Add menu options for using cv input as Clock Run/Reset #25
@ -91,18 +91,15 @@ void loop() {
|
|||||||
// Process change in state of inputs and outputs.
|
// Process change in state of inputs and outputs.
|
||||||
gravity.Process();
|
gravity.Process();
|
||||||
|
|
||||||
// Read CVs and call the update function for each channel.
|
// Check if cv run or reset is active and read cv.
|
||||||
int cv1 = gravity.cv1.Read();
|
CheckRunReset(gravity.cv1, gravity.cv2);
|
||||||
int cv2 = gravity.cv2.Read();
|
|
||||||
|
|
||||||
CheckRunReset(cv1, cv2);
|
|
||||||
|
|
||||||
for (int i = 0; i < Gravity::OUTPUT_COUNT; i++) {
|
for (int i = 0; i < Gravity::OUTPUT_COUNT; i++) {
|
||||||
auto& ch = app.channel[i];
|
auto& ch = app.channel[i];
|
||||||
// Only apply CV to the channel when the current channel has cv
|
// Only apply CV to the channel when the current channel has cv
|
||||||
// mod configured.
|
// mod configured.
|
||||||
if (ch.isCvModActive()) {
|
if (ch.isCvModActive()) {
|
||||||
ch.applyCvMod(cv1, cv2);
|
ch.applyCvMod(gravity.cv1.Read(), gravity.cv2.Read());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -173,21 +170,21 @@ void HandleExtClockTick() {
|
|||||||
app.refresh_screen = true;
|
app.refresh_screen = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void CheckRunReset(int cv1, int cv2) {
|
void CheckRunReset(AnalogInput& cv1, AnalogInput& cv2) {
|
||||||
if (app.cv_run == 1 && cv1 > 255 && !gravity.clock.IsPaused()) {
|
// Clock Run
|
||||||
gravity.clock.Stop();
|
if (app.cv_run == 1 || app.cv_run == 2) {
|
||||||
} else if (app.cv_run == 1 && cv1 < 255 && gravity.clock.IsPaused()) {
|
const int val = (app.cv_run == 1) ? cv1.Read() : cv2.Read();
|
||||||
gravity.clock.Start();
|
if (val > AnalogInput::GATE_THRESHOLD && gravity.clock.IsPaused()) {
|
||||||
} else if (app.cv_run == 2 && cv2 > 255 && !gravity.clock.IsPaused()) {
|
gravity.clock.Start();
|
||||||
gravity.clock.Stop();
|
} else if (val < AnalogInput::GATE_THRESHOLD && !gravity.clock.IsPaused()) {
|
||||||
} else if (app.cv_run == 2 && cv2 < 255 && gravity.clock.IsPaused()) {
|
gravity.clock.Stop();
|
||||||
gravity.clock.Start();
|
ResetOutputs();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Clock Reset
|
// Clock Reset
|
||||||
if (app.cv_reset == 1 && cv1 > 255) {
|
if ((app.cv_reset == 1 && cv1.IsRisingEdge(AnalogInput::GATE_THRESHOLD)) ||
|
||||||
gravity.clock.Reset();
|
(app.cv_reset == 2 && cv2.IsRisingEdge(AnalogInput::GATE_THRESHOLD))) {
|
||||||
} else if (app.cv_reset == 2 && cv2 < 255) {
|
|
||||||
gravity.clock.Reset();
|
gravity.clock.Reset();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -17,7 +17,7 @@
|
|||||||
|
|
||||||
// Define the constants for the current firmware.
|
// Define the constants for the current firmware.
|
||||||
const char StateManager::SKETCH_NAME[] = "ALT GRAVITY";
|
const char StateManager::SKETCH_NAME[] = "ALT GRAVITY";
|
||||||
const char StateManager::SEMANTIC_VERSION[] = "V2.0.0BETA3"; // NOTE: This should match the version in the library.properties file.
|
const char StateManager::SEMANTIC_VERSION[] = "V2.0.0BETA4"; // NOTE: This should match the version in the library.properties file.
|
||||||
|
|
||||||
// Number of available save slots.
|
// Number of available save slots.
|
||||||
const byte StateManager::MAX_SAVE_SLOTS = 10;
|
const byte StateManager::MAX_SAVE_SLOTS = 10;
|
||||||
@ -87,6 +87,8 @@ void StateManager::reset(AppState& app) {
|
|||||||
app.selected_channel = default_app.selected_channel;
|
app.selected_channel = default_app.selected_channel;
|
||||||
app.selected_source = default_app.selected_source;
|
app.selected_source = default_app.selected_source;
|
||||||
app.selected_pulse = default_app.selected_pulse;
|
app.selected_pulse = default_app.selected_pulse;
|
||||||
|
app.cv_run = default_app.cv_run;
|
||||||
|
app.cv_reset = default_app.cv_reset;
|
||||||
|
|
||||||
for (int i = 0; i < Gravity::OUTPUT_COUNT; i++) {
|
for (int i = 0; i < Gravity::OUTPUT_COUNT; i++) {
|
||||||
app.channel[i].Init();
|
app.channel[i].Init();
|
||||||
@ -140,6 +142,8 @@ void StateManager::_saveState(const AppState& app, byte slot_index) {
|
|||||||
save_data.selected_channel = app.selected_channel;
|
save_data.selected_channel = app.selected_channel;
|
||||||
save_data.selected_source = static_cast<byte>(app.selected_source);
|
save_data.selected_source = static_cast<byte>(app.selected_source);
|
||||||
save_data.selected_pulse = static_cast<byte>(app.selected_pulse);
|
save_data.selected_pulse = static_cast<byte>(app.selected_pulse);
|
||||||
|
save_data.cv_run = app.cv_run;
|
||||||
|
save_data.cv_reset = app.cv_reset;
|
||||||
|
|
||||||
// TODO: break this out into a separate function. Save State should be
|
// TODO: break this out into a separate function. Save State should be
|
||||||
// broken out into global / per-channel save methods. When saving via
|
// broken out into global / per-channel save methods. When saving via
|
||||||
@ -179,6 +183,8 @@ void StateManager::_loadState(AppState& app, byte slot_index) {
|
|||||||
app.selected_channel = load_data.selected_channel;
|
app.selected_channel = load_data.selected_channel;
|
||||||
app.selected_source = static_cast<Clock::Source>(load_data.selected_source);
|
app.selected_source = static_cast<Clock::Source>(load_data.selected_source);
|
||||||
app.selected_pulse = static_cast<Clock::Pulse>(load_data.selected_pulse);
|
app.selected_pulse = static_cast<Clock::Pulse>(load_data.selected_pulse);
|
||||||
|
app.cv_run = load_data.cv_run;
|
||||||
|
app.cv_reset = load_data.cv_reset;
|
||||||
|
|
||||||
for (int i = 0; i < Gravity::OUTPUT_COUNT; i++) {
|
for (int i = 0; i < Gravity::OUTPUT_COUNT; i++) {
|
||||||
auto& ch = app.channel[i];
|
auto& ch = app.channel[i];
|
||||||
|
|||||||
@ -76,6 +76,8 @@ class StateManager {
|
|||||||
byte selected_channel;
|
byte selected_channel;
|
||||||
byte selected_source;
|
byte selected_source;
|
||||||
byte selected_pulse;
|
byte selected_pulse;
|
||||||
|
byte cv_run;
|
||||||
|
byte cv_reset;
|
||||||
ChannelState channel_data[Gravity::OUTPUT_COUNT];
|
ChannelState channel_data[Gravity::OUTPUT_COUNT];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@ -19,6 +19,8 @@ const int CALIBRATED_HIGH = 512;
|
|||||||
|
|
||||||
class AnalogInput {
|
class AnalogInput {
|
||||||
public:
|
public:
|
||||||
|
static const int GATE_THRESHOLD = 0;
|
||||||
|
|
||||||
AnalogInput() {}
|
AnalogInput() {}
|
||||||
~AnalogInput() {}
|
~AnalogInput() {}
|
||||||
|
|
||||||
@ -74,6 +76,18 @@ class AnalogInput {
|
|||||||
*/
|
*/
|
||||||
inline float Voltage() { return ((read_ / 512.0) * 5.0); }
|
inline float Voltage() { return ((read_ / 512.0) * 5.0); }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks for a rising edge transition across a threshold.
|
||||||
|
*
|
||||||
|
* @param threshold The value that the input must cross.
|
||||||
|
* @return True if the value just crossed the threshold from below, false otherwise.
|
||||||
|
*/
|
||||||
|
inline bool IsRisingEdge(int16_t threshold) const {
|
||||||
|
bool was_high = old_read_ > threshold;
|
||||||
|
bool is_high = read_ > threshold;
|
||||||
|
return is_high && !was_high;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
uint8_t pin_;
|
uint8_t pin_;
|
||||||
int16_t read_;
|
int16_t read_;
|
||||||
|
|||||||
@ -82,7 +82,6 @@ class DigitalOutput {
|
|||||||
unsigned long last_triggered_;
|
unsigned long last_triggered_;
|
||||||
uint8_t trigger_duration_;
|
uint8_t trigger_duration_;
|
||||||
uint8_t cv_pin_;
|
uint8_t cv_pin_;
|
||||||
uint8_t led_pin_;
|
|
||||||
bool on_;
|
bool on_;
|
||||||
|
|
||||||
void update(uint8_t state) {
|
void update(uint8_t state) {
|
||||||
|
|||||||
Reference in New Issue
Block a user