Files
MuditaOS/module-apps/application-desktop/windows/DesktopMainWindow.cpp
2020-10-28 13:28:51 +01:00

348 lines
13 KiB
C++

// Copyright (c) 2017-2020, Mudita Sp. z.o.o. All rights reserved.
// For licensing, see https://github.com/mudita/MuditaOS/LICENSE.md
#include <memory>
#include "Alignment.hpp"
#include "BottomBar.hpp"
#include "Common.hpp"
#include "DesktopMainWindow.hpp"
#include "GuiTimer.hpp"
#include "application-desktop/ApplicationDesktop.hpp"
#include "application-desktop/data/LockPhoneData.hpp"
#include "application-desktop/data/Style.hpp"
#include "application-desktop/widgets/NotificationsBox.hpp"
#include "application-messages/ApplicationMessages.hpp"
#include "gui/widgets/Image.hpp"
#include "service-appmgr/Controller.hpp"
#include "service-time/ServiceTime.hpp"
#include "service-time/messages/TimeMessage.hpp"
#include <UiCommonActions.hpp>
#include "i18/i18.hpp"
#include "log/log.hpp"
#include <application-settings-new/ApplicationSettings.hpp>
#include <application-settings/ApplicationSettings.hpp>
#include <cassert>
#include <time/time_conversion.hpp>
namespace gui
{
void DesktopMainWindow::buildInterface()
{
auto ttime = utils::time::Time();
AppWindow::buildInterface();
bottomBar->setActive(BottomBar::Side::CENTER, true);
topBar->setActive(
{{TopBar::Elements::SIGNAL, true}, {TopBar::Elements::LOCK, true}, {TopBar::Elements::BATTERY, true}});
using namespace style::desktop;
time = new gui::Label(this, timeLabel::X, timeLabel::Y, timeLabel::Width, timeLabel::Hight);
time->setFilled(false);
time->setBorderColor(gui::ColorNoColor);
time->setFont(style::window::font::supersizemelight);
time->setText(ttime);
time->setAlignment(Alignment(gui::Alignment::Horizontal::Center, gui::Alignment::Vertical::Top));
dayText = new gui::Label(this, dayLabel::X, dayLabel::Y, dayLabel::Width, dayLabel::Hight);
dayText->setFilled(false);
dayText->setBorderColor(gui::ColorNoColor);
dayText->setFont(style::window::font::biglight);
dayText->setText(ttime.day() + ", " + ttime.str("%d %b"));
dayText->setAlignment(Alignment(gui::Alignment::Horizontal::Center, gui::Alignment::Vertical::Top));
activatedCallback = [this](Item &) {
application->switchWindow(app::window::name::desktop_menu);
return true;
};
setVisibleState();
}
void DesktopMainWindow::destroyInterface()
{
erase();
invalidate();
}
void DesktopMainWindow::invalidate() noexcept
{
time = nullptr;
dayText = nullptr;
notifications = nullptr;
}
DesktopMainWindow::DesktopMainWindow(app::Application *app) : AppWindow(app, app::window::name::desktop_main_window)
{
buildInterface();
}
void DesktopMainWindow::setVisibleState()
{
auto app = getAppDesktop();
if (app->lockHandler.lock.isLocked() && app->lockHandler.lock.getLockType() == PinLock::LockType::Screen) {
bottomBar->setText(BottomBar::Side::CENTER, utils::localize.get("app_desktop_unlock"));
topBar->setActive(TopBar::Elements::LOCK, true);
inputCallback = nullptr;
setFocusItem(nullptr);
buildNotifications(app);
sys::Bus::SendUnicast(
std::make_shared<TimersProcessingStopMessage>(), service::name::service_time, application);
}
else if (app->lockHandler.lock.isLocked()) {
application->switchWindow(app::window::name::desktop_pin_lock);
sys::Bus::SendUnicast(
std::make_shared<TimersProcessingStopMessage>(), service::name::service_time, application);
}
else {
topBar->setActive(TopBar::Elements::LOCK, false);
if (!buildNotifications(app)) {
LOG_ERROR("Couldn't fit in all notifications");
}
setActiveState(app);
if (app->need_sim_select && Store::GSM::get()->sim == Store::GSM::SIM::SIM_UNKNOWN) {
app::manager::Controller::switchApplication(
this->application, app::name_settings, app::sim_select, nullptr);
}
sys::Bus::SendUnicast(
std::make_shared<TimersProcessingStartMessage>(), service::name::service_time, application);
}
application->refreshWindow(RefreshModes::GUI_REFRESH_FAST);
}
void DesktopMainWindow::onBeforeShow(ShowMode mode, SwitchData *data)
{
// update time
time->setText(topBar->getTimeString());
// check if there was a signal to lock the pone due to inactivity.
if ((data != nullptr) && (data->getDescription() == "LockPhoneData")) {
auto app = getAppDesktop();
if (app->lockHandler.lock.isLocked()) {
return;
}
auto *lockData = dynamic_cast<LockPhoneData *>(data);
lockTimeoutApplilcation = lockData->getPreviousApplication();
application->setSuspendFlag(true);
}
setVisibleState();
}
bool DesktopMainWindow::processLongPressEvent(const InputEvent &inputEvent)
{
auto *app = getAppDesktop();
if (inputEvent.is(KeyCode::KEY_PND) && (!app->lockHandler.lock.isLocked())) {
app->lockHandler.lock.lock();
setVisibleState();
application->refreshWindow(RefreshModes::GUI_REFRESH_FAST);
application->setSuspendFlag(true);
return true;
}
else if (inputEvent.is(KeyCode::KEY_RF)) {
application->switchWindow("PowerOffWindow");
return true;
}
// long press of '0' key is translated to '+'
else if (inputEvent.is(KeyCode::KEY_0)) {
return app::prepareCall(application, "+");
}
// check if any of the lower inheritance onInput methods catch the event
return AppWindow::onInput(inputEvent);
}
bool DesktopMainWindow::processShortPressEventOnUnlocked(const InputEvent &inputEvent)
{
auto code = translator.handle(inputEvent.key, InputMode({InputMode::phone}).get());
// if numeric key was pressed record that key and send it to call application
if (code != 0) {
return app::prepareCall(application, std::string(1, static_cast<char>(code)));
}
else if (inputEvent.is(KeyCode::KEY_UP) && focusItem == nullptr) {
setFocusItem(notifications);
notifications->navigateToBottom();
return true;
}
else if (inputEvent.is(KeyCode::KEY_DOWN) && focusItem == nullptr) {
setFocusItem(notifications);
return true;
}
// check if any of the lower inheritance onInput methods catch the event
return AppWindow::onInput(inputEvent);
}
bool DesktopMainWindow::processShortPressEventOnLocked(const InputEvent &inputEvent)
{
// if enter was pressed
if (enter_cache.cached() && inputEvent.is(KeyCode::KEY_PND)) {
// if interval between enter and pnd keys is less than time defined for unlocking
// display pin lock screen or simply refresh current window to update labels
std::unique_ptr<LockPhoneData> data = std::make_unique<LockPhoneData>(LockPhoneData::Request::Pin);
// if there was no application on to before closing proceed normally to pin protection
if (lockTimeoutApplilcation.empty()) {
application->switchWindow(app::window::name::desktop_pin_lock, std::move(data));
return true;
}
else {
data->setPrevApplication(lockTimeoutApplilcation);
lockTimeoutApplilcation = "";
application->switchWindow(app::window::name::desktop_pin_lock, std::move(data));
return true;
}
}
else if (enter_cache.storeEnter(inputEvent)) {
return true;
}
// check if any of the lower inheritance onInput methods catch the event
else if (AppWindow::onInput(inputEvent)) {
return true;
}
application->switchWindow(app::window::name::desktop_locked);
return true;
}
bool DesktopMainWindow::onInput(const InputEvent &inputEvent)
{
auto *app = getAppDesktop();
if (inputEvent.isLongPress()) {
return processLongPressEvent(inputEvent);
}
else if (inputEvent.isShortPress()) {
if (app->lockHandler.lock.isLocked()) {
return processShortPressEventOnLocked(inputEvent);
}
else {
return processShortPressEventOnUnlocked(inputEvent);
}
}
return AppWindow::onInput(inputEvent);
}
void DesktopMainWindow::rebuild()
{
destroyInterface();
buildInterface();
}
bool DesktopMainWindow::updateTime(const UTF8 &timeStr)
{
auto ret = AppWindow::updateTime(timeStr);
time->setText(topBar->getTimeString());
return ret;
}
bool DesktopMainWindow::updateTime(const uint32_t &timestamp, bool mode24H)
{
auto ret = AppWindow::updateTime(timestamp, mode24H);
time->setText(topBar->getTimeString());
return ret;
}
std::list<DrawCommand *> DesktopMainWindow::buildDrawList()
{
time->setText(topBar->getTimeString());
return gui::AppWindow::buildDrawList();
}
auto DesktopMainWindow::buildNotifications(app::ApplicationDesktop *app) -> bool
{
erase(notifications);
using namespace style::desktop;
notifications = new NotificationsBox(this,
notifications::X,
notifications::Y,
notifications::Width,
bottomBar->widgetArea.pos(Axis::Y) - notifications::Y);
addWidget(notifications);
if (!notifications->visible) {
LOG_ERROR("Can't fit notifications box!");
return false;
}
auto onNotificationFocus = [this](bool isFocused) -> void {
bottomBar->setText(BottomBar::Side::CENTER, utils::localize.get("app_desktop_show"), isFocused);
bottomBar->setText(BottomBar::Side::RIGHT, utils::localize.get("app_desktop_clear"), isFocused);
};
if (app->notifications.notSeen.Calls > 0) {
notifications->addNotification(
"phone",
utils::localize.get("app_desktop_missed_calls"),
std::to_string(app->notifications.notSeen.Calls),
[app]() -> bool { return app->showCalls(); },
[app]() -> bool { return app->clearCallsNotification(); },
onNotificationFocus);
}
if (app->notifications.notSeen.SMS > 0) {
notifications->addNotification(
"mail",
utils::localize.get("app_desktop_unread_messages"),
std::to_string(app->notifications.notSeen.SMS),
[this]() -> bool {
return app::manager::Controller::switchApplication(
application, app::name_messages, gui::name::window::main_window, nullptr);
},
[app, this]() -> bool { return app->clearMessagesNotification(); },
onNotificationFocus);
}
notifications->focusChangedCallback = [this, app](Item &) -> bool {
if (notifications->focus == false) {
LOG_DEBUG("All notifications removed, notification box lost focus");
bottomBar->setActive(BottomBar::Side::RIGHT, false);
setFocusItem(nullptr);
setActiveState(app);
return true;
}
return false;
};
return true;
}
auto DesktopMainWindow::setActiveState(app::ApplicationDesktop *app) -> bool
{
bottomBar->setText(BottomBar::Side::CENTER, utils::localize.get("app_desktop_menu"));
if (app->notifications.notSeen.areEmpty() != true) {
bottomBar->setText(BottomBar::Side::RIGHT, utils::localize.get("app_desktop_clear_all"));
inputCallback = [this, app](Item &, const InputEvent &inputEvent) -> bool {
if (inputEvent.isShortPress() && inputEvent.is(KeyCode::KEY_RF) && getFocusItem() == nullptr) {
LOG_DEBUG("KEY_RF pressed to clear all notifications");
setFocusItem(notifications);
return notifications->clearAll(inputEvent);
}
return false;
};
}
else {
bottomBar->setText(BottomBar::Side::LEFT, utils::localize.get("app_desktop_calls"));
inputCallback = [app](Item &, const InputEvent &inputEvent) -> bool {
if (inputEvent.isShortPress() && inputEvent.is(gui::KeyCode::KEY_LF)) {
LOG_DEBUG("KEY_LF pressed to navigate to calls");
return app->showCalls();
}
return false;
};
}
return true;
}
app::ApplicationDesktop *DesktopMainWindow::getAppDesktop() const
{
auto *app = dynamic_cast<app::ApplicationDesktop *>(application);
assert(app);
return app;
}
} /* namespace gui */