1
0
mirror of https://github.com/AvengeMedia/DankMaterialShell.git synced 2026-06-15 23:55:21 -04:00
Files
DankMaterialShell/quickshell/Services/CalendarService.qml
T
2026-06-15 19:02:22 -04:00

356 lines
10 KiB
QML

pragma Singleton
pragma ComponentBehavior: Bound
import QtQuick
import Quickshell
import Quickshell.Io
import qs.Common
import qs.Services
Singleton {
id: root
readonly property var log: Log.scoped("CalendarService")
readonly property string backendPref: SettingsData.calendarBackend
readonly property string activeBackend: {
switch (backendPref) {
case "khal":
return "khal";
case "dankcal":
return "dankcal";
default:
if (dankBackend.connected)
return "dankcal";
if (khalBackend.installed)
return "khal";
return "none";
}
}
readonly property bool calendarAvailable: activeBackend !== "none"
readonly property bool isDankActive: activeBackend === "dankcal"
readonly property bool canCreateEvents: isDankActive && dankBackend.connected
property bool khalAvailable: true // compatibility alias - calendar card UI gate
readonly property bool dankConnected: dankBackend.connected
readonly property bool dankBinaryExists: dankBackend.binaryExists
readonly property bool dankNeedsLaunch: backendPref === "dankcal" && !dankBackend.connected && !dankBackend.socketFound
property var calendars: dankBackend.calendars
property var eventsByDate: ({})
property var taskEventsByDate: ({})
property var localTasks: ({})
property bool isLoading: khalBackend.isLoading
property string lastError: ""
property bool _rangeSet: false
property date lastStartDate
property date lastEndDate
onTaskEventsByDateChanged: mergeEvents()
onActiveBackendChanged: {
mergeEvents();
if (_rangeSet)
loadEvents(lastStartDate, lastEndDate);
}
CalendarKhalBackend {
id: khalBackend
onEventsByDateChanged: root.mergeEvents()
}
CalendarDankBackend {
id: dankBackend
enabled: root.backendPref === "dankcal" || root.backendPref === "auto"
onEventsByDateChanged: root.mergeEvents()
onConnectedChanged: {
if (connected && root._rangeSet)
root.loadEvents(root.lastStartDate, root.lastEndDate);
}
}
function loadEvents(startDate, endDate) {
root.lastStartDate = startDate;
root.lastEndDate = endDate;
root._rangeSet = true;
switch (activeBackend) {
case "dankcal":
dankBackend.loadEvents(startDate, endDate);
break;
case "khal":
khalBackend.loadEvents(startDate, endDate);
break;
}
}
function _activeBackendEventsByDate() {
switch (activeBackend) {
case "dankcal":
return dankBackend.eventsByDate;
case "khal":
return khalBackend.eventsByDate;
default:
return {};
}
}
function getEventsForDate(date) {
let dateKey = Qt.formatDate(date, "yyyy-MM-dd");
return root.eventsByDate[dateKey] || [];
}
function hasEventsForDate(date) {
return getEventsForDate(date).length > 0;
}
function writableCalendars() {
return isDankActive ? dankBackend.writableCalendars() : [];
}
function defaultCalendar() {
return isDankActive ? dankBackend.defaultCalendar() : null;
}
function launchDankCalendar() {
dankBackend.launch();
}
function createEvent(fields, callback) {
if (isDankActive) {
dankBackend.createEvent(fields, callback);
return;
}
if (callback)
callback({
"error": "read-only backend"
});
}
function updateEvent(id, fields, callback) {
if (isDankActive) {
dankBackend.updateEvent(id, fields, callback);
return;
}
if (callback)
callback({
"error": "read-only backend"
});
}
function deleteEvent(id, callback) {
if (isDankActive) {
dankBackend.deleteEvent(id, callback);
return;
}
if (callback)
callback({
"error": "read-only backend"
});
}
function loadTasks(text) {
if (!text || text.trim() === "") {
root.localTasks = {};
root.taskEventsByDate = {};
return;
}
try {
root.localTasks = JSON.parse(text);
updateTaskEvents();
} catch (error) {
log.warn("Failed to parse local tasks JSON: " + error.toString());
}
}
function saveTasks() {
let dir = Quickshell.env("HOME") + "/.config/niri-calendar-todo";
Quickshell.execDetached(["mkdir", "-p", dir]);
tasksFileView.setText(JSON.stringify(root.localTasks, null, 2));
}
function updateTaskEvents() {
let newTaskEvents = {};
for (let dateKey in root.localTasks) {
let taskList = root.localTasks[dateKey] || [];
newTaskEvents[dateKey] = [];
for (let task of taskList) {
let eventId = "task_" + task.id;
let parts = dateKey.split("-");
let taskDate = new Date(parseInt(parts[0]), parseInt(parts[1]) - 1, parseInt(parts[2]));
newTaskEvents[dateKey].push({
"id": eventId,
"title": task.text,
"completed": !!task.completed,
"start": taskDate,
"end": taskDate,
"location": "",
"description": "Task from your Planner",
"url": "",
"calendar": "Todo Planner",
"color": "#10B981",
"allDay": true,
"isMultiDay": false
});
}
}
root.taskEventsByDate = newTaskEvents;
}
function addTaskForDate(date, text) {
let dateKey = Qt.formatDate(date, "yyyy-MM-dd");
let tasks = Object.assign({}, root.localTasks);
if (!tasks[dateKey])
tasks[dateKey] = [];
let taskId = (new Date().getTime()) + "-dms";
tasks[dateKey].push({
"id": taskId,
"text": text,
"completed": false
});
root.localTasks = tasks;
updateTaskEvents();
saveTasks();
}
function toggleTask(taskId) {
let cleanId = taskId.replace("task_", "");
let tasks = Object.assign({}, root.localTasks);
let updated = false;
for (let dateKey in tasks) {
let list = tasks[dateKey];
for (let item of list) {
if (item.id === cleanId) {
item.completed = !item.completed;
updated = true;
break;
}
}
if (updated)
break;
}
if (updated) {
root.localTasks = tasks;
updateTaskEvents();
saveTasks();
}
}
function removeTask(taskId) {
let cleanId = taskId.replace("task_", "");
let tasks = Object.assign({}, root.localTasks);
let updated = false;
for (let dateKey in tasks) {
let list = tasks[dateKey];
let filtered = list.filter(item => item.id !== cleanId);
if (filtered.length !== list.length) {
if (filtered.length === 0)
delete tasks[dateKey];
else
tasks[dateKey] = filtered;
updated = true;
break;
}
}
if (updated) {
root.localTasks = tasks;
updateTaskEvents();
saveTasks();
}
}
function reorderTasksForDate(date, orderedIds) {
let dateKey = Qt.formatDate(date, "yyyy-MM-dd");
let tasks = Object.assign({}, root.localTasks);
let v = tasks[dateKey] || [];
let idToItem = {};
for (let item of v)
idToItem[item.id] = item;
let newV = [];
for (let tid of orderedIds) {
if (idToItem[tid])
newV.push(idToItem[tid]);
}
let orderedSet = new Set(orderedIds);
for (let item of v) {
if (!orderedSet.has(item.id))
newV.push(item);
}
tasks[dateKey] = newV;
root.localTasks = tasks;
updateTaskEvents();
saveTasks();
}
function editTask(taskId, newText) {
let cleanId = taskId.replace("task_", "");
let tasks = Object.assign({}, root.localTasks);
let updated = false;
for (let dateKey in tasks) {
let list = tasks[dateKey];
for (let item of list) {
if (item.id === cleanId) {
item.text = newText;
updated = true;
break;
}
}
if (updated)
break;
}
if (updated) {
root.localTasks = tasks;
updateTaskEvents();
saveTasks();
}
}
function mergeEvents() {
let merged = {};
let backendEvents = _activeBackendEventsByDate();
for (let dateKey in backendEvents)
merged[dateKey] = [].concat(backendEvents[dateKey]);
for (let dateKey in root.taskEventsByDate) {
if (!merged[dateKey])
merged[dateKey] = [];
for (let event of root.taskEventsByDate[dateKey]) {
if (!merged[dateKey].some(e => e.id === event.id))
merged[dateKey].push(event);
}
}
for (let dateKey in merged) {
let list = merged[dateKey];
for (let idx = 0; idx < list.length; idx++)
list[idx]._origIdx = idx;
list.sort((a, b) => {
let diff = a.start.getTime() - b.start.getTime();
if (diff !== 0)
return diff;
return a._origIdx - b._origIdx;
});
}
root.eventsByDate = merged;
}
FileView {
id: tasksFileView
path: Quickshell.env("HOME") + "/.config/niri-calendar-todo/tasks.json"
blockLoading: false
blockWrites: false
atomicWrites: true
watchChanges: true
printErrors: false
onLoaded: loadTasks(tasksFileView.text())
onLoadFailed: {
root.localTasks = {};
root.taskEventsByDate = {};
}
}
}