2020-01-18 12:17:52 -05:00
|
|
|
#pragma once
|
2020-08-21 05:55:59 -04:00
|
|
|
|
|
|
|
#include <cstdint>
|
2020-03-15 16:01:24 -04:00
|
|
|
#include "../TouchEvents.h"
|
2020-01-18 12:17:52 -05:00
|
|
|
|
|
|
|
namespace Pinetime {
|
|
|
|
namespace Applications {
|
2020-02-16 12:32:36 -05:00
|
|
|
class DisplayApp;
|
2020-01-18 12:17:52 -05:00
|
|
|
namespace Screens {
|
2021-02-24 14:40:24 -05:00
|
|
|
|
2021-04-18 13:28:14 -04:00
|
|
|
template <class T> class DirtyValue {
|
2021-04-24 05:00:45 -04:00
|
|
|
public:
|
2021-04-18 13:28:14 -04: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 14:40:24 -05:00
|
|
|
}
|
2021-04-18 13:28:14 -04:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2021-04-24 05:00:45 -04:00
|
|
|
private:
|
2021-04-18 13:28:14 -04:00
|
|
|
T value {}; // NSDMI - default initialise type
|
|
|
|
bool isUpdated {true}; // NSDMI - use brace initilisation
|
2021-02-24 14:40:24 -05:00
|
|
|
};
|
2020-02-23 07:44:39 -05:00
|
|
|
|
2021-04-18 13:28:14 -04:00
|
|
|
class Screen {
|
2021-04-24 05:00:45 -04:00
|
|
|
public:
|
2021-04-18 13:28:14 -04:00
|
|
|
explicit Screen(DisplayApp* app) : app {app} {
|
|
|
|
}
|
|
|
|
virtual ~Screen() = default;
|
2020-02-23 07:44:39 -05:00
|
|
|
|
2021-04-18 13:28:14 -04: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 12:17:52 -05:00
|
|
|
|
2021-04-18 13:28:14 -04: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 16:01:24 -04:00
|
|
|
|
2021-04-18 13:28:14 -04: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-03 22:08:51 -04:00
|
|
|
|
2021-04-24 05:00:45 -04:00
|
|
|
protected:
|
2021-04-18 13:28:14 -04:00
|
|
|
DisplayApp* app;
|
|
|
|
bool running = true;
|
2020-01-18 12:17:52 -05:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|