123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155 |
- /* PitchTier_to_PointProcess.cpp
- *
- * Copyright (C) 1992-2011,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 "PitchTier_to_PointProcess.h"
- #include "Pitch_to_PitchTier.h"
- autoPointProcess PitchTier_to_PointProcess (PitchTier me) {
- try {
- autoPointProcess thee = PointProcess_create (my xmin, my xmax, 1000);
- double area = 0.5; // imagine an event half a period before the beginning
- integer size = my points.size;
- if (size == 0) return thee;
- for (integer interval = 0; interval <= size; interval ++) {
- double t1 = ( interval == 0 ? my xmin : my points.at [interval] -> number );
- Melder_assert (isdefined (t1));
- double t2 = ( interval == size ? my xmax : my points.at [interval + 1] -> number );
- Melder_assert (isdefined (t2));
- double f1 = my points.at [interval == 0 ? 1 : interval] -> value;
- Melder_assert (isdefined (f1));
- double f2 = my points.at [interval == size ? size : interval + 1] -> value;
- Melder_assert (isdefined (f2));
- area += (t2 - t1) * 0.5 * (f1 + f2);
- while (area >= 1.0) {
- double slope = (f2 - f1) / (t2 - t1), discriminant;
- area -= 1.0;
- discriminant = f2 * f2 - 2.0 * area * slope;
- if (discriminant < 0.0) discriminant = 0.0; // catch rounding errors
- PointProcess_addPoint (thee.get(), t2 - 2.0 * area / (f2 + sqrt (discriminant)));
- }
- }
- return thee;
- } catch (MelderError) {
- Melder_throw (me, U": not converted to PointProcess.");
- }
- }
- autoPointProcess PitchTier_Pitch_to_PointProcess (PitchTier me, Pitch vuv) {
- try {
- autoPointProcess fullPoint = PitchTier_to_PointProcess (me);
- autoPointProcess thee = PointProcess_create (my xmin, my xmax, fullPoint -> maxnt);
- /*
- * Copy only voiced parts to result.
- */
- for (integer i = 1; i <= fullPoint -> nt; i ++) {
- double t = fullPoint -> t [i];
- if (Pitch_isVoiced_t (vuv, t)) {
- PointProcess_addPoint (thee.get(), t);
- }
- }
- return thee;
- } catch (MelderError) {
- Melder_throw (me, U" & ", vuv, U": not converted to PointProcess.");
- }
- }
- static int PointProcess_isVoiced_t (PointProcess me, double t, double maxT) {
- integer imid = PointProcess_getNearestIndex (me, t);
- if (imid == 0) return 0;
- double tmid = my t [imid];
- int leftVoiced = imid > 1 && tmid - my t [imid - 1] <= maxT;
- int rightVoiced = imid < my nt && my t [imid + 1] - tmid <= maxT;
- if ((leftVoiced && t <= tmid) || (rightVoiced && t >= tmid)) return 1;
- if (leftVoiced && t < 1.5 * tmid - 0.5 * my t [imid - 1]) return 1;
- if (rightVoiced && t > 1.5 * tmid - 0.5 * my t [imid + 1]) return 1;
- return 0;
- }
- autoPointProcess PitchTier_Point_to_PointProcess (PitchTier me, PointProcess vuv, double maxT) {
- try {
- autoPointProcess fullPoint = PitchTier_to_PointProcess (me);
- autoPointProcess thee = PointProcess_create (my xmin, my xmax, fullPoint -> maxnt);
- /*
- * Copy only voiced parts to result.
- */
- for (integer i = 1; i <= fullPoint -> nt; i ++) {
- double t = fullPoint -> t [i];
- if (PointProcess_isVoiced_t (vuv, t, maxT)) {
- PointProcess_addPoint (thee.get(), t);
- }
- }
- return thee;
- } catch (MelderError) {
- Melder_throw (me, U" & ", vuv, U": not converted to PointProcess.");
- }
- }
- autoPitchTier PointProcess_to_PitchTier (PointProcess me, double maximumInterval) {
- try {
- autoPitchTier thee = PitchTier_create (my xmin, my xmax);
- for (integer i = 1; i < my nt; i ++) {
- double interval = my t [i + 1] - my t [i];
- if (interval <= maximumInterval) {
- RealTier_addPoint (thee.get(), my t [i] + 0.5 * interval, 1.0 / interval);
- }
- }
- return thee;
- } catch (MelderError) {
- Melder_throw (me, U": not converted to PitchTier.");
- }
- }
- autoPitchTier Pitch_PointProcess_to_PitchTier (Pitch me, PointProcess pp) {
- try {
- autoPitchTier temp = Pitch_to_PitchTier (me);
- autoPitchTier thee = PitchTier_PointProcess_to_PitchTier (temp.get(), pp);
- return thee;
- } catch (MelderError) {
- Melder_throw (me, U" & ", pp, U": not converted to PitchTier.");
- }
- }
- autoPitchTier PitchTier_PointProcess_to_PitchTier (PitchTier me, PointProcess pp) {
- try {
- if (my points.size == 0) Melder_throw (U"No pitch points.");
- autoPitchTier thee = PitchTier_create (pp -> xmin, pp -> xmax);
- for (integer i = 1; i <= pp -> nt; i ++) {
- double time = pp -> t [i];
- double value = RealTier_getValueAtTime (me, time);
- RealTier_addPoint (thee.get(), time, value);
- }
- return thee;
- } catch (MelderError) {
- Melder_throw (me, U" & ", pp, U": not converted to PitchTier.");
- }
- }
- autoTableOfReal PitchTier_downto_TableOfReal (PitchTier me, int useSemitones) {
- try {
- autoTableOfReal thee = RealTier_downto_TableOfReal (me, U"Time", U"F0");
- if (useSemitones)
- for (integer i = 1; i <= thy numberOfRows; i ++)
- thy data [i] [2] = NUMhertzToSemitones (thy data [i] [2]);
- return thee;
- } catch (MelderError) {
- Melder_throw (me, U": not converted to TableOfReal.");
- }
- }
- /* End of file PitchTier_to_PointProcess.cpp */
|