1 | <?php |
||
7 | class Note |
||
8 | { |
||
9 | const ACCIDENTAL_NATURAL = ''; |
||
10 | const ACCIDENTAL_SHARP = '♯'; |
||
11 | const ACCIDENTAL_FLAT = '♭'; |
||
12 | const ACCIDENTAL_DOUBLE_SHARP = 'x'; |
||
13 | const ACCIDENTAL_DOUBLE_FLAT = '♭♭'; |
||
14 | const ACCIDENTAL_QUARTER_SHARP = '¼♯'; |
||
15 | const ACCIDENTAL_QUARTER_FLAT = '¼♭'; |
||
16 | const ACCIDENTAL_THREE_QUARTER_SHARP = '¾♯'; |
||
17 | const ACCIDENTAL_THREE_QUARTER_FLAT = '¾♭'; |
||
18 | |||
19 | const PATTERN_ACCIDENTAL_SHARP = '([♯s#]|sharp)'; |
||
20 | const PATTERN_ACCIDENTAL_FLAT = '([♭fb]|flat)'; |
||
21 | const PATTERN_ACCIDENTAL_QUARTER = '(quarter|¼|1\/4)[ -]?'; |
||
22 | const PATTERN_ACCIDENTAL_3_QUARTER = '((three|3)[ -]quarter|¾|3\/4)[ -]?'; |
||
23 | |||
24 | private static $accidentalPatterns = [ |
||
25 | '' => self::ACCIDENTAL_NATURAL, |
||
26 | self::PATTERN_ACCIDENTAL_FLAT => self::ACCIDENTAL_FLAT, |
||
27 | self::PATTERN_ACCIDENTAL_SHARP => self::ACCIDENTAL_SHARP, |
||
28 | '(\-|' . self::PATTERN_ACCIDENTAL_QUARTER . self::PATTERN_ACCIDENTAL_FLAT . ')' => self::ACCIDENTAL_QUARTER_FLAT, |
||
29 | '(\+|' . self::PATTERN_ACCIDENTAL_QUARTER . self::PATTERN_ACCIDENTAL_SHARP . ')' => self::ACCIDENTAL_QUARTER_SHARP, |
||
30 | '(𝄫|bb|double[ -]' . self::PATTERN_ACCIDENTAL_FLAT . ')' => self::ACCIDENTAL_DOUBLE_FLAT, |
||
31 | '(𝄪|♯♯|##|double[ -]' . self::PATTERN_ACCIDENTAL_SHARP . ')' => self::ACCIDENTAL_DOUBLE_SHARP, |
||
32 | '(' . self::PATTERN_ACCIDENTAL_FLAT . '\-|' . self::PATTERN_ACCIDENTAL_3_QUARTER . self::PATTERN_ACCIDENTAL_FLAT . ')' => self::ACCIDENTAL_THREE_QUARTER_FLAT, |
||
33 | '(' . self::PATTERN_ACCIDENTAL_SHARP . '\+|' . self::PATTERN_ACCIDENTAL_3_QUARTER . self::PATTERN_ACCIDENTAL_SHARP . ')' => self::ACCIDENTAL_THREE_QUARTER_SHARP, |
||
34 | ]; |
||
35 | |||
36 | private static $accidentalCents = [ |
||
37 | self::ACCIDENTAL_NATURAL => 0, |
||
38 | self::ACCIDENTAL_FLAT => -100, |
||
39 | self::ACCIDENTAL_SHARP => 100, |
||
40 | self::ACCIDENTAL_QUARTER_FLAT => -50, |
||
41 | self::ACCIDENTAL_QUARTER_SHARP => 50, |
||
42 | self::ACCIDENTAL_DOUBLE_FLAT => -200, |
||
43 | self::ACCIDENTAL_DOUBLE_SHARP => 200, |
||
44 | self::ACCIDENTAL_THREE_QUARTER_FLAT => -150, |
||
45 | self::ACCIDENTAL_THREE_QUARTER_SHARP => 150, |
||
46 | ]; |
||
47 | |||
48 | private static $preferredAccidentals = [ |
||
49 | self::ACCIDENTAL_NATURAL, |
||
50 | self::ACCIDENTAL_SHARP, |
||
51 | self::ACCIDENTAL_FLAT, |
||
52 | self::ACCIDENTAL_QUARTER_SHARP, |
||
53 | self::ACCIDENTAL_QUARTER_FLAT, |
||
54 | self::ACCIDENTAL_DOUBLE_SHARP, |
||
55 | self::ACCIDENTAL_DOUBLE_FLAT, |
||
56 | self::ACCIDENTAL_THREE_QUARTER_FLAT, |
||
57 | self::ACCIDENTAL_THREE_QUARTER_SHARP, |
||
58 | ]; |
||
59 | |||
60 | private static $names = [ |
||
61 | 'C' => 0, |
||
62 | 'D' => 200, |
||
63 | 'E' => 400, |
||
64 | 'F' => 500, |
||
65 | 'G' => 700, |
||
66 | 'A' => 900, |
||
67 | 'B' => 1100, |
||
68 | ]; |
||
69 | |||
70 | private $name; |
||
71 | private $accidental; |
||
72 | private $octave; |
||
73 | private $difference; |
||
74 | |||
75 | /** |
||
76 | * Internal constructor. Use one of the factory methods to create a Note. |
||
77 | * |
||
78 | * @see Note::fromCents() |
||
79 | * @see Note::fromFrequency() |
||
80 | * @see Note::fromName() |
||
81 | * |
||
82 | * @param string $name The note name (A-G). |
||
83 | * @param string $accidental The accidental (one of the Note::ACCIDENTAL_ |
||
84 | * constants). |
||
85 | * @param int $octave The octave, in scientific pitch notation. |
||
86 | * @param float $difference The note's difference in cents from 12-TET. |
||
87 | */ |
||
88 | private function __construct(string $name, string $accidental, int $octave, float $difference) |
||
99 | |||
100 | /** |
||
101 | * Instantiate a Note from a number of cents. |
||
102 | * |
||
103 | * @param float $cents A number of cents above C4. |
||
104 | * @param string[] $preferredAccidentals A list of accidentals in order of |
||
105 | * preference. This will be merged |
||
106 | * with a default list. |
||
107 | * |
||
108 | * @return self |
||
109 | */ |
||
110 | public static function fromCents(float $cents, array $preferredAccidentals = []): self |
||
126 | |||
127 | /** |
||
128 | * Instantiate a Note from a note name. |
||
129 | * |
||
130 | * @param string $name A note name with an accidental and an octave in |
||
131 | * scientific pitch notation, e.g. C#4 or Eb5. |
||
132 | * |
||
133 | * @return \ExtendedStrings\Strings\Note |
||
134 | */ |
||
135 | public static function fromName(string $name): self |
||
164 | |||
165 | /** |
||
166 | * Instantiate a Note from a frequency. |
||
167 | * |
||
168 | * @param float $frequency The frequency (in Hz). |
||
169 | * @param float $A4 The frequency of A4, for reference. |
||
170 | * @param array $preferredAccidentals Some preferred accidentals. |
||
171 | * |
||
172 | * @return self |
||
173 | */ |
||
174 | public static function fromFrequency($frequency, float $A4 = 440.0, array $preferredAccidentals = []): self |
||
178 | |||
179 | /** |
||
180 | * Returns the note as a number of cents above C4. |
||
181 | * |
||
182 | * @return float |
||
183 | */ |
||
184 | public function getCents(): float |
||
191 | |||
192 | /** |
||
193 | * Returns the note as a frequency. |
||
194 | * |
||
195 | * @param float $A4 The frequency of A4 (in Hz), for reference. |
||
196 | * |
||
197 | * @return float |
||
198 | */ |
||
199 | public function getFrequency(float $A4 = 440.0): float |
||
203 | |||
204 | /** |
||
205 | * Returns a string representation of the note. |
||
206 | * |
||
207 | * @return string |
||
208 | */ |
||
209 | public function __toString(): string |
||
218 | |||
219 | /** |
||
220 | * Returns the simple note name (one of A-G). |
||
221 | * |
||
222 | * @return string |
||
223 | */ |
||
224 | public function getName(): string |
||
228 | |||
229 | /** |
||
230 | * Returns the accidental (one of the Note::ACCIDENTAL_ constants). |
||
231 | * |
||
232 | * @return string |
||
233 | */ |
||
234 | public function getAccidental(): string |
||
238 | |||
239 | /** |
||
240 | * Returns the octave (in scientific pitch notation). |
||
241 | * |
||
242 | * @return int |
||
243 | */ |
||
244 | public function getOctave(): int |
||
248 | |||
249 | /** |
||
250 | * Returns the difference between the note and its 12-TET form, in cents. |
||
251 | * |
||
252 | * @return float |
||
253 | */ |
||
254 | public function getDifference(): float |
||
258 | |||
259 | /** |
||
260 | * @param string $accidental |
||
261 | * |
||
262 | * @return string |
||
263 | */ |
||
264 | private static function normalizeAccidental(string $accidental): string |
||
276 | } |
||
277 |