67
70
- pressing the button increments the field currently being set
68
71
- pressing and holding the button for a second cycles through the
69
72
fields that can be set
70
- press and holding the button for 5 seconds to finish
73
- pressing and holding the button for 5 seconds sets the time and
72
76
******************************************************************************/
82
#include "modes/analogue_clock.h"
83
#include "modes/digital_clock.h"
84
#include "modes/test_pattern.h"
85
#include "modes/settings_mode.h"
87
#include "text_renderer.h"
77
90
//_____________________________________________________________________________
81
93
// when non-zero, the time (in microseconds) of a new fan pulse that
82
94
// has just occurred, which means that segment drawing needs to be
84
static unsigned long new_pulse_at = 0;
96
static unsigned long _new_pulse_at = 0;
86
98
// the time (in microseconds) when the last fan pulse occurred
87
static unsigned long last_pulse_at = 0;
99
static unsigned long _last_pulse_at = 0;
89
101
// duration (in microseconds) that a segment should be displayed
90
static unsigned long segment_step = 0;
102
static unsigned long _segment_step = 0;
92
104
// remainder after divisor and a tally of the remainders for each segment
93
static unsigned long segment_step_sub_step = 0;
94
static unsigned long segment_step_sub = 0;
96
// flag to indicate that the drawing mode should be cycled to the next one
97
static bool inc_draw_mode = false;
99
// a bounce-managed button
100
static Bounce button( 3, 5 );
103
static int time_hours = 0;
104
static int time_minutes = 0;
105
static int time_seconds = 0;
107
// number of segments in a full display (rotation) is 60 (one per
108
// second) times the desired number of sub-divisions of a second
109
#define NUM_SECOND_SEGMENTS 5
110
#define NUM_SEGMENTS ( 60 * NUM_SECOND_SEGMENTS )
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
112
122
//_____________________________________________________________________________
116
// check for button presses
122
// notice button presses
123
if( button.risingEdge() )
124
inc_draw_mode = true;
128
// keep track of time
131
// previous time and any carried-over milliseconds
132
static unsigned long last_time = millis();
133
static unsigned long carry = 0;
135
// how many milliseonds have elapsed since we last checked?
136
unsigned long next_time = millis();
137
unsigned long delta = next_time - last_time + carry;
139
// update the previous time and carried-over milliseconds
140
last_time = next_time;
141
carry = delta % 1000;
143
// add the seconds that have passed to the time
144
time_seconds += delta / 1000;
145
while( time_seconds >= 60 ) {
148
if( time_minutes >= 60 ) {
151
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()
149
_button.set_press_mode( _major_mode != SETTINGS_MODE_IDX );
151
// give the mode a chance to init
152
switch( _major_mode ) {
153
case MAIN_MODE_IDX: activate_minor_mode(); break;
154
case SETTINGS_MODE_IDX: settings_mode_activate(); break;
159
// perform button events
160
void do_button_events()
162
// loop through pending events
163
while( int event = _button.get_event() )
169
switch( _major_mode ) {
171
switch( _minor_mode ) {
172
case ANALOGUE_CLOCK_IDX: analogue_clock_press(); break;
173
case DIGITAL_CLOCK_IDX: digital_clock_press(); break;
176
case SETTINGS_MODE_IDX: settings_mode_press(); break;
182
switch( _major_mode ) {
184
if( ++_minor_mode >= 3 )
186
activate_minor_mode();
188
case SETTINGS_MODE_IDX: settings_mode_long_press(); break;
193
// looooong press (change major mode)
194
if( ++_major_mode > 1 )
196
activate_major_mode();
158
// draw a segment for the test display
159
void drawNextSegment_test( bool reset )
203
// draw a display segment
204
void draw_next_segment( bool reset )
161
206
// keep track of segment
162
static unsigned int segment = 0;
163
if( reset ) segment = 0;
166
// turn on inside and outside LEDs
167
digitalWrite( 4, HIGH );
168
digitalWrite( 13, HIGH );
170
// display segment number in binary across in the inside LEDs,
171
// with the LED on pin 12 showing the least-significant bit
172
for( int a = 0; a < 8; a++ )
173
digitalWrite( 12 - a, ( ( segment >> a ) & 1 )? HIGH : LOW );
177
// draw a segment for the time display
178
void drawNextSegment_time( bool reset )
180
static unsigned int second = 0;
181
static unsigned int segment = 0;
183
// handle display reset
208
static int segment = ( NUM_SEGMENTS - CLOCK_SHIFT ) % NUM_SEGMENTS;
209
if( reset ) segment = ( NUM_SEGMENTS - CLOCK_SHIFT ) % NUM_SEGMENTS;
211
static int segment = NUM_SEGMENTS - 1 - CLOCK_SHIFT;
212
if( reset ) segment = NUM_SEGMENTS - 1 - CLOCK_SHIFT;
215
// reset the text renderer
216
TextRenderer::reset_buffer();
189
// what needs to be drawn?
190
bool draw_tick = !segment && second % 5 == 0;
191
bool draw_second = !segment && second == time_seconds;
192
bool draw_minute = !segment && second == time_minutes;
193
bool draw_hour = !segment && second == time_hours;
196
digitalWrite( 13, HIGH );
197
digitalWrite( 12, draw_tick || draw_minute );
198
for( int a = 10; a <= 11; a++ )
199
digitalWrite( a, draw_minute || draw_second );
200
for( int a = 4; a <= 9; a++ )
201
digitalWrite( 10, draw_minute | draw_second || draw_hour );
204
if( ++segment >= NUM_SECOND_SEGMENTS ) {
211
// draw a display segment
212
void drawNextSegment( bool reset )
214
static int draw_mode = 0;
216
// handle mode switch requests
217
if( reset && inc_draw_mode ) {
218
inc_draw_mode = false;
225
switch( draw_mode ) {
226
case 0: drawNextSegment_test( reset ); break;
227
case 1: drawNextSegment_time( reset ); break;
220
switch( _major_mode ) {
222
switch( _minor_mode ) {
223
case ANALOGUE_CLOCK_IDX: analogue_clock_draw_reset(); break;
224
case DIGITAL_CLOCK_IDX: digital_clock_draw_reset(); break;
227
case SETTINGS_MODE_IDX: settings_mode_draw_reset(); break;
230
// tell the text services we're starting a new frame
235
switch( _major_mode ) {
237
switch( _minor_mode ) {
238
case ANALOGUE_CLOCK_IDX: analogue_clock_draw( segment ); break;
239
case DIGITAL_CLOCK_IDX: digital_clock_draw( segment ); break;
240
case TEST_PATTERN_IDX: test_pattern_draw( segment ); break;
243
case SETTINGS_MODE_IDX: settings_mode_draw( segment ); break;
246
// draw any text that was rendered
247
TextRenderer::output_buffer();
250
if( ++segment >= NUM_SEGMENTS ) segment = 0;
252
if( --segment < 0 ) segment = NUM_SEGMENTS - 1;
232
257
// calculate time constants when a new pulse has occurred
233
void calculateSegmentTimes()
258
void calculate_segment_times()
235
260
// check for overflows, and only recalculate times if there isn't
236
261
// one (if there is, we'll just go with the last pulse's times)
237
if( new_pulse_at > last_pulse_at )
262
if( _new_pulse_at > _last_pulse_at )
239
264
// new segment stepping times
240
unsigned long delta = new_pulse_at - last_pulse_at;
241
segment_step = delta / NUM_SEGMENTS;
242
segment_step_sub = 0;
243
segment_step_sub_step = delta % NUM_SEGMENTS;
265
unsigned long delta = _new_pulse_at - _last_pulse_at;
266
_segment_step = delta / NUM_SEGMENTS;
267
_segment_step_sub = 0;
268
_segment_step_sub_step = delta % NUM_SEGMENTS;
246
271
// now we have dealt with this pulse, save the pulse time and
247
272
// clear new_pulse_at, ready for the next pulse
248
last_pulse_at = new_pulse_at;
273
_last_pulse_at = _new_pulse_at;
253
278
// wait until it is time to draw the next segment or a new pulse has
255
void waitTillNextSegment( bool reset )
280
void wait_till_end_of_segment( bool reset )
257
282
static unsigned long end_time = 0;
261
end_time = last_pulse_at;
286
end_time = _last_pulse_at;
263
288
// work out the time that this segment should be displayed until
264
end_time += segment_step;
265
segment_step_sub += segment_step_sub_step;
266
if( segment_step_sub >= NUM_SEGMENTS ) {
267
segment_step_sub -= NUM_SEGMENTS;
289
end_time += _segment_step;
290
_segment_step_sub += _segment_step_sub_step;
291
if( _segment_step_sub >= NUM_SEGMENTS ) {
292
_segment_step_sub -= NUM_SEGMENTS;
272
while( micros() < end_time && !new_pulse_at );
297
while( micros() < end_time && !_new_pulse_at );
276
301
// ISR to handle the pulses from the fan's tachiometer
277
void fanPulseHandler()
302
void fan_pulse_handler()
279
304
// the fan actually sends two pulses per revolution. These pulses
280
305
// may not be exactly evenly distributed around the rotation, so