2016-12-20 13:25:22 +01:00
|
|
|
package aio
|
2014-04-27 19:34:16 -07:00
|
|
|
|
|
|
|
import (
|
2014-11-16 13:45:18 -08:00
|
|
|
"time"
|
2014-11-22 12:21:28 -08:00
|
|
|
|
2016-12-08 13:24:03 +01:00
|
|
|
"gobot.io/x/gobot"
|
2014-04-27 19:34:16 -07:00
|
|
|
)
|
|
|
|
|
2015-01-02 10:42:53 -08:00
|
|
|
// AnalogSensorDriver represents an Analog Sensor
|
2014-04-27 19:34:16 -07:00
|
|
|
type AnalogSensorDriver struct {
|
2014-11-22 12:21:28 -08:00
|
|
|
name string
|
|
|
|
pin string
|
2015-01-02 10:42:53 -08:00
|
|
|
halt chan bool
|
2014-11-22 12:21:28 -08:00
|
|
|
interval time.Duration
|
2014-11-29 11:02:10 -08:00
|
|
|
connection AnalogReader
|
2014-11-22 12:21:28 -08:00
|
|
|
gobot.Eventer
|
|
|
|
gobot.Commander
|
2022-03-24 18:40:26 +01:00
|
|
|
rawValue int
|
|
|
|
value float64
|
|
|
|
scale func(input int) (value float64)
|
2014-04-27 19:34:16 -07:00
|
|
|
}
|
|
|
|
|
2015-01-02 10:42:53 -08:00
|
|
|
// NewAnalogSensorDriver returns a new AnalogSensorDriver with a polling interval of
|
2016-09-25 13:36:18 +02:00
|
|
|
// 10 Milliseconds given an AnalogReader and pin.
|
2022-03-24 18:40:26 +01:00
|
|
|
// The driver supports customizable scaling from read int value to returned float64.
|
|
|
|
// The default scaling is 1:1. An adjustable linear scaler is provided by the driver.
|
2015-01-02 10:42:53 -08:00
|
|
|
//
|
2016-07-13 10:44:47 -06:00
|
|
|
// Optionally accepts:
|
2015-01-02 10:42:53 -08:00
|
|
|
// time.Duration: Interval at which the AnalogSensor is polled for new information
|
2014-09-27 11:34:13 -07:00
|
|
|
//
|
|
|
|
// Adds the following API Commands:
|
2022-07-13 19:45:28 +02:00
|
|
|
// "Read" - See AnalogDriverSensor.Read
|
|
|
|
// "ReadRaw" - See AnalogDriverSensor.ReadRaw
|
2016-09-25 13:36:18 +02:00
|
|
|
func NewAnalogSensorDriver(a AnalogReader, pin string, v ...time.Duration) *AnalogSensorDriver {
|
2014-06-11 17:41:04 -07:00
|
|
|
d := &AnalogSensorDriver{
|
2017-02-02 15:32:37 +01:00
|
|
|
name: gobot.DefaultName("AnalogSensor"),
|
2014-11-29 11:02:10 -08:00
|
|
|
connection: a,
|
2014-11-22 12:21:28 -08:00
|
|
|
pin: pin,
|
|
|
|
Eventer: gobot.NewEventer(),
|
|
|
|
Commander: gobot.NewCommander(),
|
|
|
|
interval: 10 * time.Millisecond,
|
2015-01-02 10:42:53 -08:00
|
|
|
halt: make(chan bool),
|
2022-03-24 18:40:26 +01:00
|
|
|
scale: func(input int) (value float64) { return float64(input) },
|
2014-04-27 19:34:16 -07:00
|
|
|
}
|
2014-07-03 19:14:04 -07:00
|
|
|
|
2014-11-28 18:37:03 -08:00
|
|
|
if len(v) > 0 {
|
|
|
|
d.interval = v[0]
|
|
|
|
}
|
|
|
|
|
2014-11-29 11:02:10 -08:00
|
|
|
d.AddEvent(Data)
|
2022-03-24 18:40:26 +01:00
|
|
|
d.AddEvent(Value)
|
2014-11-29 11:02:10 -08:00
|
|
|
d.AddEvent(Error)
|
|
|
|
|
2014-07-09 18:32:27 -07:00
|
|
|
d.AddCommand("Read", func(params map[string]interface{}) interface{} {
|
2014-11-16 13:45:18 -08:00
|
|
|
val, err := d.Read()
|
|
|
|
return map[string]interface{}{"val": val, "err": err}
|
2014-06-11 17:41:04 -07:00
|
|
|
})
|
2014-07-03 19:14:04 -07:00
|
|
|
|
2022-07-13 19:45:28 +02:00
|
|
|
d.AddCommand("ReadRaw", func(params map[string]interface{}) interface{} {
|
|
|
|
val, err := d.ReadRaw()
|
2022-03-24 18:40:26 +01:00
|
|
|
return map[string]interface{}{"val": val, "err": err}
|
|
|
|
})
|
|
|
|
|
2014-06-11 17:41:04 -07:00
|
|
|
return d
|
2014-04-27 19:34:16 -07:00
|
|
|
}
|
|
|
|
|
2022-03-24 18:40:26 +01:00
|
|
|
// Start starts the AnalogSensorDriver and reads the sensor at the given interval.
|
2014-09-27 11:34:13 -07:00
|
|
|
// Emits the Events:
|
2022-03-24 18:40:26 +01:00
|
|
|
// Data int - Event is emitted on change and represents the current raw reading from the sensor.
|
|
|
|
// Value float64 - Event is emitted on change and represents the current reading from the sensor.
|
2015-01-02 10:42:53 -08:00
|
|
|
// Error error - Event is emitted on error reading from the sensor.
|
2016-11-07 14:55:21 +01:00
|
|
|
func (a *AnalogSensorDriver) Start() (err error) {
|
2022-03-24 18:40:26 +01:00
|
|
|
if a.interval == 0 {
|
|
|
|
// cyclic reading deactivated
|
|
|
|
return
|
|
|
|
}
|
|
|
|
var oldRawValue = 0
|
|
|
|
var oldValue = 0.0
|
2014-11-16 13:45:18 -08:00
|
|
|
go func() {
|
2016-10-27 03:01:51 -06:00
|
|
|
timer := time.NewTimer(a.interval)
|
|
|
|
timer.Stop()
|
2014-11-16 13:45:18 -08:00
|
|
|
for {
|
2022-07-13 19:45:28 +02:00
|
|
|
_, err := a.Read()
|
2014-11-16 13:45:18 -08:00
|
|
|
if err != nil {
|
2016-08-30 13:27:50 +02:00
|
|
|
a.Publish(a.Event(Error), err)
|
2022-03-24 18:40:26 +01:00
|
|
|
} else {
|
|
|
|
if a.rawValue != oldRawValue && a.rawValue != -1 {
|
|
|
|
a.Publish(a.Event(Data), a.rawValue)
|
|
|
|
oldRawValue = a.rawValue
|
|
|
|
}
|
|
|
|
if a.value != oldValue && a.value != -1 {
|
|
|
|
a.Publish(a.Event(Value), a.value)
|
|
|
|
oldValue = a.value
|
|
|
|
}
|
2014-11-16 13:45:18 -08:00
|
|
|
}
|
2016-10-27 03:01:51 -06:00
|
|
|
|
|
|
|
timer.Reset(a.interval)
|
2015-01-02 10:42:53 -08:00
|
|
|
select {
|
2016-10-27 03:01:51 -06:00
|
|
|
case <-timer.C:
|
2015-01-02 10:42:53 -08:00
|
|
|
case <-a.halt:
|
2016-10-27 03:01:51 -06:00
|
|
|
timer.Stop()
|
2015-01-02 10:42:53 -08:00
|
|
|
return
|
|
|
|
}
|
2014-06-28 17:18:16 -07:00
|
|
|
}
|
2014-11-16 13:45:18 -08:00
|
|
|
}()
|
2014-11-19 23:21:19 -08:00
|
|
|
return
|
2014-06-28 17:18:16 -07:00
|
|
|
}
|
2014-09-27 11:34:13 -07:00
|
|
|
|
2015-01-02 10:42:53 -08:00
|
|
|
// Halt stops polling the analog sensor for new information
|
2016-11-07 14:55:21 +01:00
|
|
|
func (a *AnalogSensorDriver) Halt() (err error) {
|
2022-03-24 18:40:26 +01:00
|
|
|
if a.interval == 0 {
|
|
|
|
// cyclic reading deactivated
|
|
|
|
return
|
|
|
|
}
|
2015-01-02 10:42:53 -08:00
|
|
|
a.halt <- true
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Name returns the AnalogSensorDrivers name
|
|
|
|
func (a *AnalogSensorDriver) Name() string { return a.name }
|
|
|
|
|
2016-09-25 13:36:18 +02:00
|
|
|
// SetName sets the AnalogSensorDrivers name
|
|
|
|
func (a *AnalogSensorDriver) SetName(n string) { a.name = n }
|
|
|
|
|
2015-01-02 10:42:53 -08:00
|
|
|
// Pin returns the AnalogSensorDrivers pin
|
|
|
|
func (a *AnalogSensorDriver) Pin() string { return a.pin }
|
|
|
|
|
|
|
|
// Connection returns the AnalogSensorDrivers Connection
|
2014-11-29 11:02:10 -08:00
|
|
|
func (a *AnalogSensorDriver) Connection() gobot.Connection { return a.connection.(gobot.Connection) }
|
2014-04-27 19:34:16 -07:00
|
|
|
|
2022-07-13 19:45:28 +02:00
|
|
|
// Read returns the current reading from the sensor
|
|
|
|
func (a *AnalogSensorDriver) Read() (val float64, err error) {
|
|
|
|
if a.rawValue, err = a.ReadRaw(); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
a.value = a.scale(a.rawValue)
|
|
|
|
return a.value, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ReadRaw returns the current reading from the sensor without scaling
|
|
|
|
func (a *AnalogSensorDriver) ReadRaw() (val int, err error) {
|
2014-11-29 11:02:10 -08:00
|
|
|
return a.connection.AnalogRead(a.Pin())
|
2014-04-27 19:34:16 -07:00
|
|
|
}
|
2022-03-24 18:40:26 +01:00
|
|
|
|
|
|
|
// SetScaler substitute the default 1:1 return value function by a new scaling function
|
|
|
|
func (a *AnalogSensorDriver) SetScaler(scaler func(int) float64) {
|
|
|
|
a.scale = scaler
|
|
|
|
}
|
|
|
|
|
|
|
|
// Value returns the last read value from the sensor
|
|
|
|
func (a *AnalogSensorDriver) Value() float64 {
|
|
|
|
return a.value
|
|
|
|
}
|
|
|
|
|
|
|
|
// RawValue returns the last read raw value from the sensor
|
|
|
|
func (a *AnalogSensorDriver) RawValue() int {
|
|
|
|
return a.rawValue
|
|
|
|
}
|
|
|
|
|
|
|
|
func AnalogSensorLinearScaler(fromMin, fromMax int, toMin, toMax float64) func(input int) (value float64) {
|
|
|
|
m := (toMax - toMin) / float64(fromMax-fromMin)
|
|
|
|
n := toMin - m*float64(fromMin)
|
|
|
|
return func(input int) (value float64) {
|
|
|
|
if input <= fromMin {
|
|
|
|
return toMin
|
|
|
|
}
|
|
|
|
if input >= fromMax {
|
|
|
|
return toMax
|
|
|
|
}
|
|
|
|
return float64(input)*m + n
|
|
|
|
}
|
|
|
|
}
|