2 Copyright (C) 2001-2006, William Joseph.
5 This file is part of GtkRadiant.
7 GtkRadiant is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 2 of the License, or
10 (at your option) any later version.
12 GtkRadiant is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with GtkRadiant; if not, write to the Free Software
19 Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
22 #include "entitylist.h"
24 #include "iselection.h"
26 #include <uilib/uilib.h>
28 #include <gtk/gtkcheckbutton.h>
29 #include <gtk/gtkvbox.h>
31 #include "string/string.h"
34 #include "signal/isignal.h"
35 #include "generic/object.h"
37 #include "gtkutil/widget.h"
38 #include "gtkutil/window.h"
39 #include "gtkutil/idledraw.h"
40 #include "gtkutil/accelerator.h"
41 #include "gtkutil/closure.h"
43 #include "treemodel.h"
45 #include "mainframe.h"
47 void RedrawEntityList();
48 typedef FreeCaller<void(), RedrawEntityList> RedrawEntityListCaller;
64 WindowPositionTracker m_positionTracker;
68 ui::TreeView m_tree_view{ui::null};
69 ui::TreeModel m_tree_model{ui::null};
70 bool m_selection_disabled;
73 m_dirty( EntityList::eDefault ),
74 m_idleDraw( RedrawEntityListCaller() ),
77 m_selection_disabled( false ){
81 return m_window.visible();
87 EntityList* g_EntityList;
89 inline EntityList& getEntityList(){
90 ASSERT_NOTNULL( g_EntityList );
96 inline Nameable* Node_getNameable( scene::Node& node ){
97 return NodeTypeCast<Nameable>::cast( node );
100 const char* node_get_name( scene::Node& node ){
101 Nameable* nameable = Node_getNameable( node );
102 return ( nameable != 0 )
107 template<typename value_type>
108 inline void gtk_tree_model_get_pointer( ui::TreeModel model, GtkTreeIter* iter, gint column, value_type** pointer ){
109 GValue value = GValue_default();
110 gtk_tree_model_get_value( model, iter, column, &value );
111 *pointer = (value_type*)g_value_get_pointer( &value );
116 void entitylist_treeviewcolumn_celldatafunc( ui::TreeViewColumn column, ui::CellRenderer renderer, ui::TreeModel model, GtkTreeIter* iter, gpointer data ){
118 gtk_tree_model_get_pointer( model, iter, 0, &node );
119 scene::Instance* instance;
120 gtk_tree_model_get_pointer( model, iter, 1, &instance );
122 gtk_cell_renderer_set_fixed_size( renderer, -1, -1 );
123 char* name = const_cast<char*>( node_get_name( *node ) );
124 g_object_set( G_OBJECT( renderer ), "text", name, "visible", TRUE, NULL );
126 //globalOutputStream() << "rendering cell " << makeQuoted(name) << "\n";
127 auto style = gtk_widget_get_style( ui::TreeView( getEntityList().m_tree_view ) );
128 if ( instance->childSelected() ) {
129 g_object_set( G_OBJECT( renderer ), "cell-background-gdk", &style->base[GTK_STATE_ACTIVE], NULL );
133 g_object_set( G_OBJECT( renderer ), "cell-background-gdk", &style->base[GTK_STATE_NORMAL], NULL );
138 gtk_cell_renderer_set_fixed_size( renderer, -1, 0 );
139 g_object_set( G_OBJECT( renderer ), "text", "", "visible", FALSE, NULL );
143 static gboolean entitylist_tree_select( ui::TreeSelection selection, ui::TreeModel model, ui::TreePath path, gboolean path_currently_selected, gpointer data ){
145 gtk_tree_model_get_iter( model, &iter, path );
147 gtk_tree_model_get_pointer( model, &iter, 0, &node );
148 scene::Instance* instance;
149 gtk_tree_model_get_pointer( model, &iter, 1, &instance );
150 Selectable* selectable = Instance_getSelectable( *instance );
153 if ( path_currently_selected != FALSE ) {
154 getEntityList().m_selection_disabled = true;
155 GlobalSelectionSystem().setSelectedAll( false );
156 getEntityList().m_selection_disabled = false;
159 else if ( selectable != 0 ) {
160 getEntityList().m_selection_disabled = true;
161 selectable->setSelected( path_currently_selected == FALSE );
162 getEntityList().m_selection_disabled = false;
163 if( gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON( getEntityList().m_check ) ) ){
172 static gboolean entitylist_tree_select_null( ui::TreeSelection selection, ui::TreeModel model, ui::TreePath path, gboolean path_currently_selected, gpointer data ){
176 void EntityList_ConnectSignals( ui::TreeView view ){
177 auto select = gtk_tree_view_get_selection( view );
178 gtk_tree_selection_set_select_function(select, reinterpret_cast<GtkTreeSelectionFunc>(entitylist_tree_select), NULL, 0 );
181 void EntityList_DisconnectSignals( ui::TreeView view ){
182 auto select = gtk_tree_view_get_selection( view );
183 gtk_tree_selection_set_select_function(select, reinterpret_cast<GtkTreeSelectionFunc>(entitylist_tree_select_null), 0, 0 );
188 gboolean treemodel_update_selection( ui::TreeModel model, ui::TreePath path, GtkTreeIter* iter, gpointer data ){
189 auto view = ui::TreeView::from( data );
191 scene::Instance* instance;
192 gtk_tree_model_get_pointer( model, iter, 1, &instance );
193 Selectable* selectable = Instance_getSelectable( *instance );
195 if ( selectable != 0 ) {
196 auto selection = gtk_tree_view_get_selection( view );
197 if ( selectable->isSelected() ) {
198 gtk_tree_selection_select_path( selection, path );
202 gtk_tree_selection_unselect_path( selection, path );
209 void EntityList_UpdateSelection( ui::TreeModel model, ui::TreeView view ){
210 EntityList_DisconnectSignals( view );
211 gtk_tree_model_foreach(model, reinterpret_cast<GtkTreeModelForeachFunc>(treemodel_update_selection), view._handle );
212 EntityList_ConnectSignals( view );
216 void RedrawEntityList(){
217 switch ( getEntityList().m_dirty )
219 case EntityList::eInsertRemove:
220 case EntityList::eSelection:
221 EntityList_UpdateSelection( getEntityList().m_tree_model, getEntityList().m_tree_view );
225 getEntityList().m_dirty = EntityList::eDefault;
228 void entitylist_queue_draw(){
229 getEntityList().m_idleDraw.queueDraw();
232 void EntityList_SelectionUpdate(){
233 if ( getEntityList().m_selection_disabled ) {
237 if ( getEntityList().m_dirty < EntityList::eSelection ) {
238 getEntityList().m_dirty = EntityList::eSelection;
240 entitylist_queue_draw();
243 void EntityList_SelectionChanged( const Selectable& selectable ){
244 EntityList_SelectionUpdate();
247 void entitylist_treeview_rowcollapsed( ui::TreeView view, GtkTreeIter* iter, ui::TreePath path, gpointer user_data ){
250 void entitylist_treeview_row_expanded( ui::TreeView view, GtkTreeIter* iter, ui::TreePath path, gpointer user_data ){
251 EntityList_SelectionUpdate();
255 void EntityList_SetShown( bool shown ){
256 getEntityList().m_window.visible(shown);
259 void EntityList_toggleShown(){
260 EntityList_SetShown( !getEntityList().visible() );
263 gint graph_tree_model_compare_name( ui::TreeModel model, GtkTreeIter *a, GtkTreeIter *b, gpointer user_data ){
265 gtk_tree_model_get( model, a, 0, (gpointer*)&first, -1 );
267 gtk_tree_model_get( model, b, 0, (gpointer*)&second, -1 );
269 if ( first != 0 && second != 0 ) {
270 result = string_compare( node_get_name( *first ), node_get_name( *second ) );
273 return ( first < second ) ? -1 : ( second < first ) ? 1 : 0;
278 extern GraphTreeModel* scene_graph_get_tree_model();
279 void AttachEntityTreeModel(){
280 getEntityList().m_tree_model = ui::TreeModel::from(scene_graph_get_tree_model());
282 gtk_tree_view_set_model( getEntityList().m_tree_view, getEntityList().m_tree_model );
285 void DetachEntityTreeModel(){
286 getEntityList().m_tree_model = ui::TreeModel(ui::null);
288 gtk_tree_view_set_model( getEntityList().m_tree_view, 0 );
291 void EntityList_constructWindow( ui::Window main_window ){
292 ASSERT_TRUE( !getEntityList().m_window );
294 auto window = ui::Window(create_persistent_floating_window( "Entity List", main_window ));
296 window.add_accel_group(global_accel);
298 getEntityList().m_positionTracker.connect( window );
301 getEntityList().m_window = window;
304 ui::VBox vbox = ui::VBox( FALSE, 0 );
305 gtk_container_set_border_width( GTK_CONTAINER( vbox ), 0 );
309 auto scr = create_scrolled_window( ui::Policy::AUTOMATIC, ui::Policy::AUTOMATIC );
310 vbox.pack_start( scr, TRUE, TRUE, 0 );
313 auto view = ui::TreeView(ui::New);
314 gtk_tree_view_set_headers_visible(view, FALSE );
316 auto renderer = ui::CellRendererText(ui::New);
317 auto column = gtk_tree_view_column_new();
318 gtk_tree_view_column_pack_start( column, renderer, TRUE );
319 gtk_tree_view_column_set_cell_data_func(column, renderer, reinterpret_cast<GtkTreeCellDataFunc>(entitylist_treeviewcolumn_celldatafunc), 0, 0 );
321 auto select = gtk_tree_view_get_selection(view );
322 gtk_tree_selection_set_mode( select, GTK_SELECTION_MULTIPLE );
324 view.connect( "row_expanded", G_CALLBACK( entitylist_treeview_row_expanded ), 0 );
325 view.connect( "row_collapsed", G_CALLBACK( entitylist_treeview_rowcollapsed ), 0 );
327 gtk_tree_view_append_column(view, column );
331 getEntityList().m_tree_view = view;
334 ui::Widget check = ui::Widget::from( gtk_check_button_new_with_label( "Focus on Selected" ) );
335 gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON( check ), FALSE );
337 vbox.pack_start( check, FALSE, FALSE, 0 );
338 getEntityList().m_check = check;
342 EntityList_ConnectSignals( getEntityList().m_tree_view );
343 AttachEntityTreeModel();
346 void EntityList_destroyWindow(){
347 DetachEntityTreeModel();
348 EntityList_DisconnectSignals( getEntityList().m_tree_view );
349 destroy_floating_window( getEntityList().m_window );
352 #include "preferencesystem.h"
354 #include "iselection.h"
358 scene::Node* nullNode = 0;
361 class NullSelectedInstance : public scene::Instance, public Selectable
365 InstanceTypeCastTable m_casts;
368 InstanceStaticCast<NullSelectedInstance, Selectable>::install( m_casts );
370 InstanceTypeCastTable& get(){
376 typedef LazyStatic<TypeCasts> StaticTypeCasts;
378 NullSelectedInstance() : Instance( scene::Path( makeReference( *nullNode ) ), 0, this, StaticTypeCasts::instance().get() ){
381 void setSelected( bool select ){
382 ERROR_MESSAGE( "error" );
384 bool isSelected() const {
389 typedef LazyStatic<NullSelectedInstance> StaticNullSelectedInstance;
392 void EntityList_Construct(){
393 graph_tree_model_insert( scene_graph_get_tree_model(), StaticNullSelectedInstance::instance() );
395 g_EntityList = new EntityList;
397 getEntityList().m_positionTracker.setPosition( c_default_window_pos );
399 GlobalPreferenceSystem().registerPreference( "EntityInfoDlg", make_property<WindowPositionTracker_String>( getEntityList().m_positionTracker ) );
401 typedef FreeCaller<void(const Selectable&), EntityList_SelectionChanged> EntityListSelectionChangedCaller;
402 GlobalSelectionSystem().addSelectionChangeCallback( EntityListSelectionChangedCaller() );
404 void EntityList_Destroy(){
407 graph_tree_model_erase( scene_graph_get_tree_model(), StaticNullSelectedInstance::instance() );