netradiant-custom/radiant/mru.cpp
2024-01-24 17:09:04 +06:00

259 lines
6.3 KiB
C++

/*
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
*/
#include "mru.h"
#include <cstring>
#include <cstdio>
#include "os/file.h"
#include "generic/callback.h"
#include "stream/stringstream.h"
#include "gtkutil/menu.h"
#include "map.h"
#include "qe3.h"
#define MRU_MAX 9
namespace {
QAction *MRU_items[MRU_MAX];
std::size_t MRU_used;
typedef CopiedString MRU_filename_t;
MRU_filename_t MRU_filenames[MRU_MAX];
typedef const char* MRU_key_t;
MRU_key_t MRU_keys[MRU_MAX] = { "File0", "File1", "File2", "File3", "File4", "File5", "File6", "File7", "File8" };
}
inline const char* MRU_GetText( std::size_t index ){
return MRU_filenames[index].c_str();
}
class EscapedMnemonic
{
StringBuffer m_buffer;
public:
EscapedMnemonic( std::size_t capacity ) : m_buffer( capacity ){
m_buffer.push_back( '&' );
}
const char* c_str() const {
return m_buffer.c_str();
}
void push_back( char c ){ // not escaped
m_buffer.push_back( c );
}
std::size_t write( const char* buffer, std::size_t length ){
for ( const char* end = buffer + length; buffer != end; ++buffer )
{
if ( *buffer == '&' ) {
m_buffer.push_back( '&' );
}
m_buffer.push_back( *buffer );
}
return length;
}
};
template<typename T>
inline EscapedMnemonic& operator<<( EscapedMnemonic& ostream, const T& t ){
return ostream_write( ostream, t );
}
void MRU_updateWidget( std::size_t index, const char *filename ){
EscapedMnemonic mnemonic( 64 );
mnemonic << index + 1 << "- " << filename;
MRU_items[index]->setText( mnemonic.c_str() );
}
void MRU_SetText( std::size_t index, const char *filename ){
MRU_filenames[index] = filename;
MRU_updateWidget( index, filename );
}
void MRU_AddFile( const char *str ){
std::size_t i;
const char* text;
// check if file is already in our list
for ( i = 0; i < MRU_used; i++ )
{
text = MRU_GetText( i );
if ( strcmp( text, str ) == 0 ) {
// reorder menu
for (; i > 0; i-- )
MRU_SetText( i, MRU_GetText( i - 1 ) );
MRU_SetText( 0, str );
return;
}
}
if ( MRU_used < MRU_MAX ) {
MRU_used++;
}
// move items down
for ( i = MRU_used - 1; i > 0; i-- )
MRU_SetText( i, MRU_GetText( i - 1 ) );
MRU_SetText( 0, str );
MRU_items[0]->setEnabled( true );
MRU_items[MRU_used - 1]->setVisible( true );
}
void MRU_Init(){
if ( MRU_used > MRU_MAX ) {
MRU_used = MRU_MAX;
}
}
void MRU_AddWidget( QAction *widget, std::size_t pos ){
if ( pos < MRU_MAX ) {
MRU_items[pos] = widget;
if ( pos < MRU_used ) {
MRU_updateWidget( pos, MRU_GetText( pos ) );
MRU_items[0]->setEnabled( true );
MRU_items[pos]->setVisible( true );
}
}
}
void MRU_Activate( std::size_t index ){
char text[1024];
strcpy( text, MRU_GetText( index ) );
if ( file_readable( text ) ) { //\todo Test 'map load succeeds' instead of 'file is readable'.
MRU_AddFile( text );
Map_Free();
Map_LoadFile( text );
}
else
{
MRU_used--;
for ( std::size_t i = index; i < MRU_used; i++ )
MRU_SetText( i, MRU_GetText( i + 1 ) );
if ( MRU_used == 0 ) {
MRU_items[0]->setText( "Recent Files" );
MRU_items[0]->setEnabled( false );
}
else
{
MRU_items[MRU_used]->setVisible( false );
}
}
}
class LoadMRU
{
std::size_t m_number;
public:
LoadMRU( std::size_t number )
: m_number( number ){
}
void load(){
if ( ConfirmModified( "Open Map" ) ) {
MRU_Activate( m_number - 1 );
}
}
};
typedef MemberCaller<LoadMRU, &LoadMRU::load> LoadMRUCaller;
LoadMRU g_load_mru1( 1 );
LoadMRU g_load_mru2( 2 );
LoadMRU g_load_mru3( 3 );
LoadMRU g_load_mru4( 4 );
LoadMRU g_load_mru5( 5 );
LoadMRU g_load_mru6( 6 );
LoadMRU g_load_mru7( 7 );
LoadMRU g_load_mru8( 8 );
LoadMRU g_load_mru9( 9 );
void MRU_constructMenu( QMenu* menu ){
{
QAction* item = create_menu_item_with_mnemonic( menu, "&1", LoadMRUCaller( g_load_mru1 ) );
item->setEnabled( false );
MRU_AddWidget( item, 0 );
}
{
QAction* item = create_menu_item_with_mnemonic( menu, "&2", LoadMRUCaller( g_load_mru2 ) );
item->setVisible( false );
MRU_AddWidget( item, 1 );
}
{
QAction* item = create_menu_item_with_mnemonic( menu, "&3", LoadMRUCaller( g_load_mru3 ) );
item->setVisible( false );
MRU_AddWidget( item, 2 );
}
{
QAction* item = create_menu_item_with_mnemonic( menu, "&4", LoadMRUCaller( g_load_mru4 ) );
item->setVisible( false );
MRU_AddWidget( item, 3 );
}
{
QAction* item = create_menu_item_with_mnemonic( menu, "&5", LoadMRUCaller( g_load_mru5 ) );
item->setVisible( false );
MRU_AddWidget( item, 4 );
}
{
QAction* item = create_menu_item_with_mnemonic( menu, "&6", LoadMRUCaller( g_load_mru6 ) );
item->setVisible( false );
MRU_AddWidget( item, 5 );
}
{
QAction* item = create_menu_item_with_mnemonic( menu, "&7", LoadMRUCaller( g_load_mru7 ) );
item->setVisible( false );
MRU_AddWidget( item, 6 );
}
{
QAction* item = create_menu_item_with_mnemonic( menu, "&8", LoadMRUCaller( g_load_mru8 ) );
item->setVisible( false );
MRU_AddWidget( item, 7 );
}
{
QAction* item = create_menu_item_with_mnemonic( menu, "&9", LoadMRUCaller( g_load_mru9 ) );
item->setVisible( false );
MRU_AddWidget( item, 8 );
}
}
#include "preferencesystem.h"
#include "stringio.h"
void MRU_Construct(){
GlobalPreferenceSystem().registerPreference( "Count", SizeImportStringCaller( MRU_used ), SizeExportStringCaller( MRU_used ) );
for ( std::size_t i = 0; i != MRU_MAX; ++i )
{
GlobalPreferenceSystem().registerPreference( MRU_keys[i], CopiedStringImportStringCaller( MRU_filenames[i] ), CopiedStringExportStringCaller( MRU_filenames[i] ) );
}
MRU_Init();
}
void MRU_Destroy(){
}