annotate base/Pitch.cpp @ 33:51e158b505da

* Rearrange spectrogram cacheing so that gain, normalization, instantaneous frequency calculations etc can be done from the cached data (increasing the size of the cache, but also the usability).
author Chris Cannam
date Thu, 23 Feb 2006 18:01:31 +0000
parents 090c22aa726a
children 39ae3dee27b9
rev   line source
Chris@19 1 /* -*- c-basic-offset: 4 -*- vi:set ts=8 sts=4 sw=4: */
Chris@19 2
Chris@19 3 /*
Chris@19 4 A waveform viewer and audio annotation editor.
Chris@19 5 Chris Cannam, Queen Mary University of London, 2005-2006
Chris@19 6
Chris@19 7 This is experimental software. Not for distribution.
Chris@19 8 */
Chris@19 9
Chris@19 10 #include "Pitch.h"
Chris@19 11
Chris@19 12 #include <cmath>
Chris@19 13
Chris@19 14 float
Chris@19 15 Pitch::getFrequencyForPitch(int midiPitch,
Chris@19 16 float centsOffset,
Chris@19 17 float concertA)
Chris@19 18 {
Chris@19 19 float p = float(midiPitch) + (centsOffset / 100);
Chris@19 20 return concertA * powf(2.0, (p - 69.0) / 12.0);
Chris@19 21 }
Chris@19 22
Chris@19 23 int
Chris@19 24 Pitch::getPitchForFrequency(float frequency,
Chris@19 25 float *centsOffsetReturn,
Chris@19 26 float concertA)
Chris@19 27 {
Chris@19 28 float p = 12.0 * (log(frequency / (concertA / 2.0)) / log(2.0)) + 57.0;
Chris@19 29
Chris@19 30 int midiPitch = int(p + 0.00001);
Chris@19 31 float centsOffset = (p - midiPitch) * 100.0;
Chris@19 32
Chris@19 33 if (centsOffset >= 50.0) {
Chris@19 34 midiPitch = midiPitch + 1;
Chris@19 35 centsOffset = -(100.0 - centsOffset);
Chris@19 36 }
Chris@19 37
Chris@19 38 if (centsOffsetReturn) *centsOffsetReturn = centsOffset;
Chris@19 39 return midiPitch;
Chris@19 40 }
Chris@19 41
Chris@19 42 static QString notes[] = {
Chris@19 43 "C%1", "C#%1", "D%1", "D#%1",
Chris@19 44 "E%1", "F%1", "F#%1", "G%1",
Chris@19 45 "G#%1", "A%1", "A#%1", "B%1"
Chris@19 46 };
Chris@19 47
Chris@26 48 static QString flatNotes[] = {
Chris@26 49 "C%1", "Db%1", "D%1", "Eb%1",
Chris@26 50 "E%1", "F%1", "Gb%1", "G%1",
Chris@26 51 "Ab%1", "A%1", "Bb%1", "B%1"
Chris@26 52 };
Chris@26 53
Chris@19 54 QString
Chris@19 55 Pitch::getPitchLabel(int midiPitch,
Chris@26 56 float centsOffset,
Chris@26 57 bool useFlats)
Chris@19 58 {
Chris@19 59 int octave = -2;
Chris@19 60
Chris@19 61 if (midiPitch < 0) {
Chris@19 62 while (midiPitch < 0) {
Chris@19 63 midiPitch += 12;
Chris@19 64 --octave;
Chris@19 65 }
Chris@19 66 } else {
Chris@19 67 octave = midiPitch / 12 - 2;
Chris@19 68 }
Chris@19 69
Chris@26 70 QString plain = (useFlats ? flatNotes : notes)[midiPitch % 12].arg(octave);
Chris@19 71
Chris@19 72 int ic = lrintf(centsOffset);
Chris@19 73 if (ic == 0) return plain;
Chris@19 74 else if (ic > 0) return QString("%1+%2c").arg(plain).arg(ic);
Chris@19 75 else return QString("%1%2c").arg(plain).arg(ic);
Chris@19 76 }
Chris@19 77
Chris@19 78 QString
Chris@19 79 Pitch::getPitchLabelForFrequency(float frequency,
Chris@26 80 float concertA,
Chris@26 81 bool useFlats)
Chris@19 82 {
Chris@19 83 float centsOffset = 0.0;
Chris@19 84 int midiPitch = getPitchForFrequency(frequency, &centsOffset, concertA);
Chris@26 85 return getPitchLabel(midiPitch, centsOffset, useFlats);
Chris@19 86 }
Chris@19 87