Robin Gareus
e42699600b
There are two cases: (A) Proxy is created first, dialog is created later on demand (B) Dialog is created and directly registers its window as proxy In (B) the dialog is usually on the stack and destroyed when the ArdourDialog instances leaves scope. In that case ~ArdourDialog() is called and the proxy remained. Destroying the proxy does destroy the registered window in ~WindowProxy() If ArdourDialog's d'tor itself deletes the proxy it would recurse into itself. Existing APIs e.g. drop_window() likewise delete the window and cannot be safely called from ~ArdourDialog.
214 lines
4.6 KiB
C++
214 lines
4.6 KiB
C++
/*
|
|
Copyright (C) 2013 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.
|
|
|
|
*/
|
|
|
|
#ifndef __gtk2_ardour_window_manager_h__
|
|
#define __gtk2_ardour_window_manager_h__
|
|
|
|
#include <string>
|
|
#include <map>
|
|
|
|
#include <boost/function.hpp>
|
|
#include <glibmm/refptr.h>
|
|
#include <sigc++/trackable.h>
|
|
|
|
#include "gtkmm2ext/bindings.h"
|
|
#include "gtkmm2ext/window_proxy.h"
|
|
|
|
class XMLNode;
|
|
|
|
namespace Gtk {
|
|
class Window;
|
|
class Action;
|
|
}
|
|
|
|
namespace Gtkmm2ext {
|
|
class VisibilityTracker;
|
|
}
|
|
|
|
namespace ARDOUR {
|
|
class Session;
|
|
class SessionHandlePtr;
|
|
}
|
|
|
|
namespace WM {
|
|
|
|
class ProxyBase;
|
|
|
|
class Manager : public ARDOUR::SessionHandlePtr
|
|
{
|
|
public:
|
|
static Manager& instance();
|
|
|
|
void register_window (ProxyBase*);
|
|
void remove (const ProxyBase*);
|
|
void toggle_window (ProxyBase*);
|
|
void show_visible () const;
|
|
void set_session (ARDOUR::Session*);
|
|
void add_state (XMLNode&) const;
|
|
|
|
/* HACK HACK HACK */
|
|
void set_transient_for (Gtk::Window*);
|
|
Gtk::Window* transient_parent() const { return current_transient_parent; }
|
|
|
|
private:
|
|
typedef std::list<ProxyBase*> Windows;
|
|
Windows _windows;
|
|
Glib::RefPtr<Gtk::ActionGroup> window_actions;
|
|
Gtk::Window* current_transient_parent;
|
|
|
|
Manager();
|
|
~Manager();
|
|
|
|
static Manager* _instance;
|
|
private:
|
|
void window_proxy_was_mapped (ProxyBase*);
|
|
void window_proxy_was_unmapped (ProxyBase*);
|
|
};
|
|
|
|
class ProxyBase : public ARDOUR::SessionHandlePtr, public Gtkmm2ext::WindowProxy
|
|
{
|
|
public:
|
|
ProxyBase (const std::string& name, const std::string& menu_name);
|
|
ProxyBase (const std::string& name, const std::string& menu_name, const XMLNode&);
|
|
|
|
virtual ARDOUR::SessionHandlePtr* session_handle () = 0;
|
|
|
|
protected:
|
|
void setup ();
|
|
};
|
|
|
|
class ProxyTemporary: public ProxyBase
|
|
{
|
|
public:
|
|
ProxyTemporary (const std::string& name, Gtk::Window* win);
|
|
|
|
Gtk::Window* get (bool create = false) {
|
|
(void) create;
|
|
return _window;
|
|
}
|
|
|
|
Gtk::Window* operator->() {
|
|
return _window;
|
|
}
|
|
|
|
ARDOUR::SessionHandlePtr* session_handle ();
|
|
|
|
void explicit_delete () { _window = 0 ; delete this; }
|
|
};
|
|
|
|
template<typename T>
|
|
class ProxyWithConstructor: public ProxyBase
|
|
{
|
|
public:
|
|
ProxyWithConstructor (const std::string& name, const std::string& menu_name, const boost::function<T*()>& c)
|
|
: ProxyBase (name, menu_name) , creator (c) {}
|
|
|
|
ProxyWithConstructor (const std::string& name, const std::string& menu_name, const boost::function<T*()>& c, const XMLNode* node)
|
|
: ProxyBase (name, menu_name, *node) , creator (c) {}
|
|
|
|
Gtk::Window* get (bool create = false) {
|
|
if (!_window) {
|
|
if (!create) {
|
|
return 0;
|
|
}
|
|
|
|
_window = dynamic_cast<Gtk::Window*> (creator ());
|
|
|
|
if (_window) {
|
|
setup ();
|
|
}
|
|
}
|
|
|
|
return _window;
|
|
}
|
|
|
|
T* operator->() {
|
|
return dynamic_cast<T*> (get (true));
|
|
}
|
|
|
|
ARDOUR::SessionHandlePtr* session_handle () {
|
|
/* may return null */
|
|
return dynamic_cast<T*> (_window);
|
|
}
|
|
|
|
void set_session(ARDOUR::Session *s) {
|
|
SessionHandlePtr::set_session (s);
|
|
ARDOUR::SessionHandlePtr* sp = session_handle ();
|
|
if (sp) {
|
|
sp->set_session (s);
|
|
dynamic_cast<T*>(_window)->set_session(s);
|
|
}
|
|
}
|
|
|
|
private:
|
|
boost::function<T*()> creator;
|
|
};
|
|
|
|
template<typename T>
|
|
class Proxy : public ProxyBase
|
|
{
|
|
public:
|
|
Proxy (const std::string& name, const std::string& menu_name)
|
|
: ProxyBase (name, menu_name) {}
|
|
|
|
Proxy (const std::string& name, const std::string& menu_name, const XMLNode* node)
|
|
: ProxyBase (name, menu_name, *node) {}
|
|
|
|
Gtk::Window* get (bool create = false) {
|
|
if (!_window) {
|
|
if (!create) {
|
|
return 0;
|
|
}
|
|
|
|
_window = new T ();
|
|
|
|
if (_window) {
|
|
setup ();
|
|
}
|
|
}
|
|
|
|
return _window;
|
|
}
|
|
|
|
T* operator->() {
|
|
return dynamic_cast<T*> (get(true));
|
|
}
|
|
|
|
ARDOUR::SessionHandlePtr* session_handle () {
|
|
/* may return null */
|
|
return dynamic_cast<T*> (_window);
|
|
}
|
|
|
|
void set_session(ARDOUR::Session *s) {
|
|
SessionHandlePtr::set_session (s);
|
|
ARDOUR::SessionHandlePtr* sp = session_handle ();
|
|
if (sp) {
|
|
sp->set_session (s);
|
|
dynamic_cast<T*>(_window)->set_session(s);
|
|
}
|
|
}
|
|
|
|
private:
|
|
boost::function<T*()> creator;
|
|
};
|
|
|
|
} /* namespace */
|
|
|
|
#endif /* __gtk2_ardour_window_manager_h__ */
|