541 lines
12 KiB
C++
541 lines
12 KiB
C++
/*
|
|
* Copyright (C) 2015 Paul Davis <paul@linuxaudiosystems.com>
|
|
* Copyright (C) 2017 Robin Gareus <robin@gareus.org>
|
|
*
|
|
* 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.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*/
|
|
|
|
#include <gtkmm/action.h>
|
|
#include <gtkmm/frame.h>
|
|
#include <gtkmm/notebook.h>
|
|
#include <gtkmm/window.h>
|
|
#include <gtkmm/stock.h>
|
|
|
|
#include "gtkmm2ext/actions.h"
|
|
#include "gtkmm2ext/gtk_ui.h"
|
|
#include "gtkmm2ext/utils.h"
|
|
#include "gtkmm2ext/visibility_tracker.h"
|
|
|
|
#include "widgets/tabbable.h"
|
|
|
|
#include "pbd/i18n.h"
|
|
|
|
using std::string;
|
|
using namespace Gtk;
|
|
using namespace Gtkmm2ext;
|
|
using namespace ArdourWidgets;
|
|
|
|
Tabbable::Tabbable (const string& visible_name, string const & nontranslatable_name, Widget* w, bool tabbed_by_default)
|
|
: WindowProxy (visible_name, nontranslatable_name)
|
|
, _parent_notebook (0)
|
|
, tab_requested_by_state (tabbed_by_default)
|
|
{
|
|
if (w) {
|
|
_contents = w;
|
|
} else {
|
|
_contents = &_content_vbox;
|
|
default_layout ();
|
|
}
|
|
}
|
|
|
|
Tabbable::~Tabbable ()
|
|
{
|
|
if (_window) {
|
|
delete _window;
|
|
_window = 0;
|
|
}
|
|
}
|
|
|
|
void
|
|
Tabbable::default_layout ()
|
|
{
|
|
#ifndef LIVETRAX
|
|
left_attachment_button.set_text (_("Left"));
|
|
right_attachment_button.set_text (_("Right"));
|
|
bottom_attachment_button.set_text (_("Btm"));
|
|
|
|
left_attachment_button.set_icon (ArdourIcon::AttachmentLeft);
|
|
right_attachment_button.set_icon (ArdourIcon::AttachmentRight);
|
|
bottom_attachment_button.set_icon (ArdourIcon::AttachmentBottom);
|
|
|
|
left_attachment_button.set_name ("lock button"); // XXX re-use netural "fill active" bg style of "lock"
|
|
right_attachment_button.set_name ("lock button"); // TODO create dedicate button style
|
|
bottom_attachment_button.set_name ("lock button");
|
|
|
|
left_attachment_button.set_tweaks (ArdourButton::ExpandtoSquare);
|
|
right_attachment_button.set_tweaks (ArdourButton::ExpandtoSquare);
|
|
bottom_attachment_button.set_tweaks (ArdourButton::ExpandtoSquare);
|
|
|
|
content_attachment_hbox.set_border_width(3);
|
|
content_attachment_hbox.set_spacing(3);
|
|
content_attachment_hbox.pack_end (right_attachment_button, false, false);
|
|
content_attachment_hbox.pack_end (bottom_attachment_button, false, false);
|
|
content_attachment_hbox.pack_end (left_attachment_button, false, false);
|
|
content_attachments.add (content_attachment_hbox);
|
|
#endif
|
|
|
|
content_header_hbox.pack_start (content_app_bar, true, true);
|
|
content_header_hbox.pack_start (content_attachments, false, false);
|
|
content_header_hbox.pack_start (content_tabbables, false, false);
|
|
|
|
#ifndef LIVETRAX
|
|
//wrap the header eboxen in a themeable frame
|
|
Gtk::Frame *toolbar_frame = manage(new Gtk::Frame);
|
|
toolbar_frame->set_name ("TransportFrame");
|
|
toolbar_frame->set_shadow_type (Gtk::SHADOW_NONE);
|
|
toolbar_frame->add (content_header_hbox);
|
|
|
|
_content_vbox.pack_start (*toolbar_frame, false, false);
|
|
#endif
|
|
_content_vbox.pack_start (content_hbox, true, true);
|
|
|
|
content_hbox.pack_start (content_att_left, false, false);
|
|
content_hbox.pack_start (content_midlevel_vbox, true, true);
|
|
|
|
content_midlevel_vbox.pack_start (content_right_pane, true, true);
|
|
content_midlevel_vbox.pack_start (content_att_bottom, false, false);
|
|
|
|
content_right_pane.add (content_inner_vbox);
|
|
content_right_pane.add (content_right_vbox);
|
|
|
|
//TODO: menu switcher here?
|
|
content_right_vbox.pack_start (content_att_right, true, true);
|
|
|
|
content_inner_vbox.pack_start (content_toolbar, false, false);
|
|
content_inner_vbox.pack_start (content_innermost_hbox, true, true);
|
|
|
|
content_right_pane.set_child_minsize (content_att_right, 160); /* rough guess at width of notebook tabs */
|
|
content_right_pane.set_check_divider_position (true);
|
|
content_right_pane.set_divider (0, 0.85);
|
|
|
|
_content_vbox.show_all();
|
|
}
|
|
|
|
void
|
|
Tabbable::add_to_notebook (Notebook& notebook)
|
|
{
|
|
_parent_notebook = ¬ebook;
|
|
|
|
if (tab_requested_by_state) {
|
|
attach ();
|
|
}
|
|
}
|
|
|
|
Window*
|
|
Tabbable::use_own_window (bool and_pack_it)
|
|
{
|
|
Gtk::Window* win = get (true);
|
|
|
|
if (and_pack_it) {
|
|
Gtk::Container* parent = _contents->get_parent();
|
|
if (parent) {
|
|
_contents->hide ();
|
|
parent->remove (*_contents);
|
|
}
|
|
_own_notebook.append_page (*_contents);
|
|
_contents->show ();
|
|
}
|
|
|
|
return win;
|
|
|
|
}
|
|
|
|
bool
|
|
Tabbable::window_visible () const
|
|
{
|
|
if (!_window) {
|
|
return false;
|
|
}
|
|
|
|
return _window->get_visible();
|
|
}
|
|
|
|
Window*
|
|
Tabbable::get (bool create)
|
|
{
|
|
if (_window) {
|
|
return _window;
|
|
}
|
|
|
|
if (!create) {
|
|
return 0;
|
|
}
|
|
|
|
/* From here on, we're creating the window
|
|
*/
|
|
|
|
if ((_window = new Window (WINDOW_TOPLEVEL)) == 0) {
|
|
return 0;
|
|
}
|
|
|
|
_window->add (_own_notebook);
|
|
_own_notebook.show ();
|
|
_own_notebook.set_show_tabs (false);
|
|
|
|
_window->signal_map().connect (sigc::mem_fun (*this, &Tabbable::window_mapped));
|
|
_window->signal_unmap().connect (sigc::mem_fun (*this, &Tabbable::window_unmapped));
|
|
|
|
/* do other window-related setup */
|
|
|
|
setup ();
|
|
|
|
/* window should be ready for derived classes to do something with it */
|
|
|
|
return _window;
|
|
}
|
|
|
|
void
|
|
Tabbable::show_own_window (bool and_pack_it)
|
|
{
|
|
Gtk::Widget* parent = _contents->get_parent();
|
|
Gtk::Allocation alloc;
|
|
|
|
if (parent) {
|
|
alloc = parent->get_allocation();
|
|
}
|
|
|
|
(void) use_own_window (and_pack_it);
|
|
|
|
if (parent) {
|
|
_window->set_default_size (alloc.get_width(), alloc.get_height());
|
|
}
|
|
|
|
tab_requested_by_state = false;
|
|
|
|
_window->present ();
|
|
}
|
|
|
|
Gtk::Notebook*
|
|
Tabbable::tab_root_drop ()
|
|
{
|
|
/* 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.
|
|
*/
|
|
|
|
show_own_window (false);
|
|
return &_own_notebook;
|
|
}
|
|
|
|
void
|
|
Tabbable::show_window ()
|
|
{
|
|
make_visible ();
|
|
|
|
if (_window && (current_toplevel() == _window)) {
|
|
if (!_visible) { /* was hidden, update status */
|
|
set_pos_and_size ();
|
|
}
|
|
}
|
|
}
|
|
|
|
/** If this Tabbable is currently parented by a tab, ensure that the tab is the
|
|
* current one. If it is parented by a window, then toggle the visibility of
|
|
* that window.
|
|
*/
|
|
void
|
|
Tabbable::change_visibility ()
|
|
{
|
|
if (tabbed()) {
|
|
_parent_notebook->set_current_page (_parent_notebook->page_num (*_contents));
|
|
return;
|
|
}
|
|
|
|
if (tab_requested_by_state) {
|
|
/* should be tabbed, but currently isn't parented by a notebook */
|
|
return;
|
|
}
|
|
|
|
if (_window && (current_toplevel() == _window)) {
|
|
/* Use WindowProxy method which will rotate then hide */
|
|
toggle();
|
|
}
|
|
}
|
|
|
|
void
|
|
Tabbable::make_visible ()
|
|
{
|
|
if (_window && (current_toplevel() == _window)) {
|
|
set_pos ();
|
|
_window->present ();
|
|
} else {
|
|
|
|
if (!tab_requested_by_state) {
|
|
show_own_window (true);
|
|
} else {
|
|
show_tab ();
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
Tabbable::make_invisible ()
|
|
{
|
|
if (_window && (current_toplevel() == _window)) {
|
|
_window->hide ();
|
|
} else {
|
|
hide_tab ();
|
|
}
|
|
}
|
|
|
|
void
|
|
Tabbable::detach ()
|
|
{
|
|
show_own_window (true);
|
|
}
|
|
|
|
void
|
|
Tabbable::attach ()
|
|
{
|
|
if (!_parent_notebook) {
|
|
return;
|
|
}
|
|
|
|
if (tabbed()) {
|
|
/* already tabbed */
|
|
return;
|
|
}
|
|
|
|
|
|
if (_window && current_toplevel() == _window) {
|
|
/* unpack Tabbable from parent, put it back in the main tabbed
|
|
* notebook
|
|
*/
|
|
|
|
save_pos_and_size ();
|
|
|
|
_contents->hide ();
|
|
_contents->get_parent()->remove (*_contents);
|
|
|
|
/* leave the window around */
|
|
|
|
_window->hide ();
|
|
}
|
|
|
|
_parent_notebook->append_page (*_contents);
|
|
_parent_notebook->set_tab_detachable (*_contents);
|
|
_parent_notebook->set_tab_reorderable (*_contents);
|
|
_parent_notebook->set_current_page (_parent_notebook->page_num (*_contents));
|
|
|
|
_contents->show ();
|
|
|
|
/* have to force this on, which is semantically correct, since
|
|
* the user has effectively asked for it.
|
|
*/
|
|
|
|
tab_requested_by_state = true;
|
|
StateChange (*this);
|
|
}
|
|
|
|
bool
|
|
Tabbable::delete_event_handler (GdkEventAny *ev)
|
|
{
|
|
_window->hide();
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
Tabbable::tabbed () const
|
|
{
|
|
if (_window && (current_toplevel() == _window)) {
|
|
return false;
|
|
}
|
|
|
|
if (_parent_notebook && (_parent_notebook->page_num (*_contents) >= 0)) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void
|
|
Tabbable::hide_tab ()
|
|
{
|
|
if (tabbed()) {
|
|
_contents->hide();
|
|
_parent_notebook->remove_page (*_contents);
|
|
StateChange (*this);
|
|
}
|
|
}
|
|
|
|
void
|
|
Tabbable::show_tab ()
|
|
{
|
|
if (!window_visible() && _parent_notebook) {
|
|
if (_contents->get_parent() == 0) {
|
|
tab_requested_by_state = true;
|
|
add_to_notebook (*_parent_notebook);
|
|
}
|
|
_parent_notebook->set_current_page (_parent_notebook->page_num (*_contents));
|
|
_contents->show ();
|
|
current_toplevel()->present ();
|
|
}
|
|
}
|
|
|
|
Gtk::Window*
|
|
Tabbable::current_toplevel () const
|
|
{
|
|
return dynamic_cast<Gtk::Window*> (contents().get_toplevel());
|
|
}
|
|
|
|
string
|
|
Tabbable::xml_node_name()
|
|
{
|
|
return WindowProxy::xml_node_name();
|
|
}
|
|
|
|
bool
|
|
Tabbable::tabbed_by_default() const
|
|
{
|
|
return tab_requested_by_state;
|
|
}
|
|
|
|
XMLNode&
|
|
Tabbable::get_state() const
|
|
{
|
|
XMLNode& node (WindowProxy::get_state());
|
|
|
|
node.set_property (X_("tabbed"), tabbed());
|
|
|
|
node.set_property (string_compose("%1%2", _menu_name, X_("-listpane-pos")).c_str(), content_right_pane.get_divider ());
|
|
|
|
return node;
|
|
}
|
|
|
|
int
|
|
Tabbable::set_state (const XMLNode& node, int version)
|
|
{
|
|
int ret;
|
|
|
|
if ((ret = WindowProxy::set_state (node, version)) != 0) {
|
|
return ret;
|
|
}
|
|
|
|
if (_visible) {
|
|
show_own_window (true);
|
|
}
|
|
|
|
XMLNodeList children = node.children ();
|
|
XMLNode* window_node = node.child ("Window");
|
|
|
|
if (window_node) {
|
|
window_node->get_property (X_("tabbed"), tab_requested_by_state);
|
|
float fract;
|
|
if ( window_node->get_property (string_compose("%1%2", _menu_name, X_("-listpane-pos")).c_str(), fract) ) {
|
|
fract = std::max (.05f, std::min (.95f, fract));
|
|
content_right_pane.set_divider (0, fract);
|
|
}
|
|
}
|
|
|
|
|
|
if (!_visible) {
|
|
if (tab_requested_by_state) {
|
|
attach ();
|
|
} else {
|
|
/* this does nothing if not tabbed */
|
|
hide_tab ();
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
Tabbable::window_mapped ()
|
|
{
|
|
StateChange (*this);
|
|
}
|
|
|
|
void
|
|
Tabbable::window_unmapped ()
|
|
{
|
|
StateChange (*this);
|
|
}
|
|
|
|
void
|
|
Tabbable::showhide_att_right (bool yn)
|
|
{
|
|
if (yn) {
|
|
content_right_vbox.show ();
|
|
} else {
|
|
content_right_vbox.hide ();
|
|
}
|
|
}
|
|
|
|
void
|
|
Tabbable::att_right_button_toggled ()
|
|
{
|
|
Glib::RefPtr<Gtk::Action> act = right_attachment_button.get_related_action();
|
|
if (act) {
|
|
Glib::RefPtr<Gtk::ToggleAction> tact = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(act);
|
|
if (tact) {
|
|
showhide_att_right (tact->get_active());
|
|
}
|
|
} else {
|
|
showhide_att_right (false);
|
|
}
|
|
}
|
|
|
|
void
|
|
Tabbable::showhide_att_left (bool yn)
|
|
{
|
|
if (yn) {
|
|
content_att_left.show ();
|
|
} else {
|
|
content_att_left.hide ();
|
|
}
|
|
}
|
|
|
|
void
|
|
Tabbable::att_left_button_toggled ()
|
|
{
|
|
Glib::RefPtr<Gtk::Action> act = left_attachment_button.get_related_action();
|
|
if (act) {
|
|
Glib::RefPtr<Gtk::ToggleAction> tact = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(act);
|
|
if (tact) {
|
|
showhide_att_left (tact->get_active());
|
|
}
|
|
} else {
|
|
showhide_att_left (false);
|
|
}
|
|
}
|
|
|
|
void
|
|
Tabbable::showhide_att_bottom (bool yn)
|
|
{
|
|
if (yn) {
|
|
content_att_bottom.show ();
|
|
} else {
|
|
content_att_bottom.hide ();
|
|
}
|
|
}
|
|
|
|
void
|
|
Tabbable::att_bottom_button_toggled ()
|
|
{
|
|
Glib::RefPtr<Gtk::Action> act = bottom_attachment_button.get_related_action();
|
|
if (act) {
|
|
Glib::RefPtr<Gtk::ToggleAction> tact = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(act);
|
|
if (tact) {
|
|
showhide_att_bottom (tact->get_active());
|
|
}
|
|
} else {
|
|
showhide_att_bottom (false);
|
|
}
|
|
}
|