1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/common/textfile.cpp 
   3 // Purpose:     implementation of wxTextFile class 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr> 
   9 // Licence:     wxWindows licence 
  10 /////////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 #include  "wx/wxprec.h" 
  22 #if !wxUSE_FILE || !wxUSE_TEXTBUFFER 
  24     #define wxUSE_TEXTFILE 0 
  30     #include "wx/string.h" 
  36 #include "wx/textfile.h" 
  37 #include "wx/filename.h" 
  38 #include "wx/buffer.h" 
  40 // ============================================================================ 
  41 // wxTextFile class implementation 
  42 // ============================================================================ 
  44 wxTextFile::wxTextFile(const wxString
& strFileName
) 
  45           : wxTextBuffer(strFileName
) 
  50 // ---------------------------------------------------------------------------- 
  52 // ---------------------------------------------------------------------------- 
  54 bool wxTextFile::OnExists() const 
  56     return wxFile::Exists(m_strBufferName
); 
  60 bool wxTextFile::OnOpen(const wxString 
&strBufferName
, wxTextBufferOpenMode OpenMode
) 
  62     wxFile::OpenMode FileOpenMode
; 
  67             wxFAIL_MSG( _T("unknown open mode in wxTextFile::Open") ); 
  71             FileOpenMode 
= wxFile::read
; 
  75             FileOpenMode 
= wxFile::write
; 
  79     return m_file
.Open(strBufferName
.c_str(), FileOpenMode
); 
  83 bool wxTextFile::OnClose() 
  85     return m_file
.Close(); 
  89 bool wxTextFile::OnRead(const wxMBConv
& conv
) 
  91     // file should be opened and we must be in it's beginning 
  92     wxASSERT( m_file
.IsOpened() && m_file
.Tell() == 0 ); 
  94     // read the entire file in memory: this is not the most efficient thing to 
  95     // do but there is no good way to avoid it in Unicode build because if we 
  96     // read the file block by block we can't convert each block to Unicode 
  97     // separately (the last multibyte char in the block might be only partially 
  98     // read and so the conversion would fail) and, as the file contents is kept 
  99     // in memory by wxTextFile anyhow, it shouldn't be a big problem to read 
 101     const size_t bufSize 
= (size_t)(m_file
.Length() + 4 /* for trailing NULs */ ); 
 103     wxCharBuffer 
buf(bufSize 
- 1 /* it adds 1 internally */); 
 106     for ( bool eof 
= false; !eof
; ) 
 108         // try to read up to the size of the entire block 
 109         ssize_t nRead 
= m_file
.Read(block
, WXSIZEOF(block
)); 
 111         if ( nRead 
== wxInvalidOffset 
) 
 113             // read error (error message already given in wxFile::Read) 
 120         // this shouldn't happen but don't overwrite the buffer if it does 
 121         wxCHECK_MSG( bufPos 
+ nRead 
<= bufSize
, false, 
 122                      _T("read more than file length?") ); 
 124         // append to the buffer 
 125         memcpy(buf
.data() + bufPos
, block
, nRead
); 
 129     const wxString 
str(buf
, conv
, bufPos
); 
 131     // this doesn't risk to happen in ANSI build 
 133     if ( bufSize 
> 4 && str
.empty() ) 
 135         wxLogError(_("Failed to convert file contents to Unicode.")); 
 138 #endif // wxUSE_UNICODE 
 140     free(buf
.release()); // we don't need this memory any more 
 143     // now break the buffer in lines 
 145     // last processed character, we need to know if it was a CR or not 
 146     wxChar chLast 
= '\0'; 
 148     // the beginning of the current line, changes inside the loop 
 149     wxString::const_iterator lineStart 
= str
.begin(); 
 150     const wxString::const_iterator end 
= str
.end(); 
 151     for ( wxString::const_iterator p 
= lineStart
; p 
!= end
; p
++ ) 
 153         const wxChar ch 
= *p
; 
 157                 // could be a DOS or Unix EOL 
 158                 if ( chLast 
== '\r' ) 
 160                     if ( p 
- 1 >= lineStart 
) 
 162                         AddLine(wxString(lineStart
, p 
- 1), wxTextFileType_Dos
); 
 166                         // there were two line endings, so add an empty line: 
 167                         AddLine(wxEmptyString
, wxTextFileType_Dos
); 
 170                 else // bare '\n', Unix style 
 172                     AddLine(wxString(lineStart
, p
), wxTextFileType_Unix
); 
 179                 if ( chLast 
== '\r' ) 
 182                     AddLine(wxEmptyString
, wxTextFileType_Mac
); 
 185                 //else: we don't know what this is yet -- could be a Mac EOL or 
 186                 //      start of DOS EOL so wait for next char 
 190                 if ( chLast 
== '\r' ) 
 192                     // Mac line termination 
 193                     if ( p 
- 1 >= lineStart 
) 
 195                         AddLine(wxString(lineStart
, p 
- 1), wxTextFileType_Mac
); 
 199                         // there were two line endings, so add an empty line: 
 200                         AddLine(wxEmptyString
, wxTextFileType_Mac
); 
 209     // anything in the last line? 
 210     if ( lineStart 
!= end 
) 
 212         // add unterminated last line 
 213         AddLine(wxString(lineStart
, end
), wxTextFileType_None
); 
 220 bool wxTextFile::OnWrite(wxTextFileType typeNew
, const wxMBConv
& conv
) 
 222     wxFileName fn 
= m_strBufferName
; 
 224     // We do NOT want wxPATH_NORM_CASE here, or the case will not 
 226     if ( !fn
.IsAbsolute() ) 
 227         fn
.Normalize(wxPATH_NORM_ENV_VARS 
| wxPATH_NORM_DOTS 
| wxPATH_NORM_TILDE 
| 
 228                      wxPATH_NORM_ABSOLUTE 
| wxPATH_NORM_LONG
); 
 230     wxTempFile 
fileTmp(fn
.GetFullPath()); 
 232     if ( !fileTmp
.IsOpened() ) { 
 233         wxLogError(_("can't write buffer '%s' to disk."), m_strBufferName
.c_str()); 
 237     size_t nCount 
= GetLineCount(); 
 238     for ( size_t n 
= 0; n 
< nCount
; n
++ ) { 
 239         fileTmp
.Write(GetLine(n
) + 
 240                       GetEOL(typeNew 
== wxTextFileType_None 
? GetLineType(n
) 
 245     // replace the old file with this one 
 246     return fileTmp
.Commit(); 
 249 #endif // wxUSE_TEXTFILE