mirai/src/UiState.cpp

336 lines
10 KiB
C++
Raw Normal View History

2024-08-16 21:35:12 +02:00
/*
* Mirai. Copyright (C) 2024 Vyn
* This file is licensed under version 3 of the GNU General Public License (GPL-3.0-only)
* The license can be found in the LICENSE file or at https://www.gnu.org/licenses/gpl-3.0.txt
*/
#include "UiState.h"
#include "Utils.h"
#include "appwindow.h"
2024-10-15 11:55:39 +02:00
#include "mirai-core/DataProvider.h"
2024-10-11 16:26:13 +02:00
#include "mirai-core/Day.h"
2024-08-16 21:35:12 +02:00
#include "mirai-core/Mirai.h"
#include "slint.h"
#include "slint_sharedvector.h"
#include "slint_string.h"
#include <algorithm>
#include <bits/chrono.h>
#include <cassert>
#include <charconv>
#include <chrono>
#include <ctime>
#include <format>
#include <iomanip>
#include <iostream>
#include <iterator>
#include <memory>
#include <optional>
#include <ostream>
#include <ranges>
#include <stdexcept>
#include <string>
#include <string_view>
#include <vector>
UiState::UiState(mirai::Mirai *miraiInstance) : miraiInstance_(miraiInstance), view_(miraiInstance)
{
sources_ = std::make_shared<slint::VectorModel<ui::Source>>();
days_ = std::make_shared<slint::VectorModel<ui::Day>>();
unscheduledTasks_ = std::make_shared<slint::VectorModel<ui::TaskData>>();
2024-08-16 21:35:12 +02:00
tags_ = std::make_shared<slint::VectorModel<slint::SharedString>>();
auto sourcesNames = std::make_shared<slint::MapModel<ui::Source, slint::SharedString>>(
sources_,
[&](const ui::Source &a) {
return a.name;
}
);
2024-08-16 21:35:12 +02:00
mainWindow_->global<ui::Backend>().set_sources(sourcesNames);
2024-08-16 21:35:12 +02:00
2024-10-11 16:26:13 +02:00
view_.setAllSources();
2024-08-16 21:35:12 +02:00
view_.update();
reloadSources();
2024-08-16 21:35:12 +02:00
reloadTasks();
setupCallbacks();
}
std::optional<ui::Date> stringToDate(const std::string &dateStr)
2024-08-16 21:35:12 +02:00
{
using std::operator""sv;
auto dateSplitView =
dateStr | std::views::split("-"sv) | std::views::transform([](auto v) -> int {
int i = 0;
std::from_chars(v.data(), v.data() + v.size(), i);
return i;
});
std::vector<int> dateSplit{dateSplitView.begin(), dateSplitView.end()};
if (dateSplit.size() != 3) {
return std::nullopt;
}
auto year = dateSplit[0];
auto month = dateSplit[1];
auto day = dateSplit[2];
return ui::Date{.year = year, .month = month, .day = day};
2024-08-16 21:35:12 +02:00
}
void UiState::setupUtilsCallbacks()
{
mainWindow_->global<ui::Backend>().on_format_date([&](const ui::Date &date) {
2024-08-16 21:35:12 +02:00
std::chrono::year_month_day chronoDate{
std::chrono::year(date.year),
std::chrono::month(date.month),
std::chrono::day(date.day),
};
return std::format("{:%B %d}", chronoDate);
2024-08-16 21:35:12 +02:00
});
}
void UiState::setupCallbacks()
{
mainWindow_->global<ui::Backend>().on_task_clicked([&](int sourceId, int taskId) {
auto source = miraiInstance_->getSourceById(sourceId);
assert(source);
auto task = source->getTaskById(taskId);
2024-08-16 21:35:12 +02:00
assert(task);
2024-10-11 16:26:13 +02:00
task->setChecked(!task->checked());
// task->getState() == mirai::DONE ? task->markAsUndone() : task->markAsDone();
2024-08-16 21:35:12 +02:00
miraiInstance_->save();
});
mainWindow_->global<ui::Backend>().on_source_clicked([&](int index) {
// index with value -1 is equal to no selection, aka "All"
if (index == -1) {
2024-10-11 16:26:13 +02:00
view_.setAllSources();
2024-08-16 21:35:12 +02:00
} else {
2024-10-11 16:26:13 +02:00
view_.removeSources();
const mirai::Source *source = miraiInstance_->getSourceById(index);
view_.addSource(*source);
2024-08-16 21:35:12 +02:00
}
mainWindow_->global<ui::Backend>().set_default_source_index(index == -1 ? 0 : index);
2024-08-16 21:35:12 +02:00
view_.update();
reloadSources();
2024-08-16 21:35:12 +02:00
reloadTasks();
});
mainWindow_->global<ui::Backend>().on_delete_task_clicked([&](int sourceId, int taskId) {
auto source = miraiInstance_->getSourceById(sourceId);
assert(source);
auto task = source->getTaskById(taskId);
2024-08-16 21:35:12 +02:00
assert(task);
2024-10-11 16:26:13 +02:00
source->removeTask(*task);
2024-08-16 21:35:12 +02:00
miraiInstance_->save();
view_.update();
reloadTasks();
});
mainWindow_->global<ui::Backend>().on_toggle_show_completed_tasks([&] {
2024-10-11 16:26:13 +02:00
view_.hideCompletedTasks(!view_.shouldHideCompletedTasks());
view_.update();
reloadTasks();
});
mainWindow_->global<ui::Backend>().set_sources_selected(sources_);
mainWindow_->global<ui::Backend>().set_tags(tags_);
mainWindow_->global<ui::Backend>().set_days(days_);
2024-10-11 16:26:13 +02:00
mainWindow_->global<ui::Backend>().on_save_task([&](ui::SaveTaskData newTaskData) {
2024-10-11 16:26:13 +02:00
auto source = miraiInstance_->getSourceById(newTaskData.sourceId);
assert(source);
2024-10-11 16:26:13 +02:00
auto task = source->getTaskById(newTaskData.id);
2024-10-15 11:55:39 +02:00
assert(task.has_value());
2024-10-11 16:26:13 +02:00
const mirai::Date &date = SlintDateToMiraiDate(newTaskData.date);
const auto dayOpt = source->getDayByDate(date);
2024-10-15 11:55:39 +02:00
task->setTitle(std::string(newTaskData.title));
2024-08-16 21:35:12 +02:00
2024-10-15 11:55:39 +02:00
if (!task.value().hasEvent() && date.year != 0) {
task->setDate(date);
2024-10-08 16:36:01 +02:00
}
2024-08-16 21:35:12 +02:00
2024-10-11 16:26:13 +02:00
miraiInstance_->save();
view_.update();
reloadTasks();
2024-08-16 21:35:12 +02:00
});
mainWindow_->global<ui::Backend>().on_create_task([&](ui::NewTaskData newTaskData) {
2024-10-11 16:26:13 +02:00
std::optional<mirai::Date> date = std::nullopt;
if (newTaskData.date.year != 0) {
date = SlintDateToMiraiDate(newTaskData.date);
}
auto source = miraiInstance_->getSourceById(newTaskData.sourceId);
2024-08-16 21:35:12 +02:00
2024-10-11 16:26:13 +02:00
std::optional<mirai::Event> event = std::nullopt;
if (newTaskData.eventId >= 0) {
event = source->getEventById(newTaskData.eventId);
2024-08-16 21:35:12 +02:00
}
2024-10-11 16:26:13 +02:00
source->createTask({
.title = std::string(newTaskData.title),
.event = event,
.date = date,
});
miraiInstance_->save();
view_.update();
reloadTasks();
2024-08-16 21:35:12 +02:00
});
mainWindow_->global<ui::Backend>().on_delete_event_clicked([&](int sourceId, int eventId) {
2024-10-11 16:26:13 +02:00
auto source = miraiInstance_->getSourceById(sourceId);
assert(source);
auto event = source->getEventById(eventId);
assert(event.has_value());
source->removeEvent(event.value());
miraiInstance_->save();
2024-08-16 21:35:12 +02:00
view_.update();
reloadTasks();
});
mainWindow_->global<ui::Backend>().on_create_event([&](ui::NewEventParams newEventParams) {
const ui::Date &date = newEventParams.date;
2024-10-11 16:26:13 +02:00
const std::string dateStr = SlintDateToStdString(date);
auto source = miraiInstance_->getSourceById(newEventParams.sourceId);
source->createEvent({
.title = std::string(newEventParams.title),
.date = SlintDateToMiraiDate(newEventParams.date),
.startsAt = SlintTimeToMiraiTime(newEventParams.startsAt),
.endsAt = SlintTimeToMiraiTime(newEventParams.endsAt),
});
miraiInstance_->save();
view_.update();
reloadTasks();
});
mainWindow_->global<ui::Backend>().on_save_event([&](ui::SaveEventParams newEventParams) {
const ui::Date &date = newEventParams.date;
2024-10-11 16:26:13 +02:00
const std::string dateStr = SlintDateToStdString(date);
auto source = miraiInstance_->getSourceById(newEventParams.sourceId);
assert(source);
auto event = source->getEventById(newEventParams.id);
assert(event);
event->setTitle(std::string(newEventParams.title));
event->setStartTime(SlintTimeToMiraiTime(newEventParams.startsAt));
event->setEndTime(SlintTimeToMiraiTime(newEventParams.endsAt));
// TODO we can't change the date of the event for now.
2024-08-16 21:35:12 +02:00
2024-10-11 16:26:13 +02:00
miraiInstance_->save();
view_.update();
reloadTasks();
});
2024-08-16 21:35:12 +02:00
setupUtilsCallbacks();
}
std::shared_ptr<slint::VectorModel<slint::SharedString>>
stdToSlintStringVector(const std::vector<std::string> &stdVector)
{
auto slintVector = std::make_shared<slint::VectorModel<slint::SharedString>>();
for (const auto &item : stdVector) {
slintVector->push_back(slint::SharedString(item));
}
return slintVector;
}
void UiState::reloadTasks()
{
days_->clear();
if (miraiInstance_->getSources().size() == 0) {
2024-08-16 21:35:12 +02:00
return;
}
auto todayDate = mirai::Date(std::chrono::system_clock::now());
2024-10-11 16:26:13 +02:00
auto dates = view_.getDates();
auto slintDays = std::make_shared<slint::VectorModel<ui::Day>>();
2024-10-11 16:26:13 +02:00
for (int dayIndex = 0; dayIndex < dates.size(); ++dayIndex) {
auto &currentDate = dates.at(dayIndex);
auto slintEvents = std::make_shared<slint::VectorModel<ui::Event>>();
auto slintDayTasks = std::make_shared<slint::VectorModel<ui::TaskData>>();
auto relativeDaysDiff = std::chrono::duration_cast<std::chrono::days>(
2024-10-11 16:26:13 +02:00
std::chrono::sys_days(currentDate.toStdChrono()) -
std::chrono::sys_days(todayDate.toStdChrono())
)
.count();
slintDays->push_back(ui::Day{
.date = MiraiDateToSlintDate(currentDate),
.events = slintEvents,
.tasks = slintDayTasks,
.isLate = currentDate < todayDate,
.isToday = currentDate == todayDate,
.relativeDaysDiff = static_cast<int>(relativeDaysDiff)
});
2024-10-11 16:26:13 +02:00
// Day's tasks
const std::vector<mirai::Task> tasksForDate = view_.getTasksForDate(currentDate);
for (int taskIndex = 0; taskIndex < tasksForDate.size(); ++taskIndex) {
auto &task = tasksForDate.at(taskIndex);
2024-08-16 21:35:12 +02:00
slintDayTasks->push_back({
2024-10-11 16:26:13 +02:00
.sourceId = task.sourceId(),
.id = task.id(),
.title = slint::SharedString(task.title()),
.checked = task.checked(),
2024-08-16 21:35:12 +02:00
});
}
2024-10-11 16:26:13 +02:00
// Day's events
const std::vector<mirai::Event> eventsForDate = view_.getEventsForDate(currentDate);
for (int eventIndex = 0; eventIndex < eventsForDate.size(); ++eventIndex) {
auto &currentEvent = eventsForDate.at(eventIndex);
auto slintTasks = std::make_shared<slint::VectorModel<ui::TaskData>>();
slintEvents->push_back(ui::Event{
2024-10-11 16:26:13 +02:00
.sourceId = currentEvent.sourceId(),
.id = currentEvent.id(),
.title = slint::SharedString(currentEvent.title()),
.startsAt = MiraiTimeToSlintTime(currentEvent.startsAt()),
.endsAt = MiraiTimeToSlintTime(currentEvent.endsAt()),
2024-08-16 21:35:12 +02:00
.tasks = slintTasks,
});
2024-10-11 16:26:13 +02:00
auto eventTasks = currentEvent.queryTasks();
for (int taskIndex = 0; taskIndex < eventTasks.size(); ++taskIndex) {
auto &task = eventTasks.at(taskIndex);
2024-08-16 21:35:12 +02:00
slintTasks->push_back({
2024-10-11 16:26:13 +02:00
.sourceId = task.sourceId(),
.id = task.id(),
.title = slint::SharedString(task.title()),
.checked = task.checked(),
2024-08-16 21:35:12 +02:00
});
}
}
}
days_ = slintDays;
mainWindow_->global<ui::Backend>().set_days(days_);
2024-10-08 16:36:01 +02:00
2024-10-11 16:26:13 +02:00
auto unscheduledTasksView = view_.getUnscheduledTasks();
2024-10-08 16:36:01 +02:00
unscheduledTasks_->clear();
2024-10-11 16:26:13 +02:00
for (int taskIndex = 0; taskIndex < unscheduledTasksView.size(); ++taskIndex) {
auto &task = unscheduledTasksView.at(taskIndex);
2024-10-08 16:36:01 +02:00
unscheduledTasks_->push_back({
2024-10-11 16:26:13 +02:00
.sourceId = task.sourceId(),
.id = task.id(),
.title = slint::SharedString(task.title()),
.checked = task.checked(),
2024-10-08 16:36:01 +02:00
});
}
mainWindow_->global<ui::Backend>().set_unscheduled_tasks(unscheduledTasks_);
2024-08-16 21:35:12 +02:00
}
void UiState::reloadSources()
2024-08-16 21:35:12 +02:00
{
sources_->clear();
2024-10-11 16:26:13 +02:00
bool noSourceSelected = miraiInstance_->getSources().size() == view_.activeSourceCount();
for (const auto &source : miraiInstance_->getSources()) {
2024-10-11 16:26:13 +02:00
bool isSourceSelected = view_.isSourceSelected(*source);
sources_->push_back(
2024-10-11 16:26:13 +02:00
{.name = slint::SharedString(source->name()),
.selected = isSourceSelected && !noSourceSelected}
);
2024-08-16 21:35:12 +02:00
}
mainWindow_->global<ui::Backend>().set_no_source_selected(noSourceSelected);
2024-08-16 21:35:12 +02:00
}
void UiState::run()
{
mainWindow_->run();
}