/* Copyright (C) 1999-2006 Id Software, Inc. and contributors. For a list of contributors, see the accompanying CONTRIBUTORS file. 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 */ // // Main Window for Q3Radiant // // Leonardo Zide (leo@lokigames.com) // #include "mainframe.h" #include "debugging/debugging.h" #include "version.h" #include "ifilesystem.h" #include "iundo.h" #include "ifilter.h" #include "itoolbar.h" #include "editable.h" #include "ientity.h" #include "ishaders.h" #include "igl.h" #include "moduleobserver.h" #include #include #include #include "commandlib.h" #include "scenelib.h" #include "stream/stringstream.h" #include "signal/isignal.h" #include "os/path.h" #include "os/file.h" #include "eclasslib.h" #include "moduleobservers.h" #include "gtkutil/clipboard.h" #include "gtkutil/container.h" #include "gtkutil/frame.h" #include "gtkutil/glfont.h" #include "gtkutil/glwidget.h" #include "gtkutil/image.h" #include "gtkutil/menu.h" #include "gtkutil/paned.h" #include "gtkutil/toolbar.h" #include "gtkutil/widget.h" #include "autosave.h" #include "build.h" #include "brushmanip.h" #include "brushmodule.h" #include "camwindow.h" #include "csg.h" #include "commands.h" #include "console.h" #include "entity.h" #include "entityinspector.h" #include "entitylist.h" #include "filters.h" #include "findtexturedialog.h" #include "grid.h" #include "groupdialog.h" #include "gtkdlgs.h" #include "gtkmisc.h" #include "help.h" #include "map.h" #include "mru.h" #include "multimon.h" #include "patchdialog.h" #include "patchmanip.h" #include "plugin.h" #include "pluginmanager.h" #include "pluginmenu.h" #include "plugintoolbar.h" #include "points.h" #include "preferences.h" #include "qe3.h" #include "qgl.h" #include "select.h" #include "server.h" #include "surfacedialog.h" #include "textures.h" #include "texwindow.h" #include "modelwindow.h" #include "url.h" #include "xywindow.h" #include "windowobservers.h" #include "renderstate.h" #include "feedback.h" #include "referencecache.h" #include "filterbar.h" #include "gtktheme.h" struct layout_globals_t { WindowPosition m_position; int nXYHeight; int nXYWidth; int nCamWidth; int nCamHeight; int nState; layout_globals_t() : m_position( -1, -1, 962, 480 ), nXYHeight( 377 ), nXYWidth( 480 ), nCamWidth( 480 ), nCamHeight( 250 ), nState( 0 ){ } }; layout_globals_t g_layout_globals; // VFS class VFSModuleObserver : public ModuleObserver { std::size_t m_unrealised; public: VFSModuleObserver() : m_unrealised( 1 ){ } void realise(){ if ( --m_unrealised == 0 ) { QE_InitVFS(); GlobalFileSystem().initialise(); } } void unrealise(){ if ( ++m_unrealised == 1 ) { GlobalFileSystem().shutdown(); } } }; VFSModuleObserver g_VFSModuleObserver; void VFS_Construct(){ Radiant_attachHomePathsObserver( g_VFSModuleObserver ); } void VFS_Destroy(){ Radiant_detachHomePathsObserver( g_VFSModuleObserver ); } // Home Paths #ifdef WIN32 #include #include const GUID qFOLDERID_SavedGames = {0x4C5C32FF, 0xBB9D, 0x43b0, {0xB5, 0xB4, 0x2D, 0x72, 0xE5, 0x4E, 0xAA, 0xA4}}; #define qREFKNOWNFOLDERID GUID #define qKF_FLAG_CREATE 0x8000 #define qKF_FLAG_NO_ALIAS 0x1000 typedef HRESULT ( WINAPI qSHGetKnownFolderPath_t )( qREFKNOWNFOLDERID rfid, DWORD dwFlags, HANDLE hToken, PWSTR *ppszPath ); static qSHGetKnownFolderPath_t *qSHGetKnownFolderPath; #endif void HomePaths_Realise(){ do { const char* prefix = g_pGameDescription->getKeyValue( "prefix" ); if ( !string_empty( prefix ) ) { StringOutputStream path( 256 ); #if defined( __APPLE__ ) path.clear(); path << DirectoryCleaned( g_get_home_dir() ) << "Library/Application Support" << ( prefix + 1 ) << "/"; if ( file_is_directory( path.c_str() ) ) { g_qeglobals.m_userEnginePath = path.c_str(); break; } path.clear(); path << DirectoryCleaned( g_get_home_dir() ) << prefix << "/"; #endif #if defined( WIN32 ) TCHAR mydocsdir[MAX_PATH + 1]; wchar_t *mydocsdirw; HMODULE shfolder = LoadLibrary( "shfolder.dll" ); if ( shfolder ) { qSHGetKnownFolderPath = (qSHGetKnownFolderPath_t *) GetProcAddress( shfolder, "SHGetKnownFolderPath" ); } else{ qSHGetKnownFolderPath = NULL; } CoInitializeEx( NULL, COINIT_APARTMENTTHREADED ); if ( qSHGetKnownFolderPath && qSHGetKnownFolderPath( qFOLDERID_SavedGames, qKF_FLAG_CREATE | qKF_FLAG_NO_ALIAS, NULL, &mydocsdirw ) == S_OK ) { memset( mydocsdir, 0, sizeof( mydocsdir ) ); wcstombs( mydocsdir, mydocsdirw, sizeof( mydocsdir ) - 1 ); CoTaskMemFree( mydocsdirw ); path.clear(); path << DirectoryCleaned( mydocsdir ) << ( prefix + 1 ) << "/"; if ( file_is_directory( path.c_str() ) ) { g_qeglobals.m_userEnginePath = path.c_str(); CoUninitialize(); FreeLibrary( shfolder ); break; } } CoUninitialize(); if ( shfolder ) { FreeLibrary( shfolder ); } if ( SUCCEEDED( SHGetFolderPath( NULL, CSIDL_PERSONAL, NULL, 0, mydocsdir ) ) ) { path.clear(); path << DirectoryCleaned( mydocsdir ) << "My Games/" << ( prefix + 1 ) << "/"; // win32: only add it if it already exists if ( file_is_directory( path.c_str() ) ) { g_qeglobals.m_userEnginePath = path.c_str(); break; } } #endif #if defined( POSIX ) path.clear(); path << DirectoryCleaned( g_get_home_dir() ) << prefix << "/"; g_qeglobals.m_userEnginePath = path.c_str(); break; #endif } g_qeglobals.m_userEnginePath = EnginePath_get(); } while ( 0 ); Q_mkdir( g_qeglobals.m_userEnginePath.c_str() ); { StringOutputStream path( 256 ); path << g_qeglobals.m_userEnginePath << gamename_get() << '/'; g_qeglobals.m_userGamePath = path.c_str(); } ASSERT_MESSAGE( !g_qeglobals.m_userGamePath.empty(), "HomePaths_Realise: user-game-path is empty" ); Q_mkdir( g_qeglobals.m_userGamePath.c_str() ); } ModuleObservers g_homePathObservers; void Radiant_attachHomePathsObserver( ModuleObserver& observer ){ g_homePathObservers.attach( observer ); } void Radiant_detachHomePathsObserver( ModuleObserver& observer ){ g_homePathObservers.detach( observer ); } class HomePathsModuleObserver : public ModuleObserver { std::size_t m_unrealised; public: HomePathsModuleObserver() : m_unrealised( 1 ){ } void realise(){ if ( --m_unrealised == 0 ) { HomePaths_Realise(); g_homePathObservers.realise(); } } void unrealise(){ if ( ++m_unrealised == 1 ) { g_homePathObservers.unrealise(); } } }; HomePathsModuleObserver g_HomePathsModuleObserver; void HomePaths_Construct(){ Radiant_attachEnginePathObserver( g_HomePathsModuleObserver ); } void HomePaths_Destroy(){ Radiant_detachEnginePathObserver( g_HomePathsModuleObserver ); } // Engine Path CopiedString g_strEnginePath; ModuleObservers g_enginePathObservers; std::size_t g_enginepath_unrealised = 1; void Radiant_attachEnginePathObserver( ModuleObserver& observer ){ g_enginePathObservers.attach( observer ); } void Radiant_detachEnginePathObserver( ModuleObserver& observer ){ g_enginePathObservers.detach( observer ); } void EnginePath_Realise(){ if ( --g_enginepath_unrealised == 0 ) { g_enginePathObservers.realise(); } } const char* EnginePath_get(){ ASSERT_MESSAGE( g_enginepath_unrealised == 0, "EnginePath_get: engine path not realised" ); return g_strEnginePath.c_str(); } void EnginePath_Unrealise(){ if ( ++g_enginepath_unrealised == 1 ) { g_enginePathObservers.unrealise(); } } void setEnginePath( CopiedString& self, const char* value ){ const auto buffer = StringOutputStream( 256 )( DirectoryCleaned( value ) ); if ( !path_equal( buffer.c_str(), self.c_str() ) ) { #if 0 while ( !ConfirmModified( "Paths Changed" ) ) { if ( Map_Unnamed( g_map ) ) { Map_SaveAs(); } else { Map_Save(); } } Map_RegionOff(); #endif ScopeDisableScreenUpdates disableScreenUpdates( "Processing...", "Changing Engine Path" ); EnginePath_Unrealise(); self = buffer.c_str(); EnginePath_Realise(); } } typedef ReferenceCaller1 EnginePathImportCaller; // Extra Resource Path CopiedString g_strExtraResourcePath; const char* ExtraResourcePath_get(){ return g_strExtraResourcePath.c_str(); } // App Path CopiedString g_strAppPath; ///< holds the full path of the executable const char* AppPath_get(){ return g_strAppPath.c_str(); } /// the path to the local rc-dir const char* LocalRcPath_get(){ static CopiedString rc_path; if ( rc_path.empty() ) { StringOutputStream stream( 256 ); stream << GlobalRadiant().getSettingsPath() << g_pGameDescription->mGameFile << "/"; rc_path = stream.c_str(); } return rc_path.c_str(); } /// directory for temp files /// NOTE: on *nix this is were we check for .pid CopiedString g_strSettingsPath; const char* SettingsPath_get(){ return g_strSettingsPath.c_str(); } /*! points to the game tools directory, for instance C:/Program Files/Quake III Arena/GtkRadiant (or other games) this is one of the main variables that are configured by the game selection on startup [GameToolsPath]/plugins [GameToolsPath]/modules and also q3map, bspc */ CopiedString g_strGameToolsPath; ///< this is set by g_GamesDialog const char* GameToolsPath_get(){ return g_strGameToolsPath.c_str(); } void Paths_constructPreferences( PreferencesPage& page ){ page.appendPathEntry( "Engine Path", true, StringImportCallback( EnginePathImportCaller( g_strEnginePath ) ), StringExportCallback( StringExportCaller( g_strEnginePath ) ) ); } void Paths_constructPage( PreferenceGroup& group ){ PreferencesPage page( group.createPage( "Paths", "Path Settings" ) ); Paths_constructPreferences( page ); page.appendPathEntry( "Extra Resource Path", true, StringImportCallback( EnginePathImportCaller( g_strExtraResourcePath ) ), StringExportCallback( StringExportCaller( g_strExtraResourcePath ) ) ); } void Paths_registerPreferencesPage(){ PreferencesDialog_addGamePage( FreeCaller1() ); } class PathsDialog : public Dialog { public: GtkWindow* BuildDialog(){ GtkFrame* frame = create_dialog_frame( "Path settings", GTK_SHADOW_ETCHED_IN ); GtkVBox* vbox2 = create_dialog_vbox( 0, 4 ); gtk_container_add( GTK_CONTAINER( frame ), GTK_WIDGET( vbox2 ) ); const char* engine; #if defined( WIN32 ) engine = g_pGameDescription->getRequiredKeyValue( "engine_win32" ); #elif defined( __linux__ ) || defined ( __FreeBSD__ ) engine = g_pGameDescription->getRequiredKeyValue( "engine_linux" ); #elif defined( __APPLE__ ) engine = g_pGameDescription->getRequiredKeyValue( "engine_macos" ); #else #error "unsupported platform" #endif StringOutputStream text( 256 ); text << "Select directory, where game executable sits (typically \"" << engine << "\")\n"; GtkLabel* label = GTK_LABEL( gtk_label_new( text.c_str() ) ); gtk_widget_show( GTK_WIDGET( label ) ); gtk_container_add( GTK_CONTAINER( vbox2 ), GTK_WIDGET( label ) ); { PreferencesPage preferencesPage( *this, GTK_WIDGET( vbox2 ) ); Paths_constructPreferences( preferencesPage ); } return create_simple_modal_dialog_window( "Engine Path Configuration", m_modal, GTK_WIDGET( frame ) ); } }; PathsDialog g_PathsDialog; bool g_strEnginePath_was_empty_1st_start = false; void EnginePath_verify(){ if ( !file_exists( g_strEnginePath.c_str() ) || g_strEnginePath_was_empty_1st_start ) { g_PathsDialog.Create(); g_PathsDialog.DoModal(); g_PathsDialog.Destroy(); } } namespace { CopiedString g_gamename; CopiedString g_gamemode; ModuleObservers g_gameNameObservers; ModuleObservers g_gameModeObservers; } void Radiant_attachGameNameObserver( ModuleObserver& observer ){ g_gameNameObservers.attach( observer ); } void Radiant_detachGameNameObserver( ModuleObserver& observer ){ g_gameNameObservers.detach( observer ); } const char* basegame_get(){ return g_pGameDescription->getRequiredKeyValue( "basegame" ); } const char* gamename_get(){ if ( g_gamename.empty() ) { return basegame_get(); } return g_gamename.c_str(); } void gamename_set( const char* gamename ){ if ( !string_equal( gamename, g_gamename.c_str() ) ) { g_gameNameObservers.unrealise(); g_gamename = gamename; g_gameNameObservers.realise(); } } void Radiant_attachGameModeObserver( ModuleObserver& observer ){ g_gameModeObservers.attach( observer ); } void Radiant_detachGameModeObserver( ModuleObserver& observer ){ g_gameModeObservers.detach( observer ); } const char* gamemode_get(){ return g_gamemode.c_str(); } void gamemode_set( const char* gamemode ){ if ( !string_equal( gamemode, g_gamemode.c_str() ) ) { g_gameModeObservers.unrealise(); g_gamemode = gamemode; g_gameModeObservers.realise(); } } #include "os/dir.h" class CLoadModule { const char* m_path; public: CLoadModule( const char* path ) : m_path( path ){ } void operator()( const char* name ) const { char fullname[1024]; ASSERT_MESSAGE( strlen( m_path ) + strlen( name ) < 1024, "" ); strcpy( fullname, m_path ); strcat( fullname, name ); globalOutputStream() << "Found '" << fullname << "'\n"; GlobalModuleServer_loadModule( fullname ); } }; const char* const c_library_extension = #if defined( WIN32 ) "dll" #elif defined ( __APPLE__ ) "dylib" #elif defined( __linux__ ) || defined ( __FreeBSD__ ) "so" #endif ; void Radiant_loadModules( const char* path ){ Directory_forEach( path, MatchFileExtension( c_library_extension, CLoadModule( path ) ) ); } void Radiant_loadModulesFromRoot( const char* directory ){ { StringOutputStream path( 256 ); path << directory << g_pluginsDir; Radiant_loadModules( path.c_str() ); } if ( !string_equal( g_pluginsDir, g_modulesDir ) ) { StringOutputStream path( 256 ); path << directory << g_modulesDir; Radiant_loadModules( path.c_str() ); } } //! Make COLOR_BRUSHES override worldspawn eclass colour. void SetWorldspawnColour( const Vector3& colour ){ EntityClass* worldspawn = GlobalEntityClassManager().findOrInsert( "worldspawn", true ); eclass_release_state( worldspawn ); worldspawn->color = colour; eclass_capture_state( worldspawn ); } class WorldspawnColourEntityClassObserver : public ModuleObserver { std::size_t m_unrealised; public: WorldspawnColourEntityClassObserver() : m_unrealised( 1 ){ } void realise(){ if ( --m_unrealised == 0 ) { SetWorldspawnColour( g_xywindow_globals.color_brushes ); } } void unrealise(){ if ( ++m_unrealised == 1 ) { } } }; WorldspawnColourEntityClassObserver g_WorldspawnColourEntityClassObserver; ModuleObservers g_gameToolsPathObservers; void Radiant_attachGameToolsPathObserver( ModuleObserver& observer ){ g_gameToolsPathObservers.attach( observer ); } void Radiant_detachGameToolsPathObserver( ModuleObserver& observer ){ g_gameToolsPathObservers.detach( observer ); } void Radiant_Initialise(){ GlobalModuleServer_Initialise(); Radiant_loadModulesFromRoot( AppPath_get() ); Preferences_Load(); bool success = Radiant_Construct( GlobalModuleServer_get() ); ASSERT_MESSAGE( success, "module system failed to initialise - see radiant.log for error messages" ); g_gameToolsPathObservers.realise(); g_gameModeObservers.realise(); g_gameNameObservers.realise(); } void Radiant_Shutdown(){ g_gameNameObservers.unrealise(); g_gameModeObservers.unrealise(); g_gameToolsPathObservers.unrealise(); if ( !g_preferences_globals.disable_ini ) { globalOutputStream() << "Start writing prefs\n"; Preferences_Save(); globalOutputStream() << "Done prefs\n"; } Radiant_Destroy(); GlobalModuleServer_Shutdown(); } void Exit(){ if ( ConfirmModified( "Exit Radiant" ) ) { gtk_main_quit(); } } #include "environment.h" #ifdef WIN32 #include #else #include /* According to the Single Unix Specification, environ is not * in any system header, although unistd.h often declares it. */ extern char **environ; #endif void Radiant_Restart(){ ConfirmModified( "Restart Radiant" ); // user can choose to not save, it's ok StringOutputStream mapname; mapname << "\"" << Map_Name( g_map ) << "\""; char *argv[] = { string_clone( environment_get_app_filepath() ), Map_Unnamed( g_map )? NULL : string_clone( mapname.c_str() ), NULL }; #ifdef WIN32 const int status = !_spawnv( P_NOWAIT, argv[0], argv ); #else const int status = posix_spawn( NULL, argv[0], NULL, NULL, argv, environ ); #endif // quit if radiant successfully started if ( status == 0 ) { gtk_main_quit(); } } void Undo(){ GlobalUndoSystem().undo(); SceneChangeNotify(); } void Redo(){ GlobalUndoSystem().redo(); SceneChangeNotify(); } void deleteSelection(){ if( GlobalSelectionSystem().Mode() == SelectionSystem::eComponent && GlobalSelectionSystem().countSelectedComponents() != 0 ){ UndoableCommand undo( "deleteSelectedComponents" ); CSG_DeleteComponents(); } else{ UndoableCommand undo( "deleteSelected" ); Select_Delete(); } } void Map_ExportSelected( TextOutputStream& ostream ){ Map_ExportSelected( ostream, Map_getFormat( g_map ) ); } void Map_ImportSelected( TextInputStream& istream ){ Map_ImportSelected( istream, Map_getFormat( g_map ) ); } void Selection_Copy(){ clipboard_copy( Map_ExportSelected ); } void Selection_Paste(){ clipboard_paste( Map_ImportSelected ); } void Copy(){ if ( SelectedFaces_empty() ) { Selection_Copy(); } else { SelectedFaces_copyTexture(); } } void Paste(){ if ( SelectedFaces_empty() ) { UndoableCommand undo( "paste" ); GlobalSelectionSystem().setSelectedAll( false ); Selection_Paste(); } else { SelectedFaces_pasteTexture(); } } void TranslateToCamera(){ CamWnd& camwnd = *g_pParentWnd->GetCamWnd(); GlobalSelectionSystem().translateSelected( vector3_snapped( Camera_getOrigin( camwnd ) - GlobalSelectionSystem().getBoundsSelected().origin, GetSnapGridSize() ) ); } void PasteToCamera(){ GlobalSelectionSystem().setSelectedAll( false ); UndoableCommand undo( "pasteToCamera" ); Selection_Paste(); TranslateToCamera(); } void MoveToCamera(){ UndoableCommand undo( "moveToCamera" ); TranslateToCamera(); } void ColorScheme_Original(){ TextureBrowser_setBackgroundColour( GlobalTextureBrowser(), Vector3( 0.25f, 0.25f, 0.25f ) ); g_camwindow_globals.color_selbrushes3d = Vector3( 1.0f, 0.0f, 0.0f ); g_camwindow_globals.color_cameraback = Vector3( 0.25f, 0.25f, 0.25f ); CamWnd_reconstructStatic(); CamWnd_Update( *g_pParentWnd->GetCamWnd() ); g_xywindow_globals.color_gridback = Vector3( 1.0f, 1.0f, 1.0f ); g_xywindow_globals.color_gridminor = Vector3( 0.75f, 0.75f, 0.75f ); g_xywindow_globals.color_gridmajor = Vector3( 0.5f, 0.5f, 0.5f ); g_xywindow_globals.color_gridblock = Vector3( 0.0f, 0.0f, 1.0f ); g_xywindow_globals.color_gridtext = Vector3( 0.0f, 0.0f, 0.0f ); g_xywindow_globals.color_selbrushes = Vector3( 1.0f, 0.0f, 0.0f ); XYWnd::recaptureStates(); g_xywindow_globals.color_clipper = Vector3( 0.0f, 0.0f, 1.0f ); Brush_clipperColourChanged(); g_xywindow_globals.color_brushes = Vector3( 0.0f, 0.0f, 0.0f ); SetWorldspawnColour( g_xywindow_globals.color_brushes ); g_xywindow_globals.color_viewname = Vector3( 0.5f, 0.0f, 0.75f ); XY_UpdateAllWindows(); } void ColorScheme_QER(){ TextureBrowser_setBackgroundColour( GlobalTextureBrowser(), Vector3( 0.25f, 0.25f, 0.25f ) ); g_camwindow_globals.color_cameraback = Vector3( 0.25f, 0.25f, 0.25f ); g_camwindow_globals.color_selbrushes3d = Vector3( 1.0f, 0.0f, 0.0f ); CamWnd_reconstructStatic(); CamWnd_Update( *g_pParentWnd->GetCamWnd() ); g_xywindow_globals.color_gridback = Vector3( 1.0f, 1.0f, 1.0f ); g_xywindow_globals.color_gridminor = Vector3( 1.0f, 1.0f, 1.0f ); g_xywindow_globals.color_gridmajor = Vector3( 0.5f, 0.5f, 0.5f ); g_xywindow_globals.color_gridblock = Vector3( 0.0f, 0.0f, 1.0f ); g_xywindow_globals.color_gridtext = Vector3( 0.0f, 0.0f, 0.0f ); g_xywindow_globals.color_selbrushes = Vector3( 1.0f, 0.0f, 0.0f ); XYWnd::recaptureStates(); g_xywindow_globals.color_clipper = Vector3( 0.0f, 0.0f, 1.0f ); Brush_clipperColourChanged(); g_xywindow_globals.color_brushes = Vector3( 0.0f, 0.0f, 0.0f ); SetWorldspawnColour( g_xywindow_globals.color_brushes ); g_xywindow_globals.color_viewname = Vector3( 0.5f, 0.0f, 0.75f ); XY_UpdateAllWindows(); } void ColorScheme_Black(){ TextureBrowser_setBackgroundColour( GlobalTextureBrowser(), Vector3( 0.25f, 0.25f, 0.25f ) ); g_camwindow_globals.color_cameraback = Vector3( 0.25f, 0.25f, 0.25f ); g_camwindow_globals.color_selbrushes3d = Vector3( 1.0f, 0.0f, 0.0f ); CamWnd_reconstructStatic(); CamWnd_Update( *g_pParentWnd->GetCamWnd() ); g_xywindow_globals.color_gridback = Vector3( 0.0f, 0.0f, 0.0f ); g_xywindow_globals.color_gridminor = Vector3( 0.2f, 0.2f, 0.2f ); g_xywindow_globals.color_gridmajor = Vector3( 0.3f, 0.5f, 0.5f ); g_xywindow_globals.color_gridblock = Vector3( 0.0f, 0.0f, 1.0f ); g_xywindow_globals.color_gridtext = Vector3( 1.0f, 1.0f, 1.0f ); g_xywindow_globals.color_selbrushes = Vector3( 1.0f, 0.0f, 0.0f ); XYWnd::recaptureStates(); g_xywindow_globals.color_clipper = Vector3( 0.0f, 0.0f, 1.0f ); Brush_clipperColourChanged(); g_xywindow_globals.color_brushes = Vector3( 1.0f, 1.0f, 1.0f ); SetWorldspawnColour( g_xywindow_globals.color_brushes ); g_xywindow_globals.color_viewname = Vector3( 0.7f, 0.7f, 0.0f ); XY_UpdateAllWindows(); } /* ydnar: to emulate maya/max/lightwave color schemes */ void ColorScheme_Ydnar(){ TextureBrowser_setBackgroundColour( GlobalTextureBrowser(), Vector3( 0.25f, 0.25f, 0.25f ) ); g_camwindow_globals.color_cameraback = Vector3( 0.25f, 0.25f, 0.25f ); g_camwindow_globals.color_selbrushes3d = Vector3( 1.0f, 0.0f, 0.0f ); CamWnd_reconstructStatic(); CamWnd_Update( *g_pParentWnd->GetCamWnd() ); g_xywindow_globals.color_gridback = Vector3( 0.77f, 0.77f, 0.77f ); g_xywindow_globals.color_gridminor = Vector3( 0.83f, 0.83f, 0.83f ); g_xywindow_globals.color_gridmajor = Vector3( 0.89f, 0.89f, 0.89f ); g_xywindow_globals.color_gridblock = Vector3( 1.0f, 1.0f, 1.0f ); g_xywindow_globals.color_gridtext = Vector3( 0.0f, 0.0f, 0.0f ); g_xywindow_globals.color_selbrushes = Vector3( 1.0f, 0.0f, 0.0f ); XYWnd::recaptureStates(); g_xywindow_globals.color_clipper = Vector3( 0.0f, 0.0f, 1.0f ); Brush_clipperColourChanged(); g_xywindow_globals.color_brushes = Vector3( 0.0f, 0.0f, 0.0f ); SetWorldspawnColour( g_xywindow_globals.color_brushes ); g_xywindow_globals.color_viewname = Vector3( 0.5f, 0.0f, 0.75f ); XY_UpdateAllWindows(); } void ColorScheme_Blender(){ TextureBrowser_setBackgroundColour( GlobalTextureBrowser(), Vector3( 0.25f, 0.25f, 0.25f ) ); g_camwindow_globals.color_cameraback = Vector3( 0.25f, 0.25f, 0.25f ); g_camwindow_globals.color_selbrushes3d = Vector3( 1.0f, 0.627451f, 0.0f ); CamWnd_reconstructStatic(); CamWnd_Update( *g_pParentWnd->GetCamWnd() ); g_xywindow_globals.color_gridback = Vector3( .225803f, .225803f, .225803f ); g_xywindow_globals.color_gridminor = Vector3( .254902f, .254902f, .254902f ); g_xywindow_globals.color_gridmajor = Vector3( .294118f, .294118f, .294118f ); g_xywindow_globals.color_gridblock = Vector3( 1.0f, 1.0f, 1.0f ); g_xywindow_globals.color_gridtext = Vector3( .972549f, .972549f, .972549f ); g_xywindow_globals.color_selbrushes = Vector3( 1.0f, 0.627451f, 0.0f ); XYWnd::recaptureStates(); g_xywindow_globals.color_clipper = Vector3( 0.0f, 0.0f, 1.0f ); Brush_clipperColourChanged(); g_xywindow_globals.color_brushes = Vector3( 0.0f, 0.0f, 0.0f ); SetWorldspawnColour( g_xywindow_globals.color_brushes ); g_xywindow_globals.color_viewname = Vector3( 0.516136f, 0.516136f, 0.516136f ); XY_UpdateAllWindows(); } /* color scheme to fit the GTK Adwaita Dark theme */ void ColorScheme_AdwaitaDark() { TextureBrowser_setBackgroundColour( GlobalTextureBrowser(), Vector3( 0.25f, 0.25f, 0.25f ) ); g_camwindow_globals.color_cameraback = Vector3( 0.25f, 0.25f, 0.25f ); g_camwindow_globals.color_selbrushes3d = Vector3( 1.0f, 0.0f, 0.0f ); CamWnd_reconstructStatic(); CamWnd_Update( *g_pParentWnd->GetCamWnd() ); g_xywindow_globals.color_gridback = Vector3( 0.25f, 0.25f, 0.25f ); g_xywindow_globals.color_gridminor = Vector3( 0.21f, 0.23f, 0.23f ); g_xywindow_globals.color_gridmajor = Vector3( 0.14f, 0.15f, 0.15f ); g_xywindow_globals.color_gridblock = Vector3( 1.0f, 1.0f, 1.0f ); g_xywindow_globals.color_gridtext = Vector3( 0.0f, 0.0f, 0.0f ); g_xywindow_globals.color_selbrushes = Vector3( 1.0f, 0.0f, 0.0f ); XYWnd::recaptureStates(); g_xywindow_globals.color_clipper = Vector3( 0.0f, 0.0f, 1.0f ); Brush_clipperColourChanged(); g_xywindow_globals.color_brushes = Vector3( 0.73f, 0.73f, 0.73f ); SetWorldspawnColour( g_xywindow_globals.color_brushes ); g_xywindow_globals.color_viewname = Vector3( 0.5f, 0.0f, 0.75f ); XY_UpdateAllWindows(); } typedef Callback1 GetColourCallback; typedef Callback1 SetColourCallback; class ChooseColour { GetColourCallback m_get; SetColourCallback m_set; public: ChooseColour( const GetColourCallback& get, const SetColourCallback& set ) : m_get( get ), m_set( set ){ } void operator()(){ Vector3 colour; m_get( colour ); color_dialog( GTK_WIDGET( MainFrame_getWindow() ), colour ); m_set( colour ); } }; void Colour_get( const Vector3& colour, Vector3& other ){ other = colour; } typedef ConstReferenceCaller1 ColourGetCaller; void Colour_set( Vector3& colour, const Vector3& other ){ colour = other; SceneChangeNotify(); } typedef ReferenceCaller1 ColourSetCaller; void BrushColour_set( const Vector3& other ){ g_xywindow_globals.color_brushes = other; SetWorldspawnColour( g_xywindow_globals.color_brushes ); SceneChangeNotify(); } typedef FreeCaller1 BrushColourSetCaller; void SelectedBrushColour_set( const Vector3& other ){ g_xywindow_globals.color_selbrushes = other; XYWnd::recaptureStates(); SceneChangeNotify(); } typedef FreeCaller1 SelectedBrushColourSetCaller; void SelectedBrush3dColour_set( const Vector3& other ){ g_camwindow_globals.color_selbrushes3d = other; CamWnd_reconstructStatic(); SceneChangeNotify(); } typedef FreeCaller1 SelectedBrush3dColourSetCaller; void ClipperColour_set( const Vector3& other ){ g_xywindow_globals.color_clipper = other; Brush_clipperColourChanged(); SceneChangeNotify(); } typedef FreeCaller1 ClipperColourSetCaller; void TextureBrowserColour_get( Vector3& other ){ other = TextureBrowser_getBackgroundColour( GlobalTextureBrowser() ); } typedef FreeCaller1 TextureBrowserColourGetCaller; void TextureBrowserColour_set( const Vector3& other ){ TextureBrowser_setBackgroundColour( GlobalTextureBrowser(), other ); } typedef FreeCaller1 TextureBrowserColourSetCaller; class ColoursMenu { public: ChooseColour m_textureback; ChooseColour m_xyback; ChooseColour m_gridmajor; ChooseColour m_gridminor; ChooseColour m_gridtext; ChooseColour m_gridblock; ChooseColour m_cameraback; ChooseColour m_brush; ChooseColour m_selectedbrush; ChooseColour m_selectedbrush3d; ChooseColour m_clipper; ChooseColour m_viewname; ColoursMenu() : m_textureback( TextureBrowserColourGetCaller(), TextureBrowserColourSetCaller() ), m_xyback( ColourGetCaller( g_xywindow_globals.color_gridback ), ColourSetCaller( g_xywindow_globals.color_gridback ) ), m_gridmajor( ColourGetCaller( g_xywindow_globals.color_gridmajor ), ColourSetCaller( g_xywindow_globals.color_gridmajor ) ), m_gridminor( ColourGetCaller( g_xywindow_globals.color_gridminor ), ColourSetCaller( g_xywindow_globals.color_gridminor ) ), m_gridtext( ColourGetCaller( g_xywindow_globals.color_gridtext ), ColourSetCaller( g_xywindow_globals.color_gridtext ) ), m_gridblock( ColourGetCaller( g_xywindow_globals.color_gridblock ), ColourSetCaller( g_xywindow_globals.color_gridblock ) ), m_cameraback( ColourGetCaller( g_camwindow_globals.color_cameraback ), ColourSetCaller( g_camwindow_globals.color_cameraback ) ), m_brush( ColourGetCaller( g_xywindow_globals.color_brushes ), BrushColourSetCaller() ), m_selectedbrush( ColourGetCaller( g_xywindow_globals.color_selbrushes ), SelectedBrushColourSetCaller() ), m_selectedbrush3d( ColourGetCaller( g_camwindow_globals.color_selbrushes3d ), SelectedBrush3dColourSetCaller() ), m_clipper( ColourGetCaller( g_xywindow_globals.color_clipper ), ClipperColourSetCaller() ), m_viewname( ColourGetCaller( g_xywindow_globals.color_viewname ), ColourSetCaller( g_xywindow_globals.color_viewname ) ){ } }; ColoursMenu g_ColoursMenu; GtkMenuItem* create_colours_menu(){ GtkMenuItem* colours_menu_item = new_sub_menu_item_with_mnemonic( "Colors" ); GtkMenu* menu_in_menu = GTK_MENU( gtk_menu_item_get_submenu( colours_menu_item ) ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu_in_menu ); } GtkMenu* menu_3 = create_sub_menu_with_mnemonic( menu_in_menu, "Themes" ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu_3 ); } create_menu_item_with_mnemonic( menu_3, "QE4 Original", "ColorSchemeOriginal" ); create_menu_item_with_mnemonic( menu_3, "Q3Radiant Original", "ColorSchemeQER" ); create_menu_item_with_mnemonic( menu_3, "Black and Green", "ColorSchemeBlackAndGreen" ); create_menu_item_with_mnemonic( menu_3, "Maya/Max/Lightwave Emulation", "ColorSchemeYdnar" ); create_menu_item_with_mnemonic( menu_3, "Blender/Dark", "ColorSchemeBlender" ); create_menu_item_with_mnemonic( menu_3, "Adwaita Dark", "ColorSchemeAdwaitaDark" ); create_menu_item_with_mnemonic( menu_in_menu, "GTK Theme...", "gtkThemeDlg" ); create_menu_item_with_mnemonic( menu_in_menu, "OpenGL Font...", "OpenGLFont" ); menu_separator( menu_in_menu ); create_menu_item_with_mnemonic( menu_in_menu, "_Texture Background...", "ChooseTextureBackgroundColor" ); create_menu_item_with_mnemonic( menu_in_menu, "Camera Background...", "ChooseCameraBackgroundColor" ); create_menu_item_with_mnemonic( menu_in_menu, "Grid Background...", "ChooseGridBackgroundColor" ); create_menu_item_with_mnemonic( menu_in_menu, "Grid Major...", "ChooseGridMajorColor" ); create_menu_item_with_mnemonic( menu_in_menu, "Grid Minor...", "ChooseGridMinorColor" ); create_menu_item_with_mnemonic( menu_in_menu, "Grid Text...", "ChooseGridTextColor" ); create_menu_item_with_mnemonic( menu_in_menu, "Grid Block...", "ChooseGridBlockColor" ); create_menu_item_with_mnemonic( menu_in_menu, "Default Brush (2D)...", "ChooseBrushColor" ); create_menu_item_with_mnemonic( menu_in_menu, "Selected Brush and Sizing (2D)...", "ChooseSelectedBrushColor" ); create_menu_item_with_mnemonic( menu_in_menu, "Selected Brush (Camera)...", "ChooseCameraSelectedBrushColor" ); create_menu_item_with_mnemonic( menu_in_menu, "Clipper...", "ChooseClipperColor" ); create_menu_item_with_mnemonic( menu_in_menu, "Active View Name and Outline...", "ChooseOrthoViewNameColor" ); return colours_menu_item; } void Restart(){ PluginsMenu_clear(); PluginToolbar_clear(); Radiant_Shutdown(); Radiant_Initialise(); PluginsMenu_populate(); PluginToolbar_populate(); } void OpenUpdateURL(){ // build the URL StringOutputStream URL( 256 ); URL << "http://www.icculus.org/netradiant/?cmd=update&data=dlupdate&query_dlup=1"; #ifdef WIN32 URL << "&OS_dlup=1"; #elif defined( __APPLE__ ) URL << "&OS_dlup=2"; #else URL << "&OS_dlup=3"; #endif URL << "&Version_dlup=" RADIANT_VERSION; g_GamesDialog.AddPacksURL( URL ); OpenURL( URL.c_str() ); } // open the Q3Rad manual void OpenHelpURL(){ // at least on win32, AppPath + "docs/index.html" StringOutputStream help( 256 ); help << AppPath_get() << "docs/index.html"; OpenURL( help.c_str() ); } void OpenBugReportURL(){ OpenURL( "http://www.icculus.org/netradiant/?cmd=bugs" ); } GtkWidget* g_page_console; void Console_ToggleShow(){ GroupDialog_showPage( g_page_console ); } GtkWidget* g_page_entity; void EntityInspector_ToggleShow(){ GroupDialog_showPage( g_page_entity ); } GtkWidget* g_page_models; void ModelBrowser_ToggleShow(){ GroupDialog_showPage( g_page_models ); } void SetClipMode( bool enable ); void ModeChangeNotify(); typedef void ( *ToolMode )(); ToolMode g_currentToolMode = 0; bool g_currentToolModeSupportsComponentEditing = false; ToolMode g_defaultToolMode = 0; void SelectionSystem_DefaultMode(){ GlobalSelectionSystem().SetMode( SelectionSystem::ePrimitive ); GlobalSelectionSystem().SetComponentMode( SelectionSystem::eDefault ); ModeChangeNotify(); } bool EdgeMode(){ return GlobalSelectionSystem().Mode() == SelectionSystem::eComponent && GlobalSelectionSystem().ComponentMode() == SelectionSystem::eEdge; } bool VertexMode(){ return GlobalSelectionSystem().Mode() == SelectionSystem::eComponent && GlobalSelectionSystem().ComponentMode() == SelectionSystem::eVertex; } bool FaceMode(){ return GlobalSelectionSystem().Mode() == SelectionSystem::eComponent && GlobalSelectionSystem().ComponentMode() == SelectionSystem::eFace; } template class BoolFunctionExport { public: static void apply( const BoolImportCallback& importCallback ){ importCallback( BoolFunction() ); } }; typedef FreeCaller1::apply> EdgeModeApplyCaller; EdgeModeApplyCaller g_edgeMode_button_caller; BoolExportCallback g_edgeMode_button_callback( g_edgeMode_button_caller ); ToggleItem g_edgeMode_button( g_edgeMode_button_callback ); typedef FreeCaller1::apply> VertexModeApplyCaller; VertexModeApplyCaller g_vertexMode_button_caller; BoolExportCallback g_vertexMode_button_callback( g_vertexMode_button_caller ); ToggleItem g_vertexMode_button( g_vertexMode_button_callback ); typedef FreeCaller1::apply> FaceModeApplyCaller; FaceModeApplyCaller g_faceMode_button_caller; BoolExportCallback g_faceMode_button_callback( g_faceMode_button_caller ); ToggleItem g_faceMode_button( g_faceMode_button_callback ); void ComponentModeChanged(){ g_edgeMode_button.update(); g_vertexMode_button.update(); g_faceMode_button.update(); } void ComponentMode_SelectionChanged( const Selectable& selectable ){ if ( GlobalSelectionSystem().Mode() == SelectionSystem::eComponent && GlobalSelectionSystem().countSelected() == 0 ) { SelectionSystem_DefaultMode(); ComponentModeChanged(); } } void SelectEdgeMode(){ #if 0 if ( GlobalSelectionSystem().Mode() == SelectionSystem::eComponent ) { GlobalSelectionSystem().Select( false ); } #endif if ( EdgeMode() ) { SelectionSystem_DefaultMode(); } else if ( GlobalSelectionSystem().countSelected() != 0 ) { if ( !g_currentToolModeSupportsComponentEditing ) { g_defaultToolMode(); } GlobalSelectionSystem().SetMode( SelectionSystem::eComponent ); GlobalSelectionSystem().SetComponentMode( SelectionSystem::eEdge ); } ComponentModeChanged(); ModeChangeNotify(); } void SelectVertexMode(){ #if 0 if ( GlobalSelectionSystem().Mode() == SelectionSystem::eComponent ) { GlobalSelectionSystem().Select( false ); } #endif if ( VertexMode() ) { SelectionSystem_DefaultMode(); } else if ( GlobalSelectionSystem().countSelected() != 0 ) { if ( !g_currentToolModeSupportsComponentEditing ) { g_defaultToolMode(); } GlobalSelectionSystem().SetMode( SelectionSystem::eComponent ); GlobalSelectionSystem().SetComponentMode( SelectionSystem::eVertex ); } ComponentModeChanged(); ModeChangeNotify(); } void SelectFaceMode(){ #if 0 if ( GlobalSelectionSystem().Mode() == SelectionSystem::eComponent ) { GlobalSelectionSystem().Select( false ); } #endif if ( FaceMode() ) { SelectionSystem_DefaultMode(); } else if ( GlobalSelectionSystem().countSelected() != 0 ) { if ( !g_currentToolModeSupportsComponentEditing ) { g_defaultToolMode(); } GlobalSelectionSystem().SetMode( SelectionSystem::eComponent ); GlobalSelectionSystem().SetComponentMode( SelectionSystem::eFace ); } ComponentModeChanged(); ModeChangeNotify(); } class CloneSelected : public scene::Graph::Walker { const bool m_makeUnique; const scene::Node* m_world; public: mutable std::vector m_cloned; CloneSelected( bool makeUnique ) : m_makeUnique( makeUnique ), m_world( Map_FindWorldspawn( g_map ) ){ } bool pre( const scene::Path& path, scene::Instance& instance ) const { if ( path.size() == 1 ) { return true; } if ( path.top().get_pointer() == m_world ) { // ignore worldspawn, but keep checking children return true; } if ( !path.top().get().isRoot() ) { if ( Instance_isSelected( instance ) ) { return false; } if( m_makeUnique && instance.childSelected() ){ /* clone selected group entity primitives to new group entity */ return false; } } return true; } void post( const scene::Path& path, scene::Instance& instance ) const { if ( path.size() == 1 ) { return; } if ( path.top().get_pointer() == m_world ) { // ignore worldspawn return; } if ( !path.top().get().isRoot() ) { if ( Instance_isSelected( instance ) ) { NodeSmartReference clone( Node_Clone( path.top() ) ); Map_gatherNamespaced( clone ); Node_getTraversable( path.parent().get() )->insert( clone ); m_cloned.push_back( clone.get_pointer() ); } else if( m_makeUnique && instance.childSelected() ){ /* clone selected group entity primitives to new group entity */ NodeSmartReference clone( Node_Clone_Selected( path.top() ) ); Map_gatherNamespaced( clone ); Node_getTraversable( path.parent().get() )->insert( clone ); m_cloned.push_back( clone.get_pointer() ); } } } }; void Scene_Clone_Selected( scene::Graph& graph, bool makeUnique ){ CloneSelected cloneSelected( makeUnique ); graph.traverse( cloneSelected ); Map_mergeClonedNames( makeUnique ); /* deselect originals */ GlobalSelectionSystem().setSelectedAll( false ); /* select cloned */ for( scene::Node *node : cloneSelected.m_cloned ) { class walker : public scene::Traversable::Walker { public: bool pre( scene::Node& node ) const override { if( scene::Instantiable *instantiable = Node_getInstantiable( node ) ){ class visitor : public scene::Instantiable::Visitor { public: void visit( scene::Instance& instance ) const override { Instance_setSelected( instance, true ); } }; instantiable->forEachInstance( visitor() ); } return true; } }; Node_traverseSubgraph( *node, walker() ); } } enum ENudgeDirection { eNudgeUp = 1, eNudgeDown = 3, eNudgeLeft = 0, eNudgeRight = 2, }; struct AxisBase { Vector3 x; Vector3 y; Vector3 z; AxisBase( const Vector3& x_, const Vector3& y_, const Vector3& z_ ) : x( x_ ), y( y_ ), z( z_ ){ } }; AxisBase AxisBase_forViewType( VIEWTYPE viewtype ){ switch ( viewtype ) { case XY: return AxisBase( g_vector3_axis_x, g_vector3_axis_y, g_vector3_axis_z ); case XZ: return AxisBase( g_vector3_axis_x, g_vector3_axis_z, g_vector3_axis_y ); case YZ: return AxisBase( g_vector3_axis_y, g_vector3_axis_z, g_vector3_axis_x ); } ERROR_MESSAGE( "invalid viewtype" ); return AxisBase( Vector3( 0, 0, 0 ), Vector3( 0, 0, 0 ), Vector3( 0, 0, 0 ) ); } Vector3 AxisBase_axisForDirection( const AxisBase& axes, ENudgeDirection direction ){ switch ( direction ) { case eNudgeLeft: return vector3_negated( axes.x ); case eNudgeUp: return axes.y; case eNudgeRight: return axes.x; case eNudgeDown: return vector3_negated( axes.y ); } ERROR_MESSAGE( "invalid direction" ); return Vector3( 0, 0, 0 ); } bool g_bNudgeAfterClone = false; void Nudge_constructPreferences( PreferencesPage& page ){ page.appendCheckBox( "", "Nudge selected after duplication", g_bNudgeAfterClone ); } void NudgeSelection( ENudgeDirection direction, float fAmount, VIEWTYPE viewtype ){ AxisBase axes( AxisBase_forViewType( viewtype ) ); Vector3 view_direction( vector3_negated( axes.z ) ); Vector3 nudge( vector3_scaled( AxisBase_axisForDirection( axes, direction ), fAmount ) ); GlobalSelectionSystem().NudgeManipulator( nudge, view_direction ); } void Selection_Clone(){ if ( GlobalSelectionSystem().Mode() == SelectionSystem::ePrimitive ) { UndoableCommand undo( "cloneSelected" ); Scene_Clone_Selected( GlobalSceneGraph(), false ); if( g_bNudgeAfterClone ){ NudgeSelection(eNudgeRight, GetGridSize(), GlobalXYWnd_getCurrentViewType()); NudgeSelection(eNudgeDown, GetGridSize(), GlobalXYWnd_getCurrentViewType()); } } } void Selection_Clone_MakeUnique(){ if ( GlobalSelectionSystem().Mode() == SelectionSystem::ePrimitive ) { UndoableCommand undo( "cloneSelectedMakeUnique" ); Scene_Clone_Selected( GlobalSceneGraph(), true ); if( g_bNudgeAfterClone ){ NudgeSelection(eNudgeRight, GetGridSize(), GlobalXYWnd_getCurrentViewType()); NudgeSelection(eNudgeDown, GetGridSize(), GlobalXYWnd_getCurrentViewType()); } } } // called when the escape key is used (either on the main window or on an inspector) void Selection_Deselect(){ if ( GlobalSelectionSystem().Mode() == SelectionSystem::eComponent ) { if ( GlobalSelectionSystem().countSelectedComponents() != 0 ) { GlobalSelectionSystem().setSelectedAllComponents( false ); } else { SelectionSystem_DefaultMode(); ComponentModeChanged(); } } else { if ( GlobalSelectionSystem().countSelectedComponents() != 0 ) { GlobalSelectionSystem().setSelectedAllComponents( false ); } else { GlobalSelectionSystem().setSelectedAll( false ); } } } void Scene_Clone_Selected(){ Scene_Clone_Selected( GlobalSceneGraph(), false ); } void RepeatTransforms(){ GlobalSelectionSystem().repeatTransforms( FreeCaller() ); } void Selection_NudgeUp(){ UndoableCommand undo( "nudgeSelectedUp" ); NudgeSelection( eNudgeUp, GetGridSize(), GlobalXYWnd_getCurrentViewType() ); } void Selection_NudgeDown(){ UndoableCommand undo( "nudgeSelectedDown" ); NudgeSelection( eNudgeDown, GetGridSize(), GlobalXYWnd_getCurrentViewType() ); } void Selection_NudgeLeft(){ UndoableCommand undo( "nudgeSelectedLeft" ); NudgeSelection( eNudgeLeft, GetGridSize(), GlobalXYWnd_getCurrentViewType() ); } void Selection_NudgeRight(){ UndoableCommand undo( "nudgeSelectedRight" ); NudgeSelection( eNudgeRight, GetGridSize(), GlobalXYWnd_getCurrentViewType() ); } void TranslateToolExport( const BoolImportCallback& importCallback ){ importCallback( GlobalSelectionSystem().ManipulatorMode() == SelectionSystem::eTranslate ); } void RotateToolExport( const BoolImportCallback& importCallback ){ importCallback( GlobalSelectionSystem().ManipulatorMode() == SelectionSystem::eRotate ); } void ScaleToolExport( const BoolImportCallback& importCallback ){ importCallback( GlobalSelectionSystem().ManipulatorMode() == SelectionSystem::eScale ); } void SkewToolExport( const BoolImportCallback& importCallback ){ importCallback( GlobalSelectionSystem().ManipulatorMode() == SelectionSystem::eSkew ); } void DragToolExport( const BoolImportCallback& importCallback ){ importCallback( GlobalSelectionSystem().ManipulatorMode() == SelectionSystem::eDrag ); } void ClipperToolExport( const BoolImportCallback& importCallback ){ importCallback( GlobalSelectionSystem().ManipulatorMode() == SelectionSystem::eClip ); } void BuildToolExport( const BoolImportCallback& importCallback ){ importCallback( GlobalSelectionSystem().ManipulatorMode() == SelectionSystem::eBuild ); } void UVToolExport( const BoolImportCallback& importCallback ){ importCallback( GlobalSelectionSystem().ManipulatorMode() == SelectionSystem::eUV ); } FreeCaller1 g_translatemode_button_caller; BoolExportCallback g_translatemode_button_callback( g_translatemode_button_caller ); ToggleItem g_translatemode_button( g_translatemode_button_callback ); FreeCaller1 g_rotatemode_button_caller; BoolExportCallback g_rotatemode_button_callback( g_rotatemode_button_caller ); ToggleItem g_rotatemode_button( g_rotatemode_button_callback ); FreeCaller1 g_scalemode_button_caller; BoolExportCallback g_scalemode_button_callback( g_scalemode_button_caller ); ToggleItem g_scalemode_button( g_scalemode_button_callback ); FreeCaller1 g_skewmode_button_caller; BoolExportCallback g_skewmode_button_callback( g_skewmode_button_caller ); ToggleItem g_skewmode_button( g_skewmode_button_callback ); FreeCaller1 g_dragmode_button_caller; BoolExportCallback g_dragmode_button_callback( g_dragmode_button_caller ); ToggleItem g_dragmode_button( g_dragmode_button_callback ); FreeCaller1 g_clipper_button_caller; BoolExportCallback g_clipper_button_callback( g_clipper_button_caller ); ToggleItem g_clipper_button( g_clipper_button_callback ); FreeCaller1 g_build_button_caller; BoolExportCallback g_build_button_callback( g_build_button_caller ); ToggleItem g_build_button( g_build_button_callback ); FreeCaller1 g_uv_button_caller; BoolExportCallback g_uv_button_callback( g_uv_button_caller ); ToggleItem g_uv_button( g_uv_button_callback ); void ToolChanged(){ g_translatemode_button.update(); g_rotatemode_button.update(); g_scalemode_button.update(); g_skewmode_button.update(); g_dragmode_button.update(); g_clipper_button.update(); g_build_button.update(); g_uv_button.update(); } const char* const c_ResizeMode_status = "QE4 Drag Tool: move and resize objects"; void DragMode(){ if ( g_currentToolMode == DragMode && g_defaultToolMode != DragMode ) { g_defaultToolMode(); } else { g_currentToolMode = DragMode; g_currentToolModeSupportsComponentEditing = true; Sys_Status( c_ResizeMode_status ); GlobalSelectionSystem().SetManipulatorMode( SelectionSystem::eDrag ); ToolChanged(); ModeChangeNotify(); } } const char* const c_TranslateMode_status = "Translate Tool: translate objects and components"; void TranslateMode(){ if ( g_currentToolMode == TranslateMode && g_defaultToolMode != TranslateMode ) { g_defaultToolMode(); } else { g_currentToolMode = TranslateMode; g_currentToolModeSupportsComponentEditing = true; Sys_Status( c_TranslateMode_status ); GlobalSelectionSystem().SetManipulatorMode( SelectionSystem::eTranslate ); ToolChanged(); ModeChangeNotify(); } } const char* const c_RotateMode_status = "Rotate Tool: rotate objects and components"; void RotateMode(){ if ( g_currentToolMode == RotateMode && g_defaultToolMode != RotateMode ) { g_defaultToolMode(); } else { g_currentToolMode = RotateMode; g_currentToolModeSupportsComponentEditing = true; Sys_Status( c_RotateMode_status ); GlobalSelectionSystem().SetManipulatorMode( SelectionSystem::eRotate ); ToolChanged(); ModeChangeNotify(); } } const char* const c_ScaleMode_status = "Scale Tool: scale objects and components"; void ScaleMode(){ if ( g_currentToolMode == ScaleMode && g_defaultToolMode != ScaleMode ) { g_defaultToolMode(); } else { g_currentToolMode = ScaleMode; g_currentToolModeSupportsComponentEditing = true; Sys_Status( c_ScaleMode_status ); GlobalSelectionSystem().SetManipulatorMode( SelectionSystem::eScale ); ToolChanged(); ModeChangeNotify(); } } const char* const c_SkewMode_status = "Transform Tool: transform objects and components"; void SkewMode(){ if ( g_currentToolMode == SkewMode && g_defaultToolMode != SkewMode ) { g_defaultToolMode(); } else { g_currentToolMode = SkewMode; g_currentToolModeSupportsComponentEditing = true; Sys_Status( c_SkewMode_status ); GlobalSelectionSystem().SetManipulatorMode( SelectionSystem::eSkew ); ToolChanged(); ModeChangeNotify(); } } const char* const c_ClipperMode_status = "Clipper Tool: apply clip planes to brushes"; void ClipperMode(){ if ( g_currentToolMode == ClipperMode && g_defaultToolMode != ClipperMode ) { g_defaultToolMode(); } else { g_currentToolMode = ClipperMode; g_currentToolModeSupportsComponentEditing = false; SelectionSystem_DefaultMode(); ComponentModeChanged(); Sys_Status( c_ClipperMode_status ); GlobalSelectionSystem().SetManipulatorMode( SelectionSystem::eClip ); ToolChanged(); ModeChangeNotify(); } } const char* const c_BuildMode_status = "Build Tool: extrude, build chains, clone"; void BuildMode(){ if ( g_currentToolMode == BuildMode && g_defaultToolMode != BuildMode ) { g_defaultToolMode(); } else { g_currentToolMode = BuildMode; g_currentToolModeSupportsComponentEditing = false; SelectionSystem_DefaultMode(); ComponentModeChanged(); Sys_Status( c_BuildMode_status ); GlobalSelectionSystem().SetManipulatorMode( SelectionSystem::eBuild ); ToolChanged(); ModeChangeNotify(); } } const char* const c_UVMode_status = "UV Tool: edit texture alignment"; void UVMode(){ if ( g_currentToolMode == UVMode && g_defaultToolMode != UVMode ) { g_defaultToolMode(); } else { g_currentToolMode = UVMode; g_currentToolModeSupportsComponentEditing = false; SelectionSystem_DefaultMode(); ComponentModeChanged(); Sys_Status( c_UVMode_status ); GlobalSelectionSystem().SetManipulatorMode( SelectionSystem::eUV ); ToolChanged(); ModeChangeNotify(); } } void ToggleRotateScaleModes(){ return g_currentToolMode == RotateMode? ScaleMode() : RotateMode(); } void ToggleDragSkewModes(){ return g_currentToolMode == DragMode? SkewMode() : DragMode(); } void Texdef_Rotate( float angle ){ StringOutputStream command; command << "brushRotateTexture -angle " << angle; UndoableCommand undo( command.c_str() ); Select_RotateTexture( angle ); } // these are actually {Anti,}Clockwise in BP mode only (AP/220 - 50/50) // TODO is possible to make really {Anti,}Clockwise void Texdef_RotateClockwise(){ Texdef_Rotate( static_cast( -fabs( g_si_globals.rotate ) ) ); } void Texdef_RotateAntiClockwise(){ Texdef_Rotate( static_cast( fabs( g_si_globals.rotate ) ) ); } void Texdef_Scale( float x, float y ){ StringOutputStream command; command << "brushScaleTexture -x " << x << " -y " << y; UndoableCommand undo( command.c_str() ); Select_ScaleTexture( x, y ); } void Texdef_ScaleUp(){ Texdef_Scale( 0, g_si_globals.scale[1] ); } void Texdef_ScaleDown(){ Texdef_Scale( 0, -g_si_globals.scale[1] ); } void Texdef_ScaleLeft(){ Texdef_Scale( -g_si_globals.scale[0],0 ); } void Texdef_ScaleRight(){ Texdef_Scale( g_si_globals.scale[0],0 ); } void Texdef_Shift( float x, float y ){ StringOutputStream command; command << "brushShiftTexture -x " << x << " -y " << y; UndoableCommand undo( command.c_str() ); Select_ShiftTexture( x, y ); } void Texdef_ShiftLeft(){ Texdef_Shift( -g_si_globals.shift[0], 0 ); } void Texdef_ShiftRight(){ Texdef_Shift( g_si_globals.shift[0], 0 ); } void Texdef_ShiftUp(){ Texdef_Shift( 0, g_si_globals.shift[1] ); } void Texdef_ShiftDown(){ Texdef_Shift( 0, -g_si_globals.shift[1] ); } class SnappableSnapToGridSelected : public scene::Graph::Walker { float m_snap; public: SnappableSnapToGridSelected( float snap ) : m_snap( snap ){ } bool pre( const scene::Path& path, scene::Instance& instance ) const { if ( path.top().get().visible() ) { Snappable* snappable = Node_getSnappable( path.top() ); if ( snappable != 0 && Instance_isSelected( instance ) ) { snappable->snapto( m_snap ); } } return true; } }; void Scene_SnapToGrid_Selected( scene::Graph& graph, float snap ){ graph.traverse( SnappableSnapToGridSelected( snap ) ); } class ComponentSnappableSnapToGridSelected : public scene::Graph::Walker { float m_snap; public: ComponentSnappableSnapToGridSelected( float snap ) : m_snap( snap ){ } bool pre( const scene::Path& path, scene::Instance& instance ) const { if ( path.top().get().visible() ) { ComponentSnappable* componentSnappable = Instance_getComponentSnappable( instance ); if ( componentSnappable != 0 && Instance_isSelected( instance ) ) { componentSnappable->snapComponents( m_snap ); } } return true; } }; void Scene_SnapToGrid_Component_Selected( scene::Graph& graph, float snap ){ graph.traverse( ComponentSnappableSnapToGridSelected( snap ) ); } void Selection_SnapToGrid(){ StringOutputStream command; command << "snapSelected -grid " << GetGridSize(); UndoableCommand undo( command.c_str() ); if ( GlobalSelectionSystem().Mode() == SelectionSystem::eComponent && GlobalSelectionSystem().countSelectedComponents() ) { Scene_SnapToGrid_Component_Selected( GlobalSceneGraph(), GetGridSize() ); } else { Scene_SnapToGrid_Selected( GlobalSceneGraph(), GetGridSize() ); } } static gboolean qe_every_second( gpointer data ){ GdkModifierType mask; gdk_window_get_pointer( 0, 0, 0, &mask ); if ( ( mask & ( GDK_BUTTON1_MASK | GDK_BUTTON2_MASK | GDK_BUTTON3_MASK ) ) == 0 ) { QE_CheckAutoSave(); } return TRUE; } guint s_qe_every_second_id = 0; void EverySecondTimer_enable(){ if ( s_qe_every_second_id == 0 ) { s_qe_every_second_id = g_timeout_add( 1000, qe_every_second, 0 ); } } void EverySecondTimer_disable(){ if ( s_qe_every_second_id != 0 ) { g_source_remove( s_qe_every_second_id ); s_qe_every_second_id = 0; } } void window_realize_remove_decoration( GtkWidget* widget, gpointer data ){ gdk_window_set_decorations( gtk_widget_get_window( widget ), (GdkWMDecoration)( GDK_DECOR_ALL | GDK_DECOR_MENU | GDK_DECOR_MINIMIZE | GDK_DECOR_MAXIMIZE ) ); } class WaitDialog { public: GtkWindow* m_window; GtkLabel* m_label; }; WaitDialog create_wait_dialog( const char* title, const char* text, bool modal ){ WaitDialog dialog; dialog.m_window = create_floating_window( title, MainFrame_getWindow() ); gtk_window_set_modal( dialog.m_window, modal ); gtk_window_set_resizable( dialog.m_window, FALSE ); gtk_container_set_border_width( GTK_CONTAINER( dialog.m_window ), 0 ); gtk_window_set_position( dialog.m_window, GTK_WIN_POS_CENTER_ON_PARENT ); gtk_window_set_accept_focus( dialog.m_window, FALSE ); gtk_window_set_focus_on_map( dialog.m_window, FALSE ); g_signal_connect( G_OBJECT( dialog.m_window ), "realize", G_CALLBACK( window_realize_remove_decoration ), 0 ); { dialog.m_label = GTK_LABEL( gtk_label_new( text ) ); gtk_misc_set_alignment( GTK_MISC( dialog.m_label ), 0.0, 0.5 ); gtk_label_set_justify( dialog.m_label, GTK_JUSTIFY_LEFT ); gtk_widget_show( GTK_WIDGET( dialog.m_label ) ); gtk_widget_set_size_request( GTK_WIDGET( dialog.m_label ), 200, -1 ); gtk_container_add( GTK_CONTAINER( dialog.m_window ), GTK_WIDGET( dialog.m_label ) ); } return dialog; } namespace { clock_t g_lastRedrawTime = 0; const clock_t c_redrawInterval = clock_t( CLOCKS_PER_SEC / 10 ); bool redrawRequired(){ clock_t currentTime = std::clock(); if ( currentTime - g_lastRedrawTime >= c_redrawInterval ) { g_lastRedrawTime = currentTime; return true; } return false; } } bool MainFrame_isActiveApp(){ //globalOutputStream() << "listing\n"; GList* list = gtk_window_list_toplevels(); for ( GList* i = list; i != 0; i = g_list_next( i ) ) { //globalOutputStream() << "toplevel.. "; if ( gtk_window_is_active( GTK_WINDOW( i->data ) ) ) { //globalOutputStream() << "is active\n"; g_list_free( list ); return true; } //globalOutputStream() << "not active\n"; } g_list_free( list ); return false; } typedef std::list StringStack; StringStack g_wait_stack; WaitDialog g_wait; bool ScreenUpdates_Enabled(){ return g_wait_stack.empty(); } void ScreenUpdates_process(){ if ( redrawRequired() && gtk_widget_get_visible( GTK_WIDGET( g_wait.m_window ) ) ) { process_gui(); } } void ScreenUpdates_Disable( const char* message, const char* title ){ if ( g_wait_stack.empty() ) { EverySecondTimer_disable(); process_gui(); bool isActiveApp = MainFrame_isActiveApp(); g_wait = create_wait_dialog( title, message, !XYWnd::m_mnuDrop || !gtk_widget_get_visible( GTK_WIDGET( XYWnd::m_mnuDrop ) ) ); //hack: avoid hiding entity menu, clicked with ctrl, by tex/model loading popup if ( isActiveApp ) { gtk_widget_show( GTK_WIDGET( g_wait.m_window ) ); ScreenUpdates_process(); } } else if ( gtk_widget_get_visible( GTK_WIDGET( g_wait.m_window ) ) ) { gtk_label_set_text( g_wait.m_label, message ); ScreenUpdates_process(); } g_wait_stack.push_back( message ); } void ScreenUpdates_Enable(){ ASSERT_MESSAGE( !ScreenUpdates_Enabled(), "screen updates already enabled" ); g_wait_stack.pop_back(); if ( g_wait_stack.empty() ) { EverySecondTimer_enable(); //gtk_widget_set_sensitive(GTK_WIDGET(MainFrame_getWindow()), TRUE); destroy_floating_window( g_wait.m_window ); g_wait.m_window = 0; //gtk_window_present(MainFrame_getWindow()); } else if ( gtk_widget_get_visible( GTK_WIDGET( g_wait.m_window ) ) ) { gtk_label_set_text( g_wait.m_label, g_wait_stack.back().c_str() ); ScreenUpdates_process(); } } void GlobalCamera_UpdateWindow(){ if ( g_pParentWnd != 0 ) { CamWnd_Update( *g_pParentWnd->GetCamWnd() ); } } void XY_UpdateAllWindows(){ if ( g_pParentWnd != 0 ) { g_pParentWnd->forEachXYWnd( []( XYWnd* xywnd ){ XYWnd_Update( *xywnd ); } ); } } void UpdateAllWindows(){ GlobalCamera_UpdateWindow(); XY_UpdateAllWindows(); } void ModeChangeNotify(){ SceneChangeNotify(); } LatchedInt g_Layout_viewStyle( 0, "Window Layout" ); LatchedBool g_Layout_enableDetachableMenus( true, "Detachable Menus" ); LatchedBool g_Layout_enableMainToolbar( true, "Main Toolbar" ); LatchedBool g_Layout_enablePatchToolbar( true, "Patch Toolbar" ); LatchedBool g_Layout_enablePluginToolbar( true, "Plugin Toolbar" ); LatchedBool g_Layout_enableFilterToolbar( true, "Filter Toolbar" ); LatchedBool g_Layout_SingleToolbar( true, "Single Scrollable Toolbar" ); GtkMenuItem* create_file_menu(){ // File menu GtkMenuItem* file_menu_item = new_sub_menu_item_with_mnemonic( "_File" ); GtkMenu* menu = GTK_MENU( gtk_menu_item_get_submenu( file_menu_item ) ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu ); } create_menu_item_with_mnemonic( menu, "_New Map", "NewMap" ); menu_separator( menu ); create_menu_item_with_mnemonic( menu, "_Open...", "OpenMap" ); create_menu_item_with_mnemonic( menu, "_Import...", "ImportMap" ); menu_separator( menu ); create_menu_item_with_mnemonic( menu, "_Save", "SaveMap" ); create_menu_item_with_mnemonic( menu, "Save _as...", "SaveMapAs" ); create_menu_item_with_mnemonic( menu, "Save s_elected...", "SaveSelected" ); create_menu_item_with_mnemonic( menu, "Save re_gion...", "SaveRegion" ); menu_separator( menu ); // menu_separator( menu ); // create_menu_item_with_mnemonic( menu, "_Refresh models", "RefreshReferences" ); // menu_separator( menu ); create_menu_item_with_mnemonic( menu, "Pro_ject settings...", "ProjectSettings" ); //menu_separator( menu ); create_menu_item_with_mnemonic( menu, "_Pointfile", "TogglePointfile" ); menu_separator( menu ); MRU_constructMenu( menu ); menu_separator( menu ); // create_menu_item_with_mnemonic( menu, "Check for NetRadiant update (web)", "CheckForUpdate" ); // FIXME create_menu_item_with_mnemonic( menu, "E_xit", "Exit" ); return file_menu_item; } GtkMenuItem* create_edit_menu(){ // Edit menu GtkMenuItem* edit_menu_item = new_sub_menu_item_with_mnemonic( "_Edit" ); GtkMenu* menu = GTK_MENU( gtk_menu_item_get_submenu( edit_menu_item ) ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu ); } create_menu_item_with_mnemonic( menu, "_Undo", "Undo" ); create_menu_item_with_mnemonic( menu, "_Redo", "Redo" ); menu_separator( menu ); create_menu_item_with_mnemonic( menu, "_Copy", "Copy" ); create_menu_item_with_mnemonic( menu, "_Paste", "Paste" ); create_menu_item_with_mnemonic( menu, "P_aste To Camera", "PasteToCamera" ); create_menu_item_with_mnemonic( menu, "Move To Camera", "MoveToCamera" ); menu_separator( menu ); create_menu_item_with_mnemonic( menu, "_Duplicate", "CloneSelection" ); create_menu_item_with_mnemonic( menu, "Duplicate, make uni_que", "CloneSelectionAndMakeUnique" ); create_menu_item_with_mnemonic( menu, "D_elete", "DeleteSelection" ); //create_menu_item_with_mnemonic( menu, "Pa_rent", "ParentSelection" ); menu_separator( menu ); create_menu_item_with_mnemonic( menu, "C_lear Selection", "UnSelectSelection" ); create_menu_item_with_mnemonic( menu, "_Invert Selection", "InvertSelection" ); create_menu_item_with_mnemonic( menu, "Select i_nside", "SelectInside" ); create_menu_item_with_mnemonic( menu, "Select _touching", "SelectTouching" ); menu_separator( menu ); // GtkMenu* convert_menu = create_sub_menu_with_mnemonic( menu, "E_xpand Selection" ); // if ( g_Layout_enableDetachableMenus.m_value ) { // menu_tearoff( convert_menu ); // } create_menu_item_with_mnemonic( menu, "Select All Of Type", "SelectAllOfType" ); create_menu_item_with_mnemonic( menu, "Select Textured", "SelectTextured" ); create_menu_item_with_mnemonic( menu, "_Expand Selection To Primitives", "ExpandSelectionToPrimitives" ); create_menu_item_with_mnemonic( menu, "_Expand Selection To Entities", "ExpandSelectionToEntities" ); create_menu_item_with_mnemonic( menu, "Select Connected Entities", "SelectConnectedEntities" ); menu_separator( menu ); create_menu_item_with_mnemonic( menu, "_Shortcuts...", FreeCaller() ); create_menu_item_with_mnemonic( menu, "Pre_ferences...", "Preferences" ); return edit_menu_item; } GtkWidget* g_toggle_z_item = 0; GtkWidget* g_toggle_console_item = 0; GtkWidget* g_toggle_entity_item = 0; GtkWidget* g_toggle_entitylist_item = 0; GtkMenuItem* create_view_menu( MainFrame::EViewStyle style ){ // View menu GtkMenuItem* view_menu_item = new_sub_menu_item_with_mnemonic( "Vie_w" ); GtkMenu* menu = GTK_MENU( gtk_menu_item_get_submenu( view_menu_item ) ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu ); } if ( style == MainFrame::eFloating ) { create_check_menu_item_with_mnemonic( menu, "Camera View", "ToggleCamera" ); create_check_menu_item_with_mnemonic( menu, "XY (Top) View", "ToggleView" ); create_check_menu_item_with_mnemonic( menu, "XZ (Front) View", "ToggleFrontView" ); create_check_menu_item_with_mnemonic( menu, "YZ (Side) View", "ToggleSideView" ); } if ( style == MainFrame::eFloating || style == MainFrame::eSplit ) { create_menu_item_with_mnemonic( menu, "Console", "ToggleConsole" ); create_menu_item_with_mnemonic( menu, "Texture Browser", "ToggleTextures" ); } create_menu_item_with_mnemonic( menu, "Model Browser", "ToggleModelBrowser" ); create_menu_item_with_mnemonic( menu, "Entity Inspector", "ToggleEntityInspector" ); create_menu_item_with_mnemonic( menu, "_Surface Inspector", "SurfaceInspector" ); create_menu_item_with_mnemonic( menu, "_Patch Inspector", "PatchInspector" ); create_menu_item_with_mnemonic( menu, "Entity List", "EntityList" ); menu_separator( menu ); { GtkMenu* camera_menu = create_sub_menu_with_mnemonic( menu, "Camera" ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( camera_menu ); } create_menu_item_with_mnemonic( camera_menu, "Focus on Selected", "CameraFocusOnSelected" ); create_menu_item_with_mnemonic( camera_menu, "_Center", "CenterView" ); create_menu_item_with_mnemonic( camera_menu, "_Up Floor", "UpFloor" ); create_menu_item_with_mnemonic( camera_menu, "_Down Floor", "DownFloor" ); menu_separator( camera_menu ); create_menu_item_with_mnemonic( camera_menu, "Far Clip Plane In", "CubicClipZoomIn" ); create_menu_item_with_mnemonic( camera_menu, "Far Clip Plane Out", "CubicClipZoomOut" ); menu_separator( camera_menu ); create_menu_item_with_mnemonic( camera_menu, "Next leak spot", "NextLeakSpot" ); create_menu_item_with_mnemonic( camera_menu, "Previous leak spot", "PrevLeakSpot" ); //cameramodel is not implemented in instances, thus useless // menu_separator( camera_menu ); // create_menu_item_with_mnemonic( camera_menu, "Look Through Selected", "LookThroughSelected" ); // create_menu_item_with_mnemonic( camera_menu, "Look Through Camera", "LookThroughCamera" ); } menu_separator( menu ); { GtkMenu* orthographic_menu = create_sub_menu_with_mnemonic( menu, "Orthographic" ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( orthographic_menu ); } if ( style == MainFrame::eRegular || style == MainFrame::eRegularLeft || style == MainFrame::eFloating ) { create_menu_item_with_mnemonic( orthographic_menu, "_Next (XY, XZ, YZ)", "NextView" ); create_menu_item_with_mnemonic( orthographic_menu, "XY (Top)", "ViewTop" ); create_menu_item_with_mnemonic( orthographic_menu, "XZ (Front)", "ViewFront" ); create_menu_item_with_mnemonic( orthographic_menu, "YZ (Side)", "ViewSide" ); menu_separator( orthographic_menu ); } else{ create_menu_item_with_mnemonic( orthographic_menu, "Center on Selected", "NextView" ); } create_menu_item_with_mnemonic( orthographic_menu, "Focus on Selected", "XYFocusOnSelected" ); create_menu_item_with_mnemonic( orthographic_menu, "Center on Selected", "CenterXYView" ); menu_separator( orthographic_menu ); create_menu_item_with_mnemonic( orthographic_menu, "_XY 100%", "Zoom100" ); create_menu_item_with_mnemonic( orthographic_menu, "XY Zoom _In", "ZoomIn" ); create_menu_item_with_mnemonic( orthographic_menu, "XY Zoom _Out", "ZoomOut" ); } menu_separator( menu ); { GtkMenu* menu_in_menu = create_sub_menu_with_mnemonic( menu, "Show" ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu_in_menu ); } create_check_menu_item_with_mnemonic( menu_in_menu, "Show Entity _Angles", "ShowAngles" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show Entity _Names", "ShowNames" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show Light Radiuses", "ShowLightRadiuses" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show Entity Boxes", "ShowBboxes" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show Entity Connections", "ShowConnections" ); menu_separator( menu_in_menu ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show 2D Size Info", "ShowSize2d" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show 3D Size Info", "ShowSize3d" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show Crosshair", "ToggleCrosshairs" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show Grid", "ToggleGrid" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show Blocks", "ShowBlocks" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show C_oordinates", "ShowCoordinates" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show Window Outline", "ShowWindowOutline" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show Axes", "ShowAxes" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show 2D Workzone", "ShowWorkzone2d" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show 3D Workzone", "ShowWorkzone3d" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Show Renderer Stats", "ShowStats" ); } { GtkMenu* menu_in_menu = create_sub_menu_with_mnemonic( menu, "Filter" ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu_in_menu ); } Filters_constructMenu( menu_in_menu ); } menu_separator( menu ); { create_check_menu_item_with_mnemonic( menu, "Hide Selected", "HideSelected" ); create_menu_item_with_mnemonic( menu, "Show Hidden", "ShowHidden" ); } menu_separator( menu ); { GtkMenu* menu_in_menu = create_sub_menu_with_mnemonic( menu, "Region" ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu_in_menu ); } create_menu_item_with_mnemonic( menu_in_menu, "_Off", "RegionOff" ); create_menu_item_with_mnemonic( menu_in_menu, "_Set XY", "RegionSetXY" ); create_menu_item_with_mnemonic( menu_in_menu, "Set _Brush", "RegionSetBrush" ); create_check_menu_item_with_mnemonic( menu_in_menu, "Set Se_lection", "RegionSetSelection" ); } //command_connect_accelerator( "CenterXYView" ); return view_menu_item; } GtkMenuItem* create_selection_menu(){ // Selection menu GtkMenuItem* selection_menu_item = new_sub_menu_item_with_mnemonic( "M_odify" ); GtkMenu* menu = GTK_MENU( gtk_menu_item_get_submenu( selection_menu_item ) ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu ); } { GtkMenu* menu_in_menu = create_sub_menu_with_mnemonic( menu, "Components" ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu_in_menu ); } create_check_menu_item_with_mnemonic( menu_in_menu, "_Edges", "DragEdges" ); create_check_menu_item_with_mnemonic( menu_in_menu, "_Vertices", "DragVertices" ); create_check_menu_item_with_mnemonic( menu_in_menu, "_Faces", "DragFaces" ); } menu_separator( menu ); create_menu_item_with_mnemonic( menu, "Snap To Grid", "SnapToGrid" ); menu_separator( menu ); { GtkMenu* menu_in_menu = create_sub_menu_with_mnemonic( menu, "Nudge" ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu_in_menu ); } create_menu_item_with_mnemonic( menu_in_menu, "Nudge Left", "SelectNudgeLeft" ); create_menu_item_with_mnemonic( menu_in_menu, "Nudge Right", "SelectNudgeRight" ); create_menu_item_with_mnemonic( menu_in_menu, "Nudge Up", "SelectNudgeUp" ); create_menu_item_with_mnemonic( menu_in_menu, "Nudge Down", "SelectNudgeDown" ); menu_separator( menu_in_menu ); create_menu_item_with_mnemonic( menu_in_menu, "Nudge +Z", "MoveSelectionUP" ); create_menu_item_with_mnemonic( menu_in_menu, "Nudge -Z", "MoveSelectionDOWN" ); } { GtkMenu* menu_in_menu = create_sub_menu_with_mnemonic( menu, "Rotate" ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu_in_menu ); } create_menu_item_with_mnemonic( menu_in_menu, "Rotate X", "RotateSelectionX" ); create_menu_item_with_mnemonic( menu_in_menu, "Rotate Y", "RotateSelectionY" ); create_menu_item_with_mnemonic( menu_in_menu, "Rotate Z", "RotateSelectionZ" ); menu_separator( menu_in_menu ); create_menu_item_with_mnemonic( menu_in_menu, "Rotate Clockwise", "RotateSelectionClockwise" ); create_menu_item_with_mnemonic( menu_in_menu, "Rotate Anticlockwise", "RotateSelectionAnticlockwise" ); } { GtkMenu* menu_in_menu = create_sub_menu_with_mnemonic( menu, "Flip" ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu_in_menu ); } create_menu_item_with_mnemonic( menu_in_menu, "Flip _X", "MirrorSelectionX" ); create_menu_item_with_mnemonic( menu_in_menu, "Flip _Y", "MirrorSelectionY" ); create_menu_item_with_mnemonic( menu_in_menu, "Flip _Z", "MirrorSelectionZ" ); menu_separator( menu_in_menu ); create_menu_item_with_mnemonic( menu_in_menu, "Flip Horizontally", "MirrorSelectionHorizontally" ); create_menu_item_with_mnemonic( menu_in_menu, "Flip Vertically", "MirrorSelectionVertically" ); } menu_separator( menu ); create_menu_item_with_mnemonic( menu, "Arbitrary rotation...", "ArbitraryRotation" ); create_menu_item_with_mnemonic( menu, "Arbitrary scale...", "ArbitraryScale" ); create_menu_item_with_mnemonic( menu, "Repeat Transforms", "RepeatTransforms" ); return selection_menu_item; } GtkMenuItem* create_bsp_menu(){ // BSP menu GtkMenuItem* bsp_menu_item = new_sub_menu_item_with_mnemonic( "_Build" ); GtkMenu* menu = GTK_MENU( gtk_menu_item_get_submenu( bsp_menu_item ) ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu ); } create_menu_item_with_mnemonic( menu, "Customize...", "BuildMenuCustomize" ); create_menu_item_with_mnemonic( menu, "Run recent build", "Build_runRecentExecutedBuild" ); menu_separator( menu ); Build_constructMenu( menu ); g_bsp_menu = menu; return bsp_menu_item; } GtkMenuItem* create_grid_menu(){ // Grid menu GtkMenuItem* grid_menu_item = new_sub_menu_item_with_mnemonic( "_Grid" ); GtkMenu* menu = GTK_MENU( gtk_menu_item_get_submenu( grid_menu_item ) ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu ); } Grid_constructMenu( menu ); return grid_menu_item; } GtkMenuItem* create_misc_menu(){ // Misc menu GtkMenuItem* misc_menu_item = new_sub_menu_item_with_mnemonic( "M_isc" ); GtkMenu* menu = GTK_MENU( gtk_menu_item_get_submenu( misc_menu_item ) ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu ); } #if 0 create_menu_item_with_mnemonic( menu, "_Benchmark", FreeCaller() ); #endif gtk_container_add( GTK_CONTAINER( menu ), GTK_WIDGET( create_colours_menu() ) ); create_menu_item_with_mnemonic( menu, "Find brush...", "FindBrush" ); create_menu_item_with_mnemonic( menu, "Map Info...", "MapInfo" ); // http://zerowing.idsoftware.com/bugzilla/show_bug.cgi?id=394 // create_menu_item_with_mnemonic(menu, "_Print XY View", FreeCaller()); create_menu_item_with_mnemonic( menu, "Set 2D _Background image...", FreeCaller() ); create_menu_item_with_mnemonic( menu, "Fullscreen", "Fullscreen" ); create_menu_item_with_mnemonic( menu, "Maximize view", "MaximizeView" ); return misc_menu_item; } GtkMenuItem* create_entity_menu(){ // Brush menu GtkMenuItem* entity_menu_item = new_sub_menu_item_with_mnemonic( "E_ntity" ); GtkMenu* menu = GTK_MENU( gtk_menu_item_get_submenu( entity_menu_item ) ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu ); } Entity_constructMenu( menu ); return entity_menu_item; } GtkMenuItem* create_brush_menu(){ // Brush menu GtkMenuItem* brush_menu_item = new_sub_menu_item_with_mnemonic( "B_rush" ); GtkMenu* menu = GTK_MENU( gtk_menu_item_get_submenu( brush_menu_item ) ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu ); } Brush_constructMenu( menu ); return brush_menu_item; } GtkMenuItem* create_patch_menu(){ // Curve menu GtkMenuItem* patch_menu_item = new_sub_menu_item_with_mnemonic( "_Curve" ); GtkMenu* menu = GTK_MENU( gtk_menu_item_get_submenu( patch_menu_item ) ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu ); } Patch_constructMenu( menu ); return patch_menu_item; } GtkMenuItem* create_help_menu(){ // Help menu GtkMenuItem* help_menu_item = new_sub_menu_item_with_mnemonic( "_Help" ); GtkMenu* menu = GTK_MENU( gtk_menu_item_get_submenu( help_menu_item ) ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu ); } // create_menu_item_with_mnemonic( menu, "Manual", "OpenManual" ); // this creates all the per-game drop downs for the game pack helps // it will take care of hooking the Sys_OpenURL calls etc. create_game_help_menu( menu ); // create_menu_item_with_mnemonic( menu, "Bug report", FreeCaller() ); create_menu_item_with_mnemonic( menu, "_About", FreeCaller() ); return help_menu_item; } GtkMenuBar* create_main_menu( MainFrame::EViewStyle style ){ GtkMenuBar* menu_bar = GTK_MENU_BAR( gtk_menu_bar_new() ); gtk_widget_show( GTK_WIDGET( menu_bar ) ); gtk_container_add( GTK_CONTAINER( menu_bar ), GTK_WIDGET( create_file_menu() ) ); gtk_container_add( GTK_CONTAINER( menu_bar ), GTK_WIDGET( create_edit_menu() ) ); gtk_container_add( GTK_CONTAINER( menu_bar ), GTK_WIDGET( create_view_menu( style ) ) ); gtk_container_add( GTK_CONTAINER( menu_bar ), GTK_WIDGET( create_selection_menu() ) ); gtk_container_add( GTK_CONTAINER( menu_bar ), GTK_WIDGET( create_bsp_menu() ) ); gtk_container_add( GTK_CONTAINER( menu_bar ), GTK_WIDGET( create_grid_menu() ) ); gtk_container_add( GTK_CONTAINER( menu_bar ), GTK_WIDGET( create_misc_menu() ) ); gtk_container_add( GTK_CONTAINER( menu_bar ), GTK_WIDGET( create_entity_menu() ) ); gtk_container_add( GTK_CONTAINER( menu_bar ), GTK_WIDGET( create_brush_menu() ) ); gtk_container_add( GTK_CONTAINER( menu_bar ), GTK_WIDGET( create_patch_menu() ) ); gtk_container_add( GTK_CONTAINER( menu_bar ), GTK_WIDGET( create_plugins_menu() ) ); gtk_container_add( GTK_CONTAINER( menu_bar ), GTK_WIDGET( create_help_menu() ) ); return menu_bar; } void PatchInspector_registerShortcuts(){ command_connect_accelerator( "PatchInspector" ); } void Patch_registerShortcuts(){ command_connect_accelerator( "InvertCurveTextureX" ); command_connect_accelerator( "InvertCurveTextureY" ); command_connect_accelerator( "PatchInsertInsertColumn" ); command_connect_accelerator( "PatchInsertInsertRow" ); command_connect_accelerator( "PatchDeleteLastColumn" ); command_connect_accelerator( "PatchDeleteLastRow" ); command_connect_accelerator( "NaturalizePatch" ); command_connect_accelerator( "CapCurrentCurve" ); } void Manipulators_registerShortcuts(){ toggle_add_accelerator( "MouseRotate" ); toggle_add_accelerator( "MouseTranslate" ); toggle_add_accelerator( "MouseScale" ); toggle_add_accelerator( "MouseTransform" ); toggle_add_accelerator( "MouseDrag" ); toggle_add_accelerator( "ToggleClipper" ); toggle_add_accelerator( "MouseBuild" ); toggle_add_accelerator( "MouseUV" ); command_connect_accelerator( "MouseRotateOrScale" ); command_connect_accelerator( "MouseDragOrTransform" ); } void TexdefNudge_registerShortcuts(){ command_connect_accelerator( "TexRotateClock" ); command_connect_accelerator( "TexRotateCounter" ); command_connect_accelerator( "TexScaleUp" ); command_connect_accelerator( "TexScaleDown" ); command_connect_accelerator( "TexScaleLeft" ); command_connect_accelerator( "TexScaleRight" ); command_connect_accelerator( "TexShiftUp" ); command_connect_accelerator( "TexShiftDown" ); command_connect_accelerator( "TexShiftLeft" ); command_connect_accelerator( "TexShiftRight" ); } void SelectNudge_registerShortcuts(){ command_connect_accelerator( "MoveSelectionDOWN" ); command_connect_accelerator( "MoveSelectionUP" ); command_connect_accelerator( "SelectNudgeLeft" ); command_connect_accelerator( "SelectNudgeRight" ); command_connect_accelerator( "SelectNudgeUp" ); command_connect_accelerator( "SelectNudgeDown" ); } void SnapToGrid_registerShortcuts(){ command_connect_accelerator( "SnapToGrid" ); } void SelectByType_registerShortcuts(){ command_connect_accelerator( "SelectAllOfType" ); } void SurfaceInspector_registerShortcuts(){ command_connect_accelerator( "FitTexture" ); command_connect_accelerator( "FitTextureWidth" ); command_connect_accelerator( "FitTextureHeight" ); command_connect_accelerator( "FitTextureWidthOnly" ); command_connect_accelerator( "FitTextureHeightOnly" ); command_connect_accelerator( "TextureProjectAxial" ); command_connect_accelerator( "TextureProjectOrtho" ); command_connect_accelerator( "TextureProjectCam" ); } void TexBro_registerShortcuts(){ command_connect_accelerator( "FindReplaceTextures" ); command_connect_accelerator( "RefreshShaders" ); toggle_add_accelerator( "SearchFromStart" ); } void Misc_registerShortcuts(){ command_connect_accelerator( "RefreshReferences" ); //refresh models command_connect_accelerator( "Redo2" ); command_connect_accelerator( "UnSelectSelection2" ); command_connect_accelerator( "DeleteSelection2" ); } void register_shortcuts(){ // PatchInspector_registerShortcuts(); // Patch_registerShortcuts(); Grid_registerShortcuts(); // XYWnd_registerShortcuts(); CamWnd_registerShortcuts(); Manipulators_registerShortcuts(); SurfaceInspector_registerShortcuts(); TexdefNudge_registerShortcuts(); // SelectNudge_registerShortcuts(); // SnapToGrid_registerShortcuts(); // SelectByType_registerShortcuts(); TexBro_registerShortcuts(); Misc_registerShortcuts(); Entity_registerShortcuts(); } void File_constructToolbar( GtkToolbar* toolbar ){ toolbar_append_button( toolbar, "Open an existing map (CTRL + O)", "file_open.png", "OpenMap" ); toolbar_append_button( toolbar, "Save the active map (CTRL + S)", "file_save.png", "SaveMap" ); } void UndoRedo_constructToolbar( GtkToolbar* toolbar ){ toolbar_append_button( toolbar, "Undo (CTRL + Z)", "undo.png", "Undo" ); toolbar_append_button( toolbar, "Redo (CTRL + Y)", "redo.png", "Redo" ); } void RotateFlip_constructToolbar( GtkToolbar* toolbar ){ // toolbar_append_button( toolbar, "x-axis Flip", "brush_flipx.png", "MirrorSelectionX" ); // toolbar_append_button( toolbar, "x-axis Rotate", "brush_rotatex.png", "RotateSelectionX" ); // toolbar_append_button( toolbar, "y-axis Flip", "brush_flipy.png", "MirrorSelectionY" ); // toolbar_append_button( toolbar, "y-axis Rotate", "brush_rotatey.png", "RotateSelectionY" ); // toolbar_append_button( toolbar, "z-axis Flip", "brush_flipz.png", "MirrorSelectionZ" ); // toolbar_append_button( toolbar, "z-axis Rotate", "brush_rotatez.png", "RotateSelectionZ" ); toolbar_append_button( toolbar, "Flip Horizontally", "brush_flip_hor.png", "MirrorSelectionHorizontally" ); toolbar_append_button( toolbar, "Flip Vertically", "brush_flip_vert.png", "MirrorSelectionVertically" ); toolbar_append_button( toolbar, "Rotate Anticlockwise", "brush_rotate_anti.png", "RotateSelectionAnticlockwise" ); toolbar_append_button( toolbar, "Rotate Clockwise", "brush_rotate_clock.png", "RotateSelectionClockwise" ); } void Select_constructToolbar( GtkToolbar* toolbar ){ toolbar_append_button( toolbar, "Select touching", "selection_selecttouching.png", "SelectTouching" ); toolbar_append_button( toolbar, "Select inside", "selection_selectinside.png", "SelectInside" ); } void CSG_constructToolbar( GtkToolbar* toolbar ){ toolbar_append_button( toolbar, "CSG Subtract (SHIFT + U)", "selection_csgsubtract.png", "CSGSubtract" ); toolbar_append_button( toolbar, "CSG Wrap Merge (CTRL + U)", "selection_csgmerge.png", "CSGWrapMerge" ); toolbar_append_button( toolbar, "Room", "selection_makeroom.png", "CSGroom" ); toolbar_append_button( toolbar, "CSG Tool", "ellipsis.png", "CSGTool" ); } void ComponentModes_constructToolbar( GtkToolbar* toolbar ){ toolbar_append_toggle_button( toolbar, "Select Vertices (V)", "modify_vertices.png", "DragVertices" ); toolbar_append_toggle_button( toolbar, "Select Edges (E)", "modify_edges.png", "DragEdges" ); toolbar_append_toggle_button( toolbar, "Select Faces (F)", "modify_faces.png", "DragFaces" ); } void XYWnd_constructToolbar( GtkToolbar* toolbar ){ toolbar_append_button( toolbar, "Change views (CTRL + TAB)", "view_change.png", "NextView" ); } void Manipulators_constructToolbar( GtkToolbar* toolbar ){ toolbar_append_toggle_button( toolbar, "Translate (W)", "select_mousetranslate.png", "MouseTranslate" ); toolbar_append_toggle_button( toolbar, "Rotate (R)", "select_mouserotate.png", "MouseRotate" ); toolbar_append_toggle_button( toolbar, "Scale", "select_mousescale.png", "MouseScale" ); toolbar_append_toggle_button( toolbar, "Transform (Q)", "select_mousetransform.png", "MouseTransform" ); toolbar_append_toggle_button( toolbar, "Resize (Q)", "select_mouseresize.png", "MouseDrag" ); toolbar_append_toggle_button( toolbar, "Clipper (X)", "select_clipper.png", "ToggleClipper" ); // toolbar_append_toggle_button( toolbar, "Build (B)", "select_mouserotate.png", "MouseBuild" ); toolbar_append_toggle_button( toolbar, "UV Tool (G)", "select_mouseuv.png", "MouseUV" ); } GtkToolbar* create_main_toolbar( MainFrame::EViewStyle style ){ GtkToolbar* toolbar = toolbar_new(); File_constructToolbar( toolbar ); toolbar_append_space( toolbar ); UndoRedo_constructToolbar( toolbar ); toolbar_append_space( toolbar ); RotateFlip_constructToolbar( toolbar ); toolbar_append_space( toolbar ); Select_constructToolbar( toolbar ); toolbar_append_space( toolbar ); CSG_constructToolbar( toolbar ); toolbar_append_space( toolbar ); ComponentModes_constructToolbar( toolbar ); toolbar_append_space( toolbar ); if ( style != MainFrame::eSplit ) { XYWnd_constructToolbar( toolbar ); toolbar_append_space( toolbar ); } CamWnd_constructToolbar( toolbar ); toolbar_append_space( toolbar ); Manipulators_constructToolbar( toolbar ); toolbar_append_space( toolbar ); if ( g_Layout_enablePatchToolbar.m_value ) { Patch_constructToolbar( toolbar ); toolbar_append_space( toolbar ); } toolbar_append_toggle_button( toolbar, "Texture Lock (SHIFT + T)", "texture_lock.png", "TogTexLock" ); toolbar_append_toggle_button( toolbar, "Texture Vertex Lock", "texture_vertexlock.png", "TogTexVertexLock" ); toolbar_append_space( toolbar ); toolbar_append_button( toolbar, "Entities (N)", "entities.png", "ToggleEntityInspector" ); // disable the console and texture button in the regular layouts if ( style != MainFrame::eRegular && style != MainFrame::eRegularLeft ) { toolbar_append_button( toolbar, "Console (O)", "console.png", "ToggleConsole" ); toolbar_append_button( toolbar, "Texture Browser (T)", "texture_browser.png", "ToggleTextures" ); } // TODO: call light inspector //GtkToolButton* g_view_lightinspector_button = toolbar_append_button(toolbar, "Light Inspector", "lightinspector.png", "ToggleLightInspector"); toolbar_append_space( toolbar ); toolbar_append_button( toolbar, "Refresh Models", "refresh_models.png", "RefreshReferences" ); return toolbar; } GtkLabel* create_main_statusbar_label(){ GtkLabel* label = GTK_LABEL( gtk_label_new( "Label" ) ); gtk_misc_set_alignment( GTK_MISC( label ), 0, 0.5 ); gtk_misc_set_padding( GTK_MISC( label ), 4, 2 ); gtk_widget_show( GTK_WIDGET( label ) ); return label; } GtkWidget* create_main_statusbar( GtkWidget *pStatusLabel[c_status__count] ){ GtkTable* table = GTK_TABLE( gtk_table_new( 1, c_status__count, FALSE ) ); gtk_widget_show( GTK_WIDGET( table ) ); { GtkLabel* label = create_main_statusbar_label(); gtk_label_set_ellipsize( label, PANGO_ELLIPSIZE_END ); gtk_table_attach_defaults( table, GTK_WIDGET( label ), 0, 1, 0, 1 ); pStatusLabel[c_status_command] = GTK_WIDGET( label ); } for ( int i = 1; i < c_status__count; ++i ) { GtkFrame* frame = GTK_FRAME( gtk_frame_new( 0 ) ); gtk_widget_show( GTK_WIDGET( frame ) ); gtk_frame_set_shadow_type( frame, GTK_SHADOW_IN ); if( i == c_status_grid || i == c_status_brushcount ) gtk_table_attach( table, GTK_WIDGET( frame ), i, i + 1, 0, 1, GTK_FILL, GTK_FILL, 0, 0 ); else gtk_table_attach_defaults( table, GTK_WIDGET( frame ), i, i + 1, 0, 1 ); if( i == c_status_brushcount ){ GtkWidget* hbox = gtk_hbox_new( FALSE, 0 ); gtk_container_add( GTK_CONTAINER( frame ), hbox ); gtk_widget_show( hbox ); const char* imgs[3] = { "status_brush.png", "patch_wireframe.png", "status_entiy.png" }; for( ; i < c_status_brushcount + 3; ++i ){ GtkImage* image = new_local_image( imgs[i - c_status_brushcount] ); gtk_widget_show( GTK_WIDGET( image ) ); gtk_box_pack_start( GTK_BOX( hbox ), GTK_WIDGET( image ), FALSE, FALSE, 0 ); GtkLabel* label = create_main_statusbar_label(); gtk_label_set_width_chars( label, 5 ); gtk_box_pack_start( GTK_BOX( hbox ), GTK_WIDGET( label ), FALSE, TRUE, 0 ); pStatusLabel[i] = GTK_WIDGET( label ); } --i; } else{ GtkLabel* label = create_main_statusbar_label(); if( i == c_status_grid ) gtk_widget_set_tooltip_markup( GTK_WIDGET( frame ), " G: Grid size\n F: map Format\n C: camera Clip distance \n L: texture Lock" ); else gtk_label_set_ellipsize( label, i == c_status_texture? PANGO_ELLIPSIZE_START : PANGO_ELLIPSIZE_END ); gtk_container_add( GTK_CONTAINER( frame ), GTK_WIDGET( label ) ); pStatusLabel[i] = GTK_WIDGET( label ); } } return GTK_WIDGET( table ); } #if 0 WidgetFocusPrinter g_mainframeWidgetFocusPrinter( "mainframe" ); class WindowFocusPrinter { const char* m_name; static gboolean frame_event( GtkWidget *widget, GdkEvent* event, WindowFocusPrinter* self ){ globalOutputStream() << self->m_name << " frame_event\n"; return FALSE; } static gboolean keys_changed( GtkWidget *widget, WindowFocusPrinter* self ){ globalOutputStream() << self->m_name << " keys_changed\n"; return FALSE; } static gboolean notify( GtkWindow* window, gpointer dummy, WindowFocusPrinter* self ){ if ( gtk_window_is_active( window ) ) { globalOutputStream() << self->m_name << " takes toplevel focus\n"; } else { globalOutputStream() << self->m_name << " loses toplevel focus\n"; } return FALSE; } public: WindowFocusPrinter( const char* name ) : m_name( name ){ } void connect( GtkWindow* toplevel_window ){ g_signal_connect( G_OBJECT( toplevel_window ), "notify::has_toplevel_focus", G_CALLBACK( notify ), this ); g_signal_connect( G_OBJECT( toplevel_window ), "notify::is_active", G_CALLBACK( notify ), this ); g_signal_connect( G_OBJECT( toplevel_window ), "keys_changed", G_CALLBACK( keys_changed ), this ); g_signal_connect( G_OBJECT( toplevel_window ), "frame_event", G_CALLBACK( frame_event ), this ); } }; WindowFocusPrinter g_mainframeFocusPrinter( "mainframe" ); #endif class MainWindowActive { static gboolean notify( GtkWindow* window, gpointer dummy, MainWindowActive* self ){ if ( g_wait.m_window != 0 && gtk_window_is_active( window ) && !gtk_widget_get_visible( GTK_WIDGET( g_wait.m_window ) ) ) { gtk_widget_show( GTK_WIDGET( g_wait.m_window ) ); } return FALSE; } public: void connect( GtkWindow* toplevel_window ){ g_signal_connect( G_OBJECT( toplevel_window ), "notify::is-active", G_CALLBACK( notify ), this ); } }; MainWindowActive g_MainWindowActive; SignalHandlerId XYWindowDestroyed_connect( const SignalHandler& handler ){ return g_pParentWnd->GetXYWnd()->onDestroyed.connectFirst( handler ); } void XYWindowDestroyed_disconnect( SignalHandlerId id ){ g_pParentWnd->GetXYWnd()->onDestroyed.disconnect( id ); } MouseEventHandlerId XYWindowMouseDown_connect( const MouseEventHandler& handler ){ return g_pParentWnd->GetXYWnd()->onMouseDown.connectFirst( handler ); } void XYWindowMouseDown_disconnect( MouseEventHandlerId id ){ g_pParentWnd->GetXYWnd()->onMouseDown.disconnect( id ); } // ============================================================================= // MainFrame class MainFrame* g_pParentWnd = 0; GtkWindow* MainFrame_getWindow(){ if ( g_pParentWnd == 0 ) { return 0; } return g_pParentWnd->m_window; } std::vector g_floating_windows; MainFrame::MainFrame() : m_window( 0 ), m_idleRedrawStatusText( RedrawStatusTextCaller( *this ) ){ m_pXYWnd = 0; m_pCamWnd = 0; m_pZWnd = 0; m_pYZWnd = 0; m_pXZWnd = 0; m_pActiveXY = 0; for ( int n = 0; n < c_status__count; ++n ) m_statusLabel[n] = 0; Create(); } MainFrame::~MainFrame(){ SaveWindowInfo(); gtk_widget_hide( GTK_WIDGET( m_window ) ); Shutdown(); for ( std::vector::iterator i = g_floating_windows.begin(); i != g_floating_windows.end(); ++i ) { gtk_widget_destroy( *i ); } gtk_widget_destroy( GTK_WIDGET( m_window ) ); } void MainFrame::SetActiveXY( XYWnd* p ){ if ( m_pActiveXY ) { m_pActiveXY->SetActive( false ); } m_pActiveXY = p; if ( m_pActiveXY ) { m_pActiveXY->SetActive( true ); } } void MainFrame_toggleFullscreen(){ GtkWindow* wnd = MainFrame_getWindow(); if( gdk_window_get_state( gtk_widget_get_window( GTK_WIDGET( wnd ) ) ) & GDK_WINDOW_STATE_FULLSCREEN ){ //some portion of buttsex, because gtk_window_unfullscreen doesn't work correctly after calling some modal window bool maximized = ( gdk_window_get_state( gtk_widget_get_window( GTK_WIDGET( wnd ) ) ) & GDK_WINDOW_STATE_MAXIMIZED ); gtk_window_unfullscreen( wnd ); if( maximized ){ gtk_window_unmaximize( wnd ); gtk_window_maximize( wnd ); } else{ gtk_window_move( wnd, g_layout_globals.m_position.x, g_layout_globals.m_position.y ); gtk_window_resize( wnd, g_layout_globals.m_position.w, g_layout_globals.m_position.h ); } } else{ gtk_window_fullscreen( wnd ); } } class MaximizeView { public: MaximizeView(): m_maximized( false ){ } void toggle(){ return m_maximized ? restore() : maximize(); } bool isMaximized(){ return m_maximized; } private: bool m_maximized; int m_vSplitPos; int m_vSplit2Pos; int m_hSplitPos; void restore(){ m_maximized = false; gtk_paned_set_position( GTK_PANED( g_pParentWnd->m_vSplit ), m_vSplitPos ); gtk_paned_set_position( GTK_PANED( g_pParentWnd->m_vSplit2 ), m_vSplit2Pos ); gtk_paned_set_position( GTK_PANED( g_pParentWnd->m_hSplit ), m_hSplitPos ); } void maximize(){ m_maximized = true; m_vSplitPos = gtk_paned_get_position( GTK_PANED( g_pParentWnd->m_vSplit ) ); m_vSplit2Pos = gtk_paned_get_position( GTK_PANED( g_pParentWnd->m_vSplit2 ) ); m_hSplitPos = gtk_paned_get_position( GTK_PANED( g_pParentWnd->m_hSplit ) ); int vSplitX, vSplitY, vSplit2X, vSplit2Y, hSplitX, hSplitY; gdk_window_get_origin( gtk_widget_get_window( g_pParentWnd->m_vSplit ), &vSplitX, &vSplitY ); gdk_window_get_origin( gtk_widget_get_window( g_pParentWnd->m_vSplit2 ), &vSplit2X, &vSplit2Y ); gdk_window_get_origin( gtk_widget_get_window( g_pParentWnd->m_hSplit ), &hSplitX, &hSplitY ); vSplitY += m_vSplitPos; vSplit2Y += m_vSplit2Pos; hSplitX += m_hSplitPos; int cur_x, cur_y; Sys_GetCursorPos( MainFrame_getWindow(), &cur_x, &cur_y ); if( cur_x > hSplitX ){ gtk_paned_set_position( GTK_PANED( g_pParentWnd->m_hSplit ), 0 ); } else{ gtk_paned_set_position( GTK_PANED( g_pParentWnd->m_hSplit ), 9999 ); } if( cur_y > vSplitY ){ gtk_paned_set_position( GTK_PANED( g_pParentWnd->m_vSplit ), 0 ); } else{ gtk_paned_set_position( GTK_PANED( g_pParentWnd->m_vSplit ), 9999 ); } if( cur_y > vSplit2Y ){ gtk_paned_set_position( GTK_PANED( g_pParentWnd->m_vSplit2 ), 0 ); } else{ gtk_paned_set_position( GTK_PANED( g_pParentWnd->m_vSplit2 ), 9999 ); } } }; MaximizeView g_maximizeview; void Maximize_View(){ if( g_pParentWnd != 0 && g_pParentWnd->m_vSplit != 0 && g_pParentWnd->m_vSplit2 != 0 && g_pParentWnd->m_hSplit != 0 ) g_maximizeview.toggle(); } GtkWindow* create_splash(){ GtkWindow* window = GTK_WINDOW( gtk_window_new( GTK_WINDOW_TOPLEVEL ) ); gtk_window_set_decorated( window, FALSE ); gtk_window_set_resizable( window, FALSE ); gtk_window_set_modal( window, TRUE ); gtk_window_set_default_size( window, -1, -1 ); gtk_window_set_position( window, GTK_WIN_POS_CENTER ); gtk_container_set_border_width( GTK_CONTAINER( window ), 0 ); GtkImage* image = new_local_image( "splash.png" ); gtk_widget_show( GTK_WIDGET( image ) ); gtk_container_add( GTK_CONTAINER( window ), GTK_WIDGET( image ) ); if( gtk_image_get_storage_type( image ) == GTK_IMAGE_PIXBUF ){ GdkBitmap* mask; GdkPixbuf* pix = gtk_image_get_pixbuf( image ); gdk_pixbuf_render_pixmap_and_mask( pix, NULL, &mask, 255 ); gtk_widget_shape_combine_mask ( GTK_WIDGET( window ), mask, 0, 0 ); } gtk_widget_set_size_request( GTK_WIDGET( window ), -1, -1 ); gtk_widget_show( GTK_WIDGET( window ) ); return window; } static GtkWindow *splash_screen = 0; void show_splash(){ splash_screen = create_splash(); process_gui(); } void hide_splash(){ gtk_widget_destroy( GTK_WIDGET( splash_screen ) ); } WindowPositionTracker g_posCamWnd; WindowPositionTracker g_posXYWnd; WindowPositionTracker g_posXZWnd; WindowPositionTracker g_posYZWnd; static gint mainframe_delete( GtkWidget *widget, GdkEvent *event, gpointer data ){ if ( ConfirmModified( "Exit Radiant" ) ) { gtk_main_quit(); } return TRUE; } gboolean toolbar_redirect_scroll( GtkWidget* widget, GdkEventScroll* event, gpointer user_data ){ //globalOutputStream() << "scroll\n"; if ( event->direction == GDK_SCROLL_UP ) { event->direction = GDK_SCROLL_RIGHT; } else if ( event->direction == GDK_SCROLL_DOWN ) { event->direction = GDK_SCROLL_LEFT; } return FALSE; } void user_shortcuts_init(){ StringOutputStream path( 256 ); path << SettingsPath_get() << g_pGameDescription->mGameFile << '/'; LoadCommandMap( path.c_str() ); SaveCommandMap( path.c_str() ); } void user_shortcuts_save(){ StringOutputStream path( 256 ); path << SettingsPath_get() << g_pGameDescription->mGameFile << '/'; SaveCommandMap( path.c_str() ); } void MainFrame::Create(){ GtkWindow* window = GTK_WINDOW( gtk_window_new( GTK_WINDOW_TOPLEVEL ) ); GlobalWindowObservers_connectTopLevel( window ); gtk_window_set_transient_for( splash_screen, window ); #if !defined( WIN32 ) { GdkPixbuf* pixbuf = pixbuf_new_from_file_with_mask( "bitmaps/icon.png" ); if ( pixbuf != 0 ) { gtk_window_set_default_icon( pixbuf ); g_object_unref( pixbuf ); } } #endif gtk_widget_add_events( GTK_WIDGET( window ), GDK_KEY_PRESS_MASK | GDK_KEY_RELEASE_MASK | GDK_FOCUS_CHANGE_MASK ); g_signal_connect( G_OBJECT( window ), "delete_event", G_CALLBACK( mainframe_delete ), this ); m_position_tracker.connect( window ); #if 0 g_mainframeWidgetFocusPrinter.connect( window ); g_mainframeFocusPrinter.connect( window ); #endif g_MainWindowActive.connect( window ); /* GlobalCommands_insert plugins commands */ GetPlugInMgr().Init( GTK_WIDGET( window ) ); /* then load shortcuts cfg */ user_shortcuts_init(); GtkWidget* vbox = gtk_vbox_new( FALSE, 0 ); gtk_container_add( GTK_CONTAINER( window ), vbox ); gtk_widget_show( vbox ); gtk_container_set_focus_chain( GTK_CONTAINER( vbox ), NULL ); global_accel_connect_window( window ); m_nCurrentStyle = (EViewStyle)g_Layout_viewStyle.m_value; register_shortcuts(); GtkMenuBar* main_menu = create_main_menu( CurrentStyle() ); gtk_box_pack_start( GTK_BOX( vbox ), GTK_WIDGET( main_menu ), FALSE, FALSE, 0 ); if( g_Layout_SingleToolbar.m_value ){ if( g_Layout_enableMainToolbar.m_value || g_Layout_enablePluginToolbar.m_value || g_Layout_enableFilterToolbar.m_value ){ GtkWidget* scr_win = gtk_scrolled_window_new( NULL, NULL ); gtk_container_set_border_width( GTK_CONTAINER( scr_win ), 0 ); gtk_scrolled_window_set_policy( GTK_SCROLLED_WINDOW( scr_win ), GTK_POLICY_AUTOMATIC, GTK_POLICY_NEVER ); GtkWidget* scrollbar = gtk_scrolled_window_get_hscrollbar( GTK_SCROLLED_WINDOW( scr_win ) ); gtk_widget_set_size_request( scrollbar, 0, 0 ); //gtk_widget_set_child_visible( scrollbar, FALSE ); //gtk_container_set_border_width( GTK_CONTAINER( scrollbar ), 0 ); gtk_scrolled_window_set_shadow_type( GTK_SCROLLED_WINDOW( scr_win ), GTK_SHADOW_NONE ); gtk_box_pack_start( GTK_BOX( vbox ), scr_win, FALSE, FALSE, 0 ); gtk_widget_show( scr_win ); g_signal_connect( G_OBJECT( scr_win ), "scroll_event", G_CALLBACK( toolbar_redirect_scroll ), 0 ); GtkWidget* hbox = gtk_hbox_new( FALSE, 3 ); gtk_widget_show( hbox ); gtk_scrolled_window_add_with_viewport( GTK_SCROLLED_WINDOW( scr_win ), hbox ); if( g_Layout_enableMainToolbar.m_value ){ GtkToolbar* main_toolbar = create_main_toolbar( CurrentStyle() ); gtk_box_pack_start( GTK_BOX( hbox ), GTK_WIDGET( main_toolbar ), TRUE, TRUE, 0 ); } if ( g_Layout_enableFilterToolbar.m_value ){ GtkToolbar* filter_toolbar = create_filter_toolbar(); gtk_box_pack_start( GTK_BOX( hbox ), GTK_WIDGET( filter_toolbar ), TRUE, TRUE, 0 ); } if ( g_Layout_enablePluginToolbar.m_value ){ GtkToolbar* plugin_toolbar = create_plugin_toolbar(); gtk_box_pack_start( GTK_BOX( hbox ), GTK_WIDGET( plugin_toolbar ), TRUE, TRUE, 0 ); } } } else{ if( g_Layout_enableMainToolbar.m_value ){ GtkToolbar* main_toolbar = create_main_toolbar( CurrentStyle() ); gtk_box_pack_start( GTK_BOX( vbox ), GTK_WIDGET( main_toolbar ), FALSE, FALSE, 0 ); } if ( g_Layout_enablePluginToolbar.m_value || g_Layout_enableFilterToolbar.m_value ){ GtkWidget* hbox = gtk_hbox_new( FALSE, 3 ); gtk_box_pack_start( GTK_BOX( vbox ), GTK_WIDGET( hbox ), FALSE, FALSE, 0 ); gtk_widget_show( hbox ); if ( g_Layout_enablePluginToolbar.m_value ){ GtkToolbar* plugin_toolbar = create_plugin_toolbar(); if ( g_Layout_enableFilterToolbar.m_value ){ gtk_box_pack_start( GTK_BOX( hbox ), GTK_WIDGET( plugin_toolbar ), FALSE, FALSE, 0 ); } else{ gtk_box_pack_start( GTK_BOX( hbox ), GTK_WIDGET( plugin_toolbar ), TRUE, TRUE, 0 ); } } if ( g_Layout_enableFilterToolbar.m_value ){ GtkToolbar* filter_toolbar = create_filter_toolbar(); gtk_box_pack_start( GTK_BOX( hbox ), GTK_WIDGET( filter_toolbar ), TRUE, TRUE, 0 ); } } } /*GtkToolbar* plugin_toolbar = create_plugin_toolbar(); if ( !g_Layout_enablePluginToolbar.m_value ) { gtk_widget_hide( GTK_WIDGET( plugin_toolbar ) ); }*/ GtkWidget* main_statusbar = create_main_statusbar( m_statusLabel ); gtk_box_pack_end( GTK_BOX( vbox ), main_statusbar, FALSE, TRUE, 2 ); GroupDialog_constructWindow( window ); /* want to realize it immediately; otherwise gtk paned splits positions wont be set correctly for floating group dlg */ gtk_widget_realize ( GTK_WIDGET( GroupDialog_getWindow() ) ); g_page_entity = GroupDialog_addPage( "Entities", EntityInspector_constructWindow( GroupDialog_getWindow() ), RawStringExportCaller( "Entities" ) ); if ( FloatingGroupDialog() ) { g_page_console = GroupDialog_addPage( "Console", Console_constructWindow( GroupDialog_getWindow() ), RawStringExportCaller( "Console" ) ); { GtkFrame* frame = create_framed_widget( TextureBrowser_constructWindow( GroupDialog_getWindow() ) ); g_page_textures = GroupDialog_addPage( "Textures", GTK_WIDGET( frame ), TextureBrowserExportTitleCaller() ); /* workaround for gtk 2.24 issue: not displayed glwidget after toggle */ g_object_set_data( G_OBJECT( g_page_textures ), "glwidget", TextureBrowser_getGLWidget() ); } } { GtkFrame* frame = create_framed_widget( ModelBrowser_constructWindow( GroupDialog_getWindow() ) ); g_page_models = GroupDialog_addPage( "Models", GTK_WIDGET( frame ), RawStringExportCaller( "Models" ) ); /* workaround for gtk 2.24 issue: not displayed glwidget after toggle */ g_object_set_data( G_OBJECT( g_page_models ), "glwidget", ModelBrowser_getGLWidget() ); } #ifdef WIN32 if ( g_multimon_globals.m_bStartOnPrimMon ) { PositionWindowOnPrimaryScreen( g_layout_globals.m_position ); } #endif window_set_position( window, g_layout_globals.m_position ); m_window = window; gtk_widget_show( GTK_WIDGET( window ) ); if ( CurrentStyle() == eRegular || CurrentStyle() == eRegularLeft ) { GtkWidget* hsplit = gtk_hpaned_new(); m_hSplit = hsplit; gtk_box_pack_start( GTK_BOX( vbox ), hsplit, TRUE, TRUE, 0 ); gtk_widget_show( hsplit ); { GtkWidget* vsplit = gtk_vpaned_new(); gtk_widget_show( vsplit ); m_vSplit = vsplit; GtkWidget* vsplit2 = gtk_vpaned_new(); gtk_widget_show( vsplit2 ); m_vSplit2 = vsplit2; if ( CurrentStyle() == eRegular ){ gtk_paned_pack1( GTK_PANED( hsplit ), vsplit, TRUE, TRUE ); gtk_paned_pack2( GTK_PANED( hsplit ), vsplit2, TRUE, TRUE ); } else{ gtk_paned_pack2( GTK_PANED( hsplit ), vsplit, TRUE, TRUE ); gtk_paned_pack1( GTK_PANED( hsplit ), vsplit2, TRUE, TRUE ); } // console GtkWidget* console_window = Console_constructWindow( window ); gtk_paned_pack2( GTK_PANED( vsplit ), console_window, TRUE, TRUE ); // xy m_pXYWnd = new XYWnd(); m_pXYWnd->SetViewType( XY ); GtkWidget* xy_window = GTK_WIDGET( create_framed_widget( m_pXYWnd->GetWidget() ) ); gtk_paned_pack1( GTK_PANED( vsplit ), xy_window, TRUE, TRUE ); { // camera m_pCamWnd = NewCamWnd(); GlobalCamera_setCamWnd( *m_pCamWnd ); CamWnd_setParent( *m_pCamWnd, window ); GtkFrame* camera_window = create_framed_widget( CamWnd_getWidget( *m_pCamWnd ) ); gtk_paned_pack1( GTK_PANED( vsplit2 ), GTK_WIDGET( camera_window ), TRUE, TRUE ); // textures GtkFrame* texture_window = create_framed_widget( TextureBrowser_constructWindow( window ) ); gtk_paned_pack2( GTK_PANED( vsplit2 ), GTK_WIDGET( texture_window ), TRUE, TRUE ); } } } else if ( CurrentStyle() == eFloating ) { { GtkWindow* window = create_persistent_floating_window( "Camera", m_window ); global_accel_connect_window( window ); g_posCamWnd.connect( window ); m_pCamWnd = NewCamWnd(); GlobalCamera_setCamWnd( *m_pCamWnd ); { GtkFrame* frame = create_framed_widget( CamWnd_getWidget( *m_pCamWnd ) ); gtk_container_add( GTK_CONTAINER( window ), GTK_WIDGET( frame ) ); } CamWnd_setParent( *m_pCamWnd, window ); CamWnd_Shown_Construct( window ); /* workaround for gtk 2.24 issue: not displayed glwidget after toggle */ g_object_set_data( G_OBJECT( window ), "glwidget", CamWnd_getWidget( *m_pCamWnd ) ); g_floating_windows.push_back( GTK_WIDGET( window ) ); } { GtkWindow* window = create_persistent_floating_window( ViewType_getTitle( XY ), m_window ); global_accel_connect_window( window ); g_posXYWnd.connect( window ); m_pXYWnd = new XYWnd(); m_pXYWnd->m_parent = window; m_pXYWnd->SetViewType( XY ); { GtkFrame* frame = create_framed_widget( m_pXYWnd->GetWidget() ); gtk_container_add( GTK_CONTAINER( window ), GTK_WIDGET( frame ) ); } XY_Top_Shown_Construct( window ); /* workaround for gtk 2.24 issue: not displayed glwidget after toggle */ g_object_set_data( G_OBJECT( window ), "glwidget", m_pXYWnd->GetWidget() ); g_floating_windows.push_back( GTK_WIDGET( window ) ); } { GtkWindow* window = create_persistent_floating_window( ViewType_getTitle( XZ ), m_window ); global_accel_connect_window( window ); g_posXZWnd.connect( window ); m_pXZWnd = new XYWnd(); m_pXZWnd->m_parent = window; m_pXZWnd->SetViewType( XZ ); { GtkFrame* frame = create_framed_widget( m_pXZWnd->GetWidget() ); gtk_container_add( GTK_CONTAINER( window ), GTK_WIDGET( frame ) ); } XZ_Front_Shown_Construct( window ); /* workaround for gtk 2.24 issue: not displayed glwidget after toggle */ g_object_set_data( G_OBJECT( window ), "glwidget", m_pXZWnd->GetWidget() ); g_floating_windows.push_back( GTK_WIDGET( window ) ); } { GtkWindow* window = create_persistent_floating_window( ViewType_getTitle( YZ ), m_window ); global_accel_connect_window( window ); g_posYZWnd.connect( window ); m_pYZWnd = new XYWnd(); m_pYZWnd->m_parent = window; m_pYZWnd->SetViewType( YZ ); { GtkFrame* frame = create_framed_widget( m_pYZWnd->GetWidget() ); gtk_container_add( GTK_CONTAINER( window ), GTK_WIDGET( frame ) ); } YZ_Side_Shown_Construct( window ); /* workaround for gtk 2.24 issue: not displayed glwidget after toggle */ g_object_set_data( G_OBJECT( window ), "glwidget", m_pYZWnd->GetWidget() ); g_floating_windows.push_back( GTK_WIDGET( window ) ); } m_vSplit = 0; m_hSplit = 0; m_vSplit2 = 0; GroupDialog_show(); } else // 4 way { m_pCamWnd = NewCamWnd(); GlobalCamera_setCamWnd( *m_pCamWnd ); CamWnd_setParent( *m_pCamWnd, window ); GtkWidget* camera = CamWnd_getWidget( *m_pCamWnd ); m_pYZWnd = new XYWnd(); m_pYZWnd->SetViewType( YZ ); GtkWidget* yz = m_pYZWnd->GetWidget(); m_pXYWnd = new XYWnd(); m_pXYWnd->SetViewType( XY ); GtkWidget* xy = m_pXYWnd->GetWidget(); m_pXZWnd = new XYWnd(); m_pXZWnd->SetViewType( XZ ); GtkWidget* xz = m_pXZWnd->GetWidget(); m_hSplit = create_split_views( camera, yz, xy, xz, m_vSplit, m_vSplit2 ); gtk_box_pack_start( GTK_BOX( vbox ), m_hSplit, TRUE, TRUE, 0 ); } EntityList_constructWindow( window ); PreferencesDialog_constructWindow( window ); FindTextureDialog_constructWindow( window ); SurfaceInspector_constructWindow( window ); PatchInspector_constructWindow( window ); SetActiveXY( m_pXYWnd ); AddGridChangeCallback( SetGridStatusCaller( *this ) ); AddGridChangeCallback( FreeCaller() ); g_defaultToolMode = DragMode; g_defaultToolMode(); SetStatusText( c_status_command, c_TranslateMode_status ); EverySecondTimer_enable(); if ( g_layout_globals.nState & GDK_WINDOW_STATE_MAXIMIZED || g_layout_globals.nState & GDK_WINDOW_STATE_ICONIFIED ) { gtk_window_maximize( window ); } if ( g_layout_globals.nState & GDK_WINDOW_STATE_FULLSCREEN ) { gtk_window_fullscreen( window ); } process_gui(); //window is maximized/fullscreened by window manager, which may be not instant if ( !FloatingGroupDialog() ) { gtk_paned_set_position( GTK_PANED( m_vSplit ), g_layout_globals.nXYHeight ); if ( CurrentStyle() == eRegular ) { gtk_paned_set_position( GTK_PANED( m_hSplit ), g_layout_globals.nXYWidth ); } else { gtk_paned_set_position( GTK_PANED( m_hSplit ), g_layout_globals.nCamWidth ); } gtk_paned_set_position( GTK_PANED( m_vSplit2 ), g_layout_globals.nCamHeight ); } //GlobalShortcuts_reportUnregistered(); } void MainFrame::SaveWindowInfo(){ //restore good state first if( gdk_window_get_state( gtk_widget_get_window( GTK_WIDGET( m_window ) ) ) & GDK_WINDOW_STATE_ICONIFIED ){ gtk_window_deiconify( m_window ); } if( g_maximizeview.isMaximized() ){ g_maximizeview.toggle(); } if ( !FloatingGroupDialog() ) { g_layout_globals.nXYHeight = gtk_paned_get_position( GTK_PANED( m_vSplit ) ); if ( CurrentStyle() != eRegular ) { g_layout_globals.nCamWidth = gtk_paned_get_position( GTK_PANED( m_hSplit ) ); } else { g_layout_globals.nXYWidth = gtk_paned_get_position( GTK_PANED( m_hSplit ) ); } g_layout_globals.nCamHeight = gtk_paned_get_position( GTK_PANED( m_vSplit2 ) ); } if( gdk_window_get_state( gtk_widget_get_window( GTK_WIDGET( m_window ) ) ) == 0 ){ g_layout_globals.m_position = m_position_tracker.getPosition(); } g_layout_globals.nState = gdk_window_get_state( gtk_widget_get_window( GTK_WIDGET( m_window ) ) ); } void MainFrame::Shutdown(){ EverySecondTimer_disable(); EntityList_destroyWindow(); delete m_pXYWnd; m_pXYWnd = 0; delete m_pYZWnd; m_pYZWnd = 0; delete m_pXZWnd; m_pXZWnd = 0; ModelBrowser_destroyWindow(); TextureBrowser_destroyWindow(); DeleteCamWnd( m_pCamWnd ); m_pCamWnd = 0; PreferencesDialog_destroyWindow(); SurfaceInspector_destroyWindow(); FindTextureDialog_destroyWindow(); PatchInspector_destroyWindow(); g_DbgDlg.destroyWindow(); // destroying group-dialog last because it may contain texture-browser GroupDialog_destroyWindow(); user_shortcuts_save(); } void MainFrame::RedrawStatusText(){ for( int i = 0; i < c_status__count; ++i ) gtk_label_set_text( GTK_LABEL( m_statusLabel[i] ), m_status[i].c_str() ); } void MainFrame::UpdateStatusText(){ m_idleRedrawStatusText.queueDraw(); } void MainFrame::SetStatusText( int status_n, const char* status ){ m_status[status_n] = status; UpdateStatusText(); } void Sys_Status( const char* status ){ if ( g_pParentWnd ) g_pParentWnd->SetStatusText( c_status_command, status ); } void brushCountChanged( const Selectable& selectable ){ QE_brushCountChanged(); } //int getRotateIncrement(){ // return static_cast( g_si_globals.rotate ); //} int getFarClipDistance(){ return g_camwindow_globals.m_nCubicScale; } float ( *GridStatus_getGridSize )() = GetGridSize; //int ( *GridStatus_getRotateIncrement )() = getRotateIncrement; int ( *GridStatus_getFarClipDistance )() = getFarClipDistance; bool ( *GridStatus_getTextureLockEnabled )(); const char* ( *GridStatus_getTexdefTypeIdLabel )(); void MainFrame::SetGridStatus(){ StringOutputStream status( 64 ); const char* lock = ( GridStatus_getTextureLockEnabled() ) ? "ON " : "OFF "; status << ( GetSnapGridSize() > 0 ? "G:" : "g:" ) << GridStatus_getGridSize() << " F:" << GridStatus_getTexdefTypeIdLabel() << " C:" << GridStatus_getFarClipDistance() << " L:" << lock; SetStatusText( c_status_grid, status.c_str() ); } void GridStatus_changed(){ if ( g_pParentWnd != 0 ) { g_pParentWnd->SetGridStatus(); } } CopiedString g_strOpenGLFont = ""; void OpenGLFont_select(){ CopiedString newfont; if( OpenGLFont_dialog( GTK_WIDGET( MainFrame_getWindow() ), g_strOpenGLFont.c_str(), newfont ) ){ { ScopeDisableScreenUpdates disableScreenUpdates( "Processing...", "Changing OpenGL Font" ); delete GlobalOpenGL().m_font; g_strOpenGLFont = newfont; GlobalOpenGL().m_font = glfont_create( g_strOpenGLFont.c_str() ); } UpdateAllWindows(); } } void GlobalGL_sharedContextCreated(){ // report OpenGL information globalOutputStream() << "GL_VENDOR: " << reinterpret_cast( glGetString( GL_VENDOR ) ) << "\n"; globalOutputStream() << "GL_RENDERER: " << reinterpret_cast( glGetString( GL_RENDERER ) ) << "\n"; globalOutputStream() << "GL_VERSION: " << reinterpret_cast( glGetString( GL_VERSION ) ) << "\n"; globalOutputStream() << "GL_EXTENSIONS: " << reinterpret_cast( glGetString( GL_EXTENSIONS ) ) << "\n"; QGL_sharedContextCreated( GlobalOpenGL() ); ShaderCache_extensionsInitialised(); GlobalShaderCache().realise(); Textures_Realise(); GlobalOpenGL().m_font = glfont_create( g_strOpenGLFont.c_str() ); } void GlobalGL_sharedContextDestroyed(){ Textures_Unrealise(); GlobalShaderCache().unrealise(); QGL_sharedContextDestroyed( GlobalOpenGL() ); } void Layout_constructPreferences( PreferencesPage& page ){ { const char* layouts[] = { "window1.png", "window2.png", "window3.png", "window4.png" }; page.appendRadioIcons( "Window Layout", StringArrayRange( layouts ), LatchedImportCaller( g_Layout_viewStyle ), IntExportCaller( g_Layout_viewStyle.m_latched ) ); } page.appendCheckBox( "", "Detachable Menus", LatchedImportCaller( g_Layout_enableDetachableMenus ), BoolExportCaller( g_Layout_enableDetachableMenus.m_latched ) ); page.appendCheckBox( "", "Main Toolbar", LatchedImportCaller( g_Layout_enableMainToolbar ), BoolExportCaller( g_Layout_enableMainToolbar.m_latched ) ); if ( !string_empty( g_pGameDescription->getKeyValue( "no_patch" ) ) ) { page.appendCheckBox( "", "Patch Toolbar", LatchedImportCaller( g_Layout_enablePatchToolbar ), BoolExportCaller( g_Layout_enablePatchToolbar.m_latched ) ); } page.appendCheckBox( "", "Plugin Toolbar", LatchedImportCaller( g_Layout_enablePluginToolbar ), BoolExportCaller( g_Layout_enablePluginToolbar.m_latched ) ); page.appendCheckBox( "", "Filter Toolbar", LatchedImportCaller( g_Layout_enableFilterToolbar ), BoolExportCaller( g_Layout_enableFilterToolbar.m_latched ) ); page.appendCheckBox( "", "Single Scrollable Toolbar", LatchedImportCaller( g_Layout_SingleToolbar ), BoolExportCaller( g_Layout_SingleToolbar.m_latched ) ); } void Layout_constructPage( PreferenceGroup& group ){ PreferencesPage page( group.createPage( "Layout", "Layout Preferences" ) ); Layout_constructPreferences( page ); } void Layout_registerPreferencesPage(){ PreferencesDialog_addInterfacePage( FreeCaller1() ); } void FocusAllViews(){ XY_Centralize(); //using centralizing here, not focusing function GlobalCamera_FocusOnSelected(); } #include "preferencesystem.h" #include "stringio.h" void MainFrame_Construct(){ GlobalCommands_insert( "OpenManual", FreeCaller(), Accelerator( GDK_KEY_F1 ) ); GlobalCommands_insert( "NewMap", FreeCaller() ); GlobalCommands_insert( "OpenMap", FreeCaller(), Accelerator( 'O', GDK_CONTROL_MASK ) ); GlobalCommands_insert( "ImportMap", FreeCaller() ); GlobalCommands_insert( "SaveMap", FreeCaller(), Accelerator( 'S', GDK_CONTROL_MASK ) ); GlobalCommands_insert( "SaveMapAs", FreeCaller() ); GlobalCommands_insert( "SaveSelected", FreeCaller() ); GlobalCommands_insert( "SaveRegion", FreeCaller() ); GlobalCommands_insert( "RefreshReferences", FreeCaller() ); GlobalCommands_insert( "ProjectSettings", FreeCaller() ); GlobalCommands_insert( "CheckForUpdate", FreeCaller() ); GlobalCommands_insert( "Exit", FreeCaller() ); GlobalCommands_insert( "Undo", FreeCaller(), Accelerator( 'Z', GDK_CONTROL_MASK ) ); GlobalCommands_insert( "Redo", FreeCaller(), Accelerator( 'Y', GDK_CONTROL_MASK ) ); GlobalCommands_insert( "Redo2", FreeCaller(), Accelerator( 'Z', (GdkModifierType)( GDK_SHIFT_MASK | GDK_CONTROL_MASK ) ) ); GlobalCommands_insert( "Copy", FreeCaller(), Accelerator( 'C', GDK_CONTROL_MASK ) ); GlobalCommands_insert( "Paste", FreeCaller(), Accelerator( 'V', GDK_CONTROL_MASK ) ); GlobalCommands_insert( "PasteToCamera", FreeCaller(), Accelerator( 'V', GDK_SHIFT_MASK ) ); GlobalCommands_insert( "MoveToCamera", FreeCaller(), Accelerator( 'V', (GdkModifierType)( GDK_SHIFT_MASK | GDK_CONTROL_MASK ) ) ); GlobalCommands_insert( "CloneSelection", FreeCaller(), Accelerator( GDK_KEY_space ) ); GlobalCommands_insert( "CloneSelectionAndMakeUnique", FreeCaller(), Accelerator( GDK_KEY_space, GDK_SHIFT_MASK ) ); GlobalCommands_insert( "DeleteSelection2", FreeCaller(), Accelerator( GDK_KEY_BackSpace ) ); GlobalCommands_insert( "DeleteSelection", FreeCaller(), Accelerator( 'Z' ) ); GlobalCommands_insert( "RepeatTransforms", FreeCaller(), Accelerator( 'R', GDK_CONTROL_MASK ) ); // GlobalCommands_insert( "ParentSelection", FreeCaller() ); GlobalCommands_insert( "UnSelectSelection2", FreeCaller(), Accelerator( GDK_KEY_Escape ) ); GlobalCommands_insert( "UnSelectSelection", FreeCaller(), Accelerator( 'C' ) ); GlobalCommands_insert( "InvertSelection", FreeCaller(), Accelerator( 'I' ) ); GlobalCommands_insert( "SelectInside", FreeCaller() ); GlobalCommands_insert( "SelectTouching", FreeCaller() ); GlobalCommands_insert( "ExpandSelectionToPrimitives", FreeCaller(), Accelerator( 'E', GDK_CONTROL_MASK ) ); GlobalCommands_insert( "ExpandSelectionToEntities", FreeCaller(), Accelerator( 'E', GDK_SHIFT_MASK ) ); GlobalCommands_insert( "SelectConnectedEntities", FreeCaller(), Accelerator( 'E', (GdkModifierType)( GDK_SHIFT_MASK | GDK_CONTROL_MASK ) ) ); GlobalCommands_insert( "Preferences", FreeCaller(), Accelerator( 'P' ) ); GlobalCommands_insert( "ToggleConsole", FreeCaller(), Accelerator( 'O' ) ); GlobalCommands_insert( "ToggleEntityInspector", FreeCaller(), Accelerator( 'N' ) ); GlobalCommands_insert( "ToggleModelBrowser", FreeCaller(), Accelerator( '/' ) ); GlobalCommands_insert( "EntityList", FreeCaller(), Accelerator( 'L' ) ); // GlobalCommands_insert( "ShowHidden", FreeCaller(), Accelerator( 'H', GDK_SHIFT_MASK ) ); // GlobalCommands_insert( "HideSelected", FreeCaller(), Accelerator( 'H' ) ); Select_registerCommands(); GlobalToggles_insert( "DragVertices", FreeCaller(), ToggleItem::AddCallbackCaller( g_vertexMode_button ), Accelerator( 'V' ) ); GlobalToggles_insert( "DragEdges", FreeCaller(), ToggleItem::AddCallbackCaller( g_edgeMode_button ), Accelerator( 'E' ) ); GlobalToggles_insert( "DragFaces", FreeCaller(), ToggleItem::AddCallbackCaller( g_faceMode_button ), Accelerator( 'F' ) ); GlobalCommands_insert( "ArbitraryRotation", FreeCaller(), Accelerator( 'R', GDK_SHIFT_MASK ) ); GlobalCommands_insert( "ArbitraryScale", FreeCaller(), Accelerator( 'S', (GdkModifierType)( GDK_SHIFT_MASK | GDK_CONTROL_MASK ) ) ); GlobalCommands_insert( "BuildMenuCustomize", FreeCaller() ); GlobalCommands_insert( "Build_runRecentExecutedBuild", FreeCaller(), Accelerator( GDK_KEY_F5 ) ); GlobalCommands_insert( "FindBrush", FreeCaller() ); GlobalCommands_insert( "MapInfo", FreeCaller(), Accelerator( 'M' ) ); GlobalToggles_insert( "ToggleClipper", FreeCaller(), ToggleItem::AddCallbackCaller( g_clipper_button ), Accelerator( 'X' ) ); GlobalToggles_insert( "MouseTranslate", FreeCaller(), ToggleItem::AddCallbackCaller( g_translatemode_button ), Accelerator( 'W' ) ); GlobalToggles_insert( "MouseRotate", FreeCaller(), ToggleItem::AddCallbackCaller( g_rotatemode_button ), Accelerator( 'R' ) ); GlobalToggles_insert( "MouseScale", FreeCaller(), ToggleItem::AddCallbackCaller( g_scalemode_button ) ); GlobalToggles_insert( "MouseTransform", FreeCaller(), ToggleItem::AddCallbackCaller( g_skewmode_button ) ); GlobalToggles_insert( "MouseDrag", FreeCaller(), ToggleItem::AddCallbackCaller( g_dragmode_button ) ); GlobalToggles_insert( "MouseBuild", FreeCaller(), ToggleItem::AddCallbackCaller( g_build_button ), Accelerator( 'B' ) ); GlobalToggles_insert( "MouseUV", FreeCaller(), ToggleItem::AddCallbackCaller( g_uv_button ), Accelerator( Accelerator( 'G' ) ) ); GlobalCommands_insert( "MouseRotateOrScale", FreeCaller() ); GlobalCommands_insert( "MouseDragOrTransform", FreeCaller(), Accelerator( 'Q' ) ); GlobalCommands_insert( "gtkThemeDlg", FreeCaller() ); GlobalCommands_insert( "OpenGLFont", FreeCaller() ); GlobalCommands_insert( "ColorSchemeOriginal", FreeCaller() ); GlobalCommands_insert( "ColorSchemeQER", FreeCaller() ); GlobalCommands_insert( "ColorSchemeBlackAndGreen", FreeCaller() ); GlobalCommands_insert( "ColorSchemeYdnar", FreeCaller() ); GlobalCommands_insert( "ColorSchemeBlender", FreeCaller() ); GlobalCommands_insert( "ColorSchemeAdwaitaDark", FreeCaller() ); GlobalCommands_insert( "ChooseTextureBackgroundColor", makeCallback( g_ColoursMenu.m_textureback ) ); GlobalCommands_insert( "ChooseGridBackgroundColor", makeCallback( g_ColoursMenu.m_xyback ) ); GlobalCommands_insert( "ChooseGridMajorColor", makeCallback( g_ColoursMenu.m_gridmajor ) ); GlobalCommands_insert( "ChooseGridMinorColor", makeCallback( g_ColoursMenu.m_gridminor ) ); GlobalCommands_insert( "ChooseGridTextColor", makeCallback( g_ColoursMenu.m_gridtext ) ); GlobalCommands_insert( "ChooseGridBlockColor", makeCallback( g_ColoursMenu.m_gridblock ) ); GlobalCommands_insert( "ChooseBrushColor", makeCallback( g_ColoursMenu.m_brush ) ); GlobalCommands_insert( "ChooseCameraBackgroundColor", makeCallback( g_ColoursMenu.m_cameraback ) ); GlobalCommands_insert( "ChooseSelectedBrushColor", makeCallback( g_ColoursMenu.m_selectedbrush ) ); GlobalCommands_insert( "ChooseCameraSelectedBrushColor", makeCallback( g_ColoursMenu.m_selectedbrush3d ) ); GlobalCommands_insert( "ChooseClipperColor", makeCallback( g_ColoursMenu.m_clipper ) ); GlobalCommands_insert( "ChooseOrthoViewNameColor", makeCallback( g_ColoursMenu.m_viewname ) ); GlobalCommands_insert( "Fullscreen", FreeCaller(), Accelerator( GDK_KEY_F11 ) ); GlobalCommands_insert( "MaximizeView", FreeCaller(), Accelerator( GDK_KEY_F12 ) ); GlobalCommands_insert( "CSGSubtract", FreeCaller(), Accelerator( 'U', GDK_SHIFT_MASK ) ); GlobalCommands_insert( "CSGMerge", FreeCaller() ); GlobalCommands_insert( "CSGWrapMerge", FreeCaller(), Accelerator( 'U', GDK_CONTROL_MASK ) ); GlobalCommands_insert( "CSGroom", FreeCaller() ); GlobalCommands_insert( "CSGTool", FreeCaller() ); Grid_registerCommands(); GlobalCommands_insert( "SnapToGrid", FreeCaller(), Accelerator( 'G', GDK_CONTROL_MASK ) ); GlobalCommands_insert( "SelectAllOfType", FreeCaller(), Accelerator( 'A', GDK_SHIFT_MASK ) ); GlobalCommands_insert( "TexRotateClock", FreeCaller(), Accelerator( GDK_KEY_Next, GDK_SHIFT_MASK ) ); GlobalCommands_insert( "TexRotateCounter", FreeCaller(), Accelerator( GDK_KEY_Prior, GDK_SHIFT_MASK ) ); GlobalCommands_insert( "TexScaleUp", FreeCaller(), Accelerator( GDK_KEY_Up, GDK_CONTROL_MASK ) ); GlobalCommands_insert( "TexScaleDown", FreeCaller(), Accelerator( GDK_KEY_Down, GDK_CONTROL_MASK ) ); GlobalCommands_insert( "TexScaleLeft", FreeCaller(), Accelerator( GDK_KEY_Left, GDK_CONTROL_MASK ) ); GlobalCommands_insert( "TexScaleRight", FreeCaller(), Accelerator( GDK_KEY_Right, GDK_CONTROL_MASK ) ); GlobalCommands_insert( "TexShiftUp", FreeCaller(), Accelerator( GDK_KEY_Up, GDK_SHIFT_MASK ) ); GlobalCommands_insert( "TexShiftDown", FreeCaller(), Accelerator( GDK_KEY_Down, GDK_SHIFT_MASK ) ); GlobalCommands_insert( "TexShiftLeft", FreeCaller(), Accelerator( GDK_KEY_Left, GDK_SHIFT_MASK ) ); GlobalCommands_insert( "TexShiftRight", FreeCaller(), Accelerator( GDK_KEY_Right, GDK_SHIFT_MASK ) ); GlobalCommands_insert( "MoveSelectionDOWN", FreeCaller(), Accelerator( GDK_KEY_KP_Subtract ) ); GlobalCommands_insert( "MoveSelectionUP", FreeCaller(), Accelerator( GDK_KEY_KP_Add ) ); GlobalCommands_insert( "SelectNudgeLeft", FreeCaller(), Accelerator( GDK_KEY_Left, GDK_MOD1_MASK ) ); GlobalCommands_insert( "SelectNudgeRight", FreeCaller(), Accelerator( GDK_KEY_Right, GDK_MOD1_MASK ) ); GlobalCommands_insert( "SelectNudgeUp", FreeCaller(), Accelerator( GDK_KEY_Up, GDK_MOD1_MASK ) ); GlobalCommands_insert( "SelectNudgeDown", FreeCaller(), Accelerator( GDK_KEY_Down, GDK_MOD1_MASK ) ); Patch_registerCommands(); XYShow_registerCommands(); typedef FreeCaller1 ComponentModeSelectionChangedCaller; GlobalSelectionSystem().addSelectionChangeCallback( ComponentModeSelectionChangedCaller() ); GlobalPreferenceSystem().registerPreference( "DetachableMenus", makeBoolStringImportCallback( LatchedAssignCaller( g_Layout_enableDetachableMenus ) ), BoolExportStringCaller( g_Layout_enableDetachableMenus.m_latched ) ); GlobalPreferenceSystem().registerPreference( "MainToolBar", makeBoolStringImportCallback( LatchedAssignCaller( g_Layout_enableMainToolbar ) ), BoolExportStringCaller( g_Layout_enableMainToolbar.m_latched ) ); GlobalPreferenceSystem().registerPreference( "PatchToolBar", makeBoolStringImportCallback( LatchedAssignCaller( g_Layout_enablePatchToolbar ) ), BoolExportStringCaller( g_Layout_enablePatchToolbar.m_latched ) ); GlobalPreferenceSystem().registerPreference( "PluginToolBar", makeBoolStringImportCallback( LatchedAssignCaller( g_Layout_enablePluginToolbar ) ), BoolExportStringCaller( g_Layout_enablePluginToolbar.m_latched ) ); GlobalPreferenceSystem().registerPreference( "FilterToolBar", makeBoolStringImportCallback( LatchedAssignCaller( g_Layout_enableFilterToolbar ) ), BoolExportStringCaller( g_Layout_enableFilterToolbar.m_latched ) ); GlobalPreferenceSystem().registerPreference( "SingleToolBar", makeBoolStringImportCallback( LatchedAssignCaller( g_Layout_SingleToolbar ) ), BoolExportStringCaller( g_Layout_SingleToolbar.m_latched ) ); GlobalPreferenceSystem().registerPreference( "QE4StyleWindows", makeIntStringImportCallback( LatchedAssignCaller( g_Layout_viewStyle ) ), IntExportStringCaller( g_Layout_viewStyle.m_latched ) ); GlobalPreferenceSystem().registerPreference( "XYHeight", IntImportStringCaller( g_layout_globals.nXYHeight ), IntExportStringCaller( g_layout_globals.nXYHeight ) ); GlobalPreferenceSystem().registerPreference( "XYWidth", IntImportStringCaller( g_layout_globals.nXYWidth ), IntExportStringCaller( g_layout_globals.nXYWidth ) ); GlobalPreferenceSystem().registerPreference( "CamWidth", IntImportStringCaller( g_layout_globals.nCamWidth ), IntExportStringCaller( g_layout_globals.nCamWidth ) ); GlobalPreferenceSystem().registerPreference( "CamHeight", IntImportStringCaller( g_layout_globals.nCamHeight ), IntExportStringCaller( g_layout_globals.nCamHeight ) ); GlobalPreferenceSystem().registerPreference( "State", IntImportStringCaller( g_layout_globals.nState ), IntExportStringCaller( g_layout_globals.nState ) ); GlobalPreferenceSystem().registerPreference( "PositionX", IntImportStringCaller( g_layout_globals.m_position.x ), IntExportStringCaller( g_layout_globals.m_position.x ) ); GlobalPreferenceSystem().registerPreference( "PositionY", IntImportStringCaller( g_layout_globals.m_position.y ), IntExportStringCaller( g_layout_globals.m_position.y ) ); GlobalPreferenceSystem().registerPreference( "Width", IntImportStringCaller( g_layout_globals.m_position.w ), IntExportStringCaller( g_layout_globals.m_position.w ) ); GlobalPreferenceSystem().registerPreference( "Height", IntImportStringCaller( g_layout_globals.m_position.h ), IntExportStringCaller( g_layout_globals.m_position.h ) ); GlobalPreferenceSystem().registerPreference( "CamWnd", WindowPositionTrackerImportStringCaller( g_posCamWnd ), WindowPositionTrackerExportStringCaller( g_posCamWnd ) ); GlobalPreferenceSystem().registerPreference( "XYWnd", WindowPositionTrackerImportStringCaller( g_posXYWnd ), WindowPositionTrackerExportStringCaller( g_posXYWnd ) ); GlobalPreferenceSystem().registerPreference( "YZWnd", WindowPositionTrackerImportStringCaller( g_posYZWnd ), WindowPositionTrackerExportStringCaller( g_posYZWnd ) ); GlobalPreferenceSystem().registerPreference( "XZWnd", WindowPositionTrackerImportStringCaller( g_posXZWnd ), WindowPositionTrackerExportStringCaller( g_posXZWnd ) ); GlobalPreferenceSystem().registerPreference( "NudgeAfterClone", BoolImportStringCaller( g_bNudgeAfterClone ), BoolExportStringCaller( g_bNudgeAfterClone ) ); GlobalPreferenceSystem().registerPreference( "OpenGLFont", CopiedStringImportStringCaller( g_strOpenGLFont ), CopiedStringExportStringCaller( g_strOpenGLFont ) ); GlobalPreferenceSystem().registerPreference( "ExtraResoucePath", CopiedStringImportStringCaller( g_strExtraResourcePath ), CopiedStringExportStringCaller( g_strExtraResourcePath ) ); GlobalPreferenceSystem().registerPreference( "EnginePath", CopiedStringImportStringCaller( g_strEnginePath ), CopiedStringExportStringCaller( g_strEnginePath ) ); if ( g_strEnginePath.empty() ) { g_strEnginePath_was_empty_1st_start = true; const char* ENGINEPATH_ATTRIBUTE = #if defined( WIN32 ) "enginepath_win32" #elif defined( __APPLE__ ) "enginepath_macos" #elif defined( __linux__ ) || defined ( __FreeBSD__ ) "enginepath_linux" #else #error "unknown platform" #endif ; g_strEnginePath = StringOutputStream( 256 )( DirectoryCleaned( g_pGameDescription->getRequiredKeyValue( ENGINEPATH_ATTRIBUTE ) ) ).c_str(); } Layout_registerPreferencesPage(); Paths_registerPreferencesPage(); PreferencesDialog_addSettingsPreferences( FreeCaller1() ); g_brushCount.setCountChangedCallback( FreeCaller() ); g_patchCount.setCountChangedCallback( FreeCaller() ); g_entityCount.setCountChangedCallback( FreeCaller() ); GlobalEntityCreator().setCounter( &g_entityCount ); GlobalSelectionSystem().addSelectionChangeCallback( FreeCaller1() ); GLWidget_sharedContextCreated = GlobalGL_sharedContextCreated; GLWidget_sharedContextDestroyed = GlobalGL_sharedContextDestroyed; GlobalEntityClassManager().attach( g_WorldspawnColourEntityClassObserver ); } void MainFrame_Destroy(){ GlobalEntityClassManager().detach( g_WorldspawnColourEntityClassObserver ); GlobalEntityCreator().setCounter( 0 ); g_entityCount.setCountChangedCallback( Callback() ); g_patchCount.setCountChangedCallback( Callback() ); g_brushCount.setCountChangedCallback( Callback() ); }