]> git.xonotic.org Git - xonotic/netradiant.git/blob - radiant/mru.cpp
radiant: some comments from NRC
[xonotic/netradiant.git] / radiant / mru.cpp
1 /*
2    Copyright (C) 1999-2006 Id Software, Inc. and contributors.
3    For a list of contributors, see the accompanying CONTRIBUTORS file.
4
5    This file is part of GtkRadiant.
6
7    GtkRadiant is free software; you can redistribute it and/or modify
8    it under the terms of the GNU General Public License as published by
9    the Free Software Foundation; either version 2 of the License, or
10    (at your option) any later version.
11
12    GtkRadiant is distributed in the hope that it will be useful,
13    but WITHOUT ANY WARRANTY; without even the implied warranty of
14    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15    GNU General Public License for more details.
16
17    You should have received a copy of the GNU General Public License
18    along with GtkRadiant; if not, write to the Free Software
19    Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
20  */
21
22 #include "mru.h"
23
24 #include <string.h>
25 #include <stdio.h>
26 #include <gtk/gtk.h>
27
28 #include "os/file.h"
29 #include "generic/callback.h"
30 #include "convert.h"
31
32 #include "gtkutil/menu.h"
33 #include "map.h"
34 #include "qe3.h"
35
36 #define MRU_MAX 9
37 namespace {
38 GtkMenuItem *MRU_items[MRU_MAX];
39 std::size_t MRU_used;
40 typedef CopiedString MRU_filename_t;
41 MRU_filename_t MRU_filenames[MRU_MAX];
42 typedef const char* MRU_key_t;
43 MRU_key_t MRU_keys[MRU_MAX] = { "File0", "File1", "File2", "File3", "File4", "File5", "File6", "File7", "File8" };
44 }
45
46 inline const char* MRU_GetText( std::size_t index ){
47         return MRU_filenames[index].c_str();
48 }
49
50 class EscapedMnemonic
51 {
52 private:
53         std::string m_buffer;
54
55 public:
56         EscapedMnemonic() : m_buffer(){
57         m_buffer.push_back( '_' );
58 }
59 const char* c_str() const {
60         return m_buffer.c_str();
61 }
62 void push_back( char c ){ // not escaped
63         m_buffer.push_back( c );
64 }
65 std::size_t write( const char* buffer, std::size_t length ){
66         for ( const char* end = buffer + length; buffer != end; ++buffer )
67         {
68                 if ( *buffer == '_' ) {
69                         m_buffer.push_back( '_' );
70                 }
71
72                 m_buffer.push_back( *buffer );
73         }
74         return length;
75 }
76 };
77
78 template<typename T>
79 inline EscapedMnemonic& operator<<( EscapedMnemonic& ostream, const T& t ){
80         return ostream_write( ostream, t );
81 }
82
83
84 void MRU_updateWidget( std::size_t index, const char *filename ){
85         EscapedMnemonic mnemonic;
86         mnemonic << Unsigned( index + 1 ) << "- " << filename;
87         gtk_label_set_text_with_mnemonic( GTK_LABEL( gtk_bin_get_child( GTK_BIN( MRU_items[index] ) ) ), mnemonic.c_str() );
88 }
89
90 void MRU_SetText( std::size_t index, const char *filename ){
91         MRU_filenames[index] = filename;
92         MRU_updateWidget( index, filename );
93 }
94
95 void MRU_AddFile( const char *str ){
96         std::size_t i;
97         const char* text;
98
99         // check if file is already in our list
100         for ( i = 0; i < MRU_used; i++ )
101         {
102                 text = MRU_GetText( i );
103
104                 if ( strcmp( text, str ) == 0 ) {
105                         // reorder menu
106                         for (; i > 0; i-- )
107                                 MRU_SetText( i, MRU_GetText( i - 1 ) );
108
109                         MRU_SetText( 0, str );
110
111                         return;
112                 }
113         }
114
115         if ( MRU_used < MRU_MAX ) {
116                 MRU_used++;
117         }
118
119         // move items down
120         for ( i = MRU_used - 1; i > 0; i-- )
121                 MRU_SetText( i, MRU_GetText( i - 1 ) );
122
123         MRU_SetText( 0, str );
124         gtk_widget_set_sensitive( ui::MenuItem::from(MRU_items[0]) , TRUE );
125         ui::MenuItem::from(MRU_items[MRU_used - 1] ).show();
126 }
127
128 void MRU_Init(){
129         if ( MRU_used > MRU_MAX ) {
130                 MRU_used = MRU_MAX;
131         }
132 }
133
134 void MRU_AddWidget( ui::MenuItem widget, std::size_t pos ){
135         if ( pos < MRU_MAX ) {
136                 MRU_items[pos] = widget;
137                 if ( pos < MRU_used ) {
138                         MRU_updateWidget( pos, MRU_GetText( pos ) );
139                         gtk_widget_set_sensitive( ui::MenuItem::from(MRU_items[0]) , TRUE );
140                         ui::MenuItem::from(MRU_items[pos]).show();
141                 }
142         }
143 }
144
145 void MRU_Activate( std::size_t index ){
146         char text[1024];
147         strcpy( text, MRU_GetText( index ) );
148
149         if ( file_readable( text ) ) { //\todo Test 'map load succeeds' instead of 'file is readable'.
150                 MRU_AddFile( text );
151                 Map_RegionOff();
152                 Map_Free();
153                 Map_LoadFile( text );
154         }
155         else
156         {
157                 MRU_used--;
158
159                 for ( std::size_t i = index; i < MRU_used; i++ )
160                         MRU_SetText( i, MRU_GetText( i + 1 ) );
161
162                 if ( MRU_used == 0 ) {
163                         auto label = ui::Label::from(gtk_bin_get_child(GTK_BIN(MRU_items[0] )) );
164                         label.text("Recent Files");
165                         gtk_widget_set_sensitive( ui::MenuItem::from(MRU_items[0]), FALSE );
166                 }
167                 else
168                 {
169                         ui::MenuItem::from(MRU_items[MRU_used]).hide();
170                 }
171         }
172 }
173
174
175 class LoadMRU
176 {
177 private:
178 std::size_t m_number;
179
180 public:
181 LoadMRU( std::size_t number )
182         : m_number( number ){
183 }
184 void load(){
185         if ( ConfirmModified( "Open Map" ) ) {
186                 MRU_Activate( m_number - 1 );
187         }
188 }
189 };
190
191 typedef MemberCaller<LoadMRU, void(), &LoadMRU::load> LoadMRUCaller;
192
193 LoadMRU g_load_mru1( 1 );
194 LoadMRU g_load_mru2( 2 );
195 LoadMRU g_load_mru3( 3 );
196 LoadMRU g_load_mru4( 4 );
197 LoadMRU g_load_mru5( 5 );
198 LoadMRU g_load_mru6( 6 );
199 LoadMRU g_load_mru7( 7 );
200 LoadMRU g_load_mru8( 8 );
201 LoadMRU g_load_mru9( 9 );
202
203 void MRU_constructMenu( ui::Menu menu ){
204         {
205                 auto item = create_menu_item_with_mnemonic( menu, "_1", LoadMRUCaller( g_load_mru1 ) );
206                 gtk_widget_set_sensitive( item , FALSE );
207                 MRU_AddWidget( item, 0 );
208         }
209         {
210                 auto item = create_menu_item_with_mnemonic( menu, "_2", LoadMRUCaller( g_load_mru2 ) );
211                 item.hide();
212                 MRU_AddWidget( item, 1 );
213         }
214         {
215                 auto item = create_menu_item_with_mnemonic( menu, "_3", LoadMRUCaller( g_load_mru3 ) );
216                 item.hide();
217                 MRU_AddWidget( item, 2 );
218         }
219         {
220                 auto item = create_menu_item_with_mnemonic( menu, "_4", LoadMRUCaller( g_load_mru4 ) );
221                 item.hide();
222                 MRU_AddWidget( item, 3 );
223         }
224         {
225                 auto item = create_menu_item_with_mnemonic( menu, "_5", LoadMRUCaller( g_load_mru5 ) );
226                 item.hide();
227                 MRU_AddWidget( item, 4 );
228         }
229         {
230                 auto item = create_menu_item_with_mnemonic( menu, "_6", LoadMRUCaller( g_load_mru6 ) );
231                 item.hide();
232                 MRU_AddWidget( item, 5 );
233         }
234         {
235                 auto item = create_menu_item_with_mnemonic( menu, "_7", LoadMRUCaller( g_load_mru7 ) );
236                 item.hide();
237                 MRU_AddWidget( item, 6 );
238         }
239         {
240                 auto item = create_menu_item_with_mnemonic( menu, "_8", LoadMRUCaller( g_load_mru8 ) );
241                 item.hide();
242                 MRU_AddWidget( item, 7 );
243         }
244         {
245                 auto item = create_menu_item_with_mnemonic( menu, "_9", LoadMRUCaller( g_load_mru9 ) );
246                 item.hide();
247                 MRU_AddWidget( item, 8 );
248         }
249 }
250
251 #include "preferencesystem.h"
252 #include "stringio.h"
253
254 void MRU_Construct(){
255         GlobalPreferenceSystem().registerPreference( "Count", make_property_string( MRU_used ) );
256
257         for ( std::size_t i = 0; i != MRU_MAX; ++i )
258         {
259                 GlobalPreferenceSystem().registerPreference( MRU_keys[i], make_property_string( MRU_filenames[i] ) );
260         }
261
262         MRU_Init();
263 }
264 void MRU_Destroy(){
265 }