TDME2 1.9.121
ModelEditorTabView.cpp
Go to the documentation of this file.
2
3#include <string>
4
5#include <tdme/tdme.h>
6#include <tdme/audio/Audio.h>
7#include <tdme/audio/Sound.h>
20#include <tdme/engine/Engine.h>
24#include <tdme/engine/Timing.h>
26#include <tdme/gui/GUI.h>
27#include <tdme/math/Vector3.h>
52
53using std::string;
54
74using tdme::gui::GUI;
100
101ModelEditorTabView::ModelEditorTabView(EditorView* editorView, const string& tabId, Prototype* prototype)
102{
103 this->editorView = editorView;
104 this->tabId = tabId;
105 this->popUps = editorView->getPopUps();
106 engine = Engine::createOffScreenInstance(512, 512, true, true, true);
109 engine->setSceneColor(Color4(125.0f / 255.0f, 125.0f / 255.0f, 125.0f / 255.0f, 1.0f));
110 audio = Audio::getInstance();
111 modelEditorTabController = nullptr;
112 prototypeDisplayView = nullptr;
113 prototypePhysicsView = nullptr;
114 prototypeSoundsView = nullptr;
116 lodLevel = 1;
117 audioStarted = -1LL;
118 audioOffset = -1LL;
122}
123
125 delete prototype;
128 delete engine;
129}
130
132 return editorView;
133}
134
136 return dynamic_cast<TabController*>(modelEditorTabController);
137}
138
140{
141 return popUps;
142}
143
145{
146 return prototype;
147}
148
150{
151 engine->reset();
152 if (this->prototype != nullptr) delete this->prototype;
153 this->prototype = prototype;
154 lodLevel = 1;
155 //
156 class SetPrototypeAction: public Action {
157 private:
158 ModelEditorTabView* modelEditorTabView;
159 public:
160 SetPrototypeAction(ModelEditorTabView* modelEditorTabView): modelEditorTabView(modelEditorTabView) {}
161 virtual void performAction() {
162 modelEditorTabView->initModel(false);
163 modelEditorTabView->modelEditorTabController->setOutlinerContent();
164 }
165 };
166 engine->enqueueAction(new SetPrototypeAction(this));
167}
168
170{
171 engine->reset();
172 //
173 class ResetPrototypeAction: public Action {
174 private:
175 ModelEditorTabView* modelEditorTabView;
176 public:
177 ResetPrototypeAction(ModelEditorTabView* modelEditorTabView): modelEditorTabView(modelEditorTabView) {}
178 virtual void performAction() {
179 modelEditorTabView->initModel(true);
180 modelEditorTabView->modelEditorTabController->setOutlinerContent();
181 }
182 };
183 engine->enqueueAction(new ResetPrototypeAction(this));
184}
185
187{
188 engine->reset();
189 //
190 class ReloadPrototypeAction: public Action {
191 private:
192 ModelEditorTabView* modelEditorTabView;
193 public:
194 ReloadPrototypeAction(ModelEditorTabView* modelEditorTabView): modelEditorTabView(modelEditorTabView) {}
195 virtual void performAction() {
196 modelEditorTabView->initModel(true);
197 modelEditorTabView->modelEditorTabController->setOutlinerContent();
198 }
199 };
200 engine->enqueueAction(new ReloadPrototypeAction(this));
201}
202
204{
205 engine->reset();
206 //
207 class ReimportPrototypeAction: public Action {
208 private:
209 ModelEditorTabView* modelEditorTabView;
210 public:
211 ReimportPrototypeAction(ModelEditorTabView* modelEditorTabView): modelEditorTabView(modelEditorTabView) {}
212 virtual void performAction() {
213 modelEditorTabView->initModel(false);
214 modelEditorTabView->modelEditorTabController->setOutlinerContent();
215 }
216 };
217 engine->enqueueAction(new ReimportPrototypeAction(this));
218}
219
221{
222 if (prototype == nullptr) return;
223 engine->removeEntity("model");
224 engine->removeEntity("attachment1");
225 if (attachment1Model != nullptr) {
226 delete attachment1Model;
227 attachment1Model = nullptr;
228 }
231 auto currentModelObject = dynamic_cast<Object3D*>(engine->getEntity("model"));
232 if (currentModelObject != nullptr) {
233 ModelStatistics modelStatistics;
234 ModelUtilities::computeModelStatistics(currentModelObject->getModel(), &modelStatistics);
235 modelEditorTabController->setStatistics(modelStatistics.opaqueFaceCount, modelStatistics.transparentFaceCount, modelStatistics.materialCount);
236 } else
237 {
239 }
240}
241
243{
244 return prototypeFileName;
245}
246
248 return lodLevel;
249}
250
252 if (this->lodLevel != lodLevel) {
253 this->lodLevel = lodLevel;
254 initModel(false);
255 }
256}
257
259 engine->reset();
261}
262
263void ModelEditorTabView::loadModel(const string& pathName, const string& fileName)
264{
265 // new model file
266 prototypeFileName = FileSystem::getInstance()->getFileName(pathName, fileName);
267 try {
268 // set model in entity
270 ModelReader::read(
271 pathName,
272 fileName
273 )
274 );
275 } catch (Exception& exception) {
276 modelEditorTabController->showErrorPopUp("Warning", (string(exception.what())));
277 }
279}
280
281void ModelEditorTabView::reimportModel(const string& pathName, const string& fileName)
282{
283 if (prototype == nullptr) return;
284 engine->removeEntity("model");
285 engine->removeEntity("attachment1");
286 if (attachment1Model != nullptr) {
287 delete attachment1Model;
288 attachment1Model = nullptr;
289 }
290 struct AnimationSetupStruct {
291 bool loop;
292 string overlayFromNodeId;
293 float speed;
294 };
295 // store old animation setups
296 map<string, AnimationSetupStruct> originalAnimationSetups;
297 for (auto animationSetupIt: prototype->getModel()->getAnimationSetups()) {
298 auto animationSetup = animationSetupIt.second;
299 originalAnimationSetups[animationSetup->getId()] = {
300 .loop = animationSetup->isLoop(),
301 .overlayFromNodeId = animationSetup->getOverlayFromNodeId(),
302 .speed = animationSetup->getSpeed()
303 };
304 }
305 // new model file
306 prototypeFileName = FileSystem::getInstance()->getFileName(pathName, fileName);
307 string log;
308 try {
309 // load model
310 auto model = ModelReader::read(
311 pathName,
312 fileName
313 );
314 // restore animation setup properties
315 for (auto originalAnimationSetupIt: originalAnimationSetups) {
316 auto originalAnimationSetupId = originalAnimationSetupIt.first;
317 auto originalAnimationSetup = originalAnimationSetupIt.second;
318 auto animationSetup = model->getAnimationSetup(originalAnimationSetupId);
319 if (animationSetup == nullptr) {
320 Console::println("ModelEditorTabView::reimportModel(): missing animation setup: " + originalAnimationSetupId);
321 log+= "Missing animation setup: " + originalAnimationSetupId + ", skipping.\n";
322 continue;
323 }
324 Console::println("ModelEditorTabView::reimportModel(): reimport animation setup: " + originalAnimationSetupId);
325 animationSetup->setLoop(originalAnimationSetup.loop);
326 animationSetup->setOverlayFromNodeId(originalAnimationSetup.overlayFromNodeId);
327 animationSetup->setSpeed(originalAnimationSetup.speed);
328 }
329 // set model in entity
330 prototype->setModel(model);
331 } catch (Exception& exception) {
332 modelEditorTabController->showErrorPopUp("Warning", (string(exception.what())));
333 }
335 if (log.size() > 0) {
337 }
338}
339
340void ModelEditorTabView::saveFile(const string& pathName, const string& fileName)
341{
342 PrototypeWriter::write(pathName, fileName, prototype);
343}
344
346{
347 engine->reset();
348 //
349 class ReloadFileAction: public Action {
350 private:
351 ModelEditorTabView* modelEditorTabView;
352 public:
353 ReloadFileAction(ModelEditorTabView* modelEditorTabView): modelEditorTabView(modelEditorTabView) {}
354 virtual void performAction() {
355 modelEditorTabView->loadModel();
356 modelEditorTabView->initModel(true);
357 modelEditorTabView->modelEditorTabController->setOutlinerContent();
358 }
359 };
360 engine->enqueueAction(new ReloadFileAction(this));
361
362}
363
364void ModelEditorTabView::pivotApply(float x, float y, float z)
365{
366 if (prototype == nullptr) return;
367 prototype->getPivot().set(x, y, z);
368}
369
371 if (prototype == nullptr || prototype->getModel() == nullptr) return;
372 engine->removeEntity("model");
373 class ComputeNormalsProgressCallback: public ProgressCallback {
374 private:
375 ProgressBarScreenController* progressBarScreenController;
376 public:
377 ComputeNormalsProgressCallback(ProgressBarScreenController* progressBarScreenController): progressBarScreenController(progressBarScreenController) {
378 }
379 virtual void progress(float value) {
380 progressBarScreenController->progress(value);
381 }
382 };
383 popUps->getProgressBarScreenController()->show("Computing smooth normals ...");
384 ModelTools::computeNormals(prototype->getModel(), new ComputeNormalsProgressCallback(popUps->getProgressBarScreenController()));
387}
388
390 if (prototype == nullptr || prototype->getModel() == nullptr) return;
391 engine->removeEntity("model");
392 prototype->setModel(ModelTools::optimizeModel(prototype->unsetModel()));
394}
395
397{
400}
401
403{
404 // audio
405 if (audioOffset > 0 && Time::getCurrentMillis() - audioStarted >= audioOffset) {
406 auto sound = audio->getEntity("sound");
407 if (sound != nullptr) sound->play();
408 audioOffset = -1LL;
409 }
410
411 //
412 auto model = static_cast<Object3D*>(engine->getEntity("model"));
413
414 // attachment1
415 auto attachment1 = static_cast<Object3D*>(engine->getEntity("attachment1"));
416 if (model != nullptr && attachment1 != nullptr) {
417 // model attachment bone matrix
418 auto transformationsMatrix = model->getNodeTransformationsMatrix(attachment1Bone);
419 transformationsMatrix*= model->getTransformations().getTransformationsMatrix();
420 attachment1->setTranslation(transformationsMatrix * Vector3(0.0f, 0.0f, 0.0f));
421 // euler angles
422 auto euler = transformationsMatrix.computeEulerAngles();
423 // rotations
424 attachment1->setRotationAngle(0, euler.getZ());
425 attachment1->setRotationAngle(1, euler.getY());
426 attachment1->setRotationAngle(2, euler.getX());
427 // scale
428 Vector3 scale;
429 transformationsMatrix.getScale(scale);
430 attachment1->setScale(scale);
431 // finally update
432 attachment1->update();
433 }
434
435 //
437
438 // rendering
441 engine->display();
442}
443
445{
446 // TODO: a.drewke
447 try {
448 Properties settings;
449 settings.load("settings", "modeleditor.properties");
450 prototypePhysicsView->setDisplayBoundingVolume(settings.get("display.boundingvolumes", "false") == "true");
451 prototypeDisplayView->setDisplayGroundPlate(settings.get("display.groundplate", "true") == "true");
452 prototypeDisplayView->setDisplayShadowing(settings.get("display.shadowing", "true") == "true");
453 } catch (Exception& exception) {
454 Console::print(string("ModelEditorTabView::loadSettings(): An error occurred: "));
455 Console::println(string(exception.what()));
456 }
457}
458
460{
461 try {
467 } catch (Exception& exception) {
468 Console::print(string("ModelEditorTabView::initialize(): An error occurred: "));
469 Console::println(string(exception.what()));
470 }
471 loadSettings();
472}
473
475{
476 // TODO: a.drewke
477 try {
478 Properties settings;
479 settings.put("display.boundingvolumes", prototypePhysicsView->isDisplayBoundingVolume() == true ? "true" : "false");
480 settings.put("display.groundplate", prototypeDisplayView->isDisplayGroundPlate() == true ? "true" : "false");
481 settings.put("display.shadowing", prototypeDisplayView->isDisplayShadowing() == true ? "true" : "false");
482 settings.store("settings", "modeleditor.properties");
483 } catch (Exception& exception) {
484 Console::print(string("ModelEditorTabView::storeSettings(): An error occurred: "));
485 Console::println(string(exception.what()));
486 }
487}
488
490
491{
493 engine->dispose();
494 audio->removeEntity("sound");
495}
496
498{
499 Console::println(string("Model file: " + prototypeFileName));
500 try {
503 FileSystem::getInstance()->getFileName(prototypeFileName),
504 "",
505 FileSystem::getInstance()->getPathName(prototypeFileName),
506 FileSystem::getInstance()->getFileName(prototypeFileName),
507 Vector3()
508 )
509 );
510 } catch (Exception& exception) {
511 popUps->getInfoDialogScreenController()->show("Warning", (exception.what()));
512 }
513}
514
515Prototype* ModelEditorTabView::loadModelPrototype(const string& name, const string& description, const string& pathName, const string& fileName, const Vector3& pivot)
516{
517 if (StringTools::endsWith(StringTools::toLowerCase(fileName), ".tmodel") == true) {
518 auto prototype = PrototypeReader::read(
519 pathName,
520 fileName
521 );
522 return prototype;
523 } else {
524 auto model = ModelReader::read(
525 pathName,
526 fileName
527 );
528 auto prototype = new Prototype(
529 Prototype::ID_NONE,
530 Prototype_Type::MODEL,
531 name,
532 description,
533 "",
534 pathName + "/" + fileName,
535 StringTools::replace(StringTools::replace(StringTools::replace(model->getId(), "\\", "_"), "/", "_"), ":", "_") + ".png",
536 model,
537 pivot
538 );
539 return prototype;
540
541 }
542 return nullptr;
543}
544
545void ModelEditorTabView::playAnimation(const string& baseAnimationId, const string& overlay1AnimationId, const string& overlay2AnimationId, const string& overlay3AnimationId) {
546 auto object = dynamic_cast<Object3D*>(engine->getEntity("model"));
547 if (object != nullptr) {
548 audio->removeEntity("sound");
549 object->removeOverlayAnimations();
550 object->setAnimation(baseAnimationId);
551 if (overlay1AnimationId.empty() == false) object->addOverlayAnimation(overlay1AnimationId);
552 if (overlay2AnimationId.empty() == false) object->addOverlayAnimation(overlay2AnimationId);
553 if (overlay3AnimationId.empty() == false) object->addOverlayAnimation(overlay3AnimationId);
554 }
555}
556
557void ModelEditorTabView::addAttachment1(const string& nodeId, const string& attachmentModelFile) {
558 engine->removeEntity("attachment1");
559 try {
560 if (attachment1Model != nullptr) delete attachment1Model;
561 attachment1Model = attachmentModelFile.empty() == true?nullptr:ModelReader::read(Tools::getPathName(attachmentModelFile), Tools::getFileName(attachmentModelFile));
562 } catch (Exception& exception) {
563 Console::print(string("ModelEditorTabView::addAttachment1(): An error occurred: "));
564 Console::println(string(exception.what()));
565 popUps->getInfoDialogScreenController()->show("Warning", (exception.what()));
566 }
567 if (attachment1Model != nullptr) {
568 Entity* attachment = nullptr;
569 engine->addEntity(attachment = new Object3D("attachment1", attachment1Model));
570 attachment->addRotation(Vector3(0.0f, 0.0f, 1.0f), 0.0f);
571 attachment->addRotation(Vector3(0.0f, 1.0f, 0.0f), 0.0f);
572 attachment->addRotation(Vector3(1.0f, 0.0f, 0.0f), 0.0f);
573 }
574 attachment1Bone = nodeId;
575}
576
577void ModelEditorTabView::playSound(const string& soundId) {
578 audio->removeEntity("sound");
579 auto object = dynamic_cast<Object3D*>(engine->getEntity("model"));
580 auto soundDefinition = prototype->getSound(soundId);
581 if (soundDefinition != nullptr && soundDefinition->getFileName().length() > 0) {
582 if (object != nullptr && soundDefinition->getAnimation().size() > 0) object->setAnimation(soundDefinition->getAnimation());
583 auto pathName = PrototypeReader::getResourcePathName(
584 Tools::getPathName(prototype->getFileName()),
585 soundDefinition->getFileName()
586 );
587 auto fileName = Tools::getFileName(soundDefinition->getFileName());
588 auto sound = new Sound(
589 "sound",
590 pathName,
591 fileName
592 );
593 sound->setGain(soundDefinition->getGain());
594 sound->setPitch(soundDefinition->getPitch());
595 sound->setLooping(soundDefinition->isLooping());
596 sound->setFixed(true);
597 audio->addEntity(sound);
598 audioStarted = Time::getCurrentMillis();
599 audioOffset = -1LL;
600 if (soundDefinition->getOffset() <= 0) {
601 sound->play();
602 } else {
603 audioOffset = soundDefinition->getOffset();
604 }
605 }
606}
607
609 audio->removeEntity("sound");
610}
611
613 auto object = dynamic_cast<Object3D*>(engine->getEntity("model"));
614 if (object == nullptr || prototype == nullptr) return;
615 engine->removeEntity("model");
616 ModelTools::prepareForShader(prototype->getModel(), prototype->getShader());
618}
619
621 auto object = dynamic_cast<Object3D*>(engine->getEntity("model"));
622 if (object == nullptr || prototype == nullptr) return;
623 auto shaderParametersDefault = Engine::getShaderParameterDefaults(prototype->getShader());
624 auto distanceShaderParametersDefault = Engine::getShaderParameterDefaults(prototype->getDistanceShader());
625 for (auto& parameterIt: shaderParametersDefault) {
626 auto& parameterName = parameterIt.first;
627 auto parameterValue = prototype->getShaderParameters().getShaderParameter(parameterName);
628 object->setShaderParameter(parameterName, parameterValue);
629 }
630 for (auto& parameterIt: distanceShaderParametersDefault) {
631 auto& parameterName = parameterIt.first;
632 auto parameterValue = prototype->getDistanceShaderParameters().getShaderParameter(parameterName);
633 object->setDistanceShaderParameter(parameterName, parameterValue);
634 }
635}
636
639}
640
643}
644
646 return engine;
647}
648
654}
655
658}
659
663}
virtual void play()=0
Plays this audio entity.
Interface to audio module.
Definition: Audio.h:30
void addEntity(AudioEntity *entity)
Adds a audio entity.
Definition: Audio.cpp:57
void removeEntity(const string &id)
Removes an audio entity.
Definition: Audio.cpp:76
AudioEntity * getEntity(const string &id)
Returns an audio entity identified by given id.
Definition: Audio.cpp:50
Sound audio entity implementation.
Definition: Sound.h:20
Engine main class.
Definition: Engine.h:122
bool removeEntity(const string &id)
Removes an entity.
Definition: Engine.cpp:451
Timing * getTiming()
Definition: Engine.h:900
void display()
Renders the scene.
Definition: Engine.cpp:1269
void enqueueAction(Action *action)
Add action to action queue to be executed before next engine update.
Definition: Engine.h:1187
void setPartition(Partition *partition)
Set partition.
Definition: Engine.cpp:358
void setShadowMapLightEyeDistanceScale(float shadowMapLightEyeDistanceScale)
Set shadow map light eye distance scale.
Definition: Engine.h:632
void addEntity(Entity *entity)
Adds an entity by id.
Definition: Engine.cpp:364
void dispose()
Shutdown the engine.
Definition: Engine.cpp:1907
Entity * getEntity(const string &id)
Returns a entity by given id.
Definition: Engine.h:981
void reset()
Removes all entities and caches.
Definition: Engine.cpp:652
void setSceneColor(const Color4 &sceneColor)
Set scene color.
Definition: Engine.h:965
const ShaderParameter getShaderParameter(const string &parameterName) const
Returns shader parameter for given parameter name, if the value does not exist, the default will be r...
TDME engine entity.
Definition: Entity.h:31
virtual void addRotation(const Vector3 &axis, const float angle)=0
Add rotation.
Object 3D to be used with engine class.
Definition: Object3D.h:60
Bogus/Simple partition implementation.
Timing class.
Definition: Timing.h:17
float getAvarageFPS()
Definition: Timing.h:100
Color 4 definition.
Definition: Color4.h:20
Representation of a 3d model.
Definition: Model.h:32
map< string, AnimationSetup * > & getAnimationSetups()
TODO: return const map.
Definition: Model.h:274
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Definition: BoundingBox.h:25
Base property model class.
Definition: BaseProperty.h:16
Prototype audio definition.
void setAnimation(const string &animation)
Set animation.
Prototype definition.
Definition: Prototype.h:49
PrototypeAudio * getSound(const string &id)
Returns sound of given sound id.
Definition: Prototype.h:488
const string & getShader()
Get shader.
Definition: Prototype.h:398
Model * unsetModel()
Unset model without deleting current one.
Definition: Prototype.h:198
const string & getDistanceShader()
Get distance shader.
Definition: Prototype.h:415
void setModel(Model *model)
Set model.
Definition: Prototype.cpp:65
const EntityShaderParameters & getDistanceShaderParameters()
Get distance shader parameters.
Definition: Prototype.h:464
const EntityShaderParameters & getShaderParameters()
Get shader parameters.
Definition: Prototype.h:448
const string & getModelFileName()
Definition: Prototype.h:162
const Matrix4x4 getNodeTransformationsMatrix(const string &id)
Returns transformation matrix for given node.
Definition: Object3DBase.h:255
GUI module class.
Definition: GUI.h:66
GUI screen node that represents a screen that can be rendered via GUI system.
Definition: GUIScreenNode.h:57
Vector3 computeEulerAngles() const
Compute Euler angles (rotation around x, y, z axes)
Definition: Matrix4x4.h:661
Matrix4x4 & setTranslation(const Vector3 &translation)
Set translation.
Definition: Matrix4x4.h:270
3D vector 3 class
Definition: Vector3.h:22
Vector3 & set(float x, float y, float z)
Set up vector.
Definition: Vector3.h:73
File system singleton class.
Definition: FileSystem.h:14
void restoreOutlinerState(const TabView::OutlinerState &outlinerState)
Restore outliner state.
void storeOutlinerState(TabView::OutlinerState &outlinerState)
Store outliner state.
void show(const string &caption, const string &message)
Shows the pop up.
void show(const string &message, bool showProgressBar=true)
Shows the pop up.
void handleInputEvents() override
Handle input events that have not yet been processed.
Pop ups controller accessor class.
Definition: PopUps.h:19
ProgressBarScreenController * getProgressBarScreenController()
Definition: PopUps.h:56
InfoDialogScreenController * getInfoDialogScreenController()
Definition: PopUps.h:49
void updateInfoText(const MutableString &text)
Update info text line.
void setOutlinerAddDropDownContent()
Set outliner add drop down content.
void setStatistics(int statsOpaqueFaces, int statsTransparentFaces, int statsMaterialCount)
Set up model statistics.
void updateDetails(const string &outlinerNode)
Update details panel.
void showErrorPopUp(const string &caption, const string &message)
Shows the error pop up.
void saveFile(const string &pathName, const string &fileName)
Saving prototype as tmodel prototype.
void playSound(const string &soundId) override
Play sound.
void onCameraRotation() override
On rotation event to be overloaded.
void addAttachment1(const string &nodeId, const string &attachmentModelFile)
Add attachment 1.
void handleInputEvents() override
Handle input events that have not yet been processed.
void setLODLevel(int lodLevel)
Set LOD level to display.
virtual Prototype * loadModelPrototype(const string &name, const string &description, const string &pathName, const string &fileName, const Vector3 &pivot)
Load model prototype.
void pivotApply(float x, float y, float z)
Apply pivot.
CameraRotationInputHandler * cameraRotationInputHandler
void playAnimation(const string &baseAnimationId, const string &overlay1AnimationId=string(), const string &overlay2AnimationId=string(), const string &overlay3AnimationId=string())
Play animation.
void setPrototype(Prototype *prototype)
Set prototype.
void updateShaderParemeters()
Update shader parameters.
void reimportModel(const string &pathName, const string &fileName)
Issue reimport model file.
void onCameraScale() override
On scale event to be overloaded.
void setDisplayShadowing(bool shadowing)
Set up shadow rendering.
void setDisplayGroundPlate(bool groundPlate)
Set up ground plate visibility.
void setDisplayBoundingVolume(bool displayBoundingVolume)
Set up bounding volume visibility.
void handleInputEvents(Prototype *prototype, const Vector3 &objectScale)
Handle input events.
EditorScreenController * getScreenController()
Definition: EditorView.h:57
Console class.
Definition: Console.h:26
Model tools functions class.
Definition: ModelTools.h:38
Mutable string class.
Definition: MutableString.h:16
Properties class, which helps out with storeing or loading key value pairs from/to property files.
Definition: Properties.h:23
void put(const string &key, const string &value)
Add property.
Definition: Properties.h:58
const string & get(const string &key, const string &defaultValue) const
Get property value by key.
Definition: Properties.h:46
void load(const string &pathName, const string &fileName, FileSystemInterface *fileSystem=nullptr)
Load property file.
Definition: Properties.cpp:26
void store(const string &pathName, const string &fileName, FileSystemInterface *fileSystem=nullptr) const
Store property file.
Definition: Properties.cpp:43
String tools class.
Definition: StringTools.h:20
std::exception Exception
Exception base class.
Definition: Exception.h:19
Tab controller, which connects UI with logic.
Definition: TabController.h:23
Playable sound view interface, which represents a view that supports playing sounds additionally.
Action Interface.
Definition: Action.h:12