make reported latency translatable
[ardour.git] / gtk2_ardour / latency_gui.cc
1 /*
2     Copyright (C) 2009 Paul Davis
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 #include <inttypes.h>
21
22 #include <iomanip>
23 #include "ardour/latent.h"
24 #include "pbd/convert.h"
25 #include "pbd/error.h"
26 #include <gtkmm2ext/utils.h>
27
28 #include "latency_gui.h"
29
30 #include "i18n.h"
31
32 using namespace PBD;
33 using namespace Gtk;
34 using namespace Gtkmm2ext;
35 using namespace ARDOUR;
36
37
38 static const gchar *_unit_strings[] = {
39         N_("sample"),
40         N_("msec"),
41         N_("period"),
42         0
43 };
44
45 std::vector<std::string> LatencyGUI::unit_strings;
46
47 std::string
48 LatencyBarController::get_label (double&)
49 {
50         double const nframes = _latency_gui->adjustment.get_value();
51         std::stringstream s;
52
53         if (nframes < (_latency_gui->sample_rate / 1000.0)) {
54                 const framepos_t nf = (framepos_t) rint (nframes);
55                 s << string_compose (P_("%1 sample", "%1 samples", nf), nf);
56         } else {
57                 s << std::fixed << std::setprecision (2) << (nframes / (_latency_gui->sample_rate / 1000.0)) << " ms";
58         }
59
60         return s.str ();
61 }
62
63 LatencyGUI::LatencyGUI (Latent& l, framepos_t sr, framepos_t psz)
64         : _latent (l),
65           initial_value (_latent.user_latency()),
66           sample_rate (sr),
67           period_size (psz),
68           ignored (new PBD::IgnorableControllable()),
69           /* max 1 second, step by frames, page by msecs */
70           adjustment (initial_value, 0.0, sample_rate, 1.0, sample_rate / 1000.0f),
71           bc (adjustment, this),
72           reset_button (_("Reset"))
73 {
74         Widget* w;
75
76         if (unit_strings.empty()) {
77                 unit_strings = I18N (_unit_strings);
78         }
79
80         set_popdown_strings (units_combo, unit_strings);
81         units_combo.set_active_text (unit_strings.front());
82
83         w = manage (new Image (Stock::ADD, ICON_SIZE_BUTTON));
84         w->show ();
85         plus_button.add (*w);
86         w = manage (new Image (Stock::REMOVE, ICON_SIZE_BUTTON));
87         w->show ();
88         minus_button.add (*w);
89
90         hbox1.pack_start (bc, true, true);
91
92         hbox2.set_homogeneous (false);
93         hbox2.set_spacing (12);
94         hbox2.pack_start (reset_button);
95         hbox2.pack_start (minus_button);
96         hbox2.pack_start (plus_button);
97         hbox2.pack_start (units_combo, true, true);
98
99         minus_button.signal_clicked().connect (sigc::bind (sigc::mem_fun (*this, &LatencyGUI::change_latency_from_button), -1));
100         plus_button.signal_clicked().connect (sigc::bind (sigc::mem_fun (*this, &LatencyGUI::change_latency_from_button), 1));
101         reset_button.signal_clicked().connect (sigc::mem_fun (*this, &LatencyGUI::reset));
102
103         adjustment.signal_value_changed().connect (sigc::mem_fun (*this, &LatencyGUI::finish));
104
105         bc.set_size_request (-1, 25);
106         bc.set_style (BarController::LeftToRight);
107         bc.set_use_parent (true);
108         bc.set_name (X_("PluginSlider"));
109
110         set_spacing (12);
111         pack_start (hbox1, true, true);
112         pack_start (hbox2, true, true);
113 }
114
115 void
116 LatencyGUI::finish ()
117 {
118         framepos_t new_value = (framepos_t) adjustment.get_value();
119         if (new_value != initial_value) {
120                 _latent.set_user_latency (new_value);
121         }
122 }
123
124 void
125 LatencyGUI::reset ()
126 {
127         _latent.set_user_latency (0);
128         adjustment.set_value (initial_value);
129 }
130
131 void
132 LatencyGUI::refresh ()
133 {
134         initial_value = _latent.signal_latency();
135         adjustment.set_value (initial_value);
136 }
137
138 void
139 LatencyGUI::change_latency_from_button (int dir)
140 {
141         std::string unitstr = units_combo.get_active_text();
142         double shift = 0.0;
143
144         if (unitstr == unit_strings[0]) {
145                 shift = 1;
146         } else if (unitstr == unit_strings[1]) {
147                 shift = (sample_rate / 1000.0);
148         } else if (unitstr == unit_strings[2]) {
149                 shift = period_size;
150         } else {
151                 fatal << string_compose (_("programming error: %1 (%2)"), X_("illegal string in latency GUI units combo"), unitstr)
152                       << endmsg;
153                 /*NOTREACHED*/
154         }
155
156         if (dir > 0) {
157                 adjustment.set_value (adjustment.get_value() + shift);
158         } else {
159                 adjustment.set_value (adjustment.get_value() - shift);
160         }
161 }
162
163 LatencyDialog::LatencyDialog (const std::string& title, Latent& l, framepos_t sr, framepos_t psz)
164         : ArdourDialog (title, false, true),
165           lwidget (l, sr, psz)
166 {
167         get_vbox()->pack_start (lwidget);
168         add_button (Stock::CLOSE, RESPONSE_CLOSE);
169
170         show_all ();
171         run ();
172 }
173
174