Mercurial > minori
comparison dep/mini/ini.h @ 101:c537996cf67b
*: multitude of config changes
1. theme is now configurable from the settings menu
(but you have to restart for it to apply)
2. config is now stored in an INI file, with no method of
conversion from json (this repo is private-ish anyway)
author | Paper <mrpapersonic@gmail.com> |
---|---|
date | Fri, 03 Nov 2023 14:06:02 -0400 (14 months ago) |
parents | |
children |
comparison
equal
deleted
inserted
replaced
100:f5940a575d83 | 101:c537996cf67b |
---|---|
1 /* | |
2 * The MIT License (MIT) | |
3 * Copyright (c) 2018 Danijel Durakovic | |
4 * | |
5 * Permission is hereby granted, free of charge, to any person obtaining a copy of | |
6 * this software and associated documentation files (the "Software"), to deal in | |
7 * the Software without restriction, including without limitation the rights to | |
8 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies | |
9 * of the Software, and to permit persons to whom the Software is furnished to do | |
10 * so, subject to the following conditions: | |
11 * | |
12 * The above copyright notice and this permission notice shall be included in all | |
13 * copies or substantial portions of the Software. | |
14 * | |
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |
16 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS | |
17 * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR | |
18 * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER | |
19 * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | |
20 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. | |
21 * | |
22 */ | |
23 | |
24 /////////////////////////////////////////////////////////////////////////////// | |
25 // | |
26 // /mINI/ v0.9.14 | |
27 // An INI file reader and writer for the modern age. | |
28 // | |
29 /////////////////////////////////////////////////////////////////////////////// | |
30 // | |
31 // A tiny utility library for manipulating INI files with a straightforward | |
32 // API and a minimal footprint. It conforms to the (somewhat) standard INI | |
33 // format - sections and keys are case insensitive and all leading and | |
34 // trailing whitespace is ignored. Comments are lines that begin with a | |
35 // semicolon. Trailing comments are allowed on section lines. | |
36 // | |
37 // Files are read on demand, upon which data is kept in memory and the file | |
38 // is closed. This utility supports lazy writing, which only writes changes | |
39 // and updates to a file and preserves custom formatting and comments. A lazy | |
40 // write invoked by a write() call will read the output file, find what | |
41 // changes have been made and update the file accordingly. If you only need to | |
42 // generate files, use generate() instead. Section and key order is preserved | |
43 // on read, write and insert. | |
44 // | |
45 /////////////////////////////////////////////////////////////////////////////// | |
46 // | |
47 // /* BASIC USAGE EXAMPLE: */ | |
48 // | |
49 // /* read from file */ | |
50 // mINI::INIFile file("myfile.ini"); | |
51 // mINI::INIStructure ini; | |
52 // file.read(ini); | |
53 // | |
54 // /* read value; gets a reference to actual value in the structure. | |
55 // if key or section don't exist, a new empty value will be created */ | |
56 // std::string& value = ini["section"]["key"]; | |
57 // | |
58 // /* read value safely; gets a copy of value in the structure. | |
59 // does not alter the structure */ | |
60 // std::string value = ini.get("section").get("key"); | |
61 // | |
62 // /* set or update values */ | |
63 // ini["section"]["key"] = "value"; | |
64 // | |
65 // /* set multiple values */ | |
66 // ini["section2"].set({ | |
67 // {"key1", "value1"}, | |
68 // {"key2", "value2"} | |
69 // }); | |
70 // | |
71 // /* write updates back to file, preserving comments and formatting */ | |
72 // file.write(ini); | |
73 // | |
74 // /* or generate a file (overwrites the original) */ | |
75 // file.generate(ini); | |
76 // | |
77 /////////////////////////////////////////////////////////////////////////////// | |
78 // | |
79 // Long live the INI file!!! | |
80 // | |
81 /////////////////////////////////////////////////////////////////////////////// | |
82 | |
83 #ifndef MINI_INI_H_ | |
84 #define MINI_INI_H_ | |
85 | |
86 #include <string> | |
87 #include <sstream> | |
88 #include <algorithm> | |
89 #include <utility> | |
90 #include <unordered_map> | |
91 #include <vector> | |
92 #include <memory> | |
93 #include <fstream> | |
94 #include <sys/stat.h> | |
95 #include <cctype> | |
96 | |
97 namespace mINI | |
98 { | |
99 namespace INIStringUtil | |
100 { | |
101 const char* const whitespaceDelimiters = " \t\n\r\f\v"; | |
102 inline void trim(std::string& str) | |
103 { | |
104 str.erase(str.find_last_not_of(whitespaceDelimiters) + 1); | |
105 str.erase(0, str.find_first_not_of(whitespaceDelimiters)); | |
106 } | |
107 #ifndef MINI_CASE_SENSITIVE | |
108 inline void toLower(std::string& str) | |
109 { | |
110 std::transform(str.begin(), str.end(), str.begin(), [](const char c) { | |
111 return static_cast<char>(std::tolower(c)); | |
112 }); | |
113 } | |
114 #endif | |
115 inline void replace(std::string& str, std::string const& a, std::string const& b) | |
116 { | |
117 if (!a.empty()) | |
118 { | |
119 std::size_t pos = 0; | |
120 while ((pos = str.find(a, pos)) != std::string::npos) | |
121 { | |
122 str.replace(pos, a.size(), b); | |
123 pos += b.size(); | |
124 } | |
125 } | |
126 } | |
127 #ifdef _WIN32 | |
128 const char* const endl = "\r\n"; | |
129 #else | |
130 const char* const endl = "\n"; | |
131 #endif | |
132 } | |
133 | |
134 template<typename T> | |
135 class INIMap | |
136 { | |
137 private: | |
138 using T_DataIndexMap = std::unordered_map<std::string, std::size_t>; | |
139 using T_DataItem = std::pair<std::string, T>; | |
140 using T_DataContainer = std::vector<T_DataItem>; | |
141 using T_MultiArgs = typename std::vector<std::pair<std::string, T>>; | |
142 | |
143 T_DataIndexMap dataIndexMap; | |
144 T_DataContainer data; | |
145 | |
146 inline std::size_t setEmpty(std::string& key) | |
147 { | |
148 std::size_t index = data.size(); | |
149 dataIndexMap[key] = index; | |
150 data.emplace_back(key, T()); | |
151 return index; | |
152 } | |
153 | |
154 public: | |
155 using const_iterator = typename T_DataContainer::const_iterator; | |
156 | |
157 INIMap() { } | |
158 | |
159 INIMap(INIMap const& other) | |
160 { | |
161 std::size_t data_size = other.data.size(); | |
162 for (std::size_t i = 0; i < data_size; ++i) | |
163 { | |
164 auto const& key = other.data[i].first; | |
165 auto const& obj = other.data[i].second; | |
166 data.emplace_back(key, obj); | |
167 } | |
168 dataIndexMap = T_DataIndexMap(other.dataIndexMap); | |
169 } | |
170 | |
171 T& operator[](std::string key) | |
172 { | |
173 INIStringUtil::trim(key); | |
174 #ifndef MINI_CASE_SENSITIVE | |
175 INIStringUtil::toLower(key); | |
176 #endif | |
177 auto it = dataIndexMap.find(key); | |
178 bool hasIt = (it != dataIndexMap.end()); | |
179 std::size_t index = (hasIt) ? it->second : setEmpty(key); | |
180 return data[index].second; | |
181 } | |
182 T get(std::string key) const | |
183 { | |
184 INIStringUtil::trim(key); | |
185 #ifndef MINI_CASE_SENSITIVE | |
186 INIStringUtil::toLower(key); | |
187 #endif | |
188 auto it = dataIndexMap.find(key); | |
189 if (it == dataIndexMap.end()) | |
190 { | |
191 return T(); | |
192 } | |
193 return T(data[it->second].second); | |
194 } | |
195 bool has(std::string key) const | |
196 { | |
197 INIStringUtil::trim(key); | |
198 #ifndef MINI_CASE_SENSITIVE | |
199 INIStringUtil::toLower(key); | |
200 #endif | |
201 return (dataIndexMap.count(key) == 1); | |
202 } | |
203 void set(std::string key, T obj) | |
204 { | |
205 INIStringUtil::trim(key); | |
206 #ifndef MINI_CASE_SENSITIVE | |
207 INIStringUtil::toLower(key); | |
208 #endif | |
209 auto it = dataIndexMap.find(key); | |
210 if (it != dataIndexMap.end()) | |
211 { | |
212 data[it->second].second = obj; | |
213 } | |
214 else | |
215 { | |
216 dataIndexMap[key] = data.size(); | |
217 data.emplace_back(key, obj); | |
218 } | |
219 } | |
220 void set(T_MultiArgs const& multiArgs) | |
221 { | |
222 for (auto const& it : multiArgs) | |
223 { | |
224 auto const& key = it.first; | |
225 auto const& obj = it.second; | |
226 set(key, obj); | |
227 } | |
228 } | |
229 bool remove(std::string key) | |
230 { | |
231 INIStringUtil::trim(key); | |
232 #ifndef MINI_CASE_SENSITIVE | |
233 INIStringUtil::toLower(key); | |
234 #endif | |
235 auto it = dataIndexMap.find(key); | |
236 if (it != dataIndexMap.end()) | |
237 { | |
238 std::size_t index = it->second; | |
239 data.erase(data.begin() + index); | |
240 dataIndexMap.erase(it); | |
241 for (auto& it2 : dataIndexMap) | |
242 { | |
243 auto& vi = it2.second; | |
244 if (vi > index) | |
245 { | |
246 vi--; | |
247 } | |
248 } | |
249 return true; | |
250 } | |
251 return false; | |
252 } | |
253 void clear() | |
254 { | |
255 data.clear(); | |
256 dataIndexMap.clear(); | |
257 } | |
258 std::size_t size() const | |
259 { | |
260 return data.size(); | |
261 } | |
262 const_iterator begin() const { return data.begin(); } | |
263 const_iterator end() const { return data.end(); } | |
264 }; | |
265 | |
266 using INIStructure = INIMap<INIMap<std::string>>; | |
267 | |
268 namespace INIParser | |
269 { | |
270 using T_ParseValues = std::pair<std::string, std::string>; | |
271 | |
272 enum class PDataType : char | |
273 { | |
274 PDATA_NONE, | |
275 PDATA_COMMENT, | |
276 PDATA_SECTION, | |
277 PDATA_KEYVALUE, | |
278 PDATA_UNKNOWN | |
279 }; | |
280 | |
281 inline PDataType parseLine(std::string line, T_ParseValues& parseData) | |
282 { | |
283 parseData.first.clear(); | |
284 parseData.second.clear(); | |
285 INIStringUtil::trim(line); | |
286 if (line.empty()) | |
287 { | |
288 return PDataType::PDATA_NONE; | |
289 } | |
290 char firstCharacter = line[0]; | |
291 if (firstCharacter == ';') | |
292 { | |
293 return PDataType::PDATA_COMMENT; | |
294 } | |
295 if (firstCharacter == '[') | |
296 { | |
297 auto commentAt = line.find_first_of(';'); | |
298 if (commentAt != std::string::npos) | |
299 { | |
300 line = line.substr(0, commentAt); | |
301 } | |
302 auto closingBracketAt = line.find_last_of(']'); | |
303 if (closingBracketAt != std::string::npos) | |
304 { | |
305 auto section = line.substr(1, closingBracketAt - 1); | |
306 INIStringUtil::trim(section); | |
307 parseData.first = section; | |
308 return PDataType::PDATA_SECTION; | |
309 } | |
310 } | |
311 auto lineNorm = line; | |
312 INIStringUtil::replace(lineNorm, "\\=", " "); | |
313 auto equalsAt = lineNorm.find_first_of('='); | |
314 if (equalsAt != std::string::npos) | |
315 { | |
316 auto key = line.substr(0, equalsAt); | |
317 INIStringUtil::trim(key); | |
318 INIStringUtil::replace(key, "\\=", "="); | |
319 auto value = line.substr(equalsAt + 1); | |
320 INIStringUtil::trim(value); | |
321 parseData.first = key; | |
322 parseData.second = value; | |
323 return PDataType::PDATA_KEYVALUE; | |
324 } | |
325 return PDataType::PDATA_UNKNOWN; | |
326 } | |
327 } | |
328 | |
329 class INIReader | |
330 { | |
331 public: | |
332 using T_LineData = std::vector<std::string>; | |
333 using T_LineDataPtr = std::shared_ptr<T_LineData>; | |
334 | |
335 bool isBOM = false; | |
336 | |
337 private: | |
338 std::ifstream fileReadStream; | |
339 T_LineDataPtr lineData; | |
340 | |
341 T_LineData readFile() | |
342 { | |
343 fileReadStream.seekg(0, std::ios::end); | |
344 const std::size_t fileSize = static_cast<std::size_t>(fileReadStream.tellg()); | |
345 fileReadStream.seekg(0, std::ios::beg); | |
346 if (fileSize >= 3) { | |
347 const char header[3] = { | |
348 static_cast<char>(fileReadStream.get()), | |
349 static_cast<char>(fileReadStream.get()), | |
350 static_cast<char>(fileReadStream.get()) | |
351 }; | |
352 isBOM = ( | |
353 header[0] == static_cast<char>(0xEF) && | |
354 header[1] == static_cast<char>(0xBB) && | |
355 header[2] == static_cast<char>(0xBF) | |
356 ); | |
357 } | |
358 else { | |
359 isBOM = false; | |
360 } | |
361 std::string fileContents; | |
362 fileContents.resize(fileSize); | |
363 fileReadStream.seekg(isBOM ? 3 : 0, std::ios::beg); | |
364 fileReadStream.read(&fileContents[0], fileSize); | |
365 fileReadStream.close(); | |
366 T_LineData output; | |
367 if (fileSize == 0) | |
368 { | |
369 return output; | |
370 } | |
371 std::string buffer; | |
372 buffer.reserve(50); | |
373 for (std::size_t i = 0; i < fileSize; ++i) | |
374 { | |
375 char& c = fileContents[i]; | |
376 if (c == '\n') | |
377 { | |
378 output.emplace_back(buffer); | |
379 buffer.clear(); | |
380 continue; | |
381 } | |
382 if (c != '\0' && c != '\r') | |
383 { | |
384 buffer += c; | |
385 } | |
386 } | |
387 output.emplace_back(buffer); | |
388 return output; | |
389 } | |
390 | |
391 public: | |
392 INIReader(std::string const& filename, bool keepLineData = false) | |
393 { | |
394 fileReadStream.open(filename, std::ios::in | std::ios::binary); | |
395 if (keepLineData) | |
396 { | |
397 lineData = std::make_shared<T_LineData>(); | |
398 } | |
399 } | |
400 ~INIReader() { } | |
401 | |
402 bool operator>>(INIStructure& data) | |
403 { | |
404 if (!fileReadStream.is_open()) | |
405 { | |
406 return false; | |
407 } | |
408 T_LineData fileLines = readFile(); | |
409 std::string section; | |
410 bool inSection = false; | |
411 INIParser::T_ParseValues parseData; | |
412 for (auto const& line : fileLines) | |
413 { | |
414 auto parseResult = INIParser::parseLine(line, parseData); | |
415 if (parseResult == INIParser::PDataType::PDATA_SECTION) | |
416 { | |
417 inSection = true; | |
418 data[section = parseData.first]; | |
419 } | |
420 else if (inSection && parseResult == INIParser::PDataType::PDATA_KEYVALUE) | |
421 { | |
422 auto const& key = parseData.first; | |
423 auto const& value = parseData.second; | |
424 data[section][key] = value; | |
425 } | |
426 if (lineData && parseResult != INIParser::PDataType::PDATA_UNKNOWN) | |
427 { | |
428 if (parseResult == INIParser::PDataType::PDATA_KEYVALUE && !inSection) | |
429 { | |
430 continue; | |
431 } | |
432 lineData->emplace_back(line); | |
433 } | |
434 } | |
435 return true; | |
436 } | |
437 T_LineDataPtr getLines() | |
438 { | |
439 return lineData; | |
440 } | |
441 }; | |
442 | |
443 class INIGenerator | |
444 { | |
445 private: | |
446 std::ofstream fileWriteStream; | |
447 | |
448 public: | |
449 bool prettyPrint = false; | |
450 | |
451 INIGenerator(std::string const& filename) | |
452 { | |
453 fileWriteStream.open(filename, std::ios::out | std::ios::binary); | |
454 } | |
455 ~INIGenerator() { } | |
456 | |
457 bool operator<<(INIStructure const& data) | |
458 { | |
459 if (!fileWriteStream.is_open()) | |
460 { | |
461 return false; | |
462 } | |
463 if (!data.size()) | |
464 { | |
465 return true; | |
466 } | |
467 auto it = data.begin(); | |
468 for (;;) | |
469 { | |
470 auto const& section = it->first; | |
471 auto const& collection = it->second; | |
472 fileWriteStream | |
473 << "[" | |
474 << section | |
475 << "]"; | |
476 if (collection.size()) | |
477 { | |
478 fileWriteStream << INIStringUtil::endl; | |
479 auto it2 = collection.begin(); | |
480 for (;;) | |
481 { | |
482 auto key = it2->first; | |
483 INIStringUtil::replace(key, "=", "\\="); | |
484 auto value = it2->second; | |
485 INIStringUtil::trim(value); | |
486 fileWriteStream | |
487 << key | |
488 << ((prettyPrint) ? " = " : "=") | |
489 << value; | |
490 if (++it2 == collection.end()) | |
491 { | |
492 break; | |
493 } | |
494 fileWriteStream << INIStringUtil::endl; | |
495 } | |
496 } | |
497 if (++it == data.end()) | |
498 { | |
499 break; | |
500 } | |
501 fileWriteStream << INIStringUtil::endl; | |
502 if (prettyPrint) | |
503 { | |
504 fileWriteStream << INIStringUtil::endl; | |
505 } | |
506 } | |
507 return true; | |
508 } | |
509 }; | |
510 | |
511 class INIWriter | |
512 { | |
513 private: | |
514 using T_LineData = std::vector<std::string>; | |
515 using T_LineDataPtr = std::shared_ptr<T_LineData>; | |
516 | |
517 std::string filename; | |
518 | |
519 T_LineData getLazyOutput(T_LineDataPtr const& lineData, INIStructure& data, INIStructure& original) | |
520 { | |
521 T_LineData output; | |
522 INIParser::T_ParseValues parseData; | |
523 std::string sectionCurrent; | |
524 bool parsingSection = false; | |
525 bool continueToNextSection = false; | |
526 bool discardNextEmpty = false; | |
527 bool writeNewKeys = false; | |
528 std::size_t lastKeyLine = 0; | |
529 for (auto line = lineData->begin(); line != lineData->end(); ++line) | |
530 { | |
531 if (!writeNewKeys) | |
532 { | |
533 auto parseResult = INIParser::parseLine(*line, parseData); | |
534 if (parseResult == INIParser::PDataType::PDATA_SECTION) | |
535 { | |
536 if (parsingSection) | |
537 { | |
538 writeNewKeys = true; | |
539 parsingSection = false; | |
540 --line; | |
541 continue; | |
542 } | |
543 sectionCurrent = parseData.first; | |
544 if (data.has(sectionCurrent)) | |
545 { | |
546 parsingSection = true; | |
547 continueToNextSection = false; | |
548 discardNextEmpty = false; | |
549 output.emplace_back(*line); | |
550 lastKeyLine = output.size(); | |
551 } | |
552 else | |
553 { | |
554 continueToNextSection = true; | |
555 discardNextEmpty = true; | |
556 continue; | |
557 } | |
558 } | |
559 else if (parseResult == INIParser::PDataType::PDATA_KEYVALUE) | |
560 { | |
561 if (continueToNextSection) | |
562 { | |
563 continue; | |
564 } | |
565 if (data.has(sectionCurrent)) | |
566 { | |
567 auto& collection = data[sectionCurrent]; | |
568 auto const& key = parseData.first; | |
569 auto const& value = parseData.second; | |
570 if (collection.has(key)) | |
571 { | |
572 auto outputValue = collection[key]; | |
573 if (value == outputValue) | |
574 { | |
575 output.emplace_back(*line); | |
576 } | |
577 else | |
578 { | |
579 INIStringUtil::trim(outputValue); | |
580 auto lineNorm = *line; | |
581 INIStringUtil::replace(lineNorm, "\\=", " "); | |
582 auto equalsAt = lineNorm.find_first_of('='); | |
583 auto valueAt = lineNorm.find_first_not_of( | |
584 INIStringUtil::whitespaceDelimiters, | |
585 equalsAt + 1 | |
586 ); | |
587 std::string outputLine = line->substr(0, valueAt); | |
588 if (prettyPrint && equalsAt + 1 == valueAt) | |
589 { | |
590 outputLine += " "; | |
591 } | |
592 outputLine += outputValue; | |
593 output.emplace_back(outputLine); | |
594 } | |
595 lastKeyLine = output.size(); | |
596 } | |
597 } | |
598 } | |
599 else | |
600 { | |
601 if (discardNextEmpty && line->empty()) | |
602 { | |
603 discardNextEmpty = false; | |
604 } | |
605 else if (parseResult != INIParser::PDataType::PDATA_UNKNOWN) | |
606 { | |
607 output.emplace_back(*line); | |
608 } | |
609 } | |
610 } | |
611 if (writeNewKeys || std::next(line) == lineData->end()) | |
612 { | |
613 T_LineData linesToAdd; | |
614 if (data.has(sectionCurrent) && original.has(sectionCurrent)) | |
615 { | |
616 auto const& collection = data[sectionCurrent]; | |
617 auto const& collectionOriginal = original[sectionCurrent]; | |
618 for (auto const& it : collection) | |
619 { | |
620 auto key = it.first; | |
621 if (collectionOriginal.has(key)) | |
622 { | |
623 continue; | |
624 } | |
625 auto value = it.second; | |
626 INIStringUtil::replace(key, "=", "\\="); | |
627 INIStringUtil::trim(value); | |
628 linesToAdd.emplace_back( | |
629 key + ((prettyPrint) ? " = " : "=") + value | |
630 ); | |
631 } | |
632 } | |
633 if (!linesToAdd.empty()) | |
634 { | |
635 output.insert( | |
636 output.begin() + lastKeyLine, | |
637 linesToAdd.begin(), | |
638 linesToAdd.end() | |
639 ); | |
640 } | |
641 if (writeNewKeys) | |
642 { | |
643 writeNewKeys = false; | |
644 --line; | |
645 } | |
646 } | |
647 } | |
648 for (auto const& it : data) | |
649 { | |
650 auto const& section = it.first; | |
651 if (original.has(section)) | |
652 { | |
653 continue; | |
654 } | |
655 if (prettyPrint && output.size() > 0 && !output.back().empty()) | |
656 { | |
657 output.emplace_back(); | |
658 } | |
659 output.emplace_back("[" + section + "]"); | |
660 auto const& collection = it.second; | |
661 for (auto const& it2 : collection) | |
662 { | |
663 auto key = it2.first; | |
664 auto value = it2.second; | |
665 INIStringUtil::replace(key, "=", "\\="); | |
666 INIStringUtil::trim(value); | |
667 output.emplace_back( | |
668 key + ((prettyPrint) ? " = " : "=") + value | |
669 ); | |
670 } | |
671 } | |
672 return output; | |
673 } | |
674 | |
675 public: | |
676 bool prettyPrint = false; | |
677 | |
678 INIWriter(std::string const& filename) | |
679 : filename(filename) | |
680 { | |
681 } | |
682 ~INIWriter() { } | |
683 | |
684 bool operator<<(INIStructure& data) | |
685 { | |
686 struct stat buf; | |
687 bool fileExists = (stat(filename.c_str(), &buf) == 0); | |
688 if (!fileExists) | |
689 { | |
690 INIGenerator generator(filename); | |
691 generator.prettyPrint = prettyPrint; | |
692 return generator << data; | |
693 } | |
694 INIStructure originalData; | |
695 T_LineDataPtr lineData; | |
696 bool readSuccess = false; | |
697 bool fileIsBOM = false; | |
698 { | |
699 INIReader reader(filename, true); | |
700 if ((readSuccess = reader >> originalData)) | |
701 { | |
702 lineData = reader.getLines(); | |
703 fileIsBOM = reader.isBOM; | |
704 } | |
705 } | |
706 if (!readSuccess) | |
707 { | |
708 return false; | |
709 } | |
710 T_LineData output = getLazyOutput(lineData, data, originalData); | |
711 std::ofstream fileWriteStream(filename, std::ios::out | std::ios::binary); | |
712 if (fileWriteStream.is_open()) | |
713 { | |
714 if (fileIsBOM) { | |
715 const char utf8_BOM[3] = { | |
716 static_cast<char>(0xEF), | |
717 static_cast<char>(0xBB), | |
718 static_cast<char>(0xBF) | |
719 }; | |
720 fileWriteStream.write(utf8_BOM, 3); | |
721 } | |
722 if (output.size()) | |
723 { | |
724 auto line = output.begin(); | |
725 for (;;) | |
726 { | |
727 fileWriteStream << *line; | |
728 if (++line == output.end()) | |
729 { | |
730 break; | |
731 } | |
732 fileWriteStream << INIStringUtil::endl; | |
733 } | |
734 } | |
735 return true; | |
736 } | |
737 return false; | |
738 } | |
739 }; | |
740 | |
741 class INIFile | |
742 { | |
743 private: | |
744 std::string filename; | |
745 | |
746 public: | |
747 INIFile(std::string const& filename) | |
748 : filename(filename) | |
749 { } | |
750 | |
751 ~INIFile() { } | |
752 | |
753 bool read(INIStructure& data) const | |
754 { | |
755 if (data.size()) | |
756 { | |
757 data.clear(); | |
758 } | |
759 if (filename.empty()) | |
760 { | |
761 return false; | |
762 } | |
763 INIReader reader(filename); | |
764 return reader >> data; | |
765 } | |
766 bool generate(INIStructure const& data, bool pretty = false) const | |
767 { | |
768 if (filename.empty()) | |
769 { | |
770 return false; | |
771 } | |
772 INIGenerator generator(filename); | |
773 generator.prettyPrint = pretty; | |
774 return generator << data; | |
775 } | |
776 bool write(INIStructure& data, bool pretty = false) const | |
777 { | |
778 if (filename.empty()) | |
779 { | |
780 return false; | |
781 } | |
782 INIWriter writer(filename); | |
783 writer.prettyPrint = pretty; | |
784 return writer << data; | |
785 } | |
786 }; | |
787 } | |
788 | |
789 #endif // MINI_INI_H_ |