278 lines
4.4 KiB
C++
Raw Normal View History

2021-01-29 12:31:58 +01:00
//
// FILE: HeartBeat.cpp
// AUTHOR: Rob Tillaart
2022-07-27 12:53:36 +02:00
// VERSION: 0.3.1
2021-01-29 12:31:58 +01:00
// PURPOSE: Arduino library for HeartBeat with frequency and dutyCycle
// DATE: 2019-06-12
// URL: https://github.com/RobTillaart/HeartBeat
//
2022-11-09 10:31:43 +01:00
// HISTORY: see changelog
2021-01-29 12:31:58 +01:00
#include "HeartBeat.h"
HeartBeat::HeartBeat()
{
}
void HeartBeat::begin(const uint8_t pin, float frequency)
{
_pin = pin;
pinMode(_pin, OUTPUT);
setFrequency(frequency);
enable();
}
void HeartBeat::setFrequency(float frequency)
{
2021-11-02 17:26:51 +01:00
_frequency = frequency;
if (_frequency < 0) _frequency = -frequency;
if (_frequency < 0.001) _frequency = 0.001;
2021-01-29 12:31:58 +01:00
_setFreqDuty();
}
void HeartBeat::setDutyCycle(float dutyCycle)
{
2022-07-27 12:53:36 +02:00
_dutyCycle = dutyCycle;
if (_dutyCycle < 0) _dutyCycle = 0;
if (_dutyCycle > 100) _dutyCycle = 100;
2021-01-29 12:31:58 +01:00
_setFreqDuty();
}
2021-05-28 13:30:27 +02:00
2022-07-27 12:53:36 +02:00
float HeartBeat::getFrequency()
{
return _frequency;
};
float HeartBeat::getDutyCycle()
{
return _dutyCycle;
};
void HeartBeat::enable()
{
_running = true;
};
void HeartBeat::disable()
{
_running = false;
};
bool HeartBeat::isEnabled()
{
return _running;
};
2021-01-29 12:31:58 +01:00
void HeartBeat::beat()
{
2022-02-05 16:53:02 +01:00
if (_running == false)
2021-01-29 12:31:58 +01:00
{
2021-11-02 17:26:51 +01:00
_state = LOW;
2021-01-29 12:31:58 +01:00
}
else
{
uint32_t now = micros();
2021-11-02 17:26:51 +01:00
if ((_state == LOW) && (now - _lastHeartBeat) < _dutyCycleLow) return;
if ((_state == HIGH) && (now - _lastHeartBeat) < _dutyCycleHigh) return;
2021-01-29 12:31:58 +01:00
_lastHeartBeat = now;
2021-11-02 17:26:51 +01:00
_state = !_state;
2021-01-29 12:31:58 +01:00
}
2021-11-02 17:26:51 +01:00
digitalWrite(_pin, _state);
2021-01-29 12:31:58 +01:00
}
2022-07-27 12:53:36 +02:00
uint8_t HeartBeat::getState()
{
return _state;
};
2022-02-05 16:53:02 +01:00
2021-01-29 12:31:58 +01:00
/////////////////////////////////
2021-11-02 17:26:51 +01:00
//
2022-02-05 16:53:02 +01:00
// PROTECTED
2021-11-02 17:26:51 +01:00
//
2021-01-29 12:31:58 +01:00
void HeartBeat::_setFreqDuty()
{
float time = 10000.0/_frequency;
_dutyCycleHigh = round(_dutyCycle * time);
_dutyCycleLow = round((100 - _dutyCycle) * time);
}
2022-02-05 16:53:02 +01:00
/////////////////////////////////////////////////////////////////////////////
//
// HEARTBEATDIAG
//
HeartBeatDiag::HeartBeatDiag():HeartBeat()
{
}
void HeartBeatDiag::beat()
{
// normal mode
if (_codeMask == 0)
{
_codeStart = 0;
HeartBeat::beat();
return;
}
// _code mode
if (_codeStart == 0)
{
// force a LOW first.
_codeStart = 1;
_lastHeartBeat = micros();
_state = LOW;
}
else
{
uint32_t period = (_dutyCycleLow + _dutyCycleHigh)/2;
uint32_t now = micros();
if ((now - _lastHeartBeat) < period) return;
_lastHeartBeat = now;
if (_state == LOW)
{
while (_codeMask > _code)
{
_codeMask /= 10;
}
if (_codeMask == 0) return;
_pulseLength = _code / _codeMask;
_code -= ( _pulseLength * _codeMask );
_state = HIGH;
}
else
{
_pulseLength--;
if (_pulseLength == 0)
{
_state = LOW;
}
}
}
digitalWrite(_pin, _state);
}
bool HeartBeatDiag::code(uint32_t pattern)
{
// already running an errorCode?
if (_code > 0) return false;
// pattern too long
if (pattern > 999999999) return false;
_code = pattern;
_codeMask = 100000000;
return true;
}
2022-07-27 12:53:36 +02:00
void HeartBeatDiag::codeOff()
{
_codeMask = 0;
}
2022-02-05 16:53:02 +01:00
/////////////////////////////////////////////////////////////////////////////
//
// HEARTBEATSL
//
// string LS version is simpler and somewhat smaller footprint,
//
#define HEARTBEATSL_SHORT 1
#define HEARTBEATSL_LONG 3
HeartBeatSL::HeartBeatSL():HeartBeat()
{
}
void HeartBeatSL::beat()
{
// normal mode
if (_codeMask == 0)
{
_codeStart = 0;
HeartBeat::beat();
return;
}
// _code mode
if (_codeStart == 0)
{
// force a LOW first.
_codeStart = 1;
_lastHeartBeat = micros();
_state = LOW;
}
else
{
uint32_t period = (_dutyCycleLow + _dutyCycleHigh)/2;
uint32_t now = micros();
if ((now - _lastHeartBeat) < period) return;
_lastHeartBeat = now;
if (_state == LOW)
{
_pulseLength = HEARTBEATSL_SHORT;
if (_code & _codeMask) // 1 ==> LONG
{
_pulseLength = HEARTBEATSL_LONG;
}
_codeMask >>= 1;
_state = HIGH;
}
else
{
_pulseLength--;
if (_pulseLength == 0)
{
_state = LOW;
}
}
}
digitalWrite(_pin, _state);
}
bool HeartBeatSL::code(const char * str)
{
// already running an errorCode?
if (_codeMask > 0) return false;
// pattern too long
uint8_t len = strlen(str);
if (len > 7) return false;
_code = 0;
_codeMask = 1 << len;
for (uint8_t i = 0; i < len; i++)
{
if (str[i] == 'L') _code |= 1;
_code <<= 1;
}
return true;
}
2022-07-27 12:53:36 +02:00
void HeartBeatSL::codeOff()
{
_codeMask = 0;
}
2021-01-29 12:31:58 +01:00
// -- END OF FILE --
2021-11-02 17:26:51 +01:00