2020-01-18 17:17:52 +00:00
|
|
|
#pragma once
|
2020-08-21 09:55:59 +00:00
|
|
|
|
|
|
|
#include <cstdint>
|
2020-03-15 20:01:24 +00:00
|
|
|
#include "../TouchEvents.h"
|
2020-01-18 17:17:52 +00:00
|
|
|
|
|
|
|
namespace Pinetime {
|
|
|
|
namespace Applications {
|
2020-02-16 17:32:36 +00:00
|
|
|
class DisplayApp;
|
2020-01-18 17:17:52 +00:00
|
|
|
namespace Screens {
|
2021-02-24 19:40:24 +00:00
|
|
|
|
2021-04-18 17:28:14 +00:00
|
|
|
template <class T> class DirtyValue {
|
2021-04-24 09:00:45 +00:00
|
|
|
public:
|
2021-04-18 17:28:14 +00:00
|
|
|
DirtyValue() = default; // Use NSDMI
|
|
|
|
explicit DirtyValue(T const& v) : value {v} {
|
|
|
|
} // Use MIL and const-lvalue-ref
|
|
|
|
bool IsUpdated() const {
|
|
|
|
return isUpdated;
|
|
|
|
}
|
|
|
|
T const& Get() {
|
|
|
|
this->isUpdated = false;
|
|
|
|
return value;
|
|
|
|
} // never expose a non-const lvalue-ref
|
|
|
|
DirtyValue& operator=(const T& other) {
|
|
|
|
if (this->value != other) {
|
|
|
|
this->value = other;
|
|
|
|
this->isUpdated = true;
|
2021-02-24 19:40:24 +00:00
|
|
|
}
|
2021-04-18 17:28:14 +00:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2021-04-24 09:00:45 +00:00
|
|
|
private:
|
2021-04-18 17:28:14 +00:00
|
|
|
T value {}; // NSDMI - default initialise type
|
|
|
|
bool isUpdated {true}; // NSDMI - use brace initilisation
|
2021-02-24 19:40:24 +00:00
|
|
|
};
|
2020-02-23 12:44:39 +00:00
|
|
|
|
2021-04-18 17:28:14 +00:00
|
|
|
class Screen {
|
2021-04-24 09:00:45 +00:00
|
|
|
public:
|
2021-04-18 17:28:14 +00:00
|
|
|
explicit Screen(DisplayApp* app) : app {app} {
|
|
|
|
}
|
|
|
|
virtual ~Screen() = default;
|
2020-02-23 12:44:39 +00:00
|
|
|
|
2021-04-18 17:28:14 +00:00
|
|
|
/**
|
|
|
|
* Most of the time, apps only react to events (touch events, for example).
|
|
|
|
* In this case you don't need to do anything in this method.
|
|
|
|
*
|
|
|
|
* For example, InfiniPaint does nothing in Refresh().
|
|
|
|
* But, if you want to update your display periodically, draw an animation...
|
|
|
|
* you cannot do it in a touch event handler because these handlers are not
|
|
|
|
* called if the user does not touch the screen.
|
|
|
|
*
|
|
|
|
* That's why Refresh() is there: update the display periodically.
|
|
|
|
*
|
|
|
|
* @return false if the app can be closed, true if it must continue to run
|
|
|
|
**/
|
|
|
|
virtual bool Refresh() = 0;
|
2020-01-18 17:17:52 +00:00
|
|
|
|
2021-04-18 17:28:14 +00:00
|
|
|
/** @return false if the button hasn't been handled by the app, true if it has been handled */
|
|
|
|
virtual bool OnButtonPushed() {
|
|
|
|
return false;
|
|
|
|
}
|
2020-03-15 20:01:24 +00:00
|
|
|
|
2021-04-18 17:28:14 +00:00
|
|
|
/** @return false if the event hasn't been handled by the app, true if it has been handled */
|
|
|
|
virtual bool OnTouchEvent(TouchEvents event) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
virtual bool OnTouchEvent(uint16_t x, uint16_t y) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-04-04 02:08:51 +00:00
|
|
|
|
2021-04-24 09:00:45 +00:00
|
|
|
protected:
|
2021-04-18 17:28:14 +00:00
|
|
|
DisplayApp* app;
|
|
|
|
bool running = true;
|
2020-01-18 17:17:52 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|