/elec/propeller-clock

To get this branch, use:
bzr branch http://bzr.ed.am/elec/propeller-clock

« back to all changes in this revision

Viewing changes to src/propeller-clock.ino

  • Committer: edam
  • Date: 2012-02-23 00:26:32 UTC
  • Revision ID: edam@waxworlds.org-20120223002632-kkwrdwijfmv45f0j
conrtol segment number from one place and reverse the order the segments are drawn (backwards clock!)

Show diffs side-by-side

added added

removed removed

76
76
******************************************************************************/
77
77
 
78
78
 
79
 
#include <button.h>
80
 
#include "config.h"
81
 
#include "time.h"
82
 
#include "mode_switcher.h"
83
 
#include "drawer.h"
 
79
#include <Bounce.h>
 
80
#include <DS1307.h>
 
81
#include <Wire.h>
84
82
 
85
83
//_____________________________________________________________________________
86
84
//                                                                         data
105
103
static bool inc_draw_mode = false;
106
104
 
107
105
// a bounce-managed button
108
 
static Button button( 3 );
 
106
static Bounce button( 3, 50 );
 
107
 
 
108
// the time
 
109
static int time_hours = 0;
 
110
static int time_minutes = 0;
 
111
static int time_seconds = 0;
 
112
 
 
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 )
 
117
 
 
118
// clock direction
 
119
#define CLOCK_FORWARD 0
109
120
 
110
121
//_____________________________________________________________________________
111
122
//                                                                         code
115
126
void checkButtons()
116
127
{
117
128
        // update buttons
118
 
        int event = button.update();
 
129
        button.update();
119
130
 
120
 
        // handle any events
121
 
        switch( event ) {
122
 
        case 1:
 
131
        // notice button presses
 
132
        if( button.risingEdge() )
123
133
                inc_draw_mode = true;
124
 
                break;
 
134
}
 
135
 
 
136
 
 
137
// keep track of time
 
138
void trackTime()
 
139
{
 
140
        // previous time and any carried-over milliseconds
 
141
        static unsigned long last_time = millis();
 
142
        static unsigned long carry = 0;
 
143
 
 
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;
 
147
 
 
148
        // update the previous time and carried-over milliseconds
 
149
        last_time = next_time;
 
150
        carry = delta % 1000;
 
151
 
 
152
        // add the seconds that have passed to the time
 
153
        time_seconds += delta / 1000;
 
154
        while( time_seconds >= 60 ) {
 
155
                time_seconds -= 60;
 
156
                time_minutes++;
 
157
                if( time_minutes >= 60 ) {
 
158
                        time_minutes -= 60;
 
159
                        time_hours++;
 
160
                        if( time_hours >= 24 )
 
161
                                time_hours -= 24;
 
162
                }
125
163
        }
126
164
}
127
165
 
135
173
        num += 4;
136
174
 
137
175
        // pin 4 needs to be inverted (it's driving a PNP)
138
 
        if( num == 4 ) on = !on;
 
176
        // NOTE: PIN 4 TEMPORARILY DISABLED
 
177
//      if( num == 4 ) on = true;
 
178
if( num == 4 ) on = !on;
139
179
 
140
180
        digitalWrite( num, on? HIGH : LOW );
141
181
}
142
182
 
143
183
 
 
184
// draw a segment for the test display
 
185
void drawNextSegment_test( int segment )
 
186
{
 
187
        // turn on inside and outside LEDs
 
188
        ledOn( 9, true );
 
189
 
 
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 );
 
194
}
 
195
 
 
196
 
 
197
// draw a segment for the time display
 
198
void drawNextSegment_time( int segment )
 
199
{
 
200
        int second = segment / NUM_SECOND_SEGMENTS;
 
201
        int second_segment = segment % NUM_SECOND_SEGMENTS;
 
202
 
 
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;
 
208
 
 
209
        // set the LEDs
 
210
        ledOn( 9, true );
 
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 );
 
216
}
 
217
 
 
218
 
144
219
// draw a display segment
145
220
void drawNextSegment( bool reset )
146
221
{
147
 
        static ModeSwitcher mode_switcher;
148
 
        static bool init = false;
149
 
 
150
 
        if( !init ) {
151
 
                init = true;
152
 
                mode_switcher.activate();
153
 
        }
 
222
        static int draw_mode = 0;
154
223
 
155
224
        // keep track of segment
156
225
#if CLOCK_FORWARD
157
 
        static int segment = ( NUM_SEGMENTS - CLOCK_SHIFT ) % NUM_SEGMENTS;
158
 
        if( reset ) segment = ( NUM_SEGMENTS - CLOCK_SHIFT ) % NUM_SEGMENTS;
 
226
        static int segment = 0;
 
227
        if( reset ) segment = 0;
159
228
#else
160
 
        static int segment = NUM_SEGMENTS - 1 - CLOCK_SHIFT;
161
 
        if( reset ) segment = NUM_SEGMENTS - 1 - CLOCK_SHIFT;
 
229
        static int segment = NUM_SEGMENTS - 1;
 
230
        if( reset ) segment = NUM_SEGMENTS - 1;
162
231
#endif
163
232
 
164
 
        // draw
165
 
        Drawer &drawer = mode_switcher.get_drawer();
166
 
        if( reset ) drawer.draw_reset();
167
 
        drawer.draw( segment );
 
233
        // handle mode switch requests
 
234
        if( reset && inc_draw_mode ) {
 
235
                inc_draw_mode = false;
 
236
                draw_mode++;
 
237
                if( draw_mode >= 2 )
 
238
                        draw_mode = 0;
 
239
        }
 
240
 
 
241
        // draw the segment
 
242
        switch( draw_mode ) {
 
243
        case 0: drawNextSegment_test( segment ); break;
 
244
        case 1: drawNextSegment_time( segment ); break;
 
245
        }
168
246
 
169
247
#if CLOCK_FORWARD
170
 
        if( ++segment >= NUM_SEGMENTS ) segment = 0;
 
248
        segment++;
171
249
#else
172
 
        if( --segment < 0 ) segment = NUM_SEGMENTS - 1;
 
250
        segment--;
173
251
#endif
174
252
}
175
253
 
249
327
        // set up mode-switch button on pin 3
250
328
        pinMode( 3, INPUT );
251
329
        digitalWrite( 3, HIGH );
252
 
        button.add_event_at( 5, 1 );
253
 
        button.add_event_at( 1000, 2 );
254
 
        button.add_event_at( 4000, 3 );
 
330
 
 
331
        // get the time from the real-time clock
 
332
        int rtc_data[ 7 ];
 
333
        RTC.get( rtc_data, true );
 
334
        time_hours = rtc_data[ DS1307_HR ];
 
335
        time_minutes = rtc_data[ DS1307_MIN ];
 
336
        time_seconds = rtc_data[ DS1307_SEC ];
255
337
 
256
338
        // serial comms
257
339
        Serial.begin( 9600 );
272
354
                checkButtons();
273
355
 
274
356
                // keep track of time
275
 
                Time &time = Time::get_instance();
276
 
                time.update();
 
357
                trackTime();
277
358
        }
278
359
 
279
360
        // draw this segment