-
-
Notifications
You must be signed in to change notification settings - Fork 388
/
Copy pathPresetFileParser.cpp
194 lines (156 loc) · 4.91 KB
/
PresetFileParser.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
#include "PresetFileParser.hpp"
#include <algorithm>
#include <fstream>
#include <sstream>
#include <vector>
namespace libprojectM {
auto PresetFileParser::Read(const std::string& presetFile) -> bool
{
std::ifstream presetStream(presetFile.c_str(), std::ios_base::in | std::ios_base::binary);
return Read(presetStream);
}
auto PresetFileParser::Read(std::istream& presetStream) -> bool
{
if (!presetStream.good())
{
return false;
}
presetStream.seekg(0, presetStream.end);
auto fileSize = presetStream.tellg();
presetStream.seekg(0, presetStream.beg);
if (static_cast<size_t>(fileSize) > maxFileSize)
{
return false;
}
std::vector<char> presetFileContents(fileSize);
presetStream.read(presetFileContents.data(), fileSize);
if (presetStream.fail() || presetStream.bad())
{
return false;
}
size_t startPos{0}; //!< Starting position of current line
size_t pos{0}; //!< Current read position
auto parseLineIfDataAvailable = [this, &pos, &startPos, &presetFileContents]() {
if (pos > startPos)
{
auto beg = presetFileContents.begin();
std::string line(beg + startPos, beg + pos);
ParseLine(line);
}
};
while (pos < presetFileContents.size())
{
switch (presetFileContents[pos])
{
case '\r':
case '\n':
// EOL, skip over CRLF
parseLineIfDataAvailable();
startPos = pos + 1;
break;
case '\0':
// Null char is not expected. Could be a random binary file.
return false;
}
++pos;
}
parseLineIfDataAvailable();
return !m_presetValues.empty();
}
auto PresetFileParser::GetCode(const std::string& keyPrefix) const -> std::string
{
auto lowerKey = ToLower(keyPrefix);
std::stringstream code; //!< The parsed code
std::string key(lowerKey.length() + 5, '\0'); //!< Allocate a string that can hold up to 5 digits.
key.replace(0, lowerKey.length(), lowerKey);
for (int index{1}; index <= 99999; ++index)
{
key.replace(lowerKey.length(), 5, std::to_string(index));
if (m_presetValues.find(key) == m_presetValues.end())
{
break;
}
auto line = m_presetValues.at(key);
// Remove backtick char in shader code
if (!line.empty() && line.at(0) == '`')
{
line.erase(0, 1);
}
code << line << std::endl;
}
auto codeStr = code.str();
return codeStr;
}
auto PresetFileParser::GetInt(const std::string& key, int defaultValue) -> int
{
auto lowerKey = ToLower(key);
if (m_presetValues.find(lowerKey) != m_presetValues.end())
{
try
{
return std::stoi(m_presetValues.at(lowerKey));
}
catch (std::logic_error&)
{
}
}
return defaultValue;
}
auto PresetFileParser::GetFloat(const std::string& key, float defaultValue) -> float
{
auto lowerKey = ToLower(key);
if (m_presetValues.find(lowerKey) != m_presetValues.end())
{
try
{
return std::stof(m_presetValues.at(lowerKey));
}
catch (std::logic_error&)
{
}
}
return defaultValue;
}
auto PresetFileParser::GetBool(const std::string& key, bool defaultValue) -> bool
{
return GetInt(key, static_cast<int>(defaultValue)) > 0;
}
auto PresetFileParser::GetString(const std::string& key, const std::string& defaultValue) -> std::string
{
auto lowerKey = ToLower(key);
if (m_presetValues.find(lowerKey) != m_presetValues.end())
{
return m_presetValues.at(lowerKey);
}
return defaultValue;
}
const std::map<std::string, std::string>& PresetFileParser::PresetValues() const
{
return m_presetValues;
}
void PresetFileParser::ParseLine(const std::string& line)
{
// Search for first delimiter, either space or equal
auto varNameDelimiterPos = line.find_first_of(" =");
if (varNameDelimiterPos == std::string::npos || varNameDelimiterPos == 0)
{
// Empty line, delimiter at start of line or no delimiter found, skip.
return;
}
// Convert key to lower case, as INI functions are not case-sensitive.
std::string varName(ToLower(std::string(line.begin(), line.begin() + varNameDelimiterPos)));
std::string value(line.begin() + varNameDelimiterPos + 1, line.end());
// Only add first occurrence to mimic Milkdrop behaviour
if (!varName.empty() && m_presetValues.find(varName) == m_presetValues.end())
{
m_presetValues.emplace(std::move(varName), std::move(value));
}
}
auto PresetFileParser::ToLower(std::string str) -> std::string
{
std::transform(str.begin(), str.end(), str.begin(),
[](unsigned char c){ return std::tolower(c); }
);
return str;
}
} // namespace libprojectM