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/switcher_major_mode.h"
 
 
83
#include "modes/settings_major_mode.h"
 
 
84
#include "modes/analogue_clock_mode.h"
 
 
85
#include "modes/digital_clock_mode.h"
 
 
86
#include "modes/info_mode.h"
 
 
87
#include "modes/test_pattern_mode.h"
 
 
89
#include "text_renderer.h"
 
77
92
//_____________________________________________________________________________
 
81
95
// when non-zero, the time (in microseconds) of a new fan pulse that
 
82
96
// has just occurred, which means that segment drawing needs to be
 
84
 
static unsigned long new_pulse_at = 0;
 
 
98
static unsigned long _new_pulse_at = 0;
 
86
100
// the time (in microseconds) when the last fan pulse occurred
 
87
 
static unsigned long last_pulse_at = 0;
 
 
101
static unsigned long _last_pulse_at = 0;
 
89
103
// duration (in microseconds) that a segment should be displayed
 
90
 
static unsigned long segment_step = 0;
 
 
104
static unsigned long _segment_step = 0;
 
92
106
// 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 )
 
 
107
static unsigned long _segment_step_sub_step = 0;
 
 
108
static unsigned long _segment_step_sub = 0;
 
 
111
static Button _button( 3 );
 
 
114
static MajorMode *_modes[ 3 ];
 
 
116
// current major mode
 
 
117
static int _mode = 0;
 
 
119
// interupt handler's "ignore every other" flag
 
 
120
static bool _pulse_ignore = true;
 
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 )
 
 
125
// perform button events
 
 
126
void do_button_events()
 
 
128
        // loop through pending events
 
 
129
        while( int event = _button.get_event() )
 
 
135
                        _modes[ _mode ]->press();
 
 
139
                        _modes[ _mode ]->long_press();
 
 
142
                        // looooong press (change major mode)
 
 
143
                        _modes[ _mode ]->deactivate();
 
 
144
                        if( !_modes[ ++_mode ] ) _mode = 0;
 
 
145
                        _modes[ _mode ]->activate();
 
 
148
                        // switch display upside-down
 
 
149
                        _pulse_ignore = !_pulse_ignore;
 
158
 
// draw a segment for the test display
 
159
 
void drawNextSegment_test( bool reset )
 
 
156
// draw a display segment
 
 
157
void draw_next_segment( bool reset )
 
161
159
        // 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
 
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_minute;
 
193
 
        bool draw_hour = !segment && second == time_hour;
 
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;
 
 
160
        static int segment = 0;
 
 
162
        if( reset ) segment = ( NUM_SEGMENTS - CLOCK_SHIFT ) % NUM_SEGMENTS;
 
 
164
        if( reset ) segment = NUM_SEGMENTS - 1 - CLOCK_SHIFT;
 
 
167
        // reset the text renderer's buffer
 
 
168
        TextRenderer::reset_buffer();
 
 
172
                _modes[ _mode ]->draw_reset();
 
 
174
                // tell the text services we're starting a new frame
 
 
179
        _modes[ _mode ]->draw( segment );
 
 
182
        Text::draw( segment );
 
 
184
        // draw text rednerer's buffer
 
 
185
        TextRenderer::output_buffer();
 
 
188
        if( ++segment >= NUM_SEGMENTS ) segment = 0;
 
 
190
        if( --segment < 0 ) segment = NUM_SEGMENTS - 1;
 
232
195
// calculate time constants when a new pulse has occurred
 
233
 
void calculateSegmentTimes()
 
 
196
void calculate_segment_times()
 
235
198
        // check for overflows, and only recalculate times if there isn't
 
236
199
        // one (if there is, we'll just go with the last pulse's times)
 
237
 
        if( new_pulse_at > last_pulse_at )
 
 
200
        if( _new_pulse_at > _last_pulse_at )
 
239
202
                // 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;
 
 
203
                unsigned long delta = _new_pulse_at - _last_pulse_at;
 
 
204
                _segment_step = delta / NUM_SEGMENTS;
 
 
205
                _segment_step_sub = 0;
 
 
206
                _segment_step_sub_step = delta % NUM_SEGMENTS;
 
246
209
        // now we have dealt with this pulse, save the pulse time and
 
247
210
        // clear new_pulse_at, ready for the next pulse
 
248
 
        last_pulse_at = new_pulse_at;
 
 
211
        _last_pulse_at = _new_pulse_at;
 
253
216
// wait until it is time to draw the next segment or a new pulse has
 
255
 
void waitTillNextSegment( bool reset )
 
 
218
void wait_till_end_of_segment( bool reset )
 
257
220
        static unsigned long end_time = 0;
 
261
 
                end_time = last_pulse_at;
 
 
224
                end_time = _last_pulse_at;
 
263
226
        // 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;
 
 
227
        end_time += _segment_step;
 
 
228
        _segment_step_sub += _segment_step_sub_step;
 
 
229
        if( _segment_step_sub >= NUM_SEGMENTS ) {
 
 
230
                _segment_step_sub -= NUM_SEGMENTS;
 
272
 
        while( micros() < end_time && !new_pulse_at );
 
 
235
        while( micros() < end_time && !_new_pulse_at );
 
276
 
// ISR to handle the pulses from the fan's tachiometer
 
277
 
void fanPulseHandler()
 
 
239
// ISR to handle the pulses from the fan's tachometer
 
 
240
void fan_pulse_handler()
 
279
242
        // the fan actually sends two pulses per revolution. These pulses
 
280
243
        // may not be exactly evenly distributed around the rotation, so
 
281
244
        // we can't recalculate times on every pulse. Instead, we ignore
 
282
245
        // every other pulse so timings are based on a complete rotation.
 
283
 
        static bool ignore = true;
 
 
246
        _pulse_ignore = !_pulse_ignore;
 
287
249
                // set a new pulse time
 
288
 
                new_pulse_at = micros();
 
 
250
                _new_pulse_at = micros();