1 | <?php |
||
7 | class Note |
||
8 | { |
||
9 | const ACCIDENTAL_NATURAL = ''; |
||
10 | const ACCIDENTAL_SHARP = '#'; |
||
11 | const ACCIDENTAL_FLAT = 'b'; |
||
12 | const ACCIDENTAL_DOUBLE_SHARP = 'x'; |
||
13 | const ACCIDENTAL_DOUBLE_FLAT = 'bb'; |
||
14 | const ACCIDENTAL_QUARTER_SHARP = '+'; |
||
15 | const ACCIDENTAL_QUARTER_FLAT = '-'; |
||
16 | const ACCIDENTAL_THREE_QUARTER_SHARP = '#+'; |
||
17 | const ACCIDENTAL_THREE_QUARTER_FLAT = 'b-'; |
||
18 | |||
19 | private static $accidentalPatterns = [ |
||
20 | '' => self::ACCIDENTAL_NATURAL, |
||
21 | "([fb]|\u{266D}|flat)" => self::ACCIDENTAL_FLAT, |
||
22 | "([s#]|\u{266F}|sharp)" => self::ACCIDENTAL_SHARP, |
||
23 | '(\-|quarter[ -]flat)' => self::ACCIDENTAL_QUARTER_FLAT, |
||
24 | '(\+|quarter[ -]sharp)' => self::ACCIDENTAL_QUARTER_SHARP, |
||
25 | '(bb|double[ -]flat)' => self::ACCIDENTAL_DOUBLE_FLAT, |
||
26 | '(##|x|double[ -]sharp)' => self::ACCIDENTAL_DOUBLE_SHARP, |
||
27 | '(b\-|(three|3)[ -]quarter[ -]flat)' => self::ACCIDENTAL_THREE_QUARTER_FLAT, |
||
28 | '(#\+|(three|3)[ -]quarter[ -]sharp)' => self::ACCIDENTAL_THREE_QUARTER_SHARP, |
||
29 | ]; |
||
30 | |||
31 | private static $accidentalCents = [ |
||
32 | self::ACCIDENTAL_NATURAL => 0, |
||
33 | self::ACCIDENTAL_FLAT => -100, |
||
34 | self::ACCIDENTAL_SHARP => 100, |
||
35 | self::ACCIDENTAL_QUARTER_FLAT => -50, |
||
36 | self::ACCIDENTAL_QUARTER_SHARP => 50, |
||
37 | self::ACCIDENTAL_DOUBLE_FLAT => -200, |
||
38 | self::ACCIDENTAL_DOUBLE_SHARP => 200, |
||
39 | self::ACCIDENTAL_THREE_QUARTER_FLAT => -150, |
||
40 | self::ACCIDENTAL_THREE_QUARTER_SHARP => 150, |
||
41 | ]; |
||
42 | |||
43 | private static $preferredAccidentals = [ |
||
44 | self::ACCIDENTAL_NATURAL, |
||
45 | self::ACCIDENTAL_SHARP, |
||
46 | self::ACCIDENTAL_FLAT, |
||
47 | self::ACCIDENTAL_QUARTER_SHARP, |
||
48 | self::ACCIDENTAL_QUARTER_FLAT, |
||
49 | self::ACCIDENTAL_DOUBLE_SHARP, |
||
50 | self::ACCIDENTAL_DOUBLE_FLAT, |
||
51 | self::ACCIDENTAL_THREE_QUARTER_FLAT, |
||
52 | self::ACCIDENTAL_THREE_QUARTER_SHARP, |
||
53 | ]; |
||
54 | |||
55 | private static $names = [ |
||
56 | 'C' => 0, |
||
57 | 'D' => 200, |
||
58 | 'E' => 400, |
||
59 | 'F' => 500, |
||
60 | 'G' => 700, |
||
61 | 'A' => 900, |
||
62 | 'B' => 1100, |
||
63 | ]; |
||
64 | |||
65 | private $name; |
||
66 | private $accidental; |
||
67 | private $octave; |
||
68 | |||
69 | /** |
||
70 | * Internal constructor: use one of the factory methods to create a Note. |
||
71 | * |
||
72 | * @param string $name The note name (A-G). |
||
73 | * @param string $accidental The accidental (one of the Note::ACCIDENTAL_ |
||
74 | * constants). |
||
75 | * @param int $octave The octave, in scientific pitch notation. |
||
76 | */ |
||
77 | private function __construct(string $name, string $accidental, int $octave) |
||
83 | |||
84 | /** |
||
85 | * Factory to create a Note from a note name. |
||
86 | * |
||
87 | * @param string $name A note name with an accidental and an octave in |
||
88 | * scientific pitch notation, e.g. C#4 or Eb5. |
||
89 | * |
||
90 | * @return \ExtendedStrings\Strings\Note |
||
91 | */ |
||
92 | public static function fromName(string $name): self |
||
115 | |||
116 | /** |
||
117 | * Factory to create a Note from a number of cents. |
||
118 | * |
||
119 | * @param int $cents A number of cents above C4. |
||
120 | * @param string[] $preferredAccidentals A list of accidentals in order of |
||
121 | * preference. This will be merged |
||
122 | * with a default list. |
||
123 | * |
||
124 | * @return self |
||
125 | */ |
||
126 | public static function fromCents(int $cents, array $preferredAccidentals = []): self |
||
140 | |||
141 | /** |
||
142 | * @param string $accidental |
||
143 | * |
||
144 | * @return string |
||
145 | */ |
||
146 | private static function normalizeAccidental(string $accidental): string |
||
158 | |||
159 | /** |
||
160 | * @return int The number of cents above C4. |
||
161 | */ |
||
162 | public function getCents(): int |
||
168 | |||
169 | /** |
||
170 | * @return string |
||
171 | */ |
||
172 | public function __toString(): string |
||
176 | } |
||
177 |