always take verbose cursor clock mode from primary clock (finally!)
[ardour.git] / gtk2_ardour / verbose_cursor.cc
1 /*
2     Copyright (C) 2000-2011 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 <string>
21 #include <gtkmm/enums.h>
22 #include "pbd/stacktrace.h"
23 #include "ardour/profile.h"
24
25 #include "canvas/debug.h"
26 #include "canvas/scroll_group.h"
27 #include "canvas/tracking_text.h"
28
29 #include "ardour_ui.h"
30 #include "audio_clock.h"
31 #include "editor.h"
32 #include "editor_drag.h"
33 #include "global_signals.h"
34 #include "main_clock.h"
35 #include "verbose_cursor.h"
36
37 #include "i18n.h"
38
39 using namespace std;
40 using namespace ARDOUR;
41
42 VerboseCursor::VerboseCursor (Editor* editor)
43         : _editor (editor)
44 {
45         _canvas_item = new ArdourCanvas::TrackingText (_editor->get_noscroll_group());
46         CANVAS_DEBUG_NAME (_canvas_item, "verbose canvas cursor");
47         _canvas_item->set_font_description (Pango::FontDescription (ARDOUR_UI::config()->get_canvasvar_LargerBoldFont()));
48         color_handler ();
49
50         ARDOUR_UI_UTILS::ColorsChanged.connect (sigc::mem_fun (*this, &VerboseCursor::color_handler));
51 }
52
53 void
54 VerboseCursor::color_handler ()
55 {
56         _canvas_item->set_color (ARDOUR_UI::config()->get_canvasvar_VerboseCanvasCursor());
57 }
58
59 ArdourCanvas::Item *
60 VerboseCursor::canvas_item () const
61 {
62         return _canvas_item;
63 }
64
65 /** Set the contents of the cursor.
66  */
67 void
68 VerboseCursor::set (string const & text)
69 {
70         _canvas_item->set (text);
71 }
72
73 void
74 VerboseCursor::show ()
75 {
76         _canvas_item->show_and_track (true, true);
77         _canvas_item->parent()->raise_to_top ();
78 }
79
80 void
81 VerboseCursor::hide ()
82 {
83         _canvas_item->hide ();
84         _canvas_item->parent()->lower_to_bottom ();
85         /* reset back to a sensible default for the next time we display the VC */
86         _canvas_item->set_offset (ArdourCanvas::Duple (10, 10));
87 }
88
89 void
90 VerboseCursor::set_offset (ArdourCanvas::Duple const & d)
91 {
92         _canvas_item->set_offset (d);
93 }
94
95 void
96 VerboseCursor::set_time (framepos_t frame)
97 {
98         char buf[128];
99         Timecode::Time timecode;
100         Timecode::BBT_Time bbt;
101         int hours, mins;
102         framepos_t frame_rate;
103         float secs;
104
105         if (_editor->_session == 0) {
106                 return;
107         }
108
109         /* Take clock mode from the primary clock */
110
111         AudioClock::Mode m = ARDOUR_UI::instance()->primary_clock->mode();
112
113         switch (m) {
114         case AudioClock::BBT:
115                 _editor->_session->bbt_time (frame, bbt);
116                 snprintf (buf, sizeof (buf), "%02" PRIu32 "|%02" PRIu32 "|%02" PRIu32, bbt.bars, bbt.beats, bbt.ticks);
117                 break;
118
119         case AudioClock::Timecode:
120                 _editor->_session->timecode_time (frame, timecode);
121                 snprintf (buf, sizeof (buf), "%02" PRId32 ":%02" PRId32 ":%02" PRId32 ":%02" PRId32, timecode.hours, timecode.minutes, timecode.seconds, timecode.frames);
122                 break;
123
124         case AudioClock::MinSec:
125                 /* XXX this is copied from show_verbose_duration_cursor() */
126                 frame_rate = _editor->_session->frame_rate();
127                 hours = frame / (frame_rate * 3600);
128                 frame = frame % (frame_rate * 3600);
129                 mins = frame / (frame_rate * 60);
130                 frame = frame % (frame_rate * 60);
131                 secs = (float) frame / (float) frame_rate;
132                 snprintf (buf, sizeof (buf), "%02" PRId32 ":%02" PRId32 ":%07.4f", hours, mins, secs);
133                 break;
134
135         default:
136                 snprintf (buf, sizeof(buf), "%" PRIi64, frame);
137                 break;
138         }
139
140         _canvas_item->set (buf);
141 }
142
143 void
144 VerboseCursor::set_duration (framepos_t start, framepos_t end)
145 {
146         char buf[128];
147         Timecode::Time timecode;
148         Timecode::BBT_Time sbbt;
149         Timecode::BBT_Time ebbt;
150         int hours, mins;
151         framepos_t distance, frame_rate;
152         float secs;
153         Meter meter_at_start (_editor->_session->tempo_map().meter_at(start));
154
155         if (_editor->_session == 0) {
156                 return;
157         }
158
159         AudioClock::Mode m;
160
161         if (Profile->get_sae() || Profile->get_small_screen() || Profile->get_trx()) {
162                 m = ARDOUR_UI::instance()->primary_clock->mode ();
163         } else {
164                 m = ARDOUR_UI::instance()->secondary_clock->mode ();
165         }
166
167         switch (m) {
168         case AudioClock::BBT:
169         {
170                 _editor->_session->bbt_time (start, sbbt);
171                 _editor->_session->bbt_time (end, ebbt);
172
173                 /* subtract */
174                 /* XXX this computation won't work well if the
175                 user makes a selection that spans any meter changes.
176                 */
177
178                 /* use signed integers for the working values so that
179                    we can underflow.
180                 */
181
182                 int ticks = ebbt.ticks;
183                 int beats = ebbt.beats;
184                 int bars = ebbt.bars;
185
186                 ticks -= sbbt.ticks;
187                 if (ticks < 0) {
188                         ticks += int (Timecode::BBT_Time::ticks_per_beat);
189                         --beats;
190                 }
191
192                 beats -= sbbt.beats;
193                 if (beats < 0) {
194                         beats += int (meter_at_start.divisions_per_bar());
195                         --bars;
196                 }
197
198                 bars -= sbbt.bars;
199
200                 snprintf (buf, sizeof (buf), "%02" PRIu32 "|%02" PRIu32 "|%02" PRIu32, bars, beats, ticks);
201                 break;
202         }
203
204         case AudioClock::Timecode:
205                 _editor->_session->timecode_duration (end - start, timecode);
206                 snprintf (buf, sizeof (buf), "%02" PRId32 ":%02" PRId32 ":%02" PRId32 ":%02" PRId32, timecode.hours, timecode.minutes, timecode.seconds, timecode.frames);
207                 break;
208
209         case AudioClock::MinSec:
210                 /* XXX this stuff should be elsewhere.. */
211                 distance = end - start;
212                 frame_rate = _editor->_session->frame_rate();
213                 hours = distance / (frame_rate * 3600);
214                 distance = distance % (frame_rate * 3600);
215                 mins = distance / (frame_rate * 60);
216                 distance = distance % (frame_rate * 60);
217                 secs = (float) distance / (float) frame_rate;
218                 snprintf (buf, sizeof (buf), "%02" PRId32 ":%02" PRId32 ":%07.4f", hours, mins, secs);
219                 break;
220
221         default:
222                 snprintf (buf, sizeof(buf), "%" PRIi64, end - start);
223                 break;
224         }
225
226         _canvas_item->set (buf);
227 }
228
229 bool
230 VerboseCursor::visible () const
231 {
232         return _canvas_item->visible();
233 }