This will facilitate sharing of this code between score.c and scherzo.c.
mnemon/mnemon.c \
pitch.c \
scherzo.c \
+ scherzo-key.c \
score.c
scherzo_modules = $(scherzo_srcs:.c=.o)
--- /dev/null
+/* scherzo - Music notation training
+ *
+ * key.c - Common structures and functions for keys
+ *
+ * Copyright © 2013 Carl Worth
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see http://www.gnu.org/licenses/ .
+ */
+
+#include <stdbool.h>
+
+#include "scherzo-key.h"
+
+#define ARRAY_SIZE(arr) ((int) (sizeof(arr) / sizeof(arr[0])))
+
+void
+scherzo_key_init (scherzo_key_t *key, pitch_t pitch)
+{
+ int i;
+
+ pitch_t sharp_keys[] = {
+ PITCH_CLASS_LITERAL (C, NATURAL),
+ PITCH_CLASS_LITERAL (G, NATURAL),
+ PITCH_CLASS_LITERAL (D, NATURAL),
+ PITCH_CLASS_LITERAL (A, NATURAL),
+ PITCH_CLASS_LITERAL (E, NATURAL),
+ PITCH_CLASS_LITERAL (B, NATURAL),
+ PITCH_CLASS_LITERAL (F, SHARP),
+ PITCH_CLASS_LITERAL (C, SHARP),
+ };
+
+ pitch_t flat_keys[] = {
+ PITCH_CLASS_LITERAL (C, NATURAL),
+ PITCH_CLASS_LITERAL (F, NATURAL),
+ PITCH_CLASS_LITERAL (B, FLAT),
+ PITCH_CLASS_LITERAL (E, FLAT),
+ PITCH_CLASS_LITERAL (A, FLAT),
+ PITCH_CLASS_LITERAL (D, FLAT),
+ PITCH_CLASS_LITERAL (G, FLAT),
+ PITCH_CLASS_LITERAL (C, FLAT)
+ };
+
+ key->pitch = PITCH_CLASS (PITCH_NAME (pitch), PITCH_ACCIDENTAL (pitch));
+
+ key->num_sharps = 0;
+ for (i = 0; i < ARRAY_SIZE (sharp_keys); i++)
+ if (sharp_keys[i] == key->pitch)
+ key->num_sharps = i;
+
+ key->num_flats = 0;
+ for (i = 0; i < ARRAY_SIZE (flat_keys); i++)
+ if (flat_keys[i] == key->pitch)
+ key->num_flats = i;
+}
+
+bool
+scherzo_key_contains_pitch (scherzo_key_t *key, pitch_t pitch)
+{
+ pitch_accidental_t accidental = PITCH_ACCIDENTAL (pitch);
+ int i;
+
+ pitch_name_t sharps_order[] = {
+ PITCH_NAME_F,
+ PITCH_NAME_C,
+ PITCH_NAME_G,
+ PITCH_NAME_D,
+ PITCH_NAME_A,
+ PITCH_NAME_E,
+ PITCH_NAME_B
+ };
+
+ pitch_name_t flats_order[] = {
+ PITCH_NAME_B,
+ PITCH_NAME_E,
+ PITCH_NAME_A,
+ PITCH_NAME_D,
+ PITCH_NAME_G,
+ PITCH_NAME_C,
+ PITCH_NAME_F
+ };
+
+ if (accidental == PITCH_ACCIDENTAL_DOUBLE_SHARP ||
+ accidental == PITCH_ACCIDENTAL_DOUBLE_FLAT)
+ {
+ return false;
+ }
+
+ if (key->num_sharps) {
+ if (accidental == PITCH_ACCIDENTAL_FLAT)
+ return false;
+ for (i = 0; i < key->num_sharps; i++) {
+ if (sharps_order[i] == PITCH_NAME (pitch)) {
+ if (accidental == PITCH_ACCIDENTAL_SHARP)
+ return true;
+ else
+ return false;
+ }
+ }
+ if (accidental == PITCH_ACCIDENTAL_SHARP)
+ return false;
+ else
+ return true;
+ }
+
+ if (key->num_flats) {
+ if (accidental == PITCH_ACCIDENTAL_SHARP)
+ return false;
+ for (i = 0; i < key->num_flats; i++) {
+ if (flats_order[i] == PITCH_NAME (pitch)) {
+ if (accidental == PITCH_ACCIDENTAL_FLAT)
+ return true;
+ else
+ return false;
+ }
+ }
+ if (accidental == PITCH_ACCIDENTAL_FLAT)
+ return false;
+ else
+ return true;
+ }
+
+ if (accidental == PITCH_ACCIDENTAL_NATURAL)
+ return true;
+ else
+ return false;
+}
--- /dev/null
+/* scherzo - Music notation training
+ *
+ * key.h - Common structures and functions for keys
+ *
+ * Copyright © 2013 Carl Worth
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see http://www.gnu.org/licenses/ .
+ */
+
+#include <stdbool.h>
+
+#ifndef SCHERZO_KEY_H
+#define SCHERZO_KEY_H
+
+#include "pitch.h"
+
+typedef struct scherzo_key
+{
+ /* Pitch class (diatonic) */
+ pitch_t pitch;
+
+ /* Number of sharps/flats in key (if any) */
+ int num_sharps;
+ int num_flats;
+} scherzo_key_t;
+
+void
+scherzo_key_init (scherzo_key_t *key, pitch_t pitch);
+
+bool
+scherzo_key_contains_pitch (scherzo_key_t *key, pitch_t pitch);
+
+#endif /* KEY_H */
int num_staves;
} score_brace_t;
-typedef struct score_key
-{
- /* Pitch class (diatonic) */
- pitch_t pitch;
-
- /* Number of sharps/flats in key (if any) */
- int num_sharps;
- int num_flats;
-} score_key_t;
-
struct score
{
/* Nominal height of a single staff (ledger lines may make it larger) */
int width;
/* Current (diatonic) key */
- score_key_t key;
+ scherzo_key_t key;
score_brace_t **braces;
int num_braces;
void
score_set_key (score_t *score, pitch_t key)
{
- int i;
-
- pitch_t sharp_keys[] = {
- PITCH_CLASS_LITERAL (C, NATURAL),
- PITCH_CLASS_LITERAL (G, NATURAL),
- PITCH_CLASS_LITERAL (D, NATURAL),
- PITCH_CLASS_LITERAL (A, NATURAL),
- PITCH_CLASS_LITERAL (E, NATURAL),
- PITCH_CLASS_LITERAL (B, NATURAL),
- PITCH_CLASS_LITERAL (F, SHARP),
- PITCH_CLASS_LITERAL (C, SHARP),
- };
-
- pitch_t flat_keys[] = {
- PITCH_CLASS_LITERAL (C, NATURAL),
- PITCH_CLASS_LITERAL (F, NATURAL),
- PITCH_CLASS_LITERAL (B, FLAT),
- PITCH_CLASS_LITERAL (E, FLAT),
- PITCH_CLASS_LITERAL (A, FLAT),
- PITCH_CLASS_LITERAL (D, FLAT),
- PITCH_CLASS_LITERAL (G, FLAT),
- PITCH_CLASS_LITERAL (C, FLAT)
- };
-
- score->key.pitch = PITCH_CLASS (PITCH_NAME (key), PITCH_ACCIDENTAL (key));
-
- score->key.num_sharps = 0;
- for (i = 0; i < ARRAY_SIZE (sharp_keys); i++)
- if (sharp_keys[i] == score->key.pitch)
- score->key.num_sharps = i;
-
- score->key.num_flats = 0;
- for (i = 0; i < ARRAY_SIZE (flat_keys); i++)
- if (flat_keys[i] == score->key.pitch)
- score->key.num_flats = i;
+ scherzo_key_init (&score->key, key);
}
-static int
-score_key_contains_pitch (score_key_t *key, pitch_t pitch)
-{
- pitch_accidental_t accidental = PITCH_ACCIDENTAL (pitch);
- int i;
-
- pitch_name_t sharps_order[] = {
- PITCH_NAME_F,
- PITCH_NAME_C,
- PITCH_NAME_G,
- PITCH_NAME_D,
- PITCH_NAME_A,
- PITCH_NAME_E,
- PITCH_NAME_B
- };
-
- pitch_name_t flats_order[] = {
- PITCH_NAME_B,
- PITCH_NAME_E,
- PITCH_NAME_A,
- PITCH_NAME_D,
- PITCH_NAME_G,
- PITCH_NAME_C,
- PITCH_NAME_F
- };
-
- if (accidental == PITCH_ACCIDENTAL_DOUBLE_SHARP ||
- accidental == PITCH_ACCIDENTAL_DOUBLE_FLAT)
- {
- return 0;
- }
-
- if (key->num_sharps) {
- if (accidental == PITCH_ACCIDENTAL_FLAT)
- return 0;
- for (i = 0; i < key->num_sharps; i++) {
- if (sharps_order[i] == PITCH_NAME (pitch)) {
- if (accidental == PITCH_ACCIDENTAL_SHARP)
- return 1;
- else
- return 0;
- }
- }
- if (accidental == PITCH_ACCIDENTAL_SHARP)
- return 0;
- else
- return 1;
- }
-
- if (key->num_flats) {
- if (accidental == PITCH_ACCIDENTAL_SHARP)
- return 0;
- for (i = 0; i < key->num_flats; i++) {
- if (flats_order[i] == PITCH_NAME (pitch)) {
- if (accidental == PITCH_ACCIDENTAL_FLAT)
- return 1;
- else
- return 0;
- }
- }
- if (accidental == PITCH_ACCIDENTAL_FLAT)
- return 0;
- else
- return 1;
- }
-
- if (accidental == PITCH_ACCIDENTAL_NATURAL)
- return 1;
- else
- return 0;
-}
/* Returns in brace_width the width of the brace */
static void
}
if (note->duration == 0 ||
- ! score_key_contains_pitch (&score->key, note->pitch))
+ ! scherzo_key_contains_pitch (&score->key, note->pitch))
{
note_glyph[num_glyphs].x = 0;
#include <talloc.h>
#include <cairo.h>
-#include "pitch.h"
+#include "scherzo-key.h"
typedef enum score_duration
{