annotate base/Pitch.cpp @ 340:516819f2b97b

* Add Erase tool and mode * Add icons for Normalize buttons in property boxes, and for Show Peaks * Add support for velocity in notes -- not yet reflected in display or editable in the note edit dialog, but they are imported from MIDI, played, and exported * Begin work on making pastes align pasted times (subtler than I thought)
author Chris Cannam
date Fri, 23 Nov 2007 16:48:23 +0000
parents e412f65884ee
children 0a44caddd9fe
rev   line source
Chris@49 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@19 2
Chris@19 3 /*
Chris@52 4 Sonic Visualiser
Chris@52 5 An audio file viewer and annotation editor.
Chris@52 6 Centre for Digital Music, Queen Mary, University of London.
Chris@52 7 This file copyright 2006 Chris Cannam.
Chris@19 8
Chris@52 9 This program is free software; you can redistribute it and/or
Chris@52 10 modify it under the terms of the GNU General Public License as
Chris@52 11 published by the Free Software Foundation; either version 2 of the
Chris@52 12 License, or (at your option) any later version. See the file
Chris@52 13 COPYING included with this distribution for more information.
Chris@19 14 */
Chris@19 15
Chris@19 16 #include "Pitch.h"
Chris@141 17 #include "Preferences.h"
Chris@19 18
Chris@19 19 #include <cmath>
Chris@19 20
Chris@19 21 float
Chris@19 22 Pitch::getFrequencyForPitch(int midiPitch,
Chris@19 23 float centsOffset,
Chris@19 24 float concertA)
Chris@19 25 {
Chris@141 26 if (concertA <= 0.0) {
Chris@141 27 concertA = Preferences::getInstance()->getTuningFrequency();
Chris@141 28 }
Chris@19 29 float p = float(midiPitch) + (centsOffset / 100);
Chris@19 30 return concertA * powf(2.0, (p - 69.0) / 12.0);
Chris@19 31 }
Chris@19 32
Chris@19 33 int
Chris@19 34 Pitch::getPitchForFrequency(float frequency,
Chris@19 35 float *centsOffsetReturn,
Chris@19 36 float concertA)
Chris@19 37 {
Chris@141 38 if (concertA <= 0.0) {
Chris@141 39 concertA = Preferences::getInstance()->getTuningFrequency();
Chris@141 40 }
Chris@19 41 float p = 12.0 * (log(frequency / (concertA / 2.0)) / log(2.0)) + 57.0;
Chris@19 42
Chris@19 43 int midiPitch = int(p + 0.00001);
Chris@19 44 float centsOffset = (p - midiPitch) * 100.0;
Chris@19 45
Chris@19 46 if (centsOffset >= 50.0) {
Chris@19 47 midiPitch = midiPitch + 1;
Chris@19 48 centsOffset = -(100.0 - centsOffset);
Chris@19 49 }
Chris@19 50
Chris@19 51 if (centsOffsetReturn) *centsOffsetReturn = centsOffset;
Chris@19 52 return midiPitch;
Chris@19 53 }
Chris@19 54
Chris@19 55 static QString notes[] = {
Chris@19 56 "C%1", "C#%1", "D%1", "D#%1",
Chris@19 57 "E%1", "F%1", "F#%1", "G%1",
Chris@19 58 "G#%1", "A%1", "A#%1", "B%1"
Chris@19 59 };
Chris@19 60
Chris@26 61 static QString flatNotes[] = {
Chris@26 62 "C%1", "Db%1", "D%1", "Eb%1",
Chris@26 63 "E%1", "F%1", "Gb%1", "G%1",
Chris@26 64 "Ab%1", "A%1", "Bb%1", "B%1"
Chris@26 65 };
Chris@26 66
Chris@19 67 QString
Chris@19 68 Pitch::getPitchLabel(int midiPitch,
Chris@26 69 float centsOffset,
Chris@26 70 bool useFlats)
Chris@19 71 {
Chris@19 72 int octave = -2;
Chris@19 73
Chris@19 74 if (midiPitch < 0) {
Chris@19 75 while (midiPitch < 0) {
Chris@19 76 midiPitch += 12;
Chris@19 77 --octave;
Chris@19 78 }
Chris@19 79 } else {
Chris@19 80 octave = midiPitch / 12 - 2;
Chris@19 81 }
Chris@19 82
Chris@26 83 QString plain = (useFlats ? flatNotes : notes)[midiPitch % 12].arg(octave);
Chris@19 84
Chris@19 85 int ic = lrintf(centsOffset);
Chris@19 86 if (ic == 0) return plain;
Chris@19 87 else if (ic > 0) return QString("%1+%2c").arg(plain).arg(ic);
Chris@19 88 else return QString("%1%2c").arg(plain).arg(ic);
Chris@19 89 }
Chris@19 90
Chris@19 91 QString
Chris@19 92 Pitch::getPitchLabelForFrequency(float frequency,
Chris@26 93 float concertA,
Chris@26 94 bool useFlats)
Chris@19 95 {
Chris@141 96 if (concertA <= 0.0) {
Chris@141 97 concertA = Preferences::getInstance()->getTuningFrequency();
Chris@141 98 }
Chris@19 99 float centsOffset = 0.0;
Chris@19 100 int midiPitch = getPitchForFrequency(frequency, &centsOffset, concertA);
Chris@26 101 return getPitchLabel(midiPitch, centsOffset, useFlats);
Chris@19 102 }
Chris@19 103
Chris@274 104 bool
Chris@274 105 Pitch::isFrequencyInMidiRange(float frequency,
Chris@274 106 float concertA)
Chris@274 107 {
Chris@274 108 float centsOffset = 0.0;
Chris@274 109 int midiPitch = getPitchForFrequency(frequency, &centsOffset, concertA);
Chris@274 110 return (midiPitch >= 0 && midiPitch < 128);
Chris@274 111 }
Chris@274 112