76
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"
86
#include "modes/info_mode.h"
88
#include "text_renderer.h"
91
83
//_____________________________________________________________________________
94
87
// when non-zero, the time (in microseconds) of a new fan pulse that
95
88
// has just occurred, which means that segment drawing needs to be
97
static unsigned long _new_pulse_at = 0;
90
static unsigned long new_pulse_at = 0;
99
92
// the time (in microseconds) when the last fan pulse occurred
100
static unsigned long _last_pulse_at = 0;
93
static unsigned long last_pulse_at = 0;
102
95
// duration (in microseconds) that a segment should be displayed
103
static unsigned long _segment_step = 0;
96
static unsigned long segment_step = 0;
105
98
// remainder after divisor and a tally of the remainders for each segment
106
static unsigned long _segment_step_sub_step = 0;
107
static unsigned long _segment_step_sub = 0;
110
static Button _button( 3 );
113
static int _major_mode = 0;
114
static int _minor_mode = 0;
116
#define MAIN_MODE_IDX 1
117
#define SETTINGS_MODE_IDX 0
119
#define ANALOGUE_CLOCK_IDX 0
120
#define DIGITAL_CLOCK_IDX 1
121
#define TEST_PATTERN_IDX 2
122
#define INFO_MODE_IDX 3
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 )
119
#define CLOCK_FORWARD 0
124
121
//_____________________________________________________________________________
128
// activate the current minor mode
129
void activate_minor_mode()
135
// give the mode a chance to init
136
switch( _minor_mode ) {
137
case ANALOGUE_CLOCK_IDX: analogue_clock_activate(); break;
138
case DIGITAL_CLOCK_IDX: digital_clock_activate(); break;
139
case INFO_MODE_IDX: info_mode_activate(); break;
144
// activate major mode
145
void activate_major_mode()
152
_button.set_press_mode( _major_mode != SETTINGS_MODE_IDX );
154
// give the mode a chance to init
155
switch( _major_mode ) {
156
case MAIN_MODE_IDX: activate_minor_mode(); break;
157
case SETTINGS_MODE_IDX: settings_mode_activate(); break;
162
// perform button events
163
void do_button_events()
165
// loop through pending events
166
while( int event = _button.get_event() )
172
switch( _major_mode ) {
174
switch( _minor_mode ) {
175
case ANALOGUE_CLOCK_IDX: analogue_clock_press(); break;
176
case DIGITAL_CLOCK_IDX: digital_clock_press(); break;
177
case INFO_MODE_IDX: info_mode_press(); break;
180
case SETTINGS_MODE_IDX: settings_mode_press(); break;
186
switch( _major_mode ) {
188
if( ++_minor_mode >= 3 )
190
activate_minor_mode();
192
case SETTINGS_MODE_IDX: settings_mode_long_press(); break;
197
// looooong press (change major mode)
198
if( ++_major_mode > 1 )
200
activate_major_mode();
125
// check for button presses
131
// notice button presses
132
if( button.risingEdge() )
133
inc_draw_mode = true;
137
// keep track of time
140
// previous time and any carried-over milliseconds
141
static unsigned long last_time = millis();
142
static unsigned long carry = 0;
144
// how many milliseonds have elapsed since we last checked?
145
unsigned long next_time = millis();
146
unsigned long delta = next_time - last_time + carry;
148
// update the previous time and carried-over milliseconds
149
last_time = next_time;
150
carry = delta % 1000;
152
// add the seconds that have passed to the time
153
time_seconds += delta / 1000;
154
while( time_seconds >= 60 ) {
157
if( time_minutes >= 60 ) {
160
if( time_hours >= 24 )
167
// turn an led on/off
168
void ledOn( int num, bool on )
170
if( num < 0 || num > 9 ) return;
172
// convert to pin no.
175
// pin 4 needs to be inverted (it's driving a PNP)
176
// NOTE: PIN 4 TEMPORARILY DISABLED
177
// if( num == 4 ) on = true;
178
if( num == 4 ) on = !on;
180
digitalWrite( num, on? HIGH : LOW );
184
// draw a segment for the test display
185
void drawNextSegment_test( int segment )
187
// turn on inside and outside LEDs
190
// display segment number in binary across in the inside LEDs,
191
// with the LED on pin 12 showing the least-significant bit
192
for( int a = 0; a < 9; a++ )
193
ledOn( 8 - a, ( segment >> a ) & 1 );
197
// draw a segment for the time display
198
void drawNextSegment_time( int segment )
200
int second = segment / NUM_SECOND_SEGMENTS;
201
int second_segment = segment % NUM_SECOND_SEGMENTS;
203
// what needs to be drawn?
204
bool draw_tick = !second_segment && second % 5 == 0;
205
bool draw_second = !second_segment && second == time_seconds;
206
bool draw_minute = !second_segment && second == time_minutes;
207
bool draw_hour = !second_segment && second == time_hours;
211
ledOn( 8, draw_tick || draw_minute );
212
for( int a = 6; a <= 7; a++ )
213
ledOn( a, draw_minute || draw_second );
214
for( int a = 0; a <= 5; a++ )
215
ledOn( a, draw_minute || draw_second || draw_hour );
207
219
// draw a display segment
208
void draw_next_segment( bool reset )
220
void drawNextSegment( bool reset )
222
static int draw_mode = 0;
210
224
// keep track of segment
211
225
#if CLOCK_FORWARD
212
static int segment = ( NUM_SEGMENTS - CLOCK_SHIFT ) % NUM_SEGMENTS;
213
if( reset ) segment = ( NUM_SEGMENTS - CLOCK_SHIFT ) % NUM_SEGMENTS;
226
static int segment = 0;
227
if( reset ) segment = 0;
215
static int segment = NUM_SEGMENTS - 1 - CLOCK_SHIFT;
216
if( reset ) segment = NUM_SEGMENTS - 1 - CLOCK_SHIFT;
229
static int segment = NUM_SEGMENTS - 1;
230
if( reset ) segment = NUM_SEGMENTS - 1;
219
// reset the text renderer
220
TextRenderer::reset_buffer();
224
switch( _major_mode ) {
226
switch( _minor_mode ) {
227
case ANALOGUE_CLOCK_IDX: analogue_clock_draw_reset(); break;
228
case DIGITAL_CLOCK_IDX: digital_clock_draw_reset(); break;
229
case INFO_MODE_IDX: info_mode_draw_reset(); break;
232
case SETTINGS_MODE_IDX: settings_mode_draw_reset(); break;
235
// tell the text services we're starting a new frame
240
switch( _major_mode ) {
242
switch( _minor_mode ) {
243
case ANALOGUE_CLOCK_IDX: analogue_clock_draw( segment ); break;
244
case DIGITAL_CLOCK_IDX: digital_clock_draw( segment ); break;
245
case TEST_PATTERN_IDX: test_pattern_draw( segment ); break;
246
case INFO_MODE_IDX: info_mode_draw( segment ); break;
249
case SETTINGS_MODE_IDX: settings_mode_draw( segment ); break;
252
// draw any text that was rendered
253
TextRenderer::output_buffer();
233
// handle mode switch requests
234
if( reset && inc_draw_mode ) {
235
inc_draw_mode = false;
242
switch( draw_mode ) {
243
case 0: drawNextSegment_test( segment ); break;
244
case 1: drawNextSegment_time( segment ); break;
255
247
#if CLOCK_FORWARD
256
if( ++segment >= NUM_SEGMENTS ) segment = 0;
258
if( --segment < 0 ) segment = NUM_SEGMENTS - 1;
263
255
// calculate time constants when a new pulse has occurred
264
void calculate_segment_times()
256
void calculateSegmentTimes()
266
258
// check for overflows, and only recalculate times if there isn't
267
259
// one (if there is, we'll just go with the last pulse's times)
268
if( _new_pulse_at > _last_pulse_at )
260
if( new_pulse_at > last_pulse_at )
270
262
// new segment stepping times
271
unsigned long delta = _new_pulse_at - _last_pulse_at;
272
_segment_step = delta / NUM_SEGMENTS;
273
_segment_step_sub = 0;
274
_segment_step_sub_step = delta % NUM_SEGMENTS;
263
unsigned long delta = new_pulse_at - last_pulse_at;
264
segment_step = delta / NUM_SEGMENTS;
265
segment_step_sub = 0;
266
segment_step_sub_step = delta % NUM_SEGMENTS;
277
269
// now we have dealt with this pulse, save the pulse time and
278
270
// clear new_pulse_at, ready for the next pulse
279
_last_pulse_at = _new_pulse_at;
271
last_pulse_at = new_pulse_at;
284
276
// wait until it is time to draw the next segment or a new pulse has
286
void wait_till_end_of_segment( bool reset )
278
void waitTillNextSegment( bool reset )
288
280
static unsigned long end_time = 0;
292
end_time = _last_pulse_at;
284
end_time = last_pulse_at;
294
286
// work out the time that this segment should be displayed until
295
end_time += _segment_step;
296
_segment_step_sub += _segment_step_sub_step;
297
if( _segment_step_sub >= NUM_SEGMENTS ) {
298
_segment_step_sub -= NUM_SEGMENTS;
287
end_time += segment_step;
288
segment_step_sub += segment_step_sub_step;
289
if( segment_step_sub >= NUM_SEGMENTS ) {
290
segment_step_sub -= NUM_SEGMENTS;
303
while( micros() < end_time && !_new_pulse_at );
295
while( micros() < end_time && !new_pulse_at );
307
// ISR to handle the pulses from the fan's tachometer
308
void fan_pulse_handler()
299
// ISR to handle the pulses from the fan's tachiometer
300
void fanPulseHandler()
310
302
// the fan actually sends two pulses per revolution. These pulses
311
303
// may not be exactly evenly distributed around the rotation, so