Add menu options for using cv input as Clock Run/Reset #25

Merged
awonak merged 4 commits from run-reset into main 2025-08-10 00:25:07 +00:00
8 changed files with 242 additions and 101 deletions
Showing only changes of commit fd2ea45509 - Show all commits

View File

@ -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();
} }
} }

View File

@ -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];

View File

@ -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];
}; };

View File

@ -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_;

View File

@ -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) {