| 
									
										
										
										
											2017-07-14 15:33:17 +00:00
										 |  |  | #include "EqualizationToolsDialog.hpp"
 | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <iterator>
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <fstream>
 | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  | #include <limits>
 | 
					
						
							|  |  |  | #include <cmath>
 | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <QDir>
 | 
					
						
							|  |  |  | #include <QVector>
 | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  | #include <QHBoxLayout>
 | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | #include <QDialog>
 | 
					
						
							|  |  |  | #include <QDialogButtonBox>
 | 
					
						
							|  |  |  | #include <QPushButton>
 | 
					
						
							|  |  |  | #include <QFileDialog>
 | 
					
						
							|  |  |  | #include <QSettings>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "SettingsGroup.hpp"
 | 
					
						
							|  |  |  | #include "qcustomplot.h"
 | 
					
						
							|  |  |  | #include "pimpl_impl.hpp"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |   float constexpr PI = 3.1415927f; | 
					
						
							| 
									
										
										
										
											2017-07-14 15:33:17 +00:00
										 |  |  |   char const * const title = "Equalization Tools"; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   size_t constexpr intervals = 144; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // plot data loaders - wraps a plot providing value_type and
 | 
					
						
							|  |  |  |   // push_back so that a std::back_inserter output iterator can be
 | 
					
						
							|  |  |  |   // used to load plot data
 | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |   template<typename T, typename A> | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   struct plot_data_loader | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |   public: | 
					
						
							|  |  |  |     typedef T value_type; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |     // the adjust argument is a function that is passed the plot
 | 
					
						
							|  |  |  |     // pointer, the graph index and a data point, it returns a
 | 
					
						
							|  |  |  |     // possibly adjusted data point and can modify the graph including
 | 
					
						
							|  |  |  |     // adding extra points or gaps (quiet_NaN)
 | 
					
						
							|  |  |  |     plot_data_loader (QCustomPlot * plot, int graph_index, A adjust) | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |       : plot_ {plot} | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |       , index_ {graph_index} | 
					
						
							|  |  |  |       , adjust_ (adjust) | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // load point into graph
 | 
					
						
							|  |  |  |     void push_back (value_type const& d) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |       plot_->graph (index_)->data ()->add (adjust_ (plot_, index_, d)); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   private: | 
					
						
							|  |  |  |     QCustomPlot * plot_; | 
					
						
							|  |  |  |     int index_; | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |     A adjust_; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |   // helper function template to make a plot_data_loader instance
 | 
					
						
							|  |  |  |   template<typename A> | 
					
						
							|  |  |  |   auto make_plot_data_loader (QCustomPlot * plot, int index, A adjust) | 
					
						
							|  |  |  |     -> plot_data_loader<QCPGraphData, decltype (adjust)> | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return plot_data_loader<QCPGraphData, decltype (adjust)> {plot, index, adjust}; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   // identity adjust function when no adjustment is needed with the
 | 
					
						
							|  |  |  |   // above instantiation helper
 | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |   QCPGraphData adjust_identity (QCustomPlot *, int, QCPGraphData const& v) {return v;} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // a plot_data_loader adjustment function that wraps Y values of
 | 
					
						
							|  |  |  |   // (-1..+1) plotting discontinuities as gaps in the graph data
 | 
					
						
							|  |  |  |   auto wrap_pi = [] (QCustomPlot * plot, int index, QCPGraphData d)  | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     double constexpr limit {1}; | 
					
						
							|  |  |  |     static unsigned wrap_count {0}; | 
					
						
							|  |  |  |     static double last_x {std::numeric_limits<double>::lowest ()}; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |     d.value += 2 * limit * wrap_count; | 
					
						
							|  |  |  |     if (d.value > limit) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         // insert a gap in the graph
 | 
					
						
							|  |  |  |         plot->graph (index)->data ()->add ({last_x + (d.key - last_x) / 2 | 
					
						
							|  |  |  |               , std::numeric_limits<double>::quiet_NaN ()}); | 
					
						
							|  |  |  |         while (d.value > limit) | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             --wrap_count; | 
					
						
							|  |  |  |             d.value -= 2 * limit; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     else if (d.value < -limit) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         // insert a gap into the graph
 | 
					
						
							|  |  |  |         plot->graph (index)->data ()->add ({last_x + (d.key - last_x) / 2 | 
					
						
							|  |  |  |               , std::numeric_limits<double>::quiet_NaN ()}); | 
					
						
							|  |  |  |         while (d.value < -limit) | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             ++wrap_count; | 
					
						
							|  |  |  |             d.value += 2 * limit; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     last_x = d.key; | 
					
						
							|  |  |  |     return d; | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |   // generate points of type R from a function of type F for X in
 | 
					
						
							|  |  |  |   // (-1..+1) with N intervals and function of type SX to scale X and
 | 
					
						
							|  |  |  |   // of type SY to scale Y
 | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   //
 | 
					
						
							|  |  |  |   // it is up to the user to call the generator sufficient times which
 | 
					
						
							|  |  |  |   // is interval+1 times to reach +1
 | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |   template<typename R, typename F, typename SX, typename SY> | 
					
						
							|  |  |  |   struct graph_generator | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   { | 
					
						
							|  |  |  |   public: | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |     graph_generator (F f, size_t intervals, SX x_scaling, SY y_scaling) | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |       : x_ {0} | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |       , f_ (f) | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |       , intervals_ {intervals} | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |       , x_scaling_ (x_scaling) | 
					
						
							|  |  |  |       , y_scaling_ (y_scaling) | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     R operator () () | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |       typename F::value_type x {x_++ * 2.f / intervals_ - 1.f}; | 
					
						
							|  |  |  |       return {x_scaling_ (x), y_scaling_ (f_ (x))}; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   private: | 
					
						
							|  |  |  |     int x_; | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |     F f_; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |     size_t intervals_; | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |     SX x_scaling_; | 
					
						
							|  |  |  |     SY y_scaling_; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |   // helper function template to make a graph_generator instance for
 | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   // QCPGraphData type points with intervals intervals
 | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |   template<typename F, typename SX, typename SY> | 
					
						
							|  |  |  |   auto make_graph_generator (F function, SX x_scaling, SY y_scaling) | 
					
						
							|  |  |  |     -> graph_generator<QCPGraphData, F, decltype (x_scaling), decltype (y_scaling)> | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return graph_generator<QCPGraphData, F, decltype (x_scaling), decltype (y_scaling)> | 
					
						
							|  |  |  |       {function, intervals, x_scaling, y_scaling}; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // template function object for a polynomial with coefficients
 | 
					
						
							|  |  |  |   template<typename C> | 
					
						
							|  |  |  |   class polynomial | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |   public: | 
					
						
							|  |  |  |     typedef typename C::value_type value_type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     explicit polynomial (C const& coefficients) | 
					
						
							|  |  |  |       : c_ {coefficients} | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     value_type operator () (value_type const& x) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       value_type y {}; | 
					
						
							|  |  |  |       for (typename C::size_type i = c_.size (); i > 0; --i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           y = c_[i - 1] + x * y; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       return y; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   private: | 
					
						
							|  |  |  |     C c_; | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  |   // helper function template to instantiate a polynomial instance
 | 
					
						
							|  |  |  |   template<typename C> | 
					
						
							|  |  |  |   auto make_polynomial (C const& coefficients) -> polynomial<C> | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |     return polynomial<C> (coefficients); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |   // template function object for a group delay with coefficients
 | 
					
						
							|  |  |  |   template<typename C> | 
					
						
							|  |  |  |   class group_delay | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |   public: | 
					
						
							|  |  |  |     typedef typename C::value_type value_type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     explicit group_delay (C const& coefficients) | 
					
						
							|  |  |  |       : c_ {coefficients} | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     value_type operator () (value_type const& x) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       value_type tau {}; | 
					
						
							|  |  |  |       for (typename C::size_type i = 2; i < c_.size (); ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           tau += i * c_[i] * std::pow (x, i - 1); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       return -1 / (2 * PI) * tau; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   private: | 
					
						
							|  |  |  |     C c_; | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  |   // helper function template to instantiate a group_delay function
 | 
					
						
							|  |  |  |   // object
 | 
					
						
							|  |  |  |   template<typename C> | 
					
						
							|  |  |  |   auto make_group_delay (C const& coefficients) -> group_delay<C> | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return group_delay<C> (coefficients); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // handy identity function
 | 
					
						
							|  |  |  |   template<typename T> T identity (T const& v) {return v;} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // a lambda that scales the X axis from normalized to (500..2500)Hz
 | 
					
						
							|  |  |  |   auto freq_scaling = [] (float v) -> float {return 1500.f + 1000.f * v;}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // a lambda that scales the phase Y axis from radians to units of Pi
 | 
					
						
							|  |  |  |   auto pi_scaling = [] (float v) -> float {return v / PI;}; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-14 15:33:17 +00:00
										 |  |  | class EqualizationToolsDialog::impl final | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   : public QDialog | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Q_OBJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2017-07-14 15:33:17 +00:00
										 |  |  |   explicit impl (EqualizationToolsDialog * self, QSettings * settings | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |                  , QDir const& data_directory, QVector<double> const& coefficients | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |                  , QWidget * parent); | 
					
						
							|  |  |  |   ~impl () {save_window_state ();} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  |   void closeEvent (QCloseEvent * e) override | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     save_window_state (); | 
					
						
							|  |  |  |     QDialog::closeEvent (e); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |   void save_window_state () | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     SettingsGroup g (settings_, title); | 
					
						
							|  |  |  |     settings_->setValue ("geometry", saveGeometry ()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   void plot_current (); | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   void plot_phase (); | 
					
						
							|  |  |  |   void plot_amplitude (); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-14 15:33:17 +00:00
										 |  |  |   EqualizationToolsDialog * self_; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   QSettings * settings_; | 
					
						
							|  |  |  |   QDir data_directory_; | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   QHBoxLayout layout_; | 
					
						
							|  |  |  |   QVector<double> current_coefficients_; | 
					
						
							|  |  |  |   QVector<double> new_coefficients_; | 
					
						
							|  |  |  |   unsigned amp_poly_low_; | 
					
						
							|  |  |  |   unsigned amp_poly_high_; | 
					
						
							|  |  |  |   QVector<double> amp_coefficients_; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   QCustomPlot plot_; | 
					
						
							|  |  |  |   QDialogButtonBox button_box_; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-14 15:33:17 +00:00
										 |  |  | #include "EqualizationToolsDialog.moc"
 | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-14 15:33:17 +00:00
										 |  |  | EqualizationToolsDialog::EqualizationToolsDialog (QSettings * settings | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |                                                   , QDir const& data_directory | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |                                                   , QVector<double> const& coefficients | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |                                                   , QWidget * parent) | 
					
						
							|  |  |  |   : m_ {this, settings, data_directory, coefficients, parent} | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-14 15:33:17 +00:00
										 |  |  | void EqualizationToolsDialog::show () | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   m_->show (); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-14 15:33:17 +00:00
										 |  |  | EqualizationToolsDialog::impl::impl (EqualizationToolsDialog * self | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |                                      , QSettings * settings | 
					
						
							|  |  |  |                                      , QDir const& data_directory | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |                                      , QVector<double> const& coefficients | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |                                      , QWidget * parent) | 
					
						
							|  |  |  |   : QDialog {parent} | 
					
						
							|  |  |  |   , self_ {self} | 
					
						
							|  |  |  |   , settings_ {settings} | 
					
						
							|  |  |  |   , data_directory_ {data_directory} | 
					
						
							|  |  |  |   , current_coefficients_ {coefficients} | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   , amp_poly_low_ {0} | 
					
						
							|  |  |  |   , amp_poly_high_ {6000} | 
					
						
							| 
									
										
										
										
											2018-06-08 17:39:57 -05:00
										 |  |  |   , button_box_ {QDialogButtonBox::Apply | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |         | QDialogButtonBox::RestoreDefaults | QDialogButtonBox::Close | 
					
						
							|  |  |  |         , Qt::Vertical} | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-13 00:55:42 +01:00
										 |  |  |   setWindowTitle (windowTitle () + ' ' + tr ("Equalization Tools")); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   resize (500, 600); | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     SettingsGroup g {settings_, title}; | 
					
						
							|  |  |  |     restoreGeometry (settings_->value ("geometry", saveGeometry ()).toByteArray ()); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   auto legend_title = new QCPTextElement {&plot_, tr ("Phase"), QFont {"sans", 9, QFont::Bold}}; | 
					
						
							|  |  |  |   legend_title->setLayer (plot_.legend->layer ()); | 
					
						
							|  |  |  |   plot_.legend->addElement (0, 0, legend_title); | 
					
						
							|  |  |  |   plot_.legend->setVisible (true); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   plot_.xAxis->setLabel (tr ("Freq (Hz)")); | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |   plot_.xAxis->setRange (500, 2500); | 
					
						
							|  |  |  |   plot_.yAxis->setLabel (tr ("Phase (Π)")); | 
					
						
							|  |  |  |   plot_.yAxis->setRange (-1, +1); | 
					
						
							|  |  |  |   plot_.yAxis2->setLabel (tr ("Delay (ms)")); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   plot_.axisRect ()->setRangeDrag (Qt::Vertical); | 
					
						
							|  |  |  |   plot_.axisRect ()->setRangeZoom (Qt::Vertical); | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |   plot_.yAxis2->setVisible (true); | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   plot_.axisRect ()->setRangeDragAxes (nullptr, plot_.yAxis2); | 
					
						
							|  |  |  |   plot_.axisRect ()->setRangeZoomAxes (nullptr, plot_.yAxis2); | 
					
						
							| 
									
										
										
										
											2017-02-22 02:20:11 +00:00
										 |  |  |   plot_.axisRect ()->insetLayout ()->setInsetAlignment (0, Qt::AlignBottom|Qt::AlignRight); | 
					
						
							|  |  |  |   plot_.setInteractions (QCP::iRangeDrag | QCP::iRangeZoom | QCP::iSelectPlottables); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   plot_.addGraph ()->setName (tr ("Measured")); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   plot_.graph ()->setPen (QPen {Qt::blue}); | 
					
						
							|  |  |  |   plot_.graph ()->setVisible (false); | 
					
						
							|  |  |  |   plot_.graph ()->removeFromLegend (); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   plot_.addGraph ()->setName (tr ("Proposed")); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   plot_.graph ()->setPen (QPen {Qt::red}); | 
					
						
							|  |  |  |   plot_.graph ()->setVisible (false); | 
					
						
							|  |  |  |   plot_.graph ()->removeFromLegend (); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   plot_.addGraph ()->setName (tr ("Current")); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   plot_.graph ()->setPen (QPen {Qt::green}); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   plot_.addGraph (plot_.xAxis, plot_.yAxis2)->setName (tr ("Group Delay")); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   plot_.graph ()->setPen (QPen {Qt::darkGreen}); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   plot_.plotLayout ()->addElement (new QCPAxisRect {&plot_}); | 
					
						
							|  |  |  |   plot_.plotLayout ()->setRowStretchFactor (1, 0.5); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   auto amp_legend = new QCPLegend; | 
					
						
							|  |  |  |   plot_.axisRect (1)->insetLayout ()->addElement (amp_legend, Qt::AlignTop | Qt::AlignRight); | 
					
						
							|  |  |  |   plot_.axisRect (1)->insetLayout ()->setMargins (QMargins {12, 12, 12, 12}); | 
					
						
							|  |  |  |   amp_legend->setVisible (true); | 
					
						
							|  |  |  |   amp_legend->setLayer (QLatin1String {"legend"}); | 
					
						
							|  |  |  |   legend_title = new QCPTextElement {&plot_, tr ("Amplitude"), QFont {"sans", 9, QFont::Bold}}; | 
					
						
							|  |  |  |   legend_title->setLayer (amp_legend->layer ()); | 
					
						
							|  |  |  |   amp_legend->addElement (0, 0, legend_title); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   plot_.axisRect (1)->axis (QCPAxis::atBottom)->setLabel (tr ("Freq (Hz)")); | 
					
						
							|  |  |  |   plot_.axisRect (1)->axis (QCPAxis::atBottom)->setRange (0, 6000); | 
					
						
							|  |  |  |   plot_.axisRect (1)->axis (QCPAxis::atLeft)->setLabel (tr ("Relative Power (dB)")); | 
					
						
							|  |  |  |   plot_.axisRect (1)->axis (QCPAxis::atLeft)->setRangeLower (0); | 
					
						
							|  |  |  |   plot_.axisRect (1)->setRangeDragAxes (nullptr, nullptr); | 
					
						
							|  |  |  |   plot_.axisRect (1)->setRangeZoomAxes (nullptr, nullptr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   plot_.addGraph (plot_.axisRect (1)->axis (QCPAxis::atBottom) | 
					
						
							|  |  |  |                   , plot_.axisRect (1)->axis (QCPAxis::atLeft))->setName (tr ("Reference")); | 
					
						
							|  |  |  |   plot_.graph ()->setPen (QPen {Qt::blue}); | 
					
						
							|  |  |  |   plot_.graph ()->removeFromLegend (); | 
					
						
							|  |  |  |   plot_.graph ()->addToLegend (amp_legend); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   layout_.addWidget (&plot_); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   auto load_phase_button = button_box_.addButton (tr ("Phase ..."), QDialogButtonBox::ActionRole); | 
					
						
							|  |  |  |   auto refresh_button = button_box_.addButton (tr ("Refresh"), QDialogButtonBox::ActionRole); | 
					
						
							| 
									
										
										
										
											2018-06-08 17:39:57 -05:00
										 |  |  |   auto discard_measured_button = button_box_.addButton (tr ("Discard Measured"), QDialogButtonBox::ActionRole); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   layout_.addWidget (&button_box_); | 
					
						
							|  |  |  |   setLayout (&layout_); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   connect (&button_box_, &QDialogButtonBox::rejected, this, &QDialog::reject); | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   connect (&button_box_, &QDialogButtonBox::clicked, [=] (QAbstractButton * button) { | 
					
						
							|  |  |  |       if (button == load_phase_button) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           plot_phase (); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       else if (button == refresh_button) | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |           plot_current (); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |       else if (button == button_box_.button (QDialogButtonBox::Apply)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           if (plot_.graph (0)->dataCount ()) // something loaded
 | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |               current_coefficients_ = new_coefficients_; | 
					
						
							|  |  |  |               Q_EMIT self_->phase_equalization_changed (current_coefficients_); | 
					
						
							|  |  |  |               plot_current (); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       else if (button == button_box_.button (QDialogButtonBox::RestoreDefaults)) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |           current_coefficients_ = QVector<double> {0., 0., 0., 0., 0.}; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |           Q_EMIT self_->phase_equalization_changed (current_coefficients_); | 
					
						
							|  |  |  |           plot_current (); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-06-08 17:39:57 -05:00
										 |  |  |       else if (button == discard_measured_button) | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |           new_coefficients_ = QVector<double> {0., 0., 0., 0., 0.}; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |           plot_.graph (0)->data ()->clear (); | 
					
						
							|  |  |  |           plot_.graph (0)->setVisible (false); | 
					
						
							|  |  |  |           plot_.graph (0)->removeFromLegend (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           plot_.graph (1)->data ()->clear (); | 
					
						
							|  |  |  |           plot_.graph (1)->setVisible (false); | 
					
						
							|  |  |  |           plot_.graph (1)->removeFromLegend (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           plot_.replot (); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   plot_current (); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  | struct PowerSpectrumPoint | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   operator QCPGraphData () const | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return QCPGraphData {freq_, power_}; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   float freq_; | 
					
						
							|  |  |  |   float power_; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // read an amplitude point line from a stream (refspec.dat)
 | 
					
						
							|  |  |  | std::istream& operator >> (std::istream& is, PowerSpectrumPoint& r) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   float y1, y3, y4;             // discard these
 | 
					
						
							|  |  |  |   is >> r.freq_ >> y1 >> r.power_ >> y3 >> y4; | 
					
						
							|  |  |  |   return is; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-14 15:33:17 +00:00
										 |  |  | void EqualizationToolsDialog::impl::plot_current () | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   auto phase_graph = make_plot_data_loader (&plot_, 2, wrap_pi); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   plot_.graph (2)->data ()->clear (); | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   std::generate_n (std::back_inserter (phase_graph), intervals + 1 | 
					
						
							|  |  |  |                    , make_graph_generator (make_polynomial (current_coefficients_), freq_scaling, pi_scaling)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   auto group_delay_graph = make_plot_data_loader (&plot_, 3, adjust_identity); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   plot_.graph (3)->data ()->clear (); | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   std::generate_n (std::back_inserter (group_delay_graph), intervals + 1 | 
					
						
							|  |  |  |                    , make_graph_generator (make_group_delay (current_coefficients_), freq_scaling, identity<double>)); | 
					
						
							|  |  |  |   plot_.graph (3)->rescaleValueAxis (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   QFileInfo refspec_file_info {data_directory_.absoluteFilePath ("refspec.dat")}; | 
					
						
							|  |  |  |   std::ifstream refspec_file (refspec_file_info.absoluteFilePath ().toLatin1 ().constData (), std::ifstream::in); | 
					
						
							|  |  |  |   unsigned n; | 
					
						
							|  |  |  |   if (refspec_file >> amp_poly_low_ >> amp_poly_high_ >> n) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       std::istream_iterator<double> isi {refspec_file}; | 
					
						
							|  |  |  |       amp_coefficients_.clear (); | 
					
						
							|  |  |  |       std::copy_n (isi, n, std::back_inserter (amp_coefficients_)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       // may be old format refspec.dat with no header so rewind
 | 
					
						
							|  |  |  |       refspec_file.clear (); | 
					
						
							|  |  |  |       refspec_file.seekg (0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   auto reference_spectrum_graph = make_plot_data_loader (&plot_, 4, adjust_identity); | 
					
						
							|  |  |  |   plot_.graph (4)->data ()->clear (); | 
					
						
							|  |  |  |   std::copy (std::istream_iterator<PowerSpectrumPoint> {refspec_file}, | 
					
						
							|  |  |  |              std::istream_iterator<PowerSpectrumPoint> {}, | 
					
						
							|  |  |  |              std::back_inserter (reference_spectrum_graph)); | 
					
						
							|  |  |  |   plot_.graph (4)->rescaleValueAxis (true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   plot_.replot (); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct PhasePoint | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   operator QCPGraphData () const | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |     return QCPGraphData {freq_, phase_}; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   double freq_; | 
					
						
							|  |  |  |   double phase_; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // read a phase point line from a stream (pcoeff file)
 | 
					
						
							|  |  |  | std::istream& operator >> (std::istream& is, PhasePoint& c) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   double pp, sigmay;            // discard these
 | 
					
						
							|  |  |  |   if (is >> c.freq_ >> pp >> c.phase_ >> sigmay) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       c.freq_ = 1500. + 1000. * c.freq_; // scale frequency to Hz
 | 
					
						
							|  |  |  |       c.phase_ /= PI;                    // scale to units of Pi
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return is; | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-14 15:33:17 +00:00
										 |  |  | void EqualizationToolsDialog::impl::plot_phase () | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |   auto const& phase_file_name = QFileDialog::getOpenFileName (this | 
					
						
							|  |  |  |                                                               , "Select Phase Response Coefficients" | 
					
						
							|  |  |  |                                                               , data_directory_.absolutePath () | 
					
						
							|  |  |  |                                                               , "Phase Coefficient Files (*.pcoeff)"); | 
					
						
							|  |  |  |   if (!phase_file_name.size ()) return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   std::ifstream phase_file (phase_file_name.toLatin1 ().constData (), std::ifstream::in); | 
					
						
							|  |  |  |   int n; | 
					
						
							|  |  |  |   float chi; | 
					
						
							|  |  |  |   float rmsdiff; | 
					
						
							|  |  |  |   unsigned freq_low; | 
					
						
							|  |  |  |   unsigned freq_high; | 
					
						
							|  |  |  |   unsigned terms; | 
					
						
							|  |  |  |   // read header information
 | 
					
						
							|  |  |  |   if (phase_file >> n >> chi >> rmsdiff >> freq_low >> freq_high >> terms) | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |       std::istream_iterator<double> isi {phase_file}; | 
					
						
							|  |  |  |       new_coefficients_.clear (); | 
					
						
							|  |  |  |       std::copy_n (isi, terms, std::back_inserter (new_coefficients_)); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |       if (phase_file) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           plot_.graph (0)->data ()->clear (); | 
					
						
							|  |  |  |           plot_.graph (1)->data ()->clear (); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |           // read the phase data and plot as graph 0
 | 
					
						
							|  |  |  |           auto graph = make_plot_data_loader (&plot_, 0, adjust_identity); | 
					
						
							|  |  |  |           std::copy_n (std::istream_iterator<PhasePoint> {phase_file}, | 
					
						
							|  |  |  |                        intervals + 1, std::back_inserter (graph)); | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |           if (phase_file) | 
					
						
							|  |  |  |             { | 
					
						
							| 
									
										
										
										
											2017-03-16 15:33:42 +00:00
										 |  |  |               plot_.graph(0)->setLineStyle(QCPGraph::lsNone); | 
					
						
							|  |  |  |               plot_.graph(0)->setScatterStyle(QCPScatterStyle(QCPScatterStyle::ssDisc, 4)); | 
					
						
							| 
									
										
										
										
											2017-02-23 16:21:26 +00:00
										 |  |  |               plot_.graph (0)->setVisible (true); | 
					
						
							|  |  |  |               plot_.graph (0)->addToLegend (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               // generate the proposed polynomial plot as graph 1
 | 
					
						
							|  |  |  |               auto graph = make_plot_data_loader (&plot_, 1, wrap_pi); | 
					
						
							|  |  |  |               std::generate_n (std::back_inserter (graph), intervals + 1 | 
					
						
							|  |  |  |                                , make_graph_generator (make_polynomial (new_coefficients_) | 
					
						
							|  |  |  |                                                        , freq_scaling, pi_scaling)); | 
					
						
							|  |  |  |               plot_.graph (1)->setVisible (true); | 
					
						
							|  |  |  |               plot_.graph (1)->addToLegend (); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           plot_.replot (); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-02-21 02:13:13 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-14 15:33:17 +00:00
										 |  |  | #include "moc_EqualizationToolsDialog.cpp"
 |