2 Copyright (C) 2015 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 <gtkmm/action.h>
21 #include <gtkmm/notebook.h>
22 #include <gtkmm/window.h>
23 #include <gtkmm/stock.h>
25 #include "gtkmm2ext/tabbable.h"
26 #include "gtkmm2ext/gtk_ui.h"
27 #include "gtkmm2ext/utils.h"
28 #include "gtkmm2ext/visibility_tracker.h"
32 using namespace Gtkmm2ext;
36 Tabbable::Tabbable (Widget& w, const string& name)
39 , tab_close_image (ArdourIcon::CloseCross, 0xffffffff)
40 , tab_requested_by_state (true)
42 /* make the image about the same size as an actual X */
43 tab_close_image.set_size_request (15,15);
44 // set_size_request_to_display_given_text (tab_close_image, "X", 0, 0);
46 _tab_box.set_spacing (2);
47 _tab_box.pack_start (_tab_label, true, true);
48 _tab_box.pack_start (_tab_close_button, false, false);
49 _tab_close_button.add (tab_close_image);
51 _tab_close_button.signal_clicked().connect (sigc::mem_fun (*this, &Tabbable::tab_close_clicked));
54 Tabbable::~Tabbable ()
63 Tabbable::tab_close_clicked ()
69 Tabbable::add_to_notebook (Notebook& notebook, const string& tab_title)
71 _parent_notebook = ¬ebook;
72 _tab_title = tab_title;
73 _tab_label.set_text (tab_title);
76 if (tab_requested_by_state) {
82 Tabbable::use_own_window (bool and_pack_it)
84 Gtk::Window* win = get (true);
87 Gtk::Container* parent = _contents.get_parent();
89 parent->remove (_contents);
91 _own_notebook.append_page (_contents, _tab_box);
99 Tabbable::window_visible ()
105 return _window->is_visible();
109 Tabbable::get (bool create)
119 /* From here on, we're creating the window
122 if ((_window = new Window (WINDOW_TOPLEVEL)) == 0) {
126 _window->add (_own_notebook);
127 _own_notebook.show ();
128 _own_notebook.set_show_tabs (false);
130 _window->signal_map().connect (sigc::mem_fun (*this, &Tabbable::window_mapped));
131 _window->signal_unmap().connect (sigc::mem_fun (*this, &Tabbable::window_unmapped));
133 /* do other window-related setup */
137 /* window should be ready for derived classes to do something with it */
143 Tabbable::show_own_window (bool and_pack_it)
145 Gtk::Widget* parent = _contents.get_parent();
146 Gtk::Allocation alloc;
149 alloc = parent->get_allocation();
152 (void) use_own_window (and_pack_it);
155 _window->set_default_size (alloc.get_width(), alloc.get_height());
158 _window->show_all ();
163 Tabbable::tab_root_drop ()
165 /* This is called after a drop of a tab onto the root window. Its
166 * responsibility xois to return the notebook that this Tabbable's
167 * contents should be packed into before the drop handling is
168 * completed. It is not responsible for actually taking care of this
172 show_own_window (false);
173 return &_own_notebook;
177 Tabbable::show_window ()
181 if (_window && (current_toplevel() == _window)) {
182 if (!_visible) { /* was hidden, update status */
189 Tabbable::make_visible ()
191 if (_window && (current_toplevel() == _window)) {
195 if (!tab_requested_by_state) {
196 show_own_window (true);
204 Tabbable::make_invisible ()
206 if (_window && (current_toplevel() == _window)) {
216 show_own_window (true);
222 if (!_parent_notebook) {
232 if (_window && current_toplevel() == _window) {
233 /* unpack Tabbable from parent, put it back in the main tabbed
237 save_pos_and_size ();
239 _contents.get_parent()->remove (_contents);
241 /* leave the window around */
246 _parent_notebook->append_page (_contents, _tab_box);
247 _parent_notebook->set_tab_detachable (_contents);
248 _parent_notebook->set_tab_reorderable (_contents);
249 _parent_notebook->set_current_page (_parent_notebook->page_num (_contents));
251 Gtkmm2ext::UI::instance()->set_tip (_tab_label,
252 string_compose (_("Drag this tab to the desktop to show %1 in its own window\n\n"
253 "To put the window back, click on its \"close\" button"), _tab_title));
256 /* have to force this on, which is semantically correct, since
257 * the user has effectively asked for it.
260 tab_requested_by_state = true;
265 Tabbable::delete_event_handler (GdkEventAny *ev)
273 Tabbable::is_tabbed () const
275 Window* toplevel = (Window*) _contents.get_toplevel();
277 if (_window && (toplevel == _window)) {
281 if (_parent_notebook && _contents.get_parent()) {
289 Tabbable::hide_tab ()
292 _parent_notebook->remove_page (_contents);
298 Tabbable::show_tab ()
300 if (!window_visible() && _parent_notebook) {
301 if (_contents.get_parent() == 0) {
302 tab_requested_by_state = true;
303 add_to_notebook (*_parent_notebook, _tab_title);
305 _parent_notebook->set_current_page (_parent_notebook->page_num (_contents));
310 Tabbable::current_toplevel () const
312 return dynamic_cast<Gtk::Window*> (contents().get_toplevel());
316 Tabbable::xml_node_name()
318 return WindowProxy::xml_node_name();
322 Tabbable::tabbed () const
324 return _parent_notebook && (_parent_notebook->page_num (_contents) >= 0);
328 Tabbable::get_state()
330 XMLNode& node (WindowProxy::get_state());
332 node.add_property (X_("tabbed"), tabbed() ? X_("yes") : X_("no"));
338 Tabbable::set_state (const XMLNode& node, int version)
342 if ((ret = WindowProxy::set_state (node, version)) == 0) {
344 if (use_own_window (true) == 0) {
350 XMLNodeList children = node.children ();
351 XMLNode* window_node = node.child ("Window");
354 const XMLProperty* prop = window_node->property (X_("tabbed"));
356 tab_requested_by_state = PBD::string_is_affirmative (prop->value());
360 if (tab_requested_by_state) {
362 std::cerr << name() << " pn " << _parent_notebook << std::endl;
363 if (_parent_notebook) {
364 std::cerr << "\t page " << _parent_notebook->page_num (_contents) << std::endl;
369 /* this does nothing if not tabbed */
377 Tabbable::window_mapped ()
383 Tabbable::window_unmapped ()