76
76
******************************************************************************/
82
#include "analogue_clock.h"
83
#include "digital_clock.h"
84
#include "test_pattern.h"
83
86
//_____________________________________________________________________________
87
89
// when non-zero, the time (in microseconds) of a new fan pulse that
88
90
// has just occurred, which means that segment drawing needs to be
90
static unsigned long new_pulse_at = 0;
92
static unsigned long _new_pulse_at = 0;
92
94
// the time (in microseconds) when the last fan pulse occurred
93
static unsigned long last_pulse_at = 0;
95
static unsigned long _last_pulse_at = 0;
95
97
// duration (in microseconds) that a segment should be displayed
96
static unsigned long segment_step = 0;
98
static unsigned long _segment_step = 0;
98
100
// remainder after divisor and a tally of the remainders for each segment
99
static unsigned long segment_step_sub_step = 0;
100
static unsigned long segment_step_sub = 0;
102
// flag to indicate that the drawing mode should be cycled to the next one
103
static bool inc_draw_mode = false;
105
// a bounce-managed button
106
static Bounce button( 3, 50 );
109
static int time_hours = 0;
110
static int time_minutes = 0;
111
static int time_seconds = 0;
113
// number of segments in a full display (rotation) is 60 (one per
114
// second) times the desired number of sub-divisions of a second
115
#define NUM_SECOND_SEGMENTS 5
116
#define NUM_SEGMENTS ( 60 * NUM_SECOND_SEGMENTS )
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
118
117
//_____________________________________________________________________________
122
// check for button presses
121
// activate the current minor mode
122
void activate_minor_mode()
128
// notice button presses
129
if( button.risingEdge() )
130
inc_draw_mode = true;
124
switch( _minor_mode ) {
125
case DIGITAL_CLOCK_IDX: digital_clock_activate(); break;
134
// keep track of time
129
// perform button events
130
void do_button_events()
137
// previous time and any carried-over milliseconds
138
static unsigned long last_time = millis();
139
static unsigned long carry = 0;
141
// how many milliseonds have elapsed since we last checked?
142
unsigned long next_time = millis();
143
unsigned long delta = next_time - last_time + carry;
145
// update the previous time and carried-over milliseconds
146
last_time = next_time;
147
carry = delta % 1000;
149
// add the seconds that have passed to the time
150
time_seconds += delta / 1000;
151
while( time_seconds >= 60 ) {
154
if( time_minutes >= 60 ) {
157
if( time_hours >= 24 )
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();
164
// turn an led on/off
165
void ledOn( int num, bool on )
167
if( num < 0 || num > 9 ) return;
169
// convert to pin no.
172
// pin 4 needs to be inverted (it's driving a PNP)
173
// NOTE: PIN 4 TEMPORARILY DISABLED
174
if( num == 4 ) on = true; //!on
176
digitalWrite( num, on? HIGH : LOW );
180
// draw a segment for the test display
181
void drawNextSegment_test( bool reset )
176
// draw a display segment
177
void draw_next_segment( bool reset )
183
179
// keep track of segment
184
static unsigned int segment = 0;
185
if( reset ) segment = 0;
188
// turn on inside and outside LEDs
192
// display segment number in binary across in the inside LEDs,
193
// with the LED on pin 12 showing the least-significant bit
194
for( int a = 0; a < 8; a++ )
195
ledOn( 8 - a, ( segment >> a ) & 1 );
199
// draw a segment for the time display
200
void drawNextSegment_time( bool reset )
202
static int second = 0;
203
static int segment = 0;
205
// handle display reset
211
// what needs to be drawn?
212
bool draw_tick = !segment && second % 5 == 0;
213
bool draw_second = !segment && second == time_seconds;
214
bool draw_minute = !segment && second == time_minutes;
215
bool draw_hour = !segment && second == time_hours;
219
ledOn( 8, draw_tick || draw_minute );
220
for( int a = 6; a <= 7; a++ )
221
ledOn( a, draw_minute || draw_second );
222
for( int a = 0; a <= 5; a++ )
223
ledOn( a, draw_minute || draw_second || draw_hour );
226
if( ++segment >= NUM_SECOND_SEGMENTS ) {
233
// draw a display segment
234
void drawNextSegment( bool reset )
236
static int draw_mode = 0;
238
// handle mode switch requests
239
if( reset && inc_draw_mode ) {
240
inc_draw_mode = false;
247
switch( draw_mode ) {
248
case 0: drawNextSegment_test( reset ); break;
249
case 1: drawNextSegment_time( reset ); break;
181
static int segment = ( NUM_SEGMENTS - CLOCK_SHIFT ) % NUM_SEGMENTS;
182
if( reset ) segment = ( NUM_SEGMENTS - CLOCK_SHIFT ) % NUM_SEGMENTS;
184
static int segment = NUM_SEGMENTS - 1 - CLOCK_SHIFT;
185
if( reset ) segment = NUM_SEGMENTS - 1 - CLOCK_SHIFT;
189
switch( _major_mode ) {
191
switch( _minor_mode ) {
192
case ANALOGUE_CLOCK_IDX: analogue_clock_draw( segment ); break;
193
case DIGITAL_CLOCK_IDX: digital_clock_draw( segment ); break;
194
case TEST_PATTERN_IDX: test_pattern_draw( segment ); break;
200
if( ++segment >= NUM_SEGMENTS ) segment = 0;
202
if( --segment < 0 ) segment = NUM_SEGMENTS - 1;
254
207
// calculate time constants when a new pulse has occurred
255
void calculateSegmentTimes()
208
void calculate_segment_times()
257
210
// check for overflows, and only recalculate times if there isn't
258
211
// one (if there is, we'll just go with the last pulse's times)
259
if( new_pulse_at > last_pulse_at )
212
if( _new_pulse_at > _last_pulse_at )
261
214
// new segment stepping times
262
unsigned long delta = new_pulse_at - last_pulse_at;
263
segment_step = delta / NUM_SEGMENTS;
264
segment_step_sub = 0;
265
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;
268
221
// now we have dealt with this pulse, save the pulse time and
269
222
// clear new_pulse_at, ready for the next pulse
270
last_pulse_at = new_pulse_at;
223
_last_pulse_at = _new_pulse_at;
275
228
// wait until it is time to draw the next segment or a new pulse has
277
void waitTillNextSegment( bool reset )
230
void wait_till_end_of_segment( bool reset )
279
232
static unsigned long end_time = 0;
283
end_time = last_pulse_at;
236
end_time = _last_pulse_at;
285
238
// work out the time that this segment should be displayed until
286
end_time += segment_step;
287
segment_step_sub += segment_step_sub_step;
288
if( segment_step_sub >= NUM_SEGMENTS ) {
289
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;
294
while( micros() < end_time && !new_pulse_at );
247
while( micros() < end_time && !_new_pulse_at );
298
251
// ISR to handle the pulses from the fan's tachiometer
299
void fanPulseHandler()
252
void fan_pulse_handler()
301
254
// the fan actually sends two pulses per revolution. These pulses
302
255
// may not be exactly evenly distributed around the rotation, so