mirror of
https://github.com/UpsilonNumworks/Upsilon.git
synced 2026-01-18 16:27:34 +01:00
109 lines
4.7 KiB
C++
109 lines
4.7 KiB
C++
#include "battery_view.h"
|
|
|
|
const uint8_t flashMask[BatteryView::k_flashHeight][BatteryView::k_flashWidth] = {
|
|
{0xDB, 0x00, 0x00, 0xFF},
|
|
{0xB7, 0x00, 0x6D, 0xFF},
|
|
{0x6D, 0x00, 0xDB, 0xFF},
|
|
{0x24, 0x00, 0x00, 0x00},
|
|
{0x00, 0x00, 0x00, 0x24},
|
|
{0xFF, 0xDB, 0x00, 0x6D},
|
|
{0xFF, 0x6D, 0x00, 0xB7},
|
|
{0xFF, 0x00, 0x00, 0xDB},
|
|
};
|
|
|
|
const uint8_t tickMask[BatteryView::k_tickHeight][BatteryView::k_tickWidth] = {
|
|
{0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xDB, 0x00, 0x24},
|
|
{0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x6D, 0x00, 0xDB},
|
|
{0x6D, 0x00, 0xB7, 0xFF, 0xB7, 0x00, 0x24, 0xFF},
|
|
{0xDB, 0x00, 0x00, 0xFF, 0x00, 0x00, 0xFF, 0xFF},
|
|
{0xFF, 0xB7, 0x00, 0x24, 0x00, 0xB7, 0xFF, 0xFF},
|
|
{0xFF, 0xFF, 0x24, 0x00, 0x24, 0xFF, 0xFF, 0xFF},
|
|
|
|
};
|
|
|
|
bool BatteryView::setChargeState(Ion::Battery::Charge chargeState) {
|
|
/* There is no specific battery picto for 'empty' battery as the whole device
|
|
* shut down. Still, there might be a redrawing of the window before shutting
|
|
* down so we handle this case as the 'low' battery one. Plus, we avoid
|
|
* trigerring a redrawing by not marking anything as dirty when switching
|
|
* from 'low' to 'empty' battery. */
|
|
chargeState = chargeState == Ion::Battery::Charge::EMPTY ? Ion::Battery::Charge::LOW : chargeState;
|
|
if (chargeState != m_chargeState) {
|
|
m_chargeState = chargeState;
|
|
markRectAsDirty(bounds());
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool BatteryView::setIsCharging(bool isCharging) {
|
|
if (m_isCharging != isCharging) {
|
|
m_isCharging = isCharging;
|
|
markRectAsDirty(bounds());
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool BatteryView::setIsPlugged(bool isPlugged) {
|
|
if (m_isPlugged != isPlugged) {
|
|
m_isPlugged = isPlugged;
|
|
markRectAsDirty(bounds());
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void BatteryView::drawRect(KDContext * ctx, KDRect rect) const {
|
|
assert(m_chargeState != Ion::Battery::Charge::EMPTY);
|
|
/* We draw from left to right. The middle part representing the battery
|
|
*'content' depends on the charge */
|
|
|
|
// Draw the left part
|
|
ctx->fillRect(KDRect(0, 1, k_elementWidth, k_batteryHeight - 2), Palette::Battery);
|
|
|
|
// Draw top and bottom part
|
|
ctx->fillRect(KDRect(1, 0, k_batteryWidth-3, 1), Palette::Battery);
|
|
ctx->fillRect(KDRect(1, k_batteryHeight-1, k_batteryWidth-3, 1), Palette::Battery);
|
|
|
|
// Draw the middle part
|
|
constexpr KDCoordinate batteryInsideX = k_elementWidth+k_separatorThickness;
|
|
constexpr KDCoordinate batteryInsideWidth = k_batteryWidth-3*k_elementWidth-2*k_separatorThickness;
|
|
if (m_isCharging) {
|
|
// Charging: Yellow background with flash
|
|
ctx->fillRect(KDRect(batteryInsideX, 2, batteryInsideWidth, k_batteryHeight-4), Palette::BatteryInCharge);
|
|
KDRect frame((k_batteryWidth-k_flashWidth)/2, 0, k_flashWidth, k_flashHeight);
|
|
KDColor flashWorkingBuffer[BatteryView::k_flashHeight*BatteryView::k_flashWidth];
|
|
ctx->blendRectWithMask(frame, Palette::Battery, (const uint8_t *)flashMask, flashWorkingBuffer);
|
|
} else if (m_chargeState == Ion::Battery::Charge::LOW) {
|
|
assert(!m_isPlugged);
|
|
// Low: Quite empty battery
|
|
ctx->fillRect(KDRect(batteryInsideX, 2, 2*k_elementWidth, k_batteryHeight-4), Palette::BatteryLow);
|
|
ctx->fillRect(KDRect(3*k_elementWidth+k_separatorThickness, 2, k_batteryWidth-5*k_elementWidth-2*k_separatorThickness, k_batteryHeight-4), KDColor::blend(Palette::Toolbar, Palette::Battery, 128));
|
|
} else if (m_chargeState == Ion::Battery::Charge::SOMEWHERE_INBETWEEN) {
|
|
assert(!m_isPlugged);
|
|
// Middle: Half full battery
|
|
constexpr KDCoordinate middleChargeWidth = batteryInsideWidth/2;
|
|
ctx->fillRect(KDRect(batteryInsideX, 2, middleChargeWidth, k_batteryHeight-4), Palette::Battery);
|
|
ctx->fillRect(KDRect(batteryInsideX+middleChargeWidth, 2, middleChargeWidth, k_batteryHeight-4), KDColor::blend(Palette::Toolbar, Palette::Battery, 128));
|
|
} else {
|
|
assert(m_chargeState == Ion::Battery::Charge::FULL);
|
|
// Full but not plugged: Full battery
|
|
ctx->fillRect(KDRect(batteryInsideX, 2, batteryInsideWidth, k_batteryHeight-4), Palette::Battery);
|
|
if (m_isPlugged) {
|
|
// Plugged and full: Full battery with tick
|
|
KDRect frame((k_batteryWidth-k_tickWidth)/2, (k_batteryHeight-k_tickHeight)/2, k_tickWidth, k_tickHeight);
|
|
KDColor tickWorkingBuffer[BatteryView::k_tickHeight*BatteryView::k_tickWidth];
|
|
ctx->blendRectWithMask(frame, Palette::Toolbar, (const uint8_t *)tickMask, tickWorkingBuffer);
|
|
}
|
|
}
|
|
|
|
// Draw the right part
|
|
ctx->fillRect(KDRect(k_batteryWidth-2*k_elementWidth, 1, k_elementWidth, k_batteryHeight-2), Palette::Battery);
|
|
ctx->fillRect(KDRect(k_batteryWidth-k_elementWidth, (k_batteryHeight-k_capHeight)/2, k_elementWidth, k_capHeight), Palette::Battery);
|
|
}
|
|
|
|
KDSize BatteryView::minimalSizeForOptimalDisplay() const {
|
|
return KDSize(k_batteryWidth, k_batteryHeight);
|
|
}
|