123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519 |
- /* Formant.cpp
- *
- * Copyright (C) 1992-2012,2014,2015,2016,2017 Paul Boersma
- *
- * This code 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 2 of the License, or (at
- * your option) any later version.
- *
- * This code 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 work. If not, see <http://www.gnu.org/licenses/>.
- */
- #include "Formant.h"
- #include "oo_DESTROY.h"
- #include "Formant_def.h"
- #include "oo_COPY.h"
- #include "Formant_def.h"
- #include "oo_EQUAL.h"
- #include "Formant_def.h"
- #include "oo_CAN_WRITE_AS_ENCODING.h"
- #include "Formant_def.h"
- #include "oo_READ_TEXT.h"
- #include "Formant_def.h"
- #include "oo_WRITE_TEXT.h"
- #include "Formant_def.h"
- #include "oo_READ_BINARY.h"
- #include "Formant_def.h"
- #include "oo_WRITE_BINARY.h"
- #include "Formant_def.h"
- #include "oo_DESCRIPTION.h"
- #include "Formant_def.h"
- #include "enums_getText.h"
- #include "Formant_enums.h"
- #include "enums_getValue.h"
- #include "Formant_enums.h"
- Thing_implement (Formant, Sampled, 2); // version 1 = with intensity, 2 = double
- void structFormant :: v_info () {
- structDaata :: v_info ();
- MelderInfo_writeLine (U"Time domain:");
- MelderInfo_writeLine (U" Start time: ", xmin, U" seconds");
- MelderInfo_writeLine (U" End time: ", xmax, U" seconds");
- MelderInfo_writeLine (U" Total duration: ", xmax - xmin, U" seconds");
- MelderInfo_writeLine (U"Time sampling:");
- MelderInfo_writeLine (U" Number of frames: ", nx);
- MelderInfo_writeLine (U" Time step: ", dx, U" seconds");
- MelderInfo_writeLine (U" First frame centred at: ", x1, U" seconds");
- }
- double structFormant :: v_getValueAtSample (integer iframe, integer which, int units) {
- Formant_Frame frame = & d_frames [iframe];
- integer iformant = which >> 1;
- if (iformant < 1 || iformant > frame -> nFormants) return undefined;
- double frequency = frame -> formant [iformant]. frequency;
- if ((which & 1) == 0) {
- return units ? NUMhertzToBark (frequency) : frequency;
- } else {
- double bandwidth = frame -> formant [iformant]. bandwidth;
- if (units) {
- double fleft = frequency - 0.5 * bandwidth, fright = frequency + 0.5 * bandwidth;
- fleft = ( fleft <= 0.0 ? 0.0 : NUMhertzToBark (fleft) ); // prevent undefined
- fright = NUMhertzToBark (fright);
- return fright - fleft;
- }
- return bandwidth;
- }
- return undefined;
- }
- autoFormant Formant_create (double tmin, double tmax, integer nt, double dt, double t1,
- integer maximumNumberOfFormants)
- {
- try {
- autoFormant me = Thing_new (Formant);
- Sampled_init (me.get(), tmin, tmax, nt, dt, t1);
- my d_frames = NUMvector <structFormant_Frame> (1, nt);
- my maxnFormants = maximumNumberOfFormants;
- return me;
- } catch (MelderError) {
- Melder_throw (U"Formant object not created.");
- }
- }
- integer Formant_getMinNumFormants (Formant me) {
- integer minNumFormants = 100000000;
- for (integer iframe = 1; iframe <= my nx; iframe ++)
- if (my d_frames [iframe]. nFormants < minNumFormants)
- minNumFormants = my d_frames [iframe]. nFormants;
- return minNumFormants;
- }
- integer Formant_getMaxNumFormants (Formant me) {
- integer maxNumFormants = 0;
- for (integer iframe = 1; iframe <= my nx; iframe ++)
- if (my d_frames [iframe]. nFormants > maxNumFormants)
- maxNumFormants = my d_frames [iframe]. nFormants;
- return maxNumFormants;
- }
- void Formant_drawTracks (Formant me, Graphics g, double tmin, double tmax, double fmax, bool garnish) {
- integer itmin, itmax, ntrack = Formant_getMinNumFormants (me);
- if (tmax <= tmin) { tmin = my xmin; tmax = my xmax; }
- if (! Sampled_getWindowSamples (me, tmin, tmax, & itmin, & itmax)) return;
- Graphics_setInner (g);
- Graphics_setWindow (g, tmin, tmax, 0.0, fmax);
- for (integer itrack = 1; itrack <= ntrack; itrack ++) {
- for (integer iframe = itmin; iframe < itmax; iframe ++) {
- Formant_Frame curFrame = & my d_frames [iframe], nextFrame = & my d_frames [iframe + 1];
- double x1 = Sampled_indexToX (me, iframe), x2 = Sampled_indexToX (me, iframe + 1);
- double f1 = curFrame -> formant [itrack]. frequency;
- double f2 = nextFrame -> formant [itrack]. frequency;
- if (isdefined (x1) && isdefined (f1) && isdefined (x2) && isdefined (f2))
- Graphics_line (g, x1, f1, x2, f2);
- }
- }
- Graphics_unsetInner (g);
- if (garnish) {
- Graphics_drawInnerBox (g);
- Graphics_textBottom (g, true, U"Time (s)");
- Graphics_textLeft (g, true, U"Formant frequency (Hz)");
- Graphics_marksBottom (g, 2, true, true, false);
- Graphics_marksLeftEvery (g, 1.0, 1000.0, true, true, true);
- }
- }
- void Formant_drawSpeckles_inside (Formant me, Graphics g, double tmin, double tmax, double fmin, double fmax,
- double suppress_dB)
- {
- integer itmin, itmax;
- double maximumIntensity = 0.0, minimumIntensity;
- if (tmax <= tmin) { tmin = my xmin; tmax = my xmax; }
- if (! Sampled_getWindowSamples (me, tmin, tmax, & itmin, & itmax)) return;
- Graphics_setWindow (g, tmin, tmax, fmin, fmax);
- for (integer iframe = itmin; iframe <= itmax; iframe ++) {
- Formant_Frame frame = & my d_frames [iframe];
- if (frame -> intensity > maximumIntensity)
- maximumIntensity = frame -> intensity;
- }
- if (maximumIntensity == 0.0 || suppress_dB <= 0.0)
- minimumIntensity = 0.0; // ignore
- else
- minimumIntensity = maximumIntensity / pow (10.0, suppress_dB / 10.0);
- for (integer iframe = itmin; iframe <= itmax; iframe ++) {
- Formant_Frame frame = & my d_frames [iframe];
- double x = Sampled_indexToX (me, iframe);
- if (frame -> intensity < minimumIntensity) continue;
- for (integer iformant = 1; iformant <= frame -> nFormants; iformant ++) {
- double frequency = frame -> formant [iformant]. frequency;
- if (frequency >= fmin && frequency <= fmax)
- Graphics_speckle (g, x, frequency);
- }
- }
- }
- void Formant_drawSpeckles (Formant me, Graphics g, double tmin, double tmax, double fmax, double suppress_dB,
- bool garnish)
- {
- Graphics_setInner (g);
- Formant_drawSpeckles_inside (me, g, tmin, tmax, 0.0, fmax, suppress_dB);
- Graphics_unsetInner (g);
- if (garnish) {
- Graphics_drawInnerBox (g);
- Graphics_textBottom (g, true, U"Time (s)");
- Graphics_textLeft (g, true, U"Formant frequency (Hz)");
- Graphics_marksBottom (g, 2, true, true, false);
- Graphics_marksLeftEvery (g, 1.0, 1000.0, true, true, true);
- }
- }
- void Formant_formula_bandwidths (Formant me, conststring32 formula, Interpreter interpreter) {
- try {
- integer nrow = Formant_getMaxNumFormants (me);
- if (nrow < 1)
- Melder_throw (U"No formants available.");
- autoMatrix mat = Matrix_create (my xmin, my xmax, my nx, my dx, my x1, 0.5, nrow + 0.5, nrow, 1.0, 1.0);
- for (integer iframe = 1; iframe <= my nx; iframe ++) {
- Formant_Frame frame = & my d_frames [iframe];
- for (integer iformant = 1; iformant <= frame -> nFormants; iformant ++)
- mat -> z [iformant] [iframe] = frame -> formant [iformant]. bandwidth;
- }
- Matrix_formula (mat.get(), formula, interpreter, nullptr);
- for (integer iframe = 1; iframe <= my nx; iframe ++) {
- Formant_Frame frame = & my d_frames [iframe];
- for (integer iformant = 1; iformant <= frame -> nFormants; iformant ++)
- frame -> formant [iformant]. bandwidth = mat -> z [iformant] [iframe];
- }
- } catch (MelderError) {
- Melder_throw (me, U": bandwidth formula not executed.");
- }
- }
- void Formant_formula_frequencies (Formant me, conststring32 formula, Interpreter interpreter) {
- try {
- integer nrow = Formant_getMaxNumFormants (me);
- if (nrow < 1)
- Melder_throw (U"No formants available.");
- autoMatrix mat = Matrix_create (my xmin, my xmax, my nx, my dx, my x1, 0.5, nrow + 0.5, nrow, 1.0, 1.0);
- for (integer iframe = 1; iframe <= my nx; iframe ++) {
- Formant_Frame frame = & my d_frames [iframe];
- for (integer iformant = 1; iformant <= frame -> nFormants; iformant ++)
- mat -> z [iformant] [iframe] = frame -> formant [iformant]. frequency;
- }
- Matrix_formula (mat.get(), formula, interpreter, nullptr);
- for (integer iframe = 1; iframe <= my nx; iframe ++) {
- Formant_Frame frame = & my d_frames [iframe];
- for (integer iformant = 1; iformant <= frame -> nFormants; iformant ++)
- frame -> formant [iformant]. frequency = mat -> z [iformant] [iframe];
- }
- } catch (MelderError) {
- Melder_throw (me, U": frequency formula not executed.");
- }
- }
- void Formant_getExtrema (Formant me, integer iformant, double tmin, double tmax, double *fmin, double *fmax) {
- if (fmin) *fmin = 0.0;
- if (fmax) *fmax = 0.0;
- if (iformant < 1) return;
- if (tmax <= tmin) { tmin = my xmin; tmax = my xmax; }
- integer itmin, itmax;
- if (! Sampled_getWindowSamples (me, tmin, tmax, & itmin, & itmax)) return;
- for (integer iframe = itmin; iframe <= itmax; iframe ++) {
- Formant_Frame frame = & my d_frames [iframe];
- if (iformant > frame -> nFormants) continue;
- double f = frame -> formant [iformant]. frequency;
- if (f == 0.0) continue;
- if (fmin) if (f < *fmin || *fmin == 0.0) *fmin = f;
- if (fmax) if (f > *fmax) *fmax = f;
- }
- }
- void Formant_getMinimumAndTime (Formant me, integer iformant, double tmin, double tmax, kFormant_unit unit, int interpolate,
- double *return_minimum, double *return_timeOfMinimum)
- {
- Sampled_getMinimumAndX (me, tmin, tmax, iformant << 1, (int) unit, interpolate, return_minimum, return_timeOfMinimum);
- if (return_minimum && *return_minimum <= 0.0) *return_minimum = undefined;
- }
- double Formant_getMinimum (Formant me, integer iformant, double tmin, double tmax, kFormant_unit unit, int interpolate) {
- double minimum;
- Formant_getMinimumAndTime (me, iformant, tmin, tmax, unit, interpolate, & minimum, nullptr);
- return minimum;
- }
- double Formant_getTimeOfMinimum (Formant me, integer iformant, double tmin, double tmax, kFormant_unit unit, int interpolate) {
- double time;
- Formant_getMinimumAndTime (me, iformant, tmin, tmax, unit, interpolate, nullptr, & time);
- return time;
- }
- void Formant_getMaximumAndTime (Formant me, integer iformant, double tmin, double tmax, kFormant_unit unit, int interpolate,
- double *return_maximum, double *return_timeOfMaximum)
- {
- Sampled_getMaximumAndX (me, tmin, tmax, iformant << 1, (int) unit, interpolate, return_maximum, return_timeOfMaximum);
- if (return_maximum && *return_maximum <= 0.0) *return_maximum = undefined; // unlikely
- }
- double Formant_getMaximum (Formant me, integer iformant, double tmin, double tmax, kFormant_unit unit, int interpolate) {
- double maximum;
- Formant_getMaximumAndTime (me, iformant, tmin, tmax, unit, interpolate, & maximum, nullptr);
- return maximum;
- }
- double Formant_getTimeOfMaximum (Formant me, integer iformant, double tmin, double tmax, kFormant_unit unit, int interpolate) {
- double time;
- Formant_getMaximumAndTime (me, iformant, tmin, tmax, unit, interpolate, nullptr, & time);
- return time;
- }
- double Formant_getQuantile (Formant me, integer iformant, double quantile, double tmin, double tmax, kFormant_unit unit) {
- return Sampled_getQuantile (me, tmin, tmax, quantile, iformant << 1, (int) unit);
- }
- double Formant_getMean (Formant me, integer iformant, double tmin, double tmax, kFormant_unit unit) {
- return Sampled_getMean (me, tmin, tmax, iformant << 1, (int) unit, true);
- }
- double Formant_getStandardDeviation (Formant me, integer iformant, double tmin, double tmax, kFormant_unit unit) {
- if (iformant < 1 || isundef (tmin) || isundef (tmax)) return undefined;
- if (tmax <= tmin) { tmin = my xmin; tmax = my xmax; }
- integer itmin, itmax;
- if (! Sampled_getWindowSamples (me, tmin, tmax, & itmin, & itmax)) return undefined;
- double mean = Formant_getMean (me, iformant, tmin, tmax, unit);
- double sum = 0.0;
- integer n = 0;
- for (integer iframe = itmin; iframe <= itmax; iframe ++) {
- Formant_Frame frame = & my d_frames [iframe];
- if (iformant > frame -> nFormants) continue;
- double f = frame -> formant [iformant]. frequency;
- if (f == 0.0) continue;
- if (unit == kFormant_unit::BARK) f = NUMhertzToBark (f);
- n += 1;
- sum += (f - mean) * (f - mean);
- }
- if (n > 1) return sqrt (sum / (n - 1));
- return undefined;
- }
- double Formant_getValueAtTime (Formant me, integer iformant, double time, kFormant_unit unit) {
- return Sampled_getValueAtX (me, time, iformant << 1, (int) unit, true);
- }
- double Formant_getBandwidthAtTime (Formant me, integer iformant, double time, kFormant_unit unit) {
- return Sampled_getValueAtX (me, time, (iformant << 1) + 1, (int) unit, true);
- }
- double Formant_getQuantileOfBandwidth (Formant me, integer iformant, double quantile, double tmin, double tmax, kFormant_unit unit) {
- return Sampled_getQuantile (me, tmin, tmax, quantile, (iformant << 1) + 1, (int) unit);
- }
- void Formant_scatterPlot (Formant me, Graphics g, double tmin, double tmax,
- integer iformant1, double fmin1, double fmax1, integer iformant2, double fmin2, double fmax2,
- double size_mm, conststring32 mark, bool garnish)
- {
- if (iformant1 < 1 || iformant2 < 1) return;
- if (tmax <= tmin) { tmin = my xmin; tmax = my xmax; }
- integer itmin, itmax;
- if (! Sampled_getWindowSamples (me, tmin, tmax, & itmin, & itmax)) return;
- if (fmax1 == fmin1)
- Formant_getExtrema (me, iformant1, tmin, tmax, & fmin1, & fmax1);
- if (fmax1 == fmin1) return;
- if (fmax2 == fmin2)
- Formant_getExtrema (me, iformant2, tmin, tmax, & fmin2, & fmax2);
- if (fmax2 == fmin2) return;
- Graphics_setInner (g);
- Graphics_setWindow (g, fmin1, fmax1, fmin2, fmax2);
- for (integer iframe = itmin; iframe <= itmax; iframe ++) {
- Formant_Frame frame = & my d_frames [iframe];
- if (iformant1 > frame -> nFormants || iformant2 > frame -> nFormants) continue;
- double x = frame -> formant [iformant1]. frequency;
- double y = frame -> formant [iformant2]. frequency;
- if (x == 0.0 || y == 0.0) continue;
- Graphics_mark (g, x, y, size_mm, mark);
- }
- Graphics_unsetInner (g);
- if (garnish) {
- Graphics_drawInnerBox (g);
- Graphics_textBottom (g, true, Melder_cat (U"%%F_", iformant1, U" (Hz)"));
- Graphics_textLeft (g, true, Melder_cat (U"%%F_", iformant2, U" (Hz)"));
- Graphics_marksBottom (g, 2, true, true, false);
- Graphics_marksLeft (g, 2, true, true, false);
- }
- }
- autoMatrix Formant_to_Matrix (Formant me, integer iformant) {
- try {
- autoMatrix thee = Matrix_create (my xmin, my xmax, my nx, my dx, my x1, 1.0, 1.0, 1, 1.0, 1.0);
- for (integer iframe = 1; iframe <= my nx; iframe ++) {
- Formant_Frame frame = & my d_frames [iframe];
- thy z [1] [iframe] = iformant <= frame -> nFormants ?
- frame -> formant [iformant]. frequency : 0.0;
- }
- return thee;
- } catch (MelderError) {
- Melder_throw (me, U": frequencies of formant ", iformant, U" not converted to Matrix.");
- }
- }
- autoMatrix Formant_to_Matrix_bandwidths (Formant me, integer iformant) {
- try {
- autoMatrix thee = Matrix_create (my xmin, my xmax, my nx, my dx, my x1, 1.0, 1.0, 1, 1.0, 1.0);
- for (integer iframe = 1; iframe <= my nx; iframe ++) {
- Formant_Frame frame = & my d_frames [iframe];
- thy z [1] [iframe] = iformant <= frame -> nFormants ?
- frame -> formant [iformant]. bandwidth : 0.0;
- }
- return thee;
- } catch (MelderError) {
- Melder_throw (me, U": bandwidths of formant ", iformant, U" not converted to Matrix.");
- }
- }
- /*** Viterbi methods. ***/
- struct fparm { Formant me, thee; double dfCost, bfCost, octaveJumpCost, refF [1 + 5]; };
- static double getLocalCost (integer iframe, integer icand, integer itrack, void *closure) {
- struct fparm *me = (struct fparm *) closure;
- Formant_Frame frame = & my my d_frames [iframe];
- Formant_Formant candidate;
- if (icand > frame -> nFormants) return 1e30;
- candidate = & frame -> formant [icand];
- /*if (candidate -> frequency <= 0.0) candidate -> frequency = 0.001;
- Melder_fatal (U"Weird formant frequency ", candidate -> frequency, U" Hz.")*/;
- Melder_assert (candidate -> bandwidth > 0.0);
- Melder_assert (itrack > 0 && itrack <= 5);
- return my dfCost * fabs (candidate -> frequency - my refF [itrack]) +
- my bfCost * candidate -> bandwidth / candidate -> frequency;
- }
- static double getTransitionCost (integer iframe, integer icand1, integer icand2, integer itrack, void *closure) {
- struct fparm *me = (struct fparm *) closure;
- Formant_Frame prevFrame = & my my d_frames [iframe - 1], curFrame = & my my d_frames [iframe];
- double f1, f2;
- (void) itrack;
- if (icand1 > prevFrame -> nFormants || icand2 > curFrame -> nFormants) return 1e30;
- f1 = prevFrame -> formant [icand1]. frequency;
- f2 = curFrame -> formant [icand2]. frequency;
- /*Melder_assert (f1 > 0.0);*/
- /*Melder_assert (f2 > 0.0);*/
- return my octaveJumpCost * fabs (NUMlog2 (f1 / f2));
- }
- static void putResult (integer iframe, integer place, integer itrack, void *closure) {
- struct fparm *me = (struct fparm *) closure;
- Melder_assert (iframe > 0 && iframe <= my my nx);
- Melder_assert (itrack > 0 && itrack <= 5);
- Melder_assert (place > 0);
- Melder_assert (place <= my my d_frames [iframe]. nFormants);
- my thy d_frames [iframe]. formant [itrack] = my my d_frames [iframe]. formant [place];
- }
- autoFormant Formant_tracker (Formant me, integer ntrack,
- double refF1, double refF2, double refF3, double refF4, double refF5,
- double dfCost, /* Per kHz. */
- double bfCost, double octaveJumpCost)
- {
- try {
- integer nformmin = Formant_getMinNumFormants (me);
- struct fparm parm;
- if (ntrack > nformmin) Melder_throw (U"Number of tracks (", ntrack, U") should not exceed minimum number of formants (", nformmin, U").");
- autoFormant thee = Formant_create (my xmin, my xmax, my nx, my dx, my x1, ntrack);
- for (integer iframe = 1; iframe <= thy nx; iframe ++) {
- thy d_frames [iframe]. formant = NUMvector <structFormant_Formant> (1, ntrack);
- thy d_frames [iframe]. nFormants = ntrack;
- thy d_frames [iframe]. intensity = my d_frames [iframe]. intensity;
- }
- /* BUG: limit costs to 1e10 or so */
- parm.me = me;
- parm.thee = thee.get();
- parm.dfCost = dfCost / 1000.0; // per Hz
- parm.bfCost = bfCost;
- parm.octaveJumpCost = octaveJumpCost;
- parm.refF [1] = refF1;
- parm.refF [2] = refF2;
- parm.refF [3] = refF3;
- parm.refF [4] = refF4;
- parm.refF [5] = refF5;
- NUM_viterbi_multi (my nx, my maxnFormants, ntrack,
- getLocalCost, getTransitionCost, putResult, & parm);
- return thee;
- } catch (MelderError) {
- Melder_throw (me, U": not tracked.");
- }
- }
- autoTable Formant_downto_Table (Formant me, bool includeFrameNumbers,
- bool includeTimes, integer timeDecimals,
- bool includeIntensity, integer intensityDecimals,
- bool includeNumberOfFormants, integer frequencyDecimals,
- bool includeBandwidths)
- {
- try {
- autoTable thee = Table_createWithoutColumnNames (my nx, includeFrameNumbers + includeTimes + includeIntensity +
- includeNumberOfFormants + my maxnFormants * (1 + includeBandwidths));
- integer icol = 0;
- if (includeFrameNumbers) Table_setColumnLabel (thee.get(), ++ icol, U"frame");
- if (includeTimes) Table_setColumnLabel (thee.get(), ++ icol, U"time(s)");
- if (includeIntensity) Table_setColumnLabel (thee.get(), ++ icol, U"intensity");
- if (includeNumberOfFormants) Table_setColumnLabel (thee.get(), ++ icol, U"nformants");
- for (integer iformant = 1; iformant <= my maxnFormants; iformant ++) {
- Table_setColumnLabel (thee.get(), ++ icol, Melder_cat (U"F", iformant, U"(Hz)"));
- if (includeBandwidths) { Table_setColumnLabel (thee.get(), ++ icol, Melder_cat (U"B", iformant, U"(Hz)")); }
- }
- for (integer iframe = 1; iframe <= my nx; iframe ++) {
- icol = 0;
- if (includeFrameNumbers)
- Table_setNumericValue (thee.get(), iframe, ++ icol, iframe);
- if (includeTimes)
- Table_setStringValue (thee.get(), iframe, ++ icol, Melder_fixed (my x1 + (iframe - 1) * my dx, timeDecimals));
- Formant_Frame frame = & my d_frames [iframe];
- if (includeIntensity)
- Table_setStringValue (thee.get(), iframe, ++ icol, Melder_fixed (frame -> intensity, intensityDecimals));
- if (includeNumberOfFormants)
- Table_setNumericValue (thee.get(), iframe, ++ icol, frame -> nFormants);
- for (integer iformant = 1; iformant <= frame -> nFormants; iformant ++) {
- Formant_Formant formant = & frame -> formant [iformant];
- Table_setStringValue (thee.get(), iframe, ++ icol, Melder_fixed (formant -> frequency, frequencyDecimals));
- if (includeBandwidths)
- Table_setStringValue (thee.get(), iframe, ++ icol, Melder_fixed (formant -> bandwidth, frequencyDecimals));
- }
- for (integer iformant = frame -> nFormants + 1; iformant <= my maxnFormants; iformant ++) {
- Table_setNumericValue (thee.get(), iframe, ++ icol, undefined);
- if (includeBandwidths)
- Table_setNumericValue (thee.get(), iframe, ++ icol, undefined);
- }
- }
- return thee;
- } catch (MelderError) {
- Melder_throw (me, U": not converted to Table.");
- }
- }
- void Formant_list (Formant me, bool includeFrameNumbers,
- bool includeTimes, integer timeDecimals,
- bool includeIntensity, integer intensityDecimals,
- bool includeNumberOfFormants, integer frequencyDecimals,
- bool includeBandwidths)
- {
- try {
- autoTable table = Formant_downto_Table (me, includeFrameNumbers, includeTimes, timeDecimals,
- includeIntensity, intensityDecimals,
- includeNumberOfFormants, frequencyDecimals, includeBandwidths);
- Table_list (table.get(), false);
- } catch (MelderError) {
- Melder_throw (me, U": not listed.");
- }
- }
- /* End of file Formant.cpp */
|