]> git.saurik.com Git - wxWidgets.git/blame - src/unix/dir.cpp
avoid duplicate move events
[wxWidgets.git] / src / unix / dir.cpp
CommitLineData
1944c6bd 1/////////////////////////////////////////////////////////////////////////////
80fdcdb9 2// Name: src/unix/dir.cpp
1944c6bd
VZ
3// Purpose: wxDir implementation for Unix/POSIX systems
4// Author: Vadim Zeitlin
5// Modified by:
6// Created: 08.12.99
1944c6bd 7// Copyright: (c) 1999 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
65571936 8// Licence: wxWindows licence
1944c6bd
VZ
9/////////////////////////////////////////////////////////////////////////////
10
11// ============================================================================
12// declarations
13// ============================================================================
14
15// ----------------------------------------------------------------------------
16// headers
17// ----------------------------------------------------------------------------
18
1944c6bd
VZ
19// For compilers that support precompilation, includes "wx.h".
20#include "wx/wxprec.h"
21
22#ifdef __BORLANDC__
23 #pragma hdrstop
24#endif
25
26#ifndef WX_PRECOMP
27 #include "wx/intl.h"
28 #include "wx/log.h"
29#endif // PCH
30
31#include "wx/dir.h"
32#include "wx/filefn.h" // for wxMatchWild
c063adeb 33#include "wx/filename.h"
1944c6bd
VZ
34
35#include <sys/types.h>
1357a7dd
VZ
36#include <sys/stat.h>
37#include <unistd.h>
1944c6bd
VZ
38
39#include <dirent.h>
40
41// ----------------------------------------------------------------------------
42// macros
43// ----------------------------------------------------------------------------
44
45#define M_DIR ((wxDirData *)m_data)
46
47// ----------------------------------------------------------------------------
48// private classes
49// ----------------------------------------------------------------------------
50
51// this class stores everything we need to enumerate the files
52class wxDirData
53{
54public:
55 wxDirData(const wxString& dirname);
56 ~wxDirData();
57
58 bool IsOk() const { return m_dir != NULL; }
59
60 void SetFileSpec(const wxString& filespec) { m_filespec = filespec; }
61 void SetFlags(int flags) { m_flags = flags; }
62
63 void Rewind() { rewinddir(m_dir); }
64 bool Read(wxString *filename);
65
35332784
VZ
66 const wxString& GetName() const { return m_dirname; }
67
1944c6bd
VZ
68private:
69 DIR *m_dir;
70
71 wxString m_dirname;
72 wxString m_filespec;
73
74 int m_flags;
75};
76
77// ============================================================================
78// implementation
79// ============================================================================
80
81// ----------------------------------------------------------------------------
82// wxDirData
83// ----------------------------------------------------------------------------
84
85#if !defined( __VMS__ ) || ( __VMS_VER >= 70000000 )
86
87wxDirData::wxDirData(const wxString& dirname)
88 : m_dirname(dirname)
89{
90 m_dir = NULL;
91
92 // throw away the trailing slashes
93 size_t n = m_dirname.length();
9a83f860 94 wxCHECK_RET( n, wxT("empty dir name in wxDir") );
1944c6bd
VZ
95
96 while ( n > 0 && m_dirname[--n] == '/' )
97 ;
98
99 m_dirname.Truncate(n + 1);
100
101 // do open the dir
102 m_dir = opendir(m_dirname.fn_str());
103}
104
105wxDirData::~wxDirData()
106{
107 if ( m_dir )
108 {
109 if ( closedir(m_dir) != 0 )
110 {
9a83f860 111 wxLogLastError(wxT("closedir"));
1944c6bd
VZ
112 }
113 }
114}
115
116bool wxDirData::Read(wxString *filename)
117{
d3b9f782 118 dirent *de = NULL; // just to silence compiler warnings
da865fdd 119 bool matches = false;
1944c6bd 120
b0d36402
VZ
121 // speed up string concatenation in the loop a bit
122 wxString path = m_dirname;
9a83f860 123 path += wxT('/');
b0d36402 124 path.reserve(path.length() + 255);
da865fdd 125
e6ccaf1a 126 wxString de_d_name;
b0d36402 127
1944c6bd
VZ
128 while ( !matches )
129 {
130 de = readdir(m_dir);
131 if ( !de )
da865fdd
WS
132 return false;
133
e6ccaf1a 134#if wxUSE_UNICODE
86501081 135 de_d_name = wxString(de->d_name, *wxConvFileName);
e6ccaf1a
RR
136#else
137 de_d_name = de->d_name;
da865fdd 138#endif
1944c6bd
VZ
139
140 // don't return "." and ".." unless asked for
141 if ( de->d_name[0] == '.' &&
142 ((de->d_name[1] == '.' && de->d_name[2] == '\0') ||
143 (de->d_name[1] == '\0')) )
144 {
145 if ( !(m_flags & wxDIR_DOTDOT) )
146 continue;
b0d36402
VZ
147
148 // we found a valid match
149 break;
1944c6bd
VZ
150 }
151
c55f46d0
VZ
152 // check the type now: notice that we may want to check the type of
153 // the path itself and not whatever it points to in case of a symlink
c063adeb 154 wxFileName fn = wxFileName::DirName(path + de_d_name);
c55f46d0
VZ
155 if ( m_flags & wxDIR_NO_FOLLOW )
156 {
157 fn.DontFollowLink();
158 }
c063adeb
VZ
159
160 if ( !(m_flags & wxDIR_FILES) && !fn.DirExists() )
1944c6bd
VZ
161 {
162 // it's a file, but we don't want them
163 continue;
164 }
c063adeb 165 else if ( !(m_flags & wxDIR_DIRS) && fn.DirExists() )
1944c6bd
VZ
166 {
167 // it's a dir, and we don't want it
168 continue;
169 }
170
171 // finally, check the name
b0d36402 172 if ( m_filespec.empty() )
1944c6bd 173 {
da865fdd 174 matches = m_flags & wxDIR_HIDDEN ? true : de->d_name[0] != '.';
1944c6bd
VZ
175 }
176 else
177 {
178 // test against the pattern
e6ccaf1a 179 matches = wxMatchWild(m_filespec, de_d_name,
1944c6bd
VZ
180 !(m_flags & wxDIR_HIDDEN));
181 }
182 }
183
e6ccaf1a 184 *filename = de_d_name;
1944c6bd 185
da865fdd 186 return true;
1944c6bd
VZ
187}
188
189#else // old VMS (TODO)
190
191wxDirData::wxDirData(const wxString& WXUNUSED(dirname))
192{
9a83f860 193 wxFAIL_MSG(wxT("not implemented"));
1944c6bd
VZ
194}
195
196wxDirData::~wxDirData()
197{
198}
199
200bool wxDirData::Read(wxString * WXUNUSED(filename))
201{
da865fdd 202 return false;
1944c6bd
VZ
203}
204
205#endif // not or new VMS/old VMS
206
1944c6bd
VZ
207// ----------------------------------------------------------------------------
208// wxDir construction/destruction
209// ----------------------------------------------------------------------------
210
211wxDir::wxDir(const wxString& dirname)
212{
213 m_data = NULL;
214
215 (void)Open(dirname);
216}
217
218bool wxDir::Open(const wxString& dirname)
219{
220 delete M_DIR;
221 m_data = new wxDirData(dirname);
222
223 if ( !M_DIR->IsOk() )
224 {
1944c6bd
VZ
225 delete M_DIR;
226 m_data = NULL;
227
da865fdd 228 return false;
1944c6bd
VZ
229 }
230
da865fdd 231 return true;
1944c6bd
VZ
232}
233
234bool wxDir::IsOpened() const
235{
236 return m_data != NULL;
237}
238
35332784
VZ
239wxString wxDir::GetName() const
240{
241 wxString name;
242 if ( m_data )
243 {
244 name = M_DIR->GetName();
bb91ff63
VZ
245
246 // Notice that we need to check for length > 1 as we shouldn't remove
247 // the last slash from the root directory!
248 if ( name.length() > 1 && (name.Last() == wxT('/')) )
35332784 249 {
bb91ff63
VZ
250 // chop off the last slash
251 name.RemoveLast();
35332784
VZ
252 }
253 }
254
255 return name;
256}
257
6619c4af 258void wxDir::Close()
1944c6bd 259{
6619c4af
VZ
260 if ( m_data )
261 {
262 delete m_data;
263 m_data = NULL;
264 }
1944c6bd
VZ
265}
266
267// ----------------------------------------------------------------------------
268// wxDir enumerating
269// ----------------------------------------------------------------------------
270
271bool wxDir::GetFirst(wxString *filename,
272 const wxString& filespec,
273 int flags) const
274{
9a83f860 275 wxCHECK_MSG( IsOpened(), false, wxT("must wxDir::Open() first") );
1944c6bd
VZ
276
277 M_DIR->Rewind();
278
279 M_DIR->SetFileSpec(filespec);
280 M_DIR->SetFlags(flags);
281
282 return GetNext(filename);
283}
284
285bool wxDir::GetNext(wxString *filename) const
286{
9a83f860 287 wxCHECK_MSG( IsOpened(), false, wxT("must wxDir::Open() first") );
1944c6bd 288
9a83f860 289 wxCHECK_MSG( filename, false, wxT("bad pointer in wxDir::GetNext()") );
1944c6bd
VZ
290
291 return M_DIR->Read(filename);
292}
1357a7dd 293
106dcc2c 294bool wxDir::HasSubDirs(const wxString& spec) const
1357a7dd 295{
9a83f860 296 wxCHECK_MSG( IsOpened(), false, wxT("must wxDir::Open() first") );
1357a7dd
VZ
297
298 if ( spec.empty() )
299 {
300 // faster check for presence of any subdirectory: normally each subdir
301 // has a hard link to the parent directory and so, knowing that there
302 // are at least "." and "..", we have a subdirectory if and only if
303 // links number is > 2 - this is just a guess but it works fairly well
304 // in practice
305 //
306 // note that we may guess wrongly in one direction only: i.e. we may
307 // return true when there are no subdirectories but this is ok as the
308 // caller will learn it soon enough when it calls GetFirst(wxDIR)
309 // anyhow
310 wxStructStat stBuf;
766fc092 311 if ( wxStat(M_DIR->GetName(), &stBuf) == 0 )
1357a7dd
VZ
312 {
313 switch ( stBuf.st_nlink )
314 {
315 case 2:
316 // just "." and ".."
da865fdd 317 return false;
1357a7dd
VZ
318
319 case 0:
320 case 1:
321 // weird filesystem, don't try to guess for it, use dumb
322 // method below
323 break;
324
325 default:
326 // assume we have subdirs - may turn out to be wrong if we
327 // have other hard links to this directory but it's not
328 // that bad as explained above
da865fdd 329 return true;
1357a7dd
VZ
330 }
331 }
332 }
333
334 // just try to find first directory
335 wxString s;
336 return GetFirst(&s, spec, wxDIR_DIRS | wxDIR_HIDDEN);
337}
338