]> git.cworth.org Git - scherzo/blobdiff - score.c
Allow computer-keyboard input of accidentals, and draw accidentals.
[scherzo] / score.c
diff --git a/score.c b/score.c
index ac0d346522715896fc3c2e478a3a215a2b32619e..25187ba06e19a7ead61a27aae5fc77b6c351828e 100644 (file)
--- a/score.c
+++ b/score.c
@@ -218,12 +218,13 @@ _draw_note (score_t *score, cairo_t *cr,
            score_staff_t *staff, score_note_t *note)
 {
     double line;
-    cairo_glyph_t note_glyph;
+    cairo_glyph_t note_glyph[2];
     static double extend_factor = 0.25;
+    cairo_text_extents_t extents;
+    int num_glyphs = 0;
 
-    void _draw_ledger_line (double line, double width) {
-       cairo_move_to (cr,
-                      - width * extend_factor / 2.0,
+    void _draw_ledger_line (double line, double offset, double width) {
+       cairo_move_to (cr, offset - extend_factor * width / 2.0,
                       score->space_height * line + score->line_width / 2.0);
        cairo_rel_line_to (cr, (1 + extend_factor) * width, 0);
        cairo_stroke (cr);
@@ -231,6 +232,11 @@ _draw_note (score_t *score, cairo_t *cr,
 
     cairo_save (cr);
 
+    /* Move right so that X==0 is natural position for non-displaced
+     * noteheads.
+     */
+    cairo_translate (cr, score->space_height, 0);
+
     /* Which line should the note appear on? Line 0 is the top line of
      * the staff and increasing downwards. (Negative values indicate a
      * note on a ledger line above the staff). Values half way between
@@ -243,12 +249,45 @@ _draw_note (score_t *score, cairo_t *cr,
 
     /* XXX: The hard-coded glyph indices here are very ugly. We should
      * figure out how to lookup glyphs by name from this font. */
+    switch (SCORE_PITCH_ACCIDENTAL (note->pitch)) {
+    case SCORE_PITCH_ACCIDENTAL_DOUBLE_FLAT:
+           note_glyph[num_glyphs].index = 77;
+           break;
+    case SCORE_PITCH_ACCIDENTAL_FLAT:
+           note_glyph[num_glyphs].index = 68;
+           break;
+    case SCORE_PITCH_ACCIDENTAL_NATURAL:
+           note_glyph[num_glyphs].index = 101;
+           break;
+    case SCORE_PITCH_ACCIDENTAL_SHARP:
+           note_glyph[num_glyphs].index = 134;
+           break;
+    case SCORE_PITCH_ACCIDENTAL_DOUBLE_SHARP:
+           note_glyph[num_glyphs].index = 142;
+           break;
+    }
+
+    if (SCORE_PITCH_ACCIDENTAL (note->pitch) != SCORE_PITCH_ACCIDENTAL_NATURAL)
+    {
+           note_glyph[num_glyphs].x = 0;
+
+           note_glyph[num_glyphs].y = score->space_height * line;
+
+           num_glyphs++;
+
+           cairo_glyph_extents (cr, note_glyph, num_glyphs, &extents);
+
+#define ACCIDENTAL_NOTE_SPACING (score->space_height * .15)
+
+           note_glyph[0].x = - (extents.width + ACCIDENTAL_NOTE_SPACING);
+    }
+
     switch (note->duration) {
     case SCORE_DURATION_1:
-       note_glyph.index = 127;
+       note_glyph[num_glyphs].index = 127;
        break;
     case SCORE_DURATION_2:
-       note_glyph.index = 85;
+       note_glyph[num_glyphs].index = 85;
        break;
     case SCORE_DURATION_4:
     case SCORE_DURATION_8:
@@ -257,24 +296,28 @@ _draw_note (score_t *score, cairo_t *cr,
     case SCORE_DURATION_64:
     case SCORE_DURATION_128:
     default:
-       note_glyph.index = 84;
+       note_glyph[num_glyphs].index = 84;
     }
 
-    note_glyph.x = 0;
-    note_glyph.y = score->space_height * line;
+    note_glyph[num_glyphs].x = 0;
+    note_glyph[num_glyphs].y = score->space_height * line;
+
+    num_glyphs++;
 
     if (line < 0 || line > 4) {
+       double offset, width;
        int i;
-       cairo_text_extents_t note_extents;
 
-       cairo_glyph_extents (cr, &note_glyph, 1, &note_extents);
+       cairo_glyph_extents (cr, note_glyph, num_glyphs, &extents);
+       offset = note_glyph[0].x + extents.x_bearing;
+       width = extents.width;
 
        if (line < 0) {
            for (i = -1; i >= line; i--)
-               _draw_ledger_line (i, note_extents.width);
+               _draw_ledger_line (i, offset, width);
        } else {
            for (i = 5; i <= line; i++)
-               _draw_ledger_line (i, note_extents.width);
+               _draw_ledger_line (i, offset, width);
        }
     }
 
@@ -282,7 +325,7 @@ _draw_note (score_t *score, cairo_t *cr,
                          note->color.r,
                          note->color.g,
                          note->color.b);
-    cairo_show_glyphs (cr, &note_glyph, 1);
+    cairo_show_glyphs (cr, note_glyph, num_glyphs);
 
     cairo_restore (cr);
 }