76
76
******************************************************************************/
80
78
#include "config.h"
82
#include "mode_switcher.h"
82
#include "analogue_clock.h"
83
#include "digital_clock.h"
84
#include "test_pattern.h"
85
86
//_____________________________________________________________________________
89
89
// when non-zero, the time (in microseconds) of a new fan pulse that
90
90
// has just occurred, which means that segment drawing needs to be
92
static unsigned long new_pulse_at = 0;
92
static unsigned long _new_pulse_at = 0;
94
94
// the time (in microseconds) when the last fan pulse occurred
95
static unsigned long last_pulse_at = 0;
95
static unsigned long _last_pulse_at = 0;
97
97
// duration (in microseconds) that a segment should be displayed
98
static unsigned long segment_step = 0;
98
static unsigned long _segment_step = 0;
100
100
// remainder after divisor and a tally of the remainders for each segment
101
static unsigned long segment_step_sub_step = 0;
102
static unsigned long segment_step_sub = 0;
104
// flag to indicate that the drawing mode should be cycled to the next one
105
static bool inc_draw_mode = false;
107
// a bounce-managed button
108
static Button button( 3 );
101
static unsigned long _segment_step_sub_step = 0;
102
static unsigned long _segment_step_sub = 0;
105
static Button _button( 3 );
108
static int _major_mode = 0;
109
static int _minor_mode = 0;
111
#define MAIN_MODE_IDX 0
113
#define ANALOGUE_CLOCK_IDX 0
114
#define DIGITAL_CLOCK_IDX 1
115
#define TEST_PATTERN_IDX 2
110
117
//_____________________________________________________________________________
114
// check for button presses
121
// activate the current minor mode
122
void activate_minor_mode()
118
int event = button.update();
123
inc_draw_mode = true;
124
switch( _minor_mode ) {
125
case DIGITAL_CLOCK_IDX: digital_clock_activate(); break;
129
// turn an led on/off
130
void ledOn( int num, bool on )
129
// perform button events
130
void do_button_events()
132
if( num < 0 || num > 9 ) return;
134
// convert to pin no.
137
// pin 4 needs to be inverted (it's driving a PNP)
138
if( num == 4 ) on = !on;
140
digitalWrite( num, on? HIGH : LOW );
132
// loop through pending events
133
while( int event = _button.get_event() )
139
switch( _major_mode ) {
141
switch( _minor_mode ) {
142
case ANALOGUE_CLOCK_IDX: analogue_clock_press(); break;
143
case DIGITAL_CLOCK_IDX: digital_clock_press(); break;
151
switch( _major_mode ) {
153
if( ++_minor_mode >= 3 )
155
switch( _minor_mode ) {
156
case DIGITAL_CLOCK_IDX: digital_clock_activate(); break;
163
// looooong press (change major mode)
164
if( ++_major_mode > 0 )
166
switch( _major_mode ) {
167
case MAIN_MODE_IDX: _minor_mode = 0; break;
169
activate_minor_mode();
144
176
// draw a display segment
145
void drawNextSegment( bool reset )
177
void draw_next_segment( bool reset )
147
static ModeSwitcher mode_switcher;
148
static bool init = false;
152
mode_switcher.activate();
155
179
// keep track of segment
156
180
#if CLOCK_FORWARD
157
181
static int segment = ( NUM_SEGMENTS - CLOCK_SHIFT ) % NUM_SEGMENTS;
177
207
// calculate time constants when a new pulse has occurred
178
void calculateSegmentTimes()
208
void calculate_segment_times()
180
210
// check for overflows, and only recalculate times if there isn't
181
211
// one (if there is, we'll just go with the last pulse's times)
182
if( new_pulse_at > last_pulse_at )
212
if( _new_pulse_at > _last_pulse_at )
184
214
// new segment stepping times
185
unsigned long delta = new_pulse_at - last_pulse_at;
186
segment_step = delta / NUM_SEGMENTS;
187
segment_step_sub = 0;
188
segment_step_sub_step = delta % NUM_SEGMENTS;
215
unsigned long delta = _new_pulse_at - _last_pulse_at;
216
_segment_step = delta / NUM_SEGMENTS;
217
_segment_step_sub = 0;
218
_segment_step_sub_step = delta % NUM_SEGMENTS;
191
221
// now we have dealt with this pulse, save the pulse time and
192
222
// clear new_pulse_at, ready for the next pulse
193
last_pulse_at = new_pulse_at;
223
_last_pulse_at = _new_pulse_at;
198
228
// wait until it is time to draw the next segment or a new pulse has
200
void waitTillNextSegment( bool reset )
230
void wait_till_end_of_segment( bool reset )
202
232
static unsigned long end_time = 0;
206
end_time = last_pulse_at;
236
end_time = _last_pulse_at;
208
238
// work out the time that this segment should be displayed until
209
end_time += segment_step;
210
segment_step_sub += segment_step_sub_step;
211
if( segment_step_sub >= NUM_SEGMENTS ) {
212
segment_step_sub -= NUM_SEGMENTS;
239
end_time += _segment_step;
240
_segment_step_sub += _segment_step_sub_step;
241
if( _segment_step_sub >= NUM_SEGMENTS ) {
242
_segment_step_sub -= NUM_SEGMENTS;
217
while( micros() < end_time && !new_pulse_at );
247
while( micros() < end_time && !_new_pulse_at );
221
251
// ISR to handle the pulses from the fan's tachiometer
222
void fanPulseHandler()
252
void fan_pulse_handler()
224
254
// the fan actually sends two pulses per revolution. These pulses
225
255
// may not be exactly evenly distributed around the rotation, so
249
279
// set up mode-switch button on pin 3
250
280
pinMode( 3, INPUT );
251
281
digitalWrite( 3, HIGH );
252
button.add_event_at( 5, 1 );
253
button.add_event_at( 1000, 2 );
254
button.add_event_at( 4000, 3 );
257
Serial.begin( 9600 );
282
static int event_times[] = { 5, 500, 4000, 0 };
283
_button.set_event_times( event_times );
288
// activate the minor mode
289
switch( _major_mode ) {
290
case MAIN_MODE_IDX: activate_minor_mode(); break;
264
298
// if there has been a new pulse, we'll be resetting the display
265
bool reset = new_pulse_at? true : false;
299
bool reset = _new_pulse_at? true : false;
267
304
// only do this stuff at the start of a display cycle, to ensure
268
305
// that no state changes mid-display
308
// calculate segment times
309
calculate_segment_times();
274
311
// keep track of time
275
Time &time = Time::get_instance();
314
// perform button events
279
318
// draw this segment
280
drawNextSegment( reset );
282
// do we need to recalculate segment times?
284
calculateSegmentTimes();
319
draw_next_segment( reset );
286
321
// wait till it's time to draw the next segment
287
waitTillNextSegment( reset );
322
wait_till_end_of_segment( reset );