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