Merge branch 'master' into saveas
[ardour.git] / gtk2_ardour / ardour_window.cc
1 /*
2     Copyright (C) 2002-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 <iostream>
21 #include <sigc++/bind.h>
22
23 #include <gtkmm2ext/doi.h>
24
25 #include "ardour_window.h"
26 #include "ardour_ui.h"
27 #include "keyboard.h"
28 #include "utils.h"
29
30 using namespace std;
31 using namespace Gtk;
32 using namespace Gtkmm2ext;
33 using namespace ARDOUR_UI_UTILS;
34
35 ArdourWindow::ArdourWindow (string title)
36         : Window ()
37         , VisibilityTracker (*((Gtk::Window*)this))
38 {
39         set_title (title);
40         init ();
41         set_position (Gtk::WIN_POS_CENTER);
42 }
43
44 ArdourWindow::ArdourWindow (Gtk::Window& parent, string /*title*/)
45         : Window ()
46         , VisibilityTracker (*((Gtk::Window*)this))
47 {
48         init ();
49         set_transient_for (parent);
50         set_position (Gtk::WIN_POS_CENTER_ON_PARENT);
51 }
52
53 ArdourWindow::~ArdourWindow ()
54 {
55         WM::Manager::instance().remove (proxy);
56 }
57
58 bool
59 ArdourWindow::on_key_press_event (GdkEventKey* ev)
60 {
61         bool handled = Gtk::Window::on_key_press_event (ev);
62
63         if (!handled) {
64                 if (!get_modal()) {
65                         handled = relay_key_press (ev, this);
66                 }
67         }
68
69         return handled;
70 }
71
72 bool
73 ArdourWindow::on_focus_in_event (GdkEventFocus *ev)
74 {
75         Keyboard::the_keyboard().focus_in_window (ev, this);
76         return Window::on_focus_in_event (ev);
77 }
78
79 bool
80 ArdourWindow::on_focus_out_event (GdkEventFocus *ev)
81 {
82         if (!get_modal()) {
83                 Keyboard::the_keyboard().focus_out_window (ev, this);
84         }
85         return Window::on_focus_out_event (ev);
86 }
87
88 void
89 ArdourWindow::on_unmap ()
90 {
91         Keyboard::the_keyboard().leave_window (0, this);
92         Window::on_unmap ();
93 }
94
95 bool
96 ArdourWindow::on_delete_event (GdkEventAny*)
97 {
98         return false;
99 }
100
101 void
102 ArdourWindow::init ()
103 {
104         set_border_width (10);
105         add_events (Gdk::FOCUS_CHANGE_MASK);
106
107       /* ArdourWindows are not dialogs (they have no "OK" or "Close" button) but
108            they should be considered part of the same "window level" as a dialog. This
109            works on X11 and Quartz, in that:
110            
111            (a) utility & dialog windows are considered to be part of the same level
112            (b) they will float above normal windows without any particular effort
113            (c) present()-ing them will make a utility float over a dialog or
114                vice versa.
115         */
116
117         if (ARDOUR_UI::config()->get_all_floating_windows_are_dialogs()) {
118                 set_type_hint (Gdk::WINDOW_TYPE_HINT_DIALOG);
119         } else {
120                 set_type_hint (Gdk::WINDOW_TYPE_HINT_UTILITY);
121         }
122
123         Gtk::Window* parent = WM::Manager::instance().transient_parent();
124
125         if (parent) {
126                 set_transient_for (*parent);
127         }
128         
129         ARDOUR_UI::CloseAllDialogs.connect (sigc::mem_fun (*this, &ArdourWindow::hide));
130
131         proxy = new WM::ProxyTemporary (get_title(), this);
132         WM::Manager::instance().register_window (proxy);
133 }
134