netradiant-custom/libs/gtkutil/widget.h

199 lines
6.0 KiB
C++

/*
Copyright (C) 2001-2006, William Joseph.
All Rights Reserved.
This file is part of GtkRadiant.
GtkRadiant 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.
GtkRadiant 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 GtkRadiant; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#if !defined( INCLUDED_GTKUTIL_WIDGET_H )
#define INCLUDED_GTKUTIL_WIDGET_H
#include <list>
#include <gtk/gtk.h>
#include "generic/callback.h"
#include "warnings.h"
#include "debugging/debugging.h"
inline bool widget_is_visible( GtkWidget* widget ){
//return GTK_WIDGET_VISIBLE( widget );
return gtk_widget_get_visible( widget );
}
inline void widget_set_visible( GtkWidget* widget, bool show ){
if ( show ) {
/* workaround for gtk 2.24 issue: not displayed glwidget after toggle */
GtkWidget* glwidget = GTK_WIDGET( g_object_get_data( G_OBJECT( widget ), "glwidget" ) );
if ( glwidget ){
gtk_widget_hide( glwidget );
gtk_widget_show( glwidget );
}
gtk_widget_show( widget );
}
else
{
gtk_widget_hide( widget );
}
}
inline void widget_toggle_visible( GtkWidget* widget ){
widget_set_visible( widget, !widget_is_visible( widget ) );
}
class ToggleItem
{
BoolExportCallback m_exportCallback;
typedef std::list<BoolImportCallback> ImportCallbacks;
ImportCallbacks m_importCallbacks;
public:
ToggleItem( const BoolExportCallback& exportCallback ) : m_exportCallback( exportCallback ){
}
void update(){
for ( ImportCallbacks::iterator i = m_importCallbacks.begin(); i != m_importCallbacks.end(); ++i )
{
m_exportCallback( *i );
}
}
void addCallback( const BoolImportCallback& callback ){
m_importCallbacks.push_back( callback );
m_exportCallback( callback );
}
typedef MemberCaller1<ToggleItem, const BoolImportCallback&, &ToggleItem::addCallback> AddCallbackCaller;
};
class ToggleShown
{
bool m_shownDeferred;
ToggleShown( const ToggleShown& other ); // NOT COPYABLE
ToggleShown& operator=( const ToggleShown& other ); // NOT ASSIGNABLE
static gboolean notify_visible( GtkWidget* widget, gpointer dummy, ToggleShown* self ){
/* destroy = notify::visible with visible = 0, thus let's filter it out */
if( gtk_main_level() > 0 ){ //== 0 at destroy time
self->m_shownDeferred = widget_is_visible( self->m_widget );
}
//globalOutputStream() << "ToggleShown::notify_visible time " << gtk_get_current_event_time() << " visible " << self->m_shownDeferred << "\n";
self->update();
return FALSE;
}
static gboolean destroy( GtkWidget* widget, ToggleShown* self ){
//globalOutputStream() << "ToggleShown::destroy time " << gtk_get_current_event_time() << " visible " << self->m_shownDeferred << "\n";
//self->m_shownDeferred = widget_is_visible( self->m_widget ); //always 0 at destroy time
self->m_widget = 0;
return FALSE;
}
public:
GtkWidget* m_widget;
ToggleItem m_item;
ToggleShown( bool shown )
: m_shownDeferred( shown ), m_widget( 0 ), m_item( ActiveCaller( *this ) ){
}
void update(){
//globalOutputStream() << "ToggleShown::update\n";
m_item.update();
}
bool active() const {
//globalOutputStream() << "ToggleShown::active\n";
if ( m_widget == 0 ) {
return m_shownDeferred;
}
else
{
return widget_is_visible( m_widget );
}
}
void exportActive( const BoolImportCallback& importCallback ){
//globalOutputStream() << "ToggleShown::exportActive\n";
importCallback( active() );
}
typedef MemberCaller1<ToggleShown, const BoolImportCallback&, &ToggleShown::exportActive> ActiveCaller;
void set( bool shown ){
//globalOutputStream() << "ToggleShown::set\n";
if ( m_widget == 0 ) {
m_shownDeferred = shown;
}
else
{
widget_set_visible( m_widget, shown );
}
}
void toggle(){
//globalOutputStream() << "ToggleShown::toggle\n";
widget_toggle_visible( m_widget );
}
typedef MemberCaller<ToggleShown, &ToggleShown::toggle> ToggleCaller;
void connect( GtkWidget* widget ){
//globalOutputStream() << "ToggleShown::connect\n";
m_widget = widget;
widget_set_visible( m_widget, m_shownDeferred );
g_signal_connect( G_OBJECT( m_widget ), "notify::visible", G_CALLBACK( notify_visible ), this );
g_signal_connect( G_OBJECT( m_widget ), "destroy", G_CALLBACK( destroy ), this );
update();
}
};
namespace{
void ToggleShown_importBool( ToggleShown& self, bool value ){
self.set( value );
}
typedef ReferenceCaller1<ToggleShown, bool, ToggleShown_importBool> ToggleShownImportBoolCaller;
void ToggleShown_exportBool( const ToggleShown& self, const BoolImportCallback& importer ){
importer( self.active() );
}
typedef ConstReferenceCaller1<ToggleShown, const BoolImportCallback&, ToggleShown_exportBool> ToggleShownExportBoolCaller;
}
inline void widget_queue_draw( GtkWidget& widget ){
gtk_widget_queue_draw( &widget );
}
typedef ReferenceCaller<GtkWidget, widget_queue_draw> WidgetQueueDrawCaller;
inline void widget_make_default( GtkWidget* widget ){
gtk_widget_set_can_default( widget, TRUE );
gtk_widget_grab_default( widget );
}
class WidgetFocusPrinter
{
const char* m_name;
static gboolean focus_in( GtkWidget *widget, GdkEventFocus *event, WidgetFocusPrinter* self ){
globalOutputStream() << self->m_name << " takes focus\n";
return FALSE;
}
static gboolean focus_out( GtkWidget *widget, GdkEventFocus *event, WidgetFocusPrinter* self ){
globalOutputStream() << self->m_name << " loses focus\n";
return FALSE;
}
public:
WidgetFocusPrinter( const char* name ) : m_name( name ){
}
void connect( GtkWidget* widget ){
g_signal_connect( G_OBJECT( widget ), "focus_in_event", G_CALLBACK( focus_in ), this );
g_signal_connect( G_OBJECT( widget ), "focus_out_event", G_CALLBACK( focus_out ), this );
}
};
#endif