-
-
Notifications
You must be signed in to change notification settings - Fork 111
/
Copy pathScaledQuadratureEncoder.cs
147 lines (129 loc) · 8.18 KB
/
ScaledQuadratureEncoder.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
using System;
using System.Device.Gpio;
namespace Iot.Device.RotaryEncoder
{
/// <summary>
/// Scaled Quadrature Rotary Controller binding.
/// </summary>
public class ScaledQuadratureEncoder : QuadratureRotaryEncoder
{
private double _rangeMax;
private double _rangeMin;
private double _pulseIncrement;
/// <summary>Gets or sets current value associated with the RotaryEncoder.</summary>
public double Value { get; set; }
/// <summary>
/// Gets or sets the AccelerationSlope property along with the AccelerationOffset property represents how the
/// increase or decrease in value should grow as the incremental encoder is turned faster.
/// </summary>
public float AccelerationSlope { get; set; } = -0.05F;
/// <summary>
/// Gets or sets the AccelerationOffset property along with the AccelerationSlope property represents how the
/// increase or decrease in value should grow as the incremental encoder is turned faster.
/// </summary>
public float AccelerationOffset { get; set; } = 6.0F;
/// <summary>
/// EventHandler to allow the notification of value changes.
/// </summary>
public event RotaryEncoderEventHandler? ValueChanged;
/// <summary>
/// Initializes a new instance of the <see cref="ScaledQuadratureEncoder" /> class.
/// </summary>
/// <param name="pinA">Pin A that is connected to the rotary encoder. Sometimes called clk.</param>
/// <param name="pinB">Pin B that is connected to the rotary encoder. Sometimes called data.</param>
/// <param name="edges">The pin event types to 'listen' for.</param>
/// <param name="pulsesPerRotation">The number of pulses to be received for every full rotation of the encoder.</param>
/// <param name="pulseIncrement">The amount that the value increases or decreases on each pulse from the rotary encoder.</param>
/// <param name="rangeMin">Minimum value permitted. The value is clamped to this.</param>
/// <param name="rangeMax">Maximum value permitted. The value is clamped to this.</param>
/// <param name="controller">GpioController that hosts Pins A and B.</param>
/// <param name="shouldDispose">Dispose the controller if true.</param>
public ScaledQuadratureEncoder(int pinA, int pinB, PinEventTypes edges, int pulsesPerRotation, double pulseIncrement, double rangeMin, double rangeMax, GpioController? controller = null, bool shouldDispose = true)
: base(pinA, pinB, edges, pulsesPerRotation, controller, shouldDispose)
{
_pulseIncrement = pulseIncrement;
_rangeMin = rangeMin;
_rangeMax = rangeMax;
Value = _rangeMin;
}
/// <summary>
/// Initializes a new instance of the <see cref="ScaledQuadratureEncoder" /> class.
/// </summary>
/// <param name="pinA">Pin A that is connected to the rotary encoder. Sometimes called clk.</param>
/// <param name="pinB">Pin B that is connected to the rotary encoder. Sometimes called data.</param>
/// <param name="edges">The pin event types to 'listen' for.</param>
/// <param name="pulsesPerRotation">The number of pulses to be received for every full rotation of the encoder.</param>
/// <param name="pulseIncrement">The amount that the value increases or decreases on each pulse from the rotary encoder.</param>
/// <param name="rangeMin">Minimum value permitted. The value is clamped to this.</param>
/// <param name="rangeMax">Maximum value permitted. The value is clamped to this.</param>
public ScaledQuadratureEncoder(int pinA, int pinB, PinEventTypes edges, int pulsesPerRotation, double pulseIncrement, double rangeMin, double rangeMax)
: this(pinA, pinB, edges, pulsesPerRotation, pulseIncrement, rangeMin, rangeMax, new GpioController(), true)
{
}
/// <summary>
/// Initializes a new instance of the <see cref="ScaledQuadratureEncoder" /> class.
/// </summary>
/// <param name="pinA">Pin A that is connected to the rotary encoder. Sometimes called clk.</param>
/// <param name="pinB">Pin B that is connected to the rotary encoder. Sometimes called data.</param>
/// <param name="edges">The pin event types to 'listen' for.</param>
/// <param name="pulsesPerRotation">The number of pulses to be received for every full rotation of the encoder.</param>
public ScaledQuadratureEncoder(int pinA, int pinB, PinEventTypes edges, int pulsesPerRotation)
: this(pinA, pinB, edges, pulsesPerRotation, new GpioController(), true)
{
}
/// <summary>
/// Initializes a new instance of the <see cref="ScaledQuadratureEncoder" /> class.
/// </summary>
/// <param name="pinA">Pin A that is connected to the rotary encoder. Sometimes called clk.</param>
/// <param name="pinB">Pin B that is connected to the rotary encoder. Sometimes called data.</param>
/// <param name="edges">The pin event types to 'listen' for.</param>
/// <param name="pulsesPerRotation">The number of pulses to be received for every full rotation of the encoder.</param>
/// <param name="controller">GpioController that hosts Pins A and B.</param>
/// <param name="shouldDispose">Dispose the controller if true.</param>
public ScaledQuadratureEncoder(int pinA, int pinB, PinEventTypes edges, int pulsesPerRotation, GpioController? controller = null, bool shouldDispose = true)
: base(pinA, pinB, edges, pulsesPerRotation, controller, shouldDispose)
{
_pulseIncrement = 1;
_rangeMin = 0;
_rangeMax = 100;
Value = _rangeMin;
}
/// <summary>
/// Read the current Value.
/// </summary>
/// <returns>The value associated with the rotary encoder.</returns>
public double ReadValue() => Value;
/// <summary>
/// Calculate the amount of acceleration to be applied to the increment of the encoder.
/// </summary>
/// <remarks>
/// This uses a straight line function output = input * AccelerationSlope + Acceleration offset but can be overridden
/// to perform different algorithms.
/// </remarks>
/// <param name="milliSecondsSinceLastPulse">The amount of time elapsed since the last data pulse from the encoder in milliseconds.</param>
/// <returns>A value that can be used to apply acceleration to the rotary encoder.</returns>
protected virtual int Acceleration(int milliSecondsSinceLastPulse)
{
// apply a straight line line function to the pulseWidth to determine the acceleration but clamp the lower value to 1
return Math.Max(1, (int)((milliSecondsSinceLastPulse * AccelerationSlope) + AccelerationOffset));
}
/// <summary>
/// Modify the current value on receipt of a pulse from the encoder.
/// </summary>
/// <param name="blnUp">When true then the value should be incremented otherwise it should be decremented.</param>
/// <param name="milliSecondsSinceLastPulse">The amount of time elapsed since the last data pulse from the encoder in milliseconds.</param>
protected override void OnPulse(bool blnUp, int milliSecondsSinceLastPulse)
{
// call the OnPulse method on the base class to ensure the pulsecount is kept up to date
base.OnPulse(blnUp, milliSecondsSinceLastPulse);
// calculate how much to change the value by
double valueChange = (blnUp ? _pulseIncrement : -_pulseIncrement) * Acceleration(milliSecondsSinceLastPulse);
// set the value to the new value clamped by the maximum and minumum of the range.
Value = Math.Max(Math.Min(Value + valueChange, _rangeMax), _rangeMin);
// fire an event if an event handler has been attached
ValueChanged?.Invoke(this, new RotaryEncoderEventArgs(Value));
}
}
}