76
76
******************************************************************************/
82
#include "analogue_clock.h"
83
#include "digital_clock.h"
84
#include "test_pattern.h"
85
#include "settings_mode.h"
87
#include "text_renderer.h"
83
90
//_____________________________________________________________________________
87
93
// when non-zero, the time (in microseconds) of a new fan pulse that
88
94
// has just occurred, which means that segment drawing needs to be
90
static unsigned long new_pulse_at = 0;
96
static unsigned long _new_pulse_at = 0;
92
98
// the time (in microseconds) when the last fan pulse occurred
93
static unsigned long last_pulse_at = 0;
99
static unsigned long _last_pulse_at = 0;
95
101
// duration (in microseconds) that a segment should be displayed
96
static unsigned long segment_step = 0;
102
static unsigned long _segment_step = 0;
98
104
// 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 )
118
// clock draw direction
119
#define CLOCK_FORWARD 0
121
// rotate display (in segments)
122
#define CLOCK_SHIFT ( 58 * NUM_SECOND_SEGMENTS - 1 )
105
static unsigned long _segment_step_sub_step = 0;
106
static unsigned long _segment_step_sub = 0;
109
static Button _button( 3 );
112
static int _major_mode = 0;
113
static int _minor_mode = 0;
115
#define MAIN_MODE_IDX 1
116
#define SETTINGS_MODE_IDX 0
118
#define ANALOGUE_CLOCK_IDX 0
119
#define DIGITAL_CLOCK_IDX 1
120
#define TEST_PATTERN_IDX 2
124
122
//_____________________________________________________________________________
128
// check for button presses
134
// notice button presses
135
if( button.risingEdge() )
136
inc_draw_mode = true;
140
// keep track of time
143
// previous time and any carried-over milliseconds
144
static unsigned long last_time = millis();
145
static unsigned long carry = 0;
147
// how many milliseonds have elapsed since we last checked?
148
unsigned long next_time = millis();
149
unsigned long delta = next_time - last_time + carry;
151
// update the previous time and carried-over milliseconds
152
last_time = next_time;
153
carry = delta % 1000;
155
// add the seconds that have passed to the time
156
time_seconds += delta / 1000;
157
while( time_seconds >= 60 ) {
160
if( time_minutes >= 60 ) {
163
if( time_hours >= 24 )
126
// activate the current minor mode
127
void activate_minor_mode()
133
// give the mode a chance to init
134
switch( _minor_mode ) {
135
case ANALOGUE_CLOCK_IDX: analogue_clock_activate(); break;
136
case DIGITAL_CLOCK_IDX: digital_clock_activate(); break;
141
// activate major mode
142
void activate_major_mode()
148
// give the mode a chance to init
149
switch( _major_mode ) {
150
case MAIN_MODE_IDX: activate_minor_mode(); break;
151
case SETTINGS_MODE_IDX: settings_mode_activate(); break;
156
// perform button events
157
void do_button_events()
159
// loop through pending events
160
while( int event = _button.get_event() )
166
switch( _major_mode ) {
168
switch( _minor_mode ) {
169
case ANALOGUE_CLOCK_IDX: analogue_clock_press(); break;
170
case DIGITAL_CLOCK_IDX: digital_clock_press(); break;
173
case SETTINGS_MODE_IDX: settings_mode_press(); break;
179
switch( _major_mode ) {
181
if( ++_minor_mode >= 3 )
183
activate_minor_mode();
185
case SETTINGS_MODE_IDX: settings_mode_long_press(); break;
190
// looooong press (change major mode)
191
if( ++_major_mode > 1 )
193
activate_major_mode();
170
// turn an led on/off
171
void ledOn( int num, bool on )
173
if( num < 0 || num > 9 ) return;
175
// convert to pin no.
178
// pin 4 needs to be inverted (it's driving a PNP)
179
if( num == 4 ) on = !on;
181
digitalWrite( num, on? HIGH : LOW );
185
// draw a segment for the test display
186
void drawNextSegment_test( int segment )
188
// turn on outside LEDs
191
// display segment number in binary across in the inside LEDs,
192
// with the LED on pin 12 showing the least-significant bit
193
for( int a = 0; a < 9; a++ )
194
ledOn( 8 - a, ( segment >> a ) & 1 );
198
// draw a segment for the time display
199
void drawNextSegment_time( int segment )
201
int second = segment / NUM_SECOND_SEGMENTS;
202
int second_segment = segment % NUM_SECOND_SEGMENTS;
204
// what needs to be drawn?
205
bool draw_tick = ( !second_segment && second % 5 == 0 && second ) ||
206
( second == 0 && second_segment == 1 ) ||
207
( second == 59 && second_segment == NUM_SECOND_SEGMENTS - 1 );
208
bool draw_second = !second_segment && second == time_seconds;
209
bool draw_minute = !second_segment && second == time_minutes;
210
bool draw_hour = segment == time_hours * 5 * NUM_SECOND_SEGMENTS +
211
( 5 * NUM_SECOND_SEGMENTS * time_minutes / 60 );
215
ledOn( 8, draw_tick || draw_second );
216
for( int a = 6; a <= 7; a++ )
217
ledOn( a, draw_minute || draw_second );
218
for( int a = 0; a <= 5; a++ )
219
ledOn( a, draw_minute || draw_second || draw_hour );
223
200
// draw a display segment
224
void drawNextSegment( bool reset )
201
void draw_next_segment( bool reset )
226
static int draw_mode = 0;
228
203
// keep track of segment
229
204
#if CLOCK_FORWARD
230
205
static int segment = ( NUM_SEGMENTS - CLOCK_SHIFT ) % NUM_SEGMENTS;
259
254
// calculate time constants when a new pulse has occurred
260
void calculateSegmentTimes()
255
void calculate_segment_times()
262
257
// check for overflows, and only recalculate times if there isn't
263
258
// one (if there is, we'll just go with the last pulse's times)
264
if( new_pulse_at > last_pulse_at )
259
if( _new_pulse_at > _last_pulse_at )
266
261
// new segment stepping times
267
unsigned long delta = new_pulse_at - last_pulse_at;
268
segment_step = delta / NUM_SEGMENTS;
269
segment_step_sub = 0;
270
segment_step_sub_step = delta % NUM_SEGMENTS;
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;
273
268
// now we have dealt with this pulse, save the pulse time and
274
269
// clear new_pulse_at, ready for the next pulse
275
last_pulse_at = new_pulse_at;
270
_last_pulse_at = _new_pulse_at;
280
275
// wait until it is time to draw the next segment or a new pulse has
282
void waitTillNextSegment( bool reset )
277
void wait_till_end_of_segment( bool reset )
284
279
static unsigned long end_time = 0;
288
end_time = last_pulse_at;
283
end_time = _last_pulse_at;
290
285
// work out the time that this segment should be displayed until
291
end_time += segment_step;
292
segment_step_sub += segment_step_sub_step;
293
if( segment_step_sub >= NUM_SEGMENTS ) {
294
segment_step_sub -= NUM_SEGMENTS;
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;
299
while( micros() < end_time && !new_pulse_at );
294
while( micros() < end_time && !_new_pulse_at );
303
298
// ISR to handle the pulses from the fan's tachiometer
304
void fanPulseHandler()
299
void fan_pulse_handler()
306
301
// the fan actually sends two pulses per revolution. These pulses
307
302
// may not be exactly evenly distributed around the rotation, so