ITK  6.0.0
Insight Toolkit
itkInternationalizationIOHelpers.h
Go to the documentation of this file.
1 /*=========================================================================
2  *
3  * Copyright NumFOCUS
4  *
5  * Licensed under the Apache License, Version 2.0 (the "License");
6  * you may not use this file except in compliance with the License.
7  * You may obtain a copy of the License at
8  *
9  * https://www.apache.org/licenses/LICENSE-2.0.txt
10  *
11  * Unless required by applicable law or agreed to in writing, software
12  * distributed under the License is distributed on an "AS IS" BASIS,
13  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  * See the License for the specific language governing permissions and
15  * limitations under the License.
16  *
17  *=========================================================================*/
18 #ifndef itkInternationalizationIOHelpers_h
19 #define itkInternationalizationIOHelpers_h
20 #include "ITKIOImageBaseExport.h"
21 
22 // This header provides some helper functions to deal with unicode filenames
23 // It is mainly directed towards being able to use utf-8 encoded filenames
24 // on windows.
25 // This should help better dealing with internationalization (a.k.a i18n)
26 
27 #include "itkMacro.h"
28 #include "itkIOConfigure.h"
29 
30 #ifdef ITK_HAVE_UNISTD_H
31 # include <unistd.h> // for unlink
32 #else
33 # include <io.h>
34 #endif
35 
36 #include <cstdio>
37 #include <fcntl.h>
38 #include <iostream>
39 #include <string>
40 #include <sys/stat.h>
41 
42 // Find out how to handle unicode filenames on windows:
43 // * VS>=8.0 has _wopen and _wfopen and can open a (i/o)fstream using a wide
44 // string
45 // * VS7.x and MinGW have _wopen and _wfopen but cannot open a
46 // (i/o)fstream using a wide string. They can however compile fdstream
47 
48 #if defined(ITK_SUPPORTS_WCHAR_T_FILENAME_CSTYLEIO) && \
49  (defined(ITK_SUPPORTS_WCHAR_T_FILENAME_IOSTREAMS_CONSTRUCTORS) || defined(ITK_SUPPORTS_FDSTREAM_HPP))
50 # define LOCAL_USE_WIN32_WOPEN 1
51 # include <windows.h> // required by winnls.h
52 # include <winnls.h> // for MultiByteToWideChar
53 #else
54 # define LOCAL_USE_WIN32_WOPEN 0
55 #endif
56 
57 #if (LOCAL_USE_WIN32_WOPEN && defined(ITK_SUPPORTS_WCHAR_T_FILENAME_IOSTREAMS_CONSTRUCTORS)) || (!LOCAL_USE_WIN32_WOPEN)
58 # define LOCAL_USE_FDSTREAM 0
59 # include <fstream>
60 #else
61 # define LOCAL_USE_FDSTREAM 1
62 # include "itkfdstream/fdstream.hxx"
63 #endif
64 
65 namespace itk
66 {
67 namespace i18n
68 {
69 // Check if the string is correctly encoded
70 #if LOCAL_USE_WIN32_WOPEN
71 inline bool
72 IsStringEncodingValid(const std::string & str)
73 {
74  // Check if the string is really encoded in utf-8 using windows API
75  // MultiByteToWideChar returns 0 if there was a problem during conversion
76  // when given the MB_ERR_INVALID_CHARS flag
77  const int utf16_size =
78  MultiByteToWideChar(CP_UTF8, MB_ERR_INVALID_CHARS, str.c_str(), static_cast<int>(str.length()), nullptr, 0);
79 
80  return (utf16_size != 0);
81 }
82 
83 #else
84 inline bool
85 IsStringEncodingValid(const std::string & itkNotUsed(str))
86 {
87  return true;
88 }
89 
90 #endif
91 
92 #if LOCAL_USE_WIN32_WOPEN
93 // Convert a utf8 encoded std::string to a utf16 encoded wstring on windows
94 inline std::wstring
95 Utf8StringToWString(const std::string & str)
96 {
97  // We do not set the MB_ERR_INVALID_CHARS to do an approximate conversion when
98  // non
99  // utf8 characters are found. An alternative would be to throw an exception
100 
101  // First get the size
102  const int utf16_size = MultiByteToWideChar(CP_UTF8, 0, str.c_str(), static_cast<int>(str.length()), nullptr, 0);
103 
104  // Now do the conversion
105  std::wstring wstr;
106 
107  wstr.resize(utf16_size);
108  MultiByteToWideChar(CP_UTF8, 0, str.c_str(), static_cast<int>(str.length()), &wstr[0], utf16_size);
109 
110  return wstr;
111 }
112 
113 #endif
114 
115 // Get a file descriptor from a filename (using utf8 to wstring
116 // on windows if requested) without specifying any specific permissions
117 inline int
118 I18nOpen(const std::string & str, const int flags)
119 {
120 #if LOCAL_USE_WIN32_WOPEN
121  // mingw has _wopen
122  // Convert to utf16
123  const std::wstring str_utf16 = Utf8StringToWString(str);
124  return _wopen(str_utf16.c_str(), flags);
125 #else
126  return open(str.c_str(), flags);
127 #endif
128 }
129 
130 // Get a file descriptor from a filename (using utf8 to wstring
131 // on windows if requested)
132 inline int
133 I18nOpen(const std::string & str, const int flags, const int mode)
134 {
135 #if LOCAL_USE_WIN32_WOPEN
136  // mingw has _wopen
137  // Convert to utf16
138  const std::wstring str_utf16 = Utf8StringToWString(str);
139  return _wopen(str_utf16.c_str(), flags, mode);
140 #else
141  return open(str.c_str(), flags, mode);
142 #endif
143 }
144 
145 // Reading wrapper around I18nOpen to avoid explicitly specifying the flags
146 inline int
147 I18nOpenForReading(const std::string & str)
148 {
149 #if LOCAL_USE_WIN32_WOPEN
150  return I18nOpen(str, _O_RDONLY | _O_BINARY);
151 #else
152  return I18nOpen(str, O_RDONLY);
153 #endif
154 }
155 
156 // Writing wrapper around I18nOpen to avoid explicitly specifying the flags
157 inline int
158 I18nOpenForWriting(const std::string & str, const bool append = false)
159 {
160 #if LOCAL_USE_WIN32_WOPEN
161  if (!append)
162  {
163  return I18nOpen(str, _O_WRONLY | _O_CREAT | _O_BINARY, _S_IREAD | _S_IWRITE);
164  }
165  else
166  {
167  return I18nOpen(str, _O_WRONLY | _O_CREAT | _O_APPEND | _O_BINARY, _S_IREAD | _S_IWRITE);
168  }
169 #elif S_IRUSR
170  if (!append)
171  {
172  return I18nOpen(str, O_WRONLY | O_CREAT, S_IRUSR | S_IWUSR);
173  }
174 
175  return I18nOpen(str, O_WRONLY | O_CREAT | O_APPEND, S_IRUSR | S_IWUSR);
176 
177 #else
178  if (!append)
179  {
180  return I18nOpen(str, O_WRONLY | O_CREAT, S_IREAD | S_IWRITE);
181  }
182  else
183  {
184  return I18nOpen(str, O_WRONLY | O_CREAT | O_APPEND, S_IREAD | S_IWRITE);
185  }
186 #endif
187 }
188 
189 // Get a FILE * pointer from a filename (using utf8 to wstring
190 // on windows if requested)
191 inline FILE *
192 I18nFopen(const std::string & str, const std::string & mode)
193 {
194 #if LOCAL_USE_WIN32_WOPEN
195  // Convert to utf16
196  const std::wstring str_utf16 = Utf8StringToWString(str);
197  const std::wstring mode_utf16 = Utf8StringToWString(mode);
198  return _wfopen(str_utf16.c_str(), mode_utf16.c_str());
199 #else
200  return fopen(str.c_str(), mode.c_str());
201 #endif
202 }
203 
204 #if LOCAL_USE_FDSTREAM
205 class I18nOfstream : public std::ostream
206 {
207 public:
208  I18nOfstream(const char * str, std::ios_base::openmode mode = std::ios_base::out)
209  : std::ostream(0)
210  , m_fd(I18nOpenForWriting(str, (mode & std::ios::app) ? true : false))
211  , m_buf(m_fd)
212  {
214  this->rdbuf(&m_buf);
215  }
216 
217  ~I18nOfstream() { this->close(); }
218 
219  bool
220  is_open()
221  {
222  return (m_fd != -1);
223  }
224 
225  void
226  close()
227  {
228  if (m_fd != -1)
229  {
230  ::close(m_fd);
231  }
232  m_fd = -1;
233  }
234 
235 private:
236  int m_fd;
237  itk::fdoutbuf m_buf;
238 };
239 
240 class I18nIfstream : public std::istream
241 {
242 public:
243  I18nIfstream(const char * str, std::ios_base::openmode itkNotused(mode) = std::ios_base::in)
244  : std::istream(0)
245  , m_fd(I18nOpenForReading(str))
246  , m_buf(m_fd)
247  {
249  this->rdbuf(&m_buf);
250  }
251 
252  ~I18nIfstream() { this->close(); }
253 
254  bool
255  is_open()
256  {
257  return (m_fd != -1);
258  }
259 
260  void
261  close()
262  {
263  if (m_fd != -1)
264  {
265  ::close(m_fd);
266  }
267  m_fd = -1;
268  }
269 
270 private:
271  int m_fd;
272  itk::fdinbuf m_buf;
273 };
274 #elif LOCAL_USE_WIN32_WOPEN
275 class I18nOfstream : public std::ofstream
276 {
277 public:
278  I18nOfstream(const char * str, std::ios_base::openmode mode = std::ios_base::out)
279  : std::ofstream(Utf8StringToWString(str).c_str(), mode)
280  {}
281 };
282 
283 class I18nIfstream : public std::ifstream
284 {
285 public:
286  I18nIfstream(const char * str, std::ios_base::openmode mode = std::ios_base::in)
287  : std::ifstream(Utf8StringToWString(str).c_str(), mode)
288  {}
289 };
290 #else
291 using I18nOfstream = std::ofstream;
292 using I18nIfstream = std::ifstream;
293 #endif
294 } // namespace i18n
295 } // namespace itk
296 
297 #undef LOCAL_USE_WIN32_WOPEN
298 #undef LOCAL_USE_FDSTREAM
299 
300 #endif /* itkInternationalizationIOHelpers_h */
itk::i18n::I18nOpen
int I18nOpen(const std::string &str, const int flags)
Definition: itkInternationalizationIOHelpers.h:118
itk::i18n::IsStringEncodingValid
bool IsStringEncodingValid(const std::string &)
Definition: itkInternationalizationIOHelpers.h:85
itk::i18n::I18nOpenForWriting
int I18nOpenForWriting(const std::string &str, const bool append=false)
Definition: itkInternationalizationIOHelpers.h:158
itkMacro.h
itk::i18n::I18nOpenForReading
int I18nOpenForReading(const std::string &str)
Definition: itkInternationalizationIOHelpers.h:147
itk::i18n::I18nFopen
FILE * I18nFopen(const std::string &str, const std::string &mode)
Definition: itkInternationalizationIOHelpers.h:192
itk
The "itk" namespace contains all Insight Segmentation and Registration Toolkit (ITK) classes....
Definition: itkAnatomicalOrientation.h:29
itk::i18n::I18nIfstream
std::ifstream I18nIfstream
Definition: itkInternationalizationIOHelpers.h:292
itk::i18n::I18nOfstream
std::ofstream I18nOfstream
Definition: itkInternationalizationIOHelpers.h:291