/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/text_renderer.cc

  • Committer: Tim Marston
  • Date: 2012-05-09 20:36:07 UTC
  • Revision ID: tim@ed.am-20120509203607-5sh14qikxjmm6p3y
updated arduino.mk

Show diffs side-by-side

added added

removed removed

24
24
#include "config.h"
25
25
#include "Arduino.h"
26
26
#include <avr/pgmspace.h>
27
 
#include "display.h"
28
 
 
29
 
 
30
 
TextRenderer &TextRenderer::get_instance()
31
 
{
32
 
        static TextRenderer text_renderer;
33
 
        return text_renderer;
34
 
}
35
 
 
36
 
 
37
 
PString &TextRenderer::get_pstring()
38
 
{
39
 
        // reset renderer
40
 
        _duration = 0;
41
 
        _last_millis = ::millis();
42
 
        _scroll = -TEXT_DISPLAY_SEGMENTS;
43
 
 
44
 
        return _message;
45
 
}
46
 
 
47
 
 
48
 
void TextRenderer::draw_reset()
49
 
{
50
 
        // how many milliseconds have elapsed since last frame?
51
 
        unsigned long millis = ::millis();
52
 
        if( millis > _last_millis )
53
 
                _duration = millis - _last_millis;
54
 
        _last_millis = millis;
55
 
 
56
 
        // move on scroll pos
57
 
        _scroll += ( _duration * TEXT_SCROLL_SPEED / 1000 );
58
 
        if( _scroll > (signed long)_message.length() * 8 * TEXT_SCALE )
59
 
                _scroll -= (signed long)_message.length() * 8 * TEXT_SCALE + TEXT_DISPLAY_SEGMENTS;
60
 
}
61
 
 
62
 
 
63
 
void TextRenderer::draw_scroll( int segment )
64
 
{
65
 
        segment = xform_segment( segment );
66
 
        if( segment < 0 )
67
 
                Display::leds_off();
68
 
        else
69
 
                render( (long)segment + _scroll );
70
 
}
71
 
 
72
 
 
73
 
TextRenderer::TextRenderer()
74
 
    :
75
 
        _message( _message_buffer, sizeof( _message_buffer ) ),
76
 
        _glyph_cache_char( '\0' ),
77
 
        _font( 0 )
78
 
{
79
 
}
80
 
 
81
 
 
82
 
int TextRenderer::xform_segment( int segment )
83
 
{
84
 
        if( segment < TEXT_DISPLAY_SEGMENTS / 2 )
85
 
                return segment + TEXT_DISPLAY_SEGMENTS / 2;
86
 
        else if( segment >= NUM_SEGMENTS - TEXT_DISPLAY_SEGMENTS / 2 )
87
 
                return segment - ( NUM_SEGMENTS - TEXT_DISPLAY_SEGMENTS / 2 );
88
 
        else
89
 
                return - 1;
90
 
}
91
 
 
92
 
 
93
 
void TextRenderer::render( int x )
94
 
{
95
 
        // handle negative x
96
 
        if( x < 0 ) {
97
 
                Display::leds_off();
98
 
                return;
99
 
        }
100
 
 
101
 
        // scale font
102
 
        x /= TEXT_SCALE;
103
 
 
104
 
        // fetch a glyph
105
 
        int pos = x / 8;
106
 
        char c = ( pos >= 0 && pos < (signed)_message.length() )? _message[ pos ] : ' ';
107
 
        if( c != _glyph_cache_char ) {
108
 
                _glyph_cache_char = c;
109
 
                get_glyph( c, _glyph_cache );
110
 
        }
111
 
 
112
 
        // draw a column
113
 
        char col_data = _glyph_cache[ x % 8 ];
114
 
        for( int a = 0; a < 8; a++ )
115
 
                Display::led( 8 - a, ( col_data & ( 1 << a ) )? true : false );
116
 
}
117
 
 
118
 
 
119
 
void TextRenderer::get_glyph( char c, char *glyph )
 
27
#include "common.h"
 
28
#include "nvram.h"
 
29
 
 
30
 
 
31
// cached glyph
 
32
static char _glyph_cache[ 8 ];
 
33
 
 
34
// the character the current cached glyph is for
 
35
static char _glyph_cache_char;
 
36
 
 
37
// selected font
 
38
static int _font = 0;
 
39
 
 
40
// output buffer
 
41
static unsigned char _output_buffer;
 
42
 
 
43
// does buffer need rendering?
 
44
static bool _need_render;
 
45
 
 
46
 
 
47
// cache a glyph
 
48
void cache_glyph( char c )
120
49
{
121
50
        static char fonts[][ 66 * 8 ] PROGMEM = {
122
51
                { // INVD-A
390
319
                }
391
320
        };
392
321
 
 
322
        // nothing to do?
 
323
        if( c == _glyph_cache_char )
 
324
                return;
 
325
        _glyph_cache_char = c;
 
326
 
 
327
        // find glyph data position
393
328
        int pos;
394
329
        if( c >= 'A' && c <= 'Z' )
395
330
                pos = c - 'A';
408
343
        else
409
344
                pos = -1;
410
345
 
 
346
        // space? or copy glyph from progmem?
411
347
        if( pos == -1 )
412
 
                memset( glyph, 0, 8 );
413
 
        else
414
 
                memcpy_P( glyph, &( fonts[ _font ][ pos * 8 ] ), 8 );
 
348
                memset( _glyph_cache, 0, 8 );
 
349
        else
 
350
                memcpy_P( _glyph_cache, &( fonts[ _font ][ pos * 8 ] ), 8 );
 
351
}
 
352
 
 
353
 
 
354
void TextRenderer::init()
 
355
{
 
356
        Nvram::load( Nvram::NVRAM_FONT, _font );
 
357
        if( _font < 0 || _font >= 4 ) _font = 0;
 
358
}
 
359
 
 
360
 
 
361
int TextRenderer::get_width( int message_len, int scale )
 
362
{
 
363
        return message_len * 8 * scale;
 
364
}
 
365
 
 
366
 
 
367
void TextRenderer::render( const char *message, int message_len, int x,
 
368
                                                   bool y_flip, int y_shift, int scale )
 
369
{
 
370
        char glyph_col = 0;
 
371
 
 
372
        // handle negative x
 
373
        if( x >= 0 )
 
374
        {
 
375
                // scale font
 
376
                x /= scale;
 
377
 
 
378
                int pos = x / 8;
 
379
                if( pos < message_len )
 
380
                {
 
381
                        // ensure correct glyph is cached and pull out column of data
 
382
                        cache_glyph( message[ pos ] );
 
383
                        glyph_col = _glyph_cache[ x % 8 ];
 
384
                }
 
385
        }
 
386
 
 
387
        // draw a column
 
388
        if( y_flip )
 
389
                for( int a = 7; a >= 0; a-- ) {
 
390
                        if( glyph_col & 1 )
 
391
                                _output_buffer |= 1 << ( a + y_shift );
 
392
                        glyph_col >>= 1;
 
393
                }
 
394
        else
 
395
                for( int a = 0; a < 8; a++ ) {
 
396
                        if( glyph_col & 1 )
 
397
                                _output_buffer |= 1 << ( a + y_shift );
 
398
                        glyph_col >>= 1;
 
399
                }
 
400
}
 
401
 
 
402
 
 
403
void TextRenderer::reset_buffer()
 
404
{
 
405
        _output_buffer = 0;
 
406
        _need_render = false;
 
407
}
 
408
 
 
409
 
 
410
void TextRenderer::buffer_in_use()
 
411
{
 
412
        _need_render = true;
 
413
}
 
414
 
 
415
 
 
416
void TextRenderer::output_buffer()
 
417
{
 
418
        if( !_need_render ) return;
 
419
 
 
420
        for( int a = 8; a >= 0; a-- ) {
 
421
                led( a, ( _output_buffer & 1 )? true : false );
 
422
                _output_buffer >>= 1;
 
423
        }
 
424
        _output_buffer = 0;
 
425
}
 
426
 
 
427
 
 
428
int TextRenderer::get_font()
 
429
{
 
430
        return _font;
 
431
}
 
432
 
 
433
 
 
434
void TextRenderer::inc_font()
 
435
{
 
436
        if( ++_font >= 4 )
 
437
                _font = 0;
 
438
        Nvram::save( Nvram::NVRAM_FONT, _font );
415
439
}