]> git.saurik.com Git - wxWidgets.git/blob - user/wxFile/filectrl.cpp
* Fixed a memory leak in wxThread
[wxWidgets.git] / user / wxFile / filectrl.cpp
1 /*
2 * Author: Robert Roebling
3 *
4 * Copyright: (C) 1997,1998 Robert Roebling
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the wxWindows Licence, which
8 * you have received with this library (see Licence.htm).
9 *
10 */
11
12
13 #ifdef __GNUG__
14 #pragma implementation "filectrl.h"
15 #endif
16
17 #include "filectrl.h"
18
19 #include "wx/dnd.h"
20
21 #include "sys/types.h"
22 #include "sys/stat.h"
23 #include "dirent.h"
24 #include "pwd.h"
25 #include "grp.h"
26 #include "time.h"
27
28 #include "folder.xpm"
29 #include "txt.xpm"
30 #include "list.xpm"
31 #include "find.xpm"
32
33 //-----------------------------------------------------------------------------
34 // wxFileData
35 //-----------------------------------------------------------------------------
36
37 IMPLEMENT_DYNAMIC_CLASS(wxFileData,wxObject);
38
39 wxFileData::wxFileData( const wxString &name, const wxString &fname )
40 {
41 m_name = name;
42 m_fileName = fname;
43
44 struct stat buff;
45 stat( m_fileName.GetData(), &buff );
46 struct stat lbuff;
47 lstat( m_fileName.GetData(), &lbuff );
48
49 struct tm *t = localtime( &lbuff.st_mtime );
50 // struct passwd *user = getpwuid( buff.st_uid );
51 // struct group *grp = getgrgid( buff.st_gid );
52
53 m_isDir = S_ISDIR( buff.st_mode );
54 m_isLink = S_ISLNK( lbuff.st_mode );
55 m_isExe = ((buff.st_mode & S_IXUSR ) == S_IXUSR );
56
57 m_size = buff.st_size;
58
59 m_hour = t->tm_hour;
60 m_minute = t->tm_min;
61 m_month = t->tm_mon+1;
62 m_day = t->tm_mday;
63 m_year = t->tm_year;
64
65 m_permissions.sprintf( "%c%c%c",
66 ((( buff.st_mode & S_IRUSR ) == S_IRUSR ) ? 'r' : '-'),
67 ((( buff.st_mode & S_IWUSR ) == S_IWUSR ) ? 'w' : '-'),
68 ((( buff.st_mode & S_IXUSR ) == S_IXUSR ) ? 'x' : '-') );
69 };
70
71 wxString wxFileData::GetName(void) const
72 {
73 return m_name;
74 };
75
76 wxString wxFileData::GetFullName(void) const
77 {
78 return m_fileName;
79 };
80
81 wxString wxFileData::GetHint(void) const
82 {
83 wxString s = m_fileName;
84 s += " ";
85 if (m_isDir) s += "<DIR> ";
86 else if (m_isLink) s += "<LINK> ";
87 else
88 {
89 s += LongToString( m_size );
90 s += " bytes ";
91 };
92 s += IntToString( m_day );
93 s += ".";
94 s += IntToString( m_month );
95 s += ".";
96 s += IntToString( m_year );
97 s += " ";
98 s += IntToString( m_hour );
99 s += ":";
100 s += IntToString( m_minute );
101 s += " ";
102 s += m_permissions;
103 return s;
104 };
105
106 wxString wxFileData::GetEntry( const int num )
107 {
108 wxString s;
109 switch (num)
110 {
111 case 0:
112 s = m_name;
113 break;
114 case 1:
115 if (m_isDir) s = "<DIR>";
116 else if (m_isLink) s = "<LINK>";
117 else s = LongToString( m_size );
118 break;
119 case 2:
120 if (m_day < 10) s = "0"; else s = "";
121 s += IntToString( m_day );
122 s += ".";
123 if (m_month < 10) s += "0";
124 s += IntToString( m_month );
125 s += ".";
126 if (m_year < 10) s += "0"; // this should happen real soon...
127 s += IntToString( m_year );
128 break;
129 case 3:
130 if (m_hour < 10) s = "0"; else s = "";
131 s += IntToString( m_hour );
132 s += ":";
133 if (m_minute < 10) s += "0";
134 s += IntToString( m_minute );
135 break;
136 case 4:
137 s = m_permissions;
138 break;
139 default:
140 s = "No entry";
141 break;
142 };
143 return s;
144 };
145
146 bool wxFileData::IsDir( void )
147 {
148 return m_isDir;
149 };
150
151 bool wxFileData::IsExe( void )
152 {
153 return m_isExe;
154 };
155
156 bool wxFileData::IsLink( void )
157 {
158 return m_isLink;
159 };
160
161 long wxFileData::GetSize( void )
162 {
163 return m_size;
164 };
165
166 bool wxFileData::NewNameIsLegal( const wxString &s )
167 {
168 wxString fileName = wxPathOnly( m_fileName );
169 fileName += "/";
170 fileName += s;
171 return (!wxFileExists( fileName ));
172 };
173
174 bool wxFileData::Rename( const wxString &s )
175 {
176 wxString fileName = wxPathOnly( m_fileName );
177 fileName += "/";
178 fileName += s;
179 bool ret = wxRenameFile( m_fileName, fileName );
180 if (ret)
181 {
182 m_fileName = fileName;
183 m_name = s;
184 };
185 return ret;
186 };
187
188 void wxFileData::MakeItem( wxListItem &item )
189 {
190 item.m_text = m_name;
191 item.m_colour = wxBLACK;
192 if (IsExe()) item.m_colour = wxRED;
193 if (IsDir()) item.m_colour = wxBLUE;
194 if (IsLink())
195 {
196 wxColour *dg = wxTheColourDatabase->FindColour( "MEDIUM GREY" );
197 item.m_colour = dg;
198 };
199 item.m_data = (long)this;
200 };
201
202 //-----------------------------------------------------------------------------
203 // wxFileCtrl
204 //-----------------------------------------------------------------------------
205
206 IMPLEMENT_DYNAMIC_CLASS(wxFileCtrl,wxListCtrl);
207
208 BEGIN_EVENT_TABLE(wxFileCtrl,wxListCtrl)
209 EVT_SET_FOCUS (wxFileCtrl::OnSetFocus)
210 END_EVENT_TABLE()
211
212 wxFileCtrl *wxFileCtrl::m_lastFocus = (wxFileCtrl *) NULL;
213
214 wxFileCtrl::wxFileCtrl( void )
215 {
216 m_dirName = "/";
217 m_showHidden = FALSE;
218 };
219
220 wxFileCtrl::wxFileCtrl( wxWindow *win, const wxWindowID id, const wxString &dirName,
221 const wxPoint &pos, const wxSize &size,
222 const long style, const wxString &name ) :
223 wxListCtrl( win, id, pos, size, style, name )
224 {
225 SetItemSpacing( 40 );
226 wxImageList *imageList = new wxImageList( 30, 30 );
227 imageList->Add( wxBitmap( folder_xpm ) );
228 imageList->Add( wxBitmap( txt_xpm ) );
229 imageList->Add( wxBitmap( list_xpm ) );
230 imageList->Add( wxBitmap( find_xpm ) );
231
232 SetImageList( imageList, wxIMAGE_LIST_NORMAL );
233
234 m_dirName = dirName;
235 m_showHidden = FALSE;
236 Update();
237
238 m_lastFocus = this;
239
240 m_dragStartX = 0;
241 m_dragStartY = 0;
242 m_dragCount = 0;
243
244 // SetDropTarget( new wxFileDropTarget() );
245 };
246
247 void wxFileCtrl::ChangeToListMode()
248 {
249 SetSingleStyle( wxLC_LIST );
250 Update();
251 };
252
253 void wxFileCtrl::ChangeToReportMode()
254 {
255 SetSingleStyle( wxLC_REPORT );
256 Update();
257 };
258
259 void wxFileCtrl::ChangeToIconMode()
260 {
261 SetSingleStyle( wxLC_ICON );
262 Update();
263 };
264
265 void wxFileCtrl::ShowHidden( bool show )
266 {
267 m_showHidden = show;
268 Update();
269 };
270
271 int ListCompare( const long data1, const long data2, const long WXUNUSED(data) )
272 {
273 wxFileData *fd1 = (wxFileData*)data1 ;
274 wxFileData *fd2 = (wxFileData*)data2 ;
275 if (fd1->IsDir() && !fd2->IsDir()) return -1;
276 if (fd2->IsDir() && !fd1->IsDir()) return 1;
277 return strcmp( fd1->GetName(), fd2->GetName() );
278 };
279
280 void wxFileCtrl::Update( void )
281 {
282 DeleteAllItems();
283 for (int i = 0; i < 5; i++) DeleteColumn( 0 );
284 long my_style = GetWindowStyleFlag();
285 if (my_style & wxLC_REPORT)
286 {
287 InsertColumn( 0, "Name", wxLIST_FORMAT_LEFT, 110 );
288 InsertColumn( 1, "Size", wxLIST_FORMAT_LEFT, 60 );
289 InsertColumn( 2, "Date", wxLIST_FORMAT_LEFT, 55 );
290 InsertColumn( 3, "Time", wxLIST_FORMAT_LEFT, 50 );
291 InsertColumn( 4, "Permissions", wxLIST_FORMAT_LEFT, 120 );
292 };
293 wxFileData *fd = (wxFileData *) NULL;
294 wxListItem item;
295 item.m_mask = wxLIST_MASK_TEXT + wxLIST_MASK_DATA;
296 if (my_style & wxLC_ICON) item.m_mask += wxLIST_MASK_IMAGE;
297 item.m_itemId = 0;
298 item.m_col = 0;
299 wxString s;
300 wxString res = m_dirName + "/*";
301 char *f = wxFindFirstFile( res.GetData(), 0 );
302 while (f)
303 {
304 res = wxFileNameFromPath( f );
305 fd = new wxFileData( res, f );
306 s = fd->GetName();
307 if (m_showHidden || (s[0] != '.'))
308 {
309 fd->MakeItem( item );
310 if (my_style & wxLC_REPORT)
311 {
312 InsertItem( item );
313 for (int i = 1; i < 5; i++) SetItem( item.m_itemId, i, fd->GetEntry( i) );
314 }
315 else if (my_style & wxLC_LIST)
316 {
317 InsertItem( item );
318 }
319 else if (my_style & wxLC_ICON)
320 {
321 if (fd->IsDir()) item.m_image = 0; else item.m_image = 1;
322 InsertItem( item );
323 };
324 item.m_itemId++;
325 };
326 f = wxFindNextFile();
327 };
328 SortItems( ListCompare, 0 );
329 };
330
331
332 int wxFileCtrl::FillList( wxStringList &list )
333 {
334 long index = -1;
335 int count = 0;
336 wxString s;
337 for (;;)
338 {
339 index = GetNextItem( index, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED );
340 if (index == -1) break;
341 wxListItem item;
342 item.m_itemId = index;
343 GetItem( item );
344 wxFileData *fd = (wxFileData*)item.m_data;
345 list.Add( fd->GetFullName() );
346 index++;
347 count++;
348 };
349 if (count == 0)
350 {
351 index = GetNextItem( -1, wxLIST_NEXT_ALL, wxLIST_STATE_FOCUSED );
352 if (index == -1) return 0;
353 wxListItem item;
354 item.m_itemId = index;
355 GetItem( item );
356 wxFileData *fd = (wxFileData*)item.m_data;
357 list.Add( fd->GetFullName() );
358 count = 1;
359 };
360 return count;
361 };
362
363 void wxFileCtrl::DeleteFiles(void)
364 {
365 /*
366 wxStringList list;
367 int count = FillList( list );
368 if (count > 0)
369 {
370 wxString s = "Delete ";
371 s += wxIntToString( count );
372 s += " selected file";
373 if (count > 1) s += "s";
374 s += " or director";
375 if (count > 1) s += "ies?"; else s+= "y?";
376 if (wxYES == wxMessageBox( s, "Delete", wxYES_NO ))
377 wxDeleteStatusDia( NULL, &list );
378 };
379 */
380 };
381
382 void wxFileCtrl::CopyFiles( char *WXUNUSED(dest) )
383 {
384 /*
385 wxStringList list;
386 int count = FillList( list );
387 wxString s = dest;
388 int ret = 0; // 0 = nix, 1 = copy, 2 = move
389 wxCopyMoveDia( (wxFrame*)GetParent(), count, &ret, &s );
390 if (ret == 1)
391 wxCopyStatusDia( NULL, s, &list );
392 */
393 };
394
395 void wxFileCtrl::MoveFiles( char *WXUNUSED(dest) )
396 {
397 };
398
399 void wxFileCtrl::RenameFile(void)
400 {
401 };
402
403 void wxFileCtrl::MakeDir(void)
404 {
405 /*
406 wxString s = wxGetTextFromUser( "Enter new directory name:", "Make directory" );
407 if (s.IsNull()) return;
408 if (s == "") return;
409 if ((s == ".") || (s == ".."))
410 {
411 wxMessageBox( "This was obviously an invalid directory name.", "Go away." );
412 return;
413 };
414 wxString dir;
415 GetDir( dir );
416 dir += "/";
417 dir += s;
418 if (wxFileExists( dir ))
419 {
420 wxMessageBox( "Filename exists already. Cannot create directoy.", "Make directory" );
421 return;
422 };
423 wxMkdir( dir );
424 Update();
425 */
426 };
427
428 void wxFileCtrl::GoToParentDir(void)
429 {
430 wxString s = m_dirName;
431 int pos = s.Last( '/' );
432 if ((pos >= 0) && (s != "/"))
433 {
434 s.Remove( pos, s.Length()-pos );
435 if (s.Length() == 0) s = "/";
436 m_dirName = s;
437 Update();
438 };
439 };
440
441 void wxFileCtrl::GoToHomeDir(void)
442 {
443 wxString s = wxGetUserHome( wxString() );
444 m_dirName = s;
445 Update();
446 };
447
448 void wxFileCtrl::GoToDir( const wxString &dir )
449 {
450 m_dirName = dir;
451 Update();
452 };
453
454 void wxFileCtrl::GetDir( wxString &dir )
455 {
456 dir = m_dirName;
457 };
458
459 /*
460 void wxFileCtrl::OnDropFiles( int WXUNUSED(n), char **WXUNUSED(data), int WXUNUSED(x), int WXUNUSED(y) )
461 {
462 wxString destDir;
463 wxPoint pt( x, y );
464 int flag = wxLIST_HITTEST_ONITEM;
465 long hit = HitTest( pt, flag );
466 if (hit > -1)
467 {
468 wxListItem li;
469 li.m_itemId = hit;
470 GetItem( li );
471 wxFileData *fd = (wxFileData*)li.m_data;
472 if (fd->IsDir()) fd->GetFullName( destDir );
473 };
474 if (destDir.IsNull()) destDir = m_dirName;
475 int ret = 0; // 0 = nix, 1 = copy, 2 = move
476 wxCopyMoveDia( (wxFrame*)GetParent(), n, &ret, &destDir );
477 if (ret == 1)
478 {
479 wxStringList slist;
480 for (int i = 0; i < n; i++) slist.Add( data[i] );
481 wxCopyStatusDia( NULL, destDir.GetData(), &slist );
482 Update();
483 };
484 };
485 */
486
487 void wxFileCtrl::OnListDeleteItem( wxListEvent &event )
488 {
489 wxFileData *fd = (wxFileData*)event.m_item.m_data;
490 delete fd;
491 };
492
493 void wxFileCtrl::OnListKeyDown( wxListEvent &event )
494 {
495 wxFileData *fd = (wxFileData*)event.m_item.m_data;
496 if (fd->IsDir())
497 {
498 m_dirName = fd->GetFullName();
499 Update();
500 Refresh();
501 return;
502 };
503 if (fd->IsExe())
504 {
505 wxExecute( fd->GetFullName() );
506 return;
507 };
508 };
509
510 void wxFileCtrl::OnListEndLabelEdit( wxListEvent &event )
511 {
512 wxFileData *fd = (wxFileData*)event.m_item.m_data;
513 wxString newName = event.m_item.m_text;
514 if (fd->NewNameIsLegal( newName ))
515 {
516 if (fd->Rename( newName ))
517 {
518 Update();
519 }
520 else
521 {
522 wxString s = "Could not rename file to ";
523 s += newName;
524 s += ".";
525 wxMessageBox( s, "FileMaker", wxOK );
526 };
527 }
528 else
529 {
530 wxString s = "File name ";
531 s += newName;
532 s += " exists already or is invalid.\n";
533 s += "Could not rename file.";
534 wxMessageBox( s, "FileMaker", wxOK );
535 };
536 return;
537 };
538
539 void wxFileCtrl::OnSetFocus( wxFocusEvent &event )
540 {
541 m_lastFocus = this;
542 event.Skip();
543 };
544