2015-07-03 15:06:29 -04:00
|
|
|
/*
|
|
|
|
Copyright (C) 2015 Paul Davis
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <gtkmm/action.h>
|
|
|
|
#include <gtkmm/notebook.h>
|
|
|
|
#include <gtkmm/window.h>
|
2015-07-21 21:10:46 -04:00
|
|
|
#include <gtkmm/stock.h>
|
2015-07-03 15:06:29 -04:00
|
|
|
|
|
|
|
#include "gtkmm2ext/tabbable.h"
|
2015-07-09 13:27:39 -04:00
|
|
|
#include "gtkmm2ext/gtk_ui.h"
|
2015-07-21 22:37:39 -04:00
|
|
|
#include "gtkmm2ext/utils.h"
|
2015-07-03 15:06:29 -04:00
|
|
|
#include "gtkmm2ext/visibility_tracker.h"
|
|
|
|
|
2015-07-09 12:40:51 -04:00
|
|
|
#include "i18n.h"
|
|
|
|
|
2015-07-03 15:06:29 -04:00
|
|
|
using namespace Gtkmm2ext;
|
|
|
|
using namespace Gtk;
|
|
|
|
using std::string;
|
|
|
|
|
2015-07-07 22:12:21 -04:00
|
|
|
Tabbable::Tabbable (Widget& w, const string& name)
|
|
|
|
: WindowProxy (name)
|
|
|
|
, _contents (w)
|
2015-07-21 22:37:39 -04:00
|
|
|
, tab_close_image (ArdourIcon::CloseCross, 0xffffffff)
|
2015-07-03 15:06:29 -04:00
|
|
|
{
|
2015-07-21 22:37:39 -04:00
|
|
|
/* make the image about the same size as an actual X */
|
|
|
|
set_size_request_to_display_given_text (tab_close_image, "X", 0, 0);
|
|
|
|
|
|
|
|
_tab_box.set_spacing (2);
|
2015-07-21 21:10:46 -04:00
|
|
|
_tab_box.pack_start (_tab_label, true, true);
|
|
|
|
_tab_box.pack_start (_tab_close_button, false, false);
|
|
|
|
_tab_close_button.add (tab_close_image);
|
2015-07-21 22:37:39 -04:00
|
|
|
|
2015-07-21 21:10:46 -04:00
|
|
|
_tab_close_button.signal_clicked().connect (sigc::mem_fun (*this, &Tabbable::tab_close_clicked));
|
2015-07-03 15:06:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
Tabbable::~Tabbable ()
|
|
|
|
{
|
|
|
|
if (_window) {
|
|
|
|
delete _window;
|
|
|
|
_window = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-21 21:10:46 -04:00
|
|
|
void
|
|
|
|
Tabbable::tab_close_clicked ()
|
|
|
|
{
|
|
|
|
/* for this to happen, the tab must be visible so we
|
|
|
|
can assume that the contents are displayed in the
|
|
|
|
parent notebook
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (_parent_notebook) {
|
|
|
|
_parent_notebook->remove_page (_contents);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-03 15:06:29 -04:00
|
|
|
void
|
2015-07-09 13:27:39 -04:00
|
|
|
Tabbable::add_to_notebook (Notebook& notebook, const string& tab_title)
|
2015-07-03 15:06:29 -04:00
|
|
|
{
|
2015-07-21 21:10:46 -04:00
|
|
|
_tab_label.set_text (tab_title);
|
|
|
|
_tab_box.show_all ();
|
2015-07-21 22:37:39 -04:00
|
|
|
|
2015-07-21 21:10:46 -04:00
|
|
|
notebook.append_page (_contents, _tab_box);
|
2015-07-09 13:27:39 -04:00
|
|
|
|
2015-07-21 21:10:46 -04:00
|
|
|
Gtkmm2ext::UI::instance()->set_tip (_tab_label,
|
|
|
|
string_compose (_("Drag this tab to the desktop to show %1 in its own window\n\n"
|
|
|
|
"To put the window back, click on its \"close\" button"), tab_title));
|
2015-07-09 13:27:39 -04:00
|
|
|
|
2015-07-03 15:06:29 -04:00
|
|
|
notebook.set_tab_detachable (_contents);
|
2015-07-08 13:22:29 -04:00
|
|
|
notebook.set_tab_reorderable (_contents);
|
2015-07-03 15:06:29 -04:00
|
|
|
|
|
|
|
_parent_notebook = ¬ebook;
|
|
|
|
_tab_title = tab_title;
|
|
|
|
}
|
|
|
|
|
|
|
|
Window*
|
2015-07-09 12:40:51 -04:00
|
|
|
Tabbable::use_own_window (bool and_pack_it)
|
2015-07-03 15:06:29 -04:00
|
|
|
{
|
2015-07-09 12:40:51 -04:00
|
|
|
Gtk::Window* win = get (true);
|
|
|
|
|
|
|
|
if (and_pack_it) {
|
|
|
|
Gtk::Container* parent = _contents.get_parent();
|
|
|
|
if (parent) {
|
|
|
|
parent->remove (_contents);
|
|
|
|
}
|
2015-07-21 21:10:46 -04:00
|
|
|
_own_notebook.append_page (_contents, _tab_box);
|
2015-07-09 12:40:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return win;
|
|
|
|
|
2015-07-03 15:06:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Tabbable::window_visible ()
|
|
|
|
{
|
|
|
|
if (!own_window()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return visible();
|
|
|
|
}
|
|
|
|
|
|
|
|
Window*
|
|
|
|
Tabbable::get (bool create)
|
|
|
|
{
|
|
|
|
if (_window) {
|
|
|
|
return _window;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!create) {
|
|
|
|
return 0;
|
|
|
|
}
|
2015-07-07 22:12:21 -04:00
|
|
|
|
|
|
|
/* From here on, we're creating the window
|
|
|
|
*/
|
2015-07-03 15:06:29 -04:00
|
|
|
|
|
|
|
if ((_window = new Window (WINDOW_TOPLEVEL)) == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-07-07 22:12:21 -04:00
|
|
|
_window->add (_own_notebook);
|
|
|
|
_own_notebook.show ();
|
|
|
|
_own_notebook.set_show_tabs (false);
|
|
|
|
|
|
|
|
/* do other window-related setup */
|
|
|
|
|
|
|
|
setup ();
|
2015-07-03 15:06:29 -04:00
|
|
|
|
2015-07-07 22:12:21 -04:00
|
|
|
/* window should be ready for derived classes to do something with it */
|
|
|
|
|
2015-07-03 15:06:29 -04:00
|
|
|
return _window;
|
|
|
|
}
|
|
|
|
|
2015-07-07 22:12:21 -04:00
|
|
|
Gtk::Notebook*
|
|
|
|
Tabbable::tab_root_drop ()
|
|
|
|
{
|
2015-07-08 13:13:47 -04:00
|
|
|
Gtk::Allocation alloc;
|
|
|
|
|
|
|
|
alloc = _contents.get_parent()->get_allocation();
|
|
|
|
|
2015-07-09 12:40:51 -04:00
|
|
|
(void) use_own_window (false);
|
2015-07-07 22:12:21 -04:00
|
|
|
|
|
|
|
/* This is called after a drop of a tab onto the root window. Its
|
|
|
|
* responsibility is to return the notebook that this Tabbable's
|
|
|
|
* contents should be packed into before the drop handling is
|
|
|
|
* completed. It is not responsible for actually taking care of this
|
|
|
|
* packing.
|
|
|
|
*/
|
2015-07-08 13:13:47 -04:00
|
|
|
|
|
|
|
_window->set_default_size (alloc.get_width(), alloc.get_height());
|
2015-07-07 22:12:21 -04:00
|
|
|
_window->show_all ();
|
|
|
|
_window->present ();
|
|
|
|
|
|
|
|
return &_own_notebook;
|
|
|
|
}
|
|
|
|
|
2015-07-03 15:06:29 -04:00
|
|
|
void
|
|
|
|
Tabbable::show_window ()
|
|
|
|
{
|
2015-07-21 21:10:46 -04:00
|
|
|
make_visible ();
|
2015-07-03 15:06:29 -04:00
|
|
|
|
2015-07-21 21:10:46 -04:00
|
|
|
if (_window && (current_toplevel() == _window)) {
|
|
|
|
if (!_visible) { /* was hidden, update status */
|
|
|
|
set_pos_and_size ();
|
2015-07-03 15:06:29 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Tabbable::delete_event_handler (GdkEventAny *ev)
|
|
|
|
{
|
|
|
|
Window* toplevel = dynamic_cast<Window*> (_contents.get_toplevel());
|
|
|
|
|
|
|
|
if (_window == toplevel) {
|
|
|
|
|
|
|
|
/* unpack Tabbable from parent, put it back in the main tabbed
|
|
|
|
* notebook
|
|
|
|
*/
|
|
|
|
|
|
|
|
save_pos_and_size ();
|
|
|
|
|
|
|
|
_contents.get_parent()->remove (_contents);
|
|
|
|
|
|
|
|
/* leave the window around */
|
|
|
|
|
|
|
|
_window->hide ();
|
|
|
|
|
|
|
|
if (_parent_notebook) {
|
2015-07-07 22:12:21 -04:00
|
|
|
|
2015-07-21 21:10:46 -04:00
|
|
|
_parent_notebook->append_page (_contents, _tab_box);
|
2015-07-03 15:06:29 -04:00
|
|
|
_parent_notebook->set_tab_detachable (_contents);
|
2015-07-08 13:22:29 -04:00
|
|
|
_parent_notebook->set_tab_reorderable (_contents);
|
2015-07-08 13:13:47 -04:00
|
|
|
_parent_notebook->set_current_page (_parent_notebook->page_num (_contents));
|
2015-07-03 15:06:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* don't let anything else handle this */
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* nothing to do */
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Tabbable::is_tabbed () const
|
|
|
|
{
|
|
|
|
Window* toplevel = (Window*) _contents.get_toplevel();
|
|
|
|
|
|
|
|
if (_window && (toplevel == _window)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-21 21:10:46 -04:00
|
|
|
if (_parent_notebook && _contents.get_parent()) {
|
2015-07-03 15:06:29 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2015-07-07 22:12:21 -04:00
|
|
|
|
|
|
|
void
|
|
|
|
Tabbable::show_tab ()
|
|
|
|
{
|
|
|
|
if (!window_visible() && _parent_notebook) {
|
2015-07-21 21:10:46 -04:00
|
|
|
if (_contents.get_parent() == 0) {
|
|
|
|
add_to_notebook (*_parent_notebook, _tab_title);
|
|
|
|
}
|
2015-07-07 22:12:21 -04:00
|
|
|
_parent_notebook->set_current_page (_parent_notebook->page_num (_contents));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Gtk::Window*
|
|
|
|
Tabbable::current_toplevel () const
|
|
|
|
{
|
|
|
|
return dynamic_cast<Gtk::Window*> (contents().get_toplevel());
|
|
|
|
}
|
2015-07-09 12:40:51 -04:00
|
|
|
|
|
|
|
string
|
|
|
|
Tabbable::xml_node_name()
|
|
|
|
{
|
|
|
|
return WindowProxy::xml_node_name();
|
|
|
|
}
|
|
|
|
|
|
|
|
XMLNode&
|
|
|
|
Tabbable::get_state()
|
|
|
|
{
|
|
|
|
XMLNode& node (WindowProxy::get_state());
|
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
Tabbable::set_state (const XMLNode& node, int version)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if ((ret = WindowProxy::set_state (node, version)) == 0) {
|
|
|
|
if (_visible) {
|
|
|
|
if (use_own_window (true) == 0) {
|
|
|
|
ret = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2015-07-15 16:23:11 -04:00
|
|
|
|
|
|
|
void
|
|
|
|
Tabbable::make_visible ()
|
|
|
|
{
|
2015-07-21 21:10:46 -04:00
|
|
|
if (_window && (current_toplevel() == _window)) {
|
2015-07-15 16:23:11 -04:00
|
|
|
_window->present ();
|
|
|
|
} else {
|
|
|
|
show_tab ();
|
|
|
|
}
|
|
|
|
}
|