TDME2 1.9.121
TextEditorTabView.cpp
Go to the documentation of this file.
2
3#include <cctype>
4#include <string>
5
6#include <tdme/tdme.h>
11#include <tdme/gui/GUI.h>
21
22using std::string;
23
25
30using tdme::gui::GUI;
39
40TextEditorTabView::TextEditorTabView(EditorView* editorView, const string& tabId, GUIScreenNode* screenNode, const string& extension)
41{
42 this->editorView = editorView;
43 this->tabId = tabId;
44 this->popUps = editorView->getPopUps();
45 engine = Engine::createOffScreenInstance(512, 512, false, false, false);
46 engine->setSceneColor(Color4(125.0f / 255.0f, 125.0f / 255.0f, 125.0f / 255.0f, 1.0f));
47 engine->getGUI()->addScreen(screenNode->getId(), screenNode);
48 engine->getGUI()->addRenderScreen(screenNode->getId());
49 auto multiLineTextNode = required_dynamic_cast<GUIStyledTextNode*>(screenNode->getNodeById("text"));
50 if (std::find(xmlLanguage.extensions.begin(), xmlLanguage.extensions.end(), extension) != xmlLanguage.extensions.end()) {
51 auto& language = xmlLanguage;
52 auto code = multiLineTextNode->getText().getString();
53 auto startIdx = 0;
54 auto endIdx = -1;
55 auto lc = '\0';
56 auto llc = '\0';
57 auto lllc = '\0';
58 auto nc = '\0';
59 auto inlineComment = false;
60 auto quote = '\0';
61 auto tag = false;
62 auto tagMode = 0;
63 auto attribute = false;
64 for (auto i = 0; i < code.size(); i++) {
65 auto c = code[i];
66 auto nc = i + 1 < code.size()?code[i + 1]:'\0';
67 auto nnc = i + 2 < code.size()?code[i + 2]:'\0';
68 auto nnnc = i + 3 < code.size()?code[i + 3]:'\0';
69 if (tag == false && inlineComment == false && quote == '\0') {
70 if (language.commentInlineStart.empty() == false &&
71 c == language.commentInlineStart[0] &&
72 nc == language.commentInlineStart[1] &&
73 nnc == language.commentInlineStart[2] &&
74 nnnc == language.commentInlineStart[3]) {
75 inlineComment = true;
76 startIdx = i - 1;
77 endIdx = -1;
78 } else
79 if (c == '<') {
80 tag = true;
81 tagMode = 0;
82 startIdx = i;
83 endIdx = -1;
84 }
85 } else
86 if (inlineComment == true) {
87 if (language.commentInlineEnd.empty() == false &&
88 llc == language.commentInlineEnd[0] &&
89 lc == language.commentInlineEnd[1] &&
90 c == language.commentInlineEnd[2]) {
91 inlineComment = false;
92 endIdx = i + 1;
93 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, commentInlineColor);
94 startIdx = endIdx + 2;
95 endIdx = -1;
96 }
97 } else
98 if (quote != '\0') {
99 if (c == quote) {
100 quote = '\0';
101 endIdx = i + 1;
102 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, literalColor);
103 startIdx = endIdx;
104 endIdx = -1;
105 }
106 } else
107 if (tag == true) {
108 if (quote == '\0' && language.quotes.find(c) != string::npos) {
109 endIdx = i;
110 if (startIdx != endIdx) {
111 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, tagMode == 0?keyword1Color:keyword2Color);
112 tagMode++;
113 }
114 quote = c;
115 startIdx = i;
116 endIdx = -1;
117 } else
118 if (lc == '/' && c == '>') {
119 endIdx = i - 1;
120 if (startIdx != endIdx) {
121 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, tagMode == 0?keyword1Color:keyword2Color);
122 }
123 multiLineTextNode->setTextStyle(endIdx, endIdx - 1 + 2, keyword1Color);
124 startIdx = i + 1;
125 endIdx = -1;
126 tag = false;
127 tagMode = 0;
128 } else
129 if (c == '>') {
130 endIdx = i;
131 if (startIdx != endIdx) {
132 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, tagMode == 0?keyword1Color:keyword2Color);
133 }
134 multiLineTextNode->setTextStyle(endIdx, endIdx - 1 + 1, keyword1Color);
135 startIdx = i + 1;
136 endIdx = -1;
137 tag = false;
138 tagMode = 0;
139 } else
140 if (language.delimiters.find(c) != string::npos) {
141 endIdx = i;
142 if (startIdx != endIdx) {
143 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, tagMode == 0?keyword1Color:keyword2Color);
144 }
145 startIdx = i + 1;
146 endIdx = -1;
147 tagMode++;
148 }
149 }
150 lllc = llc;
151 llc = lc;
152 lc = c;
153 }
154 } else
155 if (std::find(propertiesLanguage.extensions.begin(), propertiesLanguage.extensions.end(), extension) != propertiesLanguage.extensions.end()) {
156 auto& language = propertiesLanguage;
157 auto code = multiLineTextNode->getText().getString();
158 auto commentCount = 0;
159 auto delimiterCount = 0;
160 auto nonWhitespaceCount = 0;
161 auto startIdx = -1;
162 auto endIdx = -1;
163 auto commentLine = false;
164 for (auto i = 0; i < code.size(); i++) {
165 auto c = code[i];
166 if (c == '\n') {
167 endIdx = i;
168 if (commentLine == true) {
169 if (startIdx != -1 && endIdx != -1 && startIdx != endIdx) {
170 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, commentLineColor);
171 }
172 } else
173 if (startIdx != -1 && endIdx != -1 && startIdx != endIdx) {
174 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, literalColor);
175 }
176 commentCount = 0;
177 delimiterCount = 0;
178 nonWhitespaceCount = 0;
179 startIdx = -1;
180 endIdx = -1;
181 commentLine = false;
182 } else
183 if (language.whitespaces.find(c) != string::npos) {
184 // no op
185 } else
186 if (c == language.comment && nonWhitespaceCount == 0) {
187 startIdx = i;
188 endIdx = -1;
189 commentLine = true;
190 nonWhitespaceCount++;
191 } else
192 if (c == language.delimiter && nonWhitespaceCount > 0) {
193 endIdx = i;
194 if (startIdx != -1 && startIdx != endIdx) {
195 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, keyword2Color);
196 }
197 startIdx = i + 1;
198 endIdx = -1;
199 nonWhitespaceCount++;
200 } else {
201 if (nonWhitespaceCount == 0) startIdx = i;
202 nonWhitespaceCount++;
203 }
204 }
205 endIdx = code.size();
206 if (commentLine == true) {
207 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, commentLineColor);
208 } else
209 if (startIdx != -1 && startIdx != endIdx) {
210 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, literalColor);
211 }
212 } else {
213 for (auto& language: languages) {
214 if (std::find(language.extensions.begin(), language.extensions.end(), extension) != language.extensions.end()) {
215 auto code = multiLineTextNode->getText().getString();
216 auto keywords1 = StringTools::tokenize(language.keywords1, " ");
217 auto keywords2 = StringTools::tokenize(language.keywords2, " ");
218 auto preprocessorLineKeywords = StringTools::tokenize(language.preprocessorLineKeywords, " ");
219 auto datatypeLiteralSuffixes = StringTools::tokenize(language.datatypeLiteralSuffixes, " ");
220 auto startIdx = 0;
221 auto endIdx = -1;
222 auto lc = '\0';
223 auto llc = '\0';
224 auto nc = '\0';
225 auto inlineComment = false;
226 auto lineComment = false;
227 auto preprocessorLine = false;
228 auto quote = '\0';
229 for (auto i = 0; i < code.size(); i++) {
230 auto c = code[i];
231 auto nc = i + 1 < code.size()?code[i + 1]:'\0';
232 if (inlineComment == false && lineComment == false && preprocessorLine == false && quote == '\0') {
233 if (language.commentLine.empty() == false && (language.commentLine.size() == 1 || c == language.commentLine[0]) && nc == language.commentLine[language.commentLine.size() - 1]) {
234 lineComment = true;
235 startIdx = i;
236 endIdx = -1;
237 } else
238 if (language.commentInlineStart.empty() == false && (language.commentInlineStart.size() == 1 || c == language.commentInlineStart[0]) && nc == language.commentInlineStart[language.commentInlineStart.size() - 1]) {
239 inlineComment = true;
240 startIdx = i;
241 endIdx = -1;
242 } else
243 if (quote == '\0' && language.keywordQuotes.find(c) != string::npos) {
244 quote = c;
245 startIdx = i;
246 endIdx = -1;
247 } else {
248 // delimiter
249 if (language.keywordDelimiters.find(c) != string::npos) {
250 endIdx = i;
251 } else
252 if (i == code.size() - 1) {
253 endIdx = code.size();
254 }
255 if (startIdx != -1 && endIdx != -1 && startIdx != endIdx) {
256 while (code[startIdx] == ' ' || code[startIdx] == '\t') startIdx++;
257 auto word = StringTools::trim(StringTools::substring(code, startIdx, endIdx));
258 if (word.empty() == true) continue;
259 auto literalWord = word;
260 for (auto& datatypeLiteralSuffix: datatypeLiteralSuffixes) {
261 if (StringTools::endsWith(word, datatypeLiteralSuffix) == true) {
262 auto dotCount = 0;
263 auto valid = true;
264 for (auto j = 0; j < word.size() - datatypeLiteralSuffix.size(); j++) {
265 if (word[j] == '.') {
266 dotCount++;
267 if (dotCount > 1) {
268 valid = false;
269 break;
270 }
271 } else
272 if (isdigit(word[j]) == false) {
273 valid = false;
274 break;
275 }
276 }
277 if (valid == true) {
278 literalWord = StringTools::substring(word, 0, word.size() - datatypeLiteralSuffix.size());
279 break;
280 }
281 }
282 }
283 if (Integer::is(literalWord) == true || Float::is(literalWord) == true) {
284 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, literalColor);
285 } else {
286 for (auto& keyword: keywords1) {
287 if (word == keyword) {
288 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, keyword1Color);
289 break;
290 }
291 }
292 for (auto& keyword: keywords2) {
293 if (word == keyword) {
294 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, keyword2Color);
295 break;
296 }
297 }
298 for (auto& keyword: preprocessorLineKeywords) {
299 if (word == keyword) {
300 if (c == '\n' || i == code.size() - 1) {
301 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, preprocessorColor);
302 } else {
303 preprocessorLine = true;
304 endIdx = startIdx - 1;
305 }
306 break;
307 }
308 }
309 }
310 startIdx = endIdx + 1;
311 endIdx = -1;
312 }
313 }
314 } else
315 if (lineComment == true) {
316 if (c == '\n' || i == code.size() - 1) {
317 lineComment = false;
318 endIdx = i;
319 multiLineTextNode->setTextStyle(startIdx, endIdx, commentLineColor);
320 startIdx = endIdx + 1;
321 endIdx = -1;
322 }
323 } else
324 if (inlineComment == true) {
325 if (language.commentInlineEnd.empty() == false && (language.commentInlineEnd.size() == 1 || lc == language.commentInlineEnd[0]) && c == language.commentInlineEnd[language.commentInlineEnd.size() - 1]) {
326 inlineComment = false;
327 endIdx = i;
328 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, commentInlineColor);
329 startIdx = endIdx + 1;
330 endIdx = -1;
331 }
332 } else
333 if (preprocessorLine == true) {
334 if (c == '\n' || i == code.size() - 1) {
335 preprocessorLine = false;
336 endIdx = i;
337 multiLineTextNode->setTextStyle(startIdx, endIdx, preprocessorColor);
338 startIdx = endIdx + 1;
339 endIdx = -1;
340 }
341 } else
342 if (quote != '\0') {
343 if (c == quote && (lc != '\\' || llc == '\\')) {
344 quote = '\0';endIdx = -1;
345 endIdx = i + 1;
346 multiLineTextNode->setTextStyle(startIdx, endIdx - 1, literalColor);
347 startIdx = endIdx + 1;
348 endIdx = -1;
349 }
350 }
351 llc = lc;
352 lc = c;
353 }
354
355 // done
356 break;
357 }
358 }
359 }
360}
361
364 delete engine;
365}
366
368{
370}
371
373{
374 engine->display();
375 engine->getGUI()->render();
376}
377
379{
380 try {
383 } catch (Exception& exception) {
384 Console::print(string("TextEditorTabView::initialize(): An error occurred: "));
385 Console::println(string(exception.what()));
386 }
387 // TODO: load settings
388}
389
391{
392 engine->dispose();
393}
394
396}
397
399 return engine;
400}
401
407}
408
411}
412
416}
417
Engine main class.
Definition: Engine.h:122
void display()
Renders the scene.
Definition: Engine.cpp:1269
void dispose()
Shutdown the engine.
Definition: Engine.cpp:1907
void setSceneColor(const Color4 &sceneColor)
Set scene color.
Definition: Engine.h:965
Color 4 definition.
Definition: Color4.h:20
GUI module class.
Definition: GUI.h:66
void handleEvents()
Handle screen events.
Definition: GUI.cpp:588
void render()
Render GUIs.
Definition: GUI.cpp:414
void addRenderScreen(const string &screenId)
Add render screen.
Definition: GUI.cpp:254
void addScreen(const string &id, GUIScreenNode *screen)
Add screen.
Definition: GUI.cpp:186
const string & getId()
Definition: GUINode.h:329
GUI screen node that represents a screen that can be rendered via GUI system.
Definition: GUIScreenNode.h:57
GUINode * getNodeById(const string &nodeId)
Get GUI node by id.
void restoreOutlinerState(const TabView::OutlinerState &outlinerState)
Restore outliner state.
void storeOutlinerState(TabView::OutlinerState &outlinerState)
Store outliner state.
void setDetailsContent(const string &xml)
Set details content.
void setOutlinerAddDropDownContent()
Set outliner add drop down content.
void initialize(GUIScreenNode *screenNode) override
Init.
struct tdme::tools::editor::tabviews::TextEditorTabView::XMLLanguage xmlLanguage
void initialize() override
Initiates the view.
struct tdme::tools::editor::tabviews::TextEditorTabView::PropertiesLanguage propertiesLanguage
void handleInputEvents() override
Handle input events that have not yet been processed.
void updateRendering() override
Update rendering.
EditorScreenController * getScreenController()
Definition: EditorView.h:57
Console class.
Definition: Console.h:26
Float class.
Definition: Float.h:23
Integer class.
Definition: Integer.h:26
String tools class.
Definition: StringTools.h:20
std::exception Exception
Exception base class.
Definition: Exception.h:19