134 lines
2.7 KiB
C++
Raw Normal View History

2020-02-17 17:22:11 +01:00
//
// FILE: ML8511.cpp
// AUTHOR: Rob.Tillaart@gmail.com
2021-11-09 20:05:19 +01:00
// VERSION: 0.1.7
2020-02-17 17:22:11 +01:00
// PURPOSE: ML8511 - UV sensor - library for Arduino
//
2021-01-29 12:31:58 +01:00
// HISTORY:
// 0.1.0 2020-02-03 initial version
// 0.1.1 2020-02-17 added _voltPerStep() to support more boards
// 0.1.2 2020-06-21 refactor; add estimateDUVindex()
2021-06-20 09:13:49 +02:00
// 0.1.3 2021-01-01 Arduino-ci + unit test
2021-04-25 19:56:44 +02:00
// 0.1.4 2021-04-23 fix for platformIO
2021-06-20 09:13:49 +02:00
// 0.1.5 2021-05-27 fix Arduino-lint
// 0.1.6 2021-06-19 add get/setDUVfactor(),
// rewrite estimateDUVindex(),
// add reset();
2021-11-09 20:05:19 +01:00
// 0.1.7 2021-11-09 update vuild-CI, badges
// add voltage2mW() for external ADC
2021-04-25 19:56:44 +02:00
2020-02-17 17:22:11 +01:00
#include "ML8511.h"
2021-04-25 19:56:44 +02:00
2020-02-17 17:22:11 +01:00
/////////////////////////////////////////////////////
//
// PUBLIC
//
ML8511::ML8511(uint8_t analogPin, uint8_t enablePin)
{
2021-06-20 09:13:49 +02:00
_analogPin = analogPin;
_enablePin = enablePin;
2020-11-27 11:20:37 +01:00
if (enablePin != 0xFF)
{
pinMode(_enablePin, OUTPUT);
digitalWrite(_enablePin, LOW);
_enabled = false;
enable();
}
else
{
_enabled = true;
}
2021-06-20 09:13:49 +02:00
reset();
2020-02-17 17:22:11 +01:00
};
2020-11-27 11:20:37 +01:00
2021-06-20 09:13:49 +02:00
void ML8511::reset()
{
_voltsPerStep = 5.0/1023;
_DUVfactor = 1.61; // https://github.com/RobTillaart/ML8511/issues/4
}
2020-02-17 17:22:11 +01:00
float ML8511::getUV(uint8_t energyMode)
{
2020-11-27 11:20:37 +01:00
if (!_enabled)
{
enable();
// datasheet page 5 states wait for max 1 millisecond
uint32_t start = micros();
while (micros() - start < 1000) yield();
}
2021-11-09 20:05:19 +01:00
// read the sensor
2020-11-27 11:20:37 +01:00
float voltage = analogRead(_analogPin) * _voltsPerStep;
2021-11-09 20:05:19 +01:00
// go to low power mode?
2020-11-27 11:20:37 +01:00
if (energyMode == LOW)
{
disable();
}
2020-02-17 17:22:11 +01:00
2021-11-09 20:05:19 +01:00
return voltage2mW(voltage);
}
// to be used by external ADC
float ML8511::voltage2mW(float voltage)
{
2020-11-27 11:20:37 +01:00
// see datasheet - page 4
// mW/cm2 @ 365 nm
2021-06-20 09:13:49 +02:00
// @ 25 Celsius
2020-11-27 11:20:37 +01:00
// formula estimated on graph
if (voltage <= 1.0)
{
return 0.0;
}
voltage -= 1.0; // subtract for zero point.
float mWcm2 = voltage * (15.0 / 1.8);
return mWcm2;
2020-02-17 17:22:11 +01:00
}
2021-05-28 13:39:01 +02:00
2021-06-20 09:13:49 +02:00
// experimental estimate DUV index ( ==> USE WITH CARE !!)
// use setDUVfactor(float w) to calibrate
//
2020-11-27 11:20:37 +01:00
// input is power in mW per cm2
float ML8511::estimateDUVindex(float mWcm2)
{
2021-06-20 09:13:49 +02:00
// rewrite in 0.1.6
// https://github.com/RobTillaart/ML8511/issues/4
return mWcm2 * _DUVfactor;
};
2020-11-27 11:20:37 +01:00
2021-06-20 09:13:49 +02:00
2021-11-09 20:05:19 +01:00
bool ML8511::setDUVfactor(float factor)
2021-06-20 09:13:49 +02:00
{
2021-11-09 20:05:19 +01:00
if (factor < 0.01) return false; // enforce positive values
_DUVfactor = factor;
2021-06-20 09:13:49 +02:00
return true;
2020-11-27 11:20:37 +01:00
};
2021-05-28 13:39:01 +02:00
2020-02-17 17:22:11 +01:00
void ML8511::setVoltsPerStep(float voltage, uint32_t steps)
{
2020-11-27 11:20:37 +01:00
if (steps == 0) return;
if (voltage > 0.0) _voltsPerStep = voltage / steps;
2020-02-17 17:22:11 +01:00
}
2020-11-27 11:20:37 +01:00
2020-02-17 17:22:11 +01:00
void ML8511::enable()
{
2021-01-29 12:31:58 +01:00
if (_enablePin != 0xFF) digitalWrite(_enablePin, HIGH);
2020-11-27 11:20:37 +01:00
_enabled = true;
2020-02-17 17:22:11 +01:00
};
2020-11-27 11:20:37 +01:00
2020-02-17 17:22:11 +01:00
void ML8511::disable()
{
2021-01-29 12:31:58 +01:00
if (_enablePin != 0xFF) digitalWrite(_enablePin, LOW);
2020-11-27 11:20:37 +01:00
_enabled = false;
2020-02-17 17:22:11 +01:00
};
2021-05-28 13:39:01 +02:00
2020-11-27 11:20:37 +01:00
// -- END OF FILE --