various fixes to MidiRegionView selection handling, key handling, drawing of ghost...
[ardour2.git] / gtk2_ardour / visual_time_axis.cc
blob10c8513ec8741753c4601e3a4e9f0b15038dd5a5
1 /*
2 Copyright (C) 2003 Paul Davis
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.
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.
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.
20 #include <cstdlib>
21 #include <cmath>
22 #include <algorithm>
23 #include <string>
24 #include <vector>
26 #include "pbd/error.h"
27 #include "pbd/stl_delete.h"
28 #include "pbd/whitespace.h"
30 #include <gtkmm2ext/utils.h>
31 #include <gtkmm2ext/selector.h>
32 #include <gtkmm2ext/gtk_ui.h>
33 #include <gtkmm2ext/choice.h>
35 #include "ardour/session.h"
36 #include "ardour/utils.h"
37 #include "ardour/processor.h"
38 #include "ardour/location.h"
40 #include "ardour_ui.h"
41 #include "public_editor.h"
42 #include "imageframe_time_axis.h"
43 #include "imageframe_time_axis_view.h"
44 #include "marker_time_axis_view.h"
45 #include "imageframe_view.h"
46 #include "marker_time_axis.h"
47 #include "marker_view.h"
48 #include "utils.h"
49 #include "prompter.h"
50 #include "rgb_macros.h"
51 #include "canvas_impl.h"
53 #include "i18n.h"
55 using namespace ARDOUR;
56 using namespace PBD;
57 using namespace Gtk;
59 /**
60 * Abstract Constructor for base visual time axis classes
62 * @param name the name/Id of thie TimeAxis
63 * @param ed the Ardour PublicEditor
64 * @param sess the current session
65 * @param canvas the parent canvas object
67 VisualTimeAxis::VisualTimeAxis(const string & name, PublicEditor& ed, ARDOUR::Session* sess, Canvas& canvas)
68 : AxisView(sess),
69 TimeAxisView(sess,ed,(TimeAxisView*) 0, canvas),
70 visual_button (_("v")),
71 size_button (_("h"))
73 time_axis_name = name ;
74 _color = unique_random_color() ;
76 name_entry.signal_activate().connect(sigc::mem_fun(*this, &VisualTimeAxis::name_entry_changed)) ;
77 name_entry.signal_button_press_event().connect(sigc::mem_fun(*this, &VisualTimeAxis::name_entry_button_press_handler)) ;
78 name_entry.signal_button_release_event().connect(sigc::mem_fun(*this, &VisualTimeAxis::name_entry_button_release_handler)) ;
79 name_entry.signal_key_release_event().connect(sigc::mem_fun(*this, &VisualTimeAxis::name_entry_key_release_handler)) ;
81 size_button.set_name("TrackSizeButton") ;
82 visual_button.set_name("TrackVisualButton") ;
83 hide_button.set_name("TrackRemoveButton") ;
84 hide_button.add(*(Gtk::manage(new Gtk::Image(get_xpm("small_x.xpm")))));
85 size_button.signal_button_release_event().connect (sigc::mem_fun (*this, &VisualTimeAxis::size_click)) ;
86 visual_button.signal_clicked().connect (sigc::mem_fun (*this, &VisualTimeAxis::visual_click)) ;
87 hide_button.signal_clicked().connect (sigc::mem_fun (*this, &VisualTimeAxis::hide_click)) ;
88 ARDOUR_UI::instance()->set_tip(size_button,_("Display Height")) ;
89 ARDOUR_UI::instance()->set_tip(visual_button, _("Visual options")) ;
90 ARDOUR_UI::instance()->set_tip(hide_button, _("Hide this track")) ;
92 controls_table.attach (hide_button, 0, 1, 1, 2, Gtk::FILL|Gtk::EXPAND, Gtk::FILL|Gtk::EXPAND);
93 controls_table.attach (visual_button, 1, 2, 1, 2, Gtk::FILL|Gtk::EXPAND, Gtk::FILL|Gtk::EXPAND);
94 controls_table.attach (size_button, 2, 3, 1, 2, Gtk::FILL|Gtk::EXPAND, Gtk::FILL|Gtk::EXPAND);
96 /* remove focus from the buttons */
97 size_button.unset_flags(Gtk::CAN_FOCUS) ;
98 hide_button.unset_flags(Gtk::CAN_FOCUS) ;
99 visual_button.unset_flags(Gtk::CAN_FOCUS) ;
101 set_height (hNormal) ;
105 * VisualTimeAxis Destructor
108 VisualTimeAxis::~VisualTimeAxis()
113 //---------------------------------------------------------------------------------------//
114 // Name/Id Accessors/Mutators
116 void
117 VisualTimeAxis::set_time_axis_name(const string & name, void* src)
119 std::string old_name = time_axis_name ;
121 if(name != time_axis_name)
123 time_axis_name = name ;
124 label_view() ;
125 editor.route_name_changed(this) ;
127 NameChanged(time_axis_name, old_name, src) ; /* EMIT_SIGNAL */
131 std::string
132 VisualTimeAxis::name() const
134 return(time_axis_name) ;
138 //---------------------------------------------------------------------------------------//
139 // ui methods & data
142 * Sets the height of this TrackView to one of the defined TrackHeghts
144 * @param h
146 void
147 VisualTimeAxis::set_height(uint32_t h)
149 TimeAxisView::set_height(h);
151 if (h >= hNormal) {
152 hide_name_label ();
153 show_name_entry ();
154 other_button_hbox.show_all() ;
155 } else if (h >= hSmaller) {
156 hide_name_label ();
157 show_name_entry ();
158 other_button_hbox.hide_all() ;
159 } else if (h >= hSmall) {
160 hide_name_entry ();
161 show_name_label ();
162 other_button_hbox.hide_all() ;
167 * Handle the visuals button click
170 void
171 VisualTimeAxis::visual_click()
173 popup_display_menu(0);
178 * Handle the hide buttons click
181 void
182 VisualTimeAxis::hide_click()
184 // LAME fix for hide_button display refresh
185 hide_button.set_sensitive(false);
187 editor.hide_track_in_display (*this);
189 hide_button.set_sensitive(true);
194 * Allows the selection of a new color for this TimeAxis
197 void
198 VisualTimeAxis::select_track_color ()
200 if(choose_time_axis_color())
202 //Does nothing at this abstract point
207 * Provides a color chooser for the selection of a new time axis color.
210 bool
211 VisualTimeAxis::choose_time_axis_color()
213 bool picked ;
214 Gdk::Color color ;
215 gdouble current[4] ;
216 Gdk::Color current_color ;
218 current[0] = _color.get_red() / 65535.0 ;
219 current[1] = _color.get_green() / 65535.0 ;
220 current[2] = _color.get_blue() / 65535.0 ;
221 current[3] = 1.0 ;
223 current_color.set_rgb_p (current[0],current[1],current[2]);
224 color = Gtkmm2ext::UI::instance()->get_color(_("Color Selection"),picked, &current_color) ;
226 if (picked)
228 set_time_axis_color(color) ;
230 return(picked) ;
234 * Sets the color of this TimeAxis to the specified color c
236 * @param c the new TimeAxis color
238 void
239 VisualTimeAxis::set_time_axis_color(Gdk::Color c)
241 _color = c ;
244 void
245 VisualTimeAxis::set_selected_regionviews (RegionSelection& regions)
247 // Not handled by purely visual TimeAxis
250 //---------------------------------------------------------------------------------------//
251 // Handle time axis removal
254 * Handles the Removal of this VisualTimeAxis
256 * @param src the identity of the object that initiated the change
258 void
259 VisualTimeAxis::remove_this_time_axis(void* src)
261 vector<string> choices;
263 std::string prompt = string_compose (_("Do you really want to remove track \"%1\" ?\n\nYou may also lose the playlist used by this track.\n\n(This action cannot be undone, and the session file will be overwritten)"), time_axis_name);
265 choices.push_back (_("No, do nothing."));
266 choices.push_back (_("Yes, remove it."));
268 Gtkmm2ext::Choice prompter (prompt, choices);
270 if (prompter.run () == 1) {
272 defer to idle loop, otherwise we'll delete this object
273 while we're still inside this function ...
275 Glib::signal_idle().connect(sigc::bind(sigc::ptr_fun(&VisualTimeAxis::idle_remove_this_time_axis), this, src));
280 * Callback used to remove this time axis during the gtk idle loop
281 * This is used to avoid deleting the obejct while inside the remove_this_time_axis
282 * method
284 * @param ta the VisualTimeAxis to remove
285 * @param src the identity of the object that initiated the change
287 gint
288 VisualTimeAxis::idle_remove_this_time_axis(VisualTimeAxis* ta, void* src)
290 ta->VisualTimeAxisRemoved(ta->name(), src) ; /* EMIT_SIGNAL */
291 delete ta ;
292 ta = 0 ;
293 return(false) ;
299 //---------------------------------------------------------------------------------------//
300 // Handle TimeAxis rename
303 * Construct a new prompt to receive a new name for this TimeAxis
305 * @see finish_time_axis_rename()
307 void
308 VisualTimeAxis::start_time_axis_rename()
310 ArdourPrompter name_prompter;
312 name_prompter.set_prompt (_("new name: ")) ;
313 name_prompter.add_button (_("Rename"), Gtk::RESPONSE_ACCEPT);
314 name_prompter.set_response_sensitive (Gtk::RESPONSE_ACCEPT, false);
315 name_prompter.show_all() ;
317 switch (name_prompter.run ()) {
318 case Gtk::RESPONSE_ACCEPT:
319 string result;
320 name_prompter.get_result (result);
321 if (result.length()) {
322 if (editor.get_named_time_axis(result) != 0) {
323 ARDOUR_UI::instance()->popup_error (_("A track already exists with that name"));
324 return ;
327 set_time_axis_name(result, this) ;
330 label_view() ;
334 * Handles the new name for this TimeAxis from the name prompt
336 * @see start_time_axis_rename()
339 void
340 VisualTimeAxis::label_view()
342 name_label.set_text(time_axis_name) ;
343 name_entry.set_text(time_axis_name) ;
344 ARDOUR_UI::instance()->set_tip(name_entry, time_axis_name) ;
348 //---------------------------------------------------------------------------------------//
349 // Handle name entry signals
351 void
352 VisualTimeAxis::name_entry_changed()
354 string x = name_entry.get_text ();
356 if (x == time_axis_name) {
357 return;
360 strip_whitespace_edges(x);
362 if (x.length() == 0) {
363 name_entry.set_text (time_axis_name);
364 return;
367 if (!editor.get_named_time_axis(x)) {
368 set_time_axis_name(x, this);
369 } else {
370 ARDOUR_UI::instance()->popup_error (_("A track already exists with that name"));
371 name_entry.set_text(time_axis_name);
375 bool
376 VisualTimeAxis::name_entry_button_press_handler(GdkEventButton *ev)
378 if (ev->button == 3) {
379 return true;
381 return false
384 bool
385 VisualTimeAxis::name_entry_button_release_handler(GdkEventButton *ev)
387 return false;
390 bool
391 VisualTimeAxis::name_entry_key_release_handler(GdkEventKey* ev)
393 switch (ev->keyval) {
394 case GDK_Tab:
395 case GDK_Up:
396 case GDK_Down:
397 name_entry_changed ();
398 return true;
400 default:
401 break;
404 return false;
408 //---------------------------------------------------------------------------------------//
409 // Super class methods not handled by VisualTimeAxis
411 void
412 VisualTimeAxis::show_timestretch (framepos_t start, framepos_t end)
414 // Not handled by purely visual TimeAxis
417 void
418 VisualTimeAxis::hide_timestretch()
420 // Not handled by purely visual TimeAxis