2022-12-04 17:55:37 +01:00
|
|
|
|
|
|
|
[![Arduino CI](https://github.com/RobTillaart/RAIN/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
|
|
|
|
[![Arduino-lint](https://github.com/RobTillaart/RAIN/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/RAIN/actions/workflows/arduino-lint.yml)
|
|
|
|
[![JSON check](https://github.com/RobTillaart/RAIN/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/RAIN/actions/workflows/jsoncheck.yml)
|
|
|
|
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/RAIN/blob/master/LICENSE)
|
|
|
|
[![GitHub release](https://img.shields.io/github/release/RobTillaart/RAIN.svg?maxAge=3600)](https://github.com/RobTillaart/RAIN/releases)
|
|
|
|
|
|
|
|
|
|
|
|
# RAIN
|
|
|
|
|
2022-12-06 11:26:48 +01:00
|
|
|
RAIN is an Arduino library for a rain sensor (analog).
|
2022-12-04 17:55:37 +01:00
|
|
|
|
|
|
|
|
|
|
|
## Description
|
|
|
|
|
2023-01-20 17:42:39 +01:00
|
|
|
A rain sensor like the FC-37, YL-83, HM-RD a.o. is a relative simple device.
|
|
|
|
It measures the resistance between wires when these are put in a liquid, water or in the ground.
|
2022-12-06 11:26:48 +01:00
|
|
|
The device converts the resistance to a voltage typical 0 .. 5 Volt.
|
2022-12-06 16:57:55 +01:00
|
|
|
The more the wires are covered by the liquid, the lower the voltage.
|
|
|
|
0.0 Volt is WET, and a high voltage is DRY.
|
2022-12-04 17:55:37 +01:00
|
|
|
|
2023-01-20 17:42:39 +01:00
|
|
|
The breakout (LM393 comparator) I used to test also has a digital output,
|
2022-12-06 16:57:55 +01:00
|
|
|
which goes LOW if a threshold (to be set with a potentiometer on the breakout) is reached.
|
2022-12-04 17:55:37 +01:00
|
|
|
|
2023-01-20 17:42:39 +01:00
|
|
|
The library is EXPERIMENTAL as it needs more testing.
|
2022-12-06 11:26:48 +01:00
|
|
|
(changes of the interface are definitely possible).
|
2022-12-04 17:55:37 +01:00
|
|
|
|
2023-01-20 17:42:39 +01:00
|
|
|
|
2022-12-06 16:57:55 +01:00
|
|
|
## Hardware connection
|
|
|
|
|
|
|
|
Typical connection
|
|
|
|
|
|
|
|
```
|
|
|
|
Processor LM393 SENSOR FC-37
|
|
|
|
+-------------+ +----------+ +---------------------+
|
|
|
|
| | | | | |
|
|
|
|
| GND |----->| GND | | |
|
|
|
|
| powerPin |----->| 5V |-----| |
|
|
|
|
| analogIn |<-----| AO | | |
|
|
|
|
| | | |-----| |
|
|
|
|
| digital in |<-----| DO | | |
|
|
|
|
| | | | | |
|
|
|
|
+-------------+ +----------+ +---------------------+
|
|
|
|
|
|
|
|
```
|
|
|
|
|
2022-12-04 17:55:37 +01:00
|
|
|
|
|
|
|
## Interface
|
|
|
|
|
2023-01-20 17:42:39 +01:00
|
|
|
- **RAIN(uint8_t analogPort, uint8_t powerPin = 255)** constructor.
|
|
|
|
analogPort is the internal analog port to use.
|
2022-12-06 16:57:55 +01:00
|
|
|
powerPin is optional, but recommended.
|
2023-01-20 17:42:39 +01:00
|
|
|
The default 255 means **NO** powerPin selected.
|
|
|
|
- **bool begin(float maxVoltage, uint16_t maxSteps)** sets the ADC parameters.
|
|
|
|
Allows to be changed runtime, e.g. if voltage fluctuates the math can be adapted by calling **begin()** again.
|
2022-12-04 17:55:37 +01:00
|
|
|
Might be a separate **setVoltage()** is more efficient.
|
|
|
|
- **float raw(uint8_t times = 1)** makes 1 or more measurements and averages them.
|
|
|
|
returned value is the average number of ADC steps.
|
|
|
|
- **float read(uint8_t times = 1)** makes 1 or more measurements, averages them and convert the average to a voltage.
|
2022-12-06 16:57:55 +01:00
|
|
|
This voltage is returned, and also cached for **percentage()** and **getLevel()**.
|
|
|
|
|
|
|
|
|
|
|
|
#### powerControl
|
|
|
|
|
|
|
|
Will only work if the **powerPin** is set in the constructor.
|
|
|
|
|
|
|
|
- **void powerOn()** switch the sensor on.
|
|
|
|
- **void powerOff()** switch the sensor off.
|
2023-01-20 17:42:39 +01:00
|
|
|
- **void setPowerDelay(uint8_t powerDelay = 100)** delay in microseconds
|
|
|
|
in powerOn() to stabilize the rain sensor.
|
|
|
|
Default is 100 microseconds.
|
|
|
|
- **uint8_t getPowerDelay()** returns the set value.
|
2022-12-06 16:57:55 +01:00
|
|
|
|
|
|
|
The powerPin is used to switch the LM393 ON and OFF so the sensor is powered
|
2023-01-20 17:42:39 +01:00
|
|
|
only when the sensor is read.
|
|
|
|
This reduces corrosion and minimizes the usage of energy.
|
2022-12-06 16:57:55 +01:00
|
|
|
Note: when the power is OFF, the digital-out cannot be used e.g. for interrupts.
|
|
|
|
So check what your project needs.
|
|
|
|
|
2023-01-20 17:42:39 +01:00
|
|
|
Note: the powerOn() default delays for 100 us to give the LM393 time to stabilize.
|
|
|
|
Adjust this time if you want to optimize performance, power consumption or corrosion.
|
|
|
|
Values can be 0 .. 255.
|
2022-12-06 16:57:55 +01:00
|
|
|
|
2022-12-04 17:55:37 +01:00
|
|
|
|
|
|
|
#### Analysis
|
|
|
|
|
2022-12-06 16:57:55 +01:00
|
|
|
- **void setDryReference(float dryRef)** used to calibrate the voltage when the sensor is dry.
|
|
|
|
Use **read()** to read / calibrate the voltage when the sensor is dry.
|
|
|
|
If not explicitly set, the max ADC voltage os used.
|
|
|
|
- **void getDryReference()** returns the set value.
|
2022-12-04 17:55:37 +01:00
|
|
|
- **float percentage()** returns the last **read()** to a percentage.
|
|
|
|
Note one needs to call read() again to get a new value as this uses a cached value.
|
2022-12-06 11:26:48 +01:00
|
|
|
- **float delta()** returns the delta voltage compared to previous read.
|
|
|
|
It give the first derivative of the signal. How fast does it rise.
|
2022-12-06 16:57:55 +01:00
|
|
|
- **bool setLevel(uint8_t nr, uint16_t millivolts)** allows a user to set 4 voltage levels in milliVolts.
|
2022-12-06 11:26:48 +01:00
|
|
|
- **uint8_t getLevel()**
|
2023-01-20 17:42:39 +01:00
|
|
|
Returns the level of the current cached voltage.
|
2022-12-04 17:55:37 +01:00
|
|
|
See example.
|
|
|
|
|
2022-12-06 16:57:55 +01:00
|
|
|
The library allows the user to set 4 thresholds or levels for the **getLevel()** function.
|
2023-01-20 17:42:39 +01:00
|
|
|
These 4 levels + 1 zero level can help to control behaviour of a system at a certain level.
|
|
|
|
Typical levels are almost empty, almost full and full.
|
|
|
|
The level do not need to be on a linear mapping like 20% steps, if your project need
|
2022-12-06 11:26:48 +01:00
|
|
|
other levels you can define these.
|
|
|
|
|
|
|
|
Note it is possible to adjust the levels runTime with **setLevel()**
|
|
|
|
|
2022-12-04 17:55:37 +01:00
|
|
|
|
|
|
|
#### MultiMap
|
|
|
|
|
2023-01-20 17:42:39 +01:00
|
|
|
For a continuous mapping one can use the **MultiMap** library.
|
|
|
|
It allows to map the voltage to any other useful unit as it can handle
|
2022-12-06 11:26:48 +01:00
|
|
|
even non-linearities well.
|
2022-12-04 17:55:37 +01:00
|
|
|
See https://github.com/RobTillaart/MultiMap
|
|
|
|
|
|
|
|
|
|
|
|
## Operation
|
|
|
|
|
2022-12-06 16:57:55 +01:00
|
|
|
The examples show the basic working of the functions of the library.
|
|
|
|
|
2023-01-20 17:42:39 +01:00
|
|
|
The rain sensor can be used in different types of projects.
|
2022-12-06 16:57:55 +01:00
|
|
|
Every project has a typical orientation of the sensor.
|
|
|
|
|
|
|
|
| project | orientation | measurement |
|
|
|
|
|:-----------------|:----------------:|:-------------:|
|
|
|
|
| rain sensor | angle e.g. 45° | polling
|
|
|
|
| leak detection | horizontal | interrupt - digital OUT
|
|
|
|
| water level | vertical | polling
|
|
|
|
|
2023-01-20 17:42:39 +01:00
|
|
|
For other applications it depends.
|
2022-12-04 17:55:37 +01:00
|
|
|
|
|
|
|
|
|
|
|
## Future
|
|
|
|
|
|
|
|
#### Must
|
2023-01-20 17:42:39 +01:00
|
|
|
|
2022-12-04 17:55:37 +01:00
|
|
|
- update documentation
|
|
|
|
- links etc.
|
|
|
|
|
|
|
|
|
|
|
|
#### Should
|
2023-01-20 17:42:39 +01:00
|
|
|
|
2022-12-06 11:26:48 +01:00
|
|
|
- optimizations
|
2022-12-06 16:57:55 +01:00
|
|
|
- a lot of floats...==> more uint16_t millivolts? (0.2.0)
|
2023-01-20 17:42:39 +01:00
|
|
|
- getLevel() split into level() + getLevel(nr); // semantics
|
|
|
|
- breaking change == 0.2.0
|
|
|
|
- level(0) should be settable too
|
2022-12-04 17:55:37 +01:00
|
|
|
|
|
|
|
|
|
|
|
#### Could
|
2023-01-20 17:42:39 +01:00
|
|
|
|
2022-12-06 11:26:48 +01:00
|
|
|
- add unit-tests
|
|
|
|
- **float readExt(float voltage)** for external ADC
|
2023-01-20 17:42:39 +01:00
|
|
|
- See **ACS712** library how this can be done.
|
2022-12-06 11:26:48 +01:00
|
|
|
- investigate: **getLevel()** should it do a read()?
|
|
|
|
- **setForcedRead(bool flag)** + getter
|
|
|
|
- investigate "a scale of wetness"
|
|
|
|
- investigate
|
|
|
|
- different salinity
|
|
|
|
- different liquids? which?
|
|
|
|
- how linear is the device?
|
2022-12-04 17:55:37 +01:00
|
|
|
|
2022-12-06 16:57:55 +01:00
|
|
|
|
|
|
|
#### Won't (unless requested)
|
2023-01-20 17:42:39 +01:00
|
|
|
|
2022-12-06 11:26:48 +01:00
|
|
|
- example with multiMap
|
|
|
|
- see multiMap library.
|
2022-12-06 16:57:55 +01:00
|
|
|
- **incrLevel(nr, amount = 1)** + **decrLevel(nr, amount = 1)**
|
|
|
|
to allow easier runtime tuning
|
2023-01-20 17:42:39 +01:00
|
|
|
- investigate level-changed "event"
|
|
|
|
- user should poll ==> keeps the lib simple.
|
|
|
|
- make the number of levels configurable
|
|
|
|
- dynamic array allocation.?
|
2022-12-04 17:55:37 +01:00
|
|
|
|