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") property bool khalAvailable: true // Always true to enable DMS calendar card UI property bool khalInstalled: false // Tracks if khal is actually on the system property var eventsByDate: ({}) property var khalEventsByDate: ({}) property var taskEventsByDate: ({}) property var localTasks: ({}) property bool isLoading: false property string lastError: "" property date lastStartDate property date lastEndDate property string khalDateFormat: "MM/dd/yyyy" onKhalEventsByDateChanged: mergeEvents() onTaskEventsByDateChanged: mergeEvents() function checkKhalAvailability() { if (!khalCheckProcess.running) khalCheckProcess.running = true; } function detectKhalDateFormat() { if (!khalFormatProcess.running) khalFormatProcess.running = true; } function parseKhalDateFormat(formatExample) { let qtFormat = formatExample.replace("12", "MM").replace("21", "dd").replace("2013", "yyyy"); return { format: qtFormat, parser: null }; } function loadCurrentMonth() { if (!root.khalAvailable) return; let today = new Date(); let firstDay = new Date(today.getFullYear(), today.getMonth(), 1); let lastDay = new Date(today.getFullYear(), today.getMonth() + 1, 0); // Add padding let startDate = new Date(firstDay); startDate.setDate(startDate.getDate() - firstDay.getDay() - 7); let endDate = new Date(lastDay); endDate.setDate(endDate.getDate() + (6 - lastDay.getDay()) + 7); loadEvents(startDate, endDate); } function loadEvents(startDate, endDate) { if (!root.khalInstalled) { return; } if (eventsProcess.running) { return; } // Store last requested date range for refresh timer root.lastStartDate = startDate; root.lastEndDate = endDate; root.isLoading = true; // Format dates for khal using detected format let startDateStr = Qt.formatDate(startDate, root.khalDateFormat); let endDateStr = Qt.formatDate(endDate, root.khalDateFormat); eventsProcess.requestStartDate = startDate; eventsProcess.requestEndDate = endDate; eventsProcess.command = ["khal", "list", "--json", "title", "--json", "description", "--json", "start-date", "--json", "start-time", "--json", "end-date", "--json", "end-time", "--json", "all-day", "--json", "location", "--json", "url", startDateStr, endDateStr]; eventsProcess.running = true; } function getEventsForDate(date) { let dateKey = Qt.formatDate(date, "yyyy-MM-dd"); return root.eventsByDate[dateKey] || []; } function hasEventsForDate(date) { let events = getEventsForDate(date); return events.length > 0; } // In-memory Task CRUD methods 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" // Pastel Green , "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 = {}; // Merge khal events for (let dateKey in root.khalEventsByDate) { merged[dateKey] = [].concat(root.khalEventsByDate[dateKey]); } // Merge task events 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); } } } // Sort events within each date 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; } // Initialize on component completion Component.onCompleted: { detectKhalDateFormat(); } // Atomic file view for tasks 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 = {}; } } // Process for detecting khal date format Process { id: khalFormatProcess command: ["khal", "printformats"] running: false onExited: exitCode => { if (exitCode !== 0) { checkKhalAvailability(); } } stdout: StdioCollector { onStreamFinished: { let lines = text.split('\n'); for (let line of lines) { if (line.startsWith('dateformat:')) { let formatExample = line.substring(line.indexOf(':') + 1).trim(); let formatInfo = parseKhalDateFormat(formatExample); root.khalDateFormat = formatInfo.format; break; } } checkKhalAvailability(); } } } // Process for checking khal configuration Process { id: khalCheckProcess command: ["khal", "list", "today"] running: false onExited: exitCode => { root.khalInstalled = (exitCode === 0); if (root.khalInstalled) { loadCurrentMonth(); } else { loadEvents(root.lastStartDate || new Date(), root.lastEndDate || new Date()); } } } // Process for loading events Process { id: eventsProcess property date requestStartDate property date requestEndDate property string rawOutput: "" running: false onExited: exitCode => { root.isLoading = false; if (exitCode !== 0) { root.lastError = "Failed to load events (exit code: " + exitCode + ")"; return; } try { let newEventsByDate = {}; let lines = eventsProcess.rawOutput.split('\n'); for (let line of lines) { line = line.trim(); if (!line || line === "[]") continue; // Parse JSON line let dayEvents = JSON.parse(line); // Process each event in this day's array for (let event of dayEvents) { if (!event.title) continue; // Parse start and end dates using detected format let startDate, endDate; if (event['start-date']) { startDate = Date.fromLocaleString(I18n.locale(), event['start-date'], root.khalDateFormat); } else { startDate = new Date(); } if (event['end-date']) { endDate = Date.fromLocaleString(I18n.locale(), event['end-date'], root.khalDateFormat); } else { endDate = new Date(startDate); } // Create start/end times let startTime = new Date(startDate); let endTime = new Date(endDate); if (event['start-time'] && event['all-day'] !== "True") { // Parse time if available and not all-day let timeStr = event['start-time']; if (timeStr) { // Match time with optional seconds and AM/PM let timeParts = timeStr.match(/(\d+):(\d+)(?::\d+)?\s*(AM|PM)?/i); if (timeParts) { let hours = parseInt(timeParts[1]); let minutes = parseInt(timeParts[2]); // Handle AM/PM conversion if present if (timeParts[3]) { let period = timeParts[3].toUpperCase(); if (period === 'PM' && hours !== 12) { hours += 12; } else if (period === 'AM' && hours === 12) { hours = 0; } } startTime.setHours(hours, minutes); if (event['end-time']) { let endTimeParts = event['end-time'].match(/(\d+):(\d+)(?::\d+)?\s*(AM|PM)?/i); if (endTimeParts) { let endHours = parseInt(endTimeParts[1]); let endMinutes = parseInt(endTimeParts[2]); // Handle AM/PM conversion if present if (endTimeParts[3]) { let endPeriod = endTimeParts[3].toUpperCase(); if (endPeriod === 'PM' && endHours !== 12) { endHours += 12; } else if (endPeriod === 'AM' && endHours === 12) { endHours = 0; } } endTime.setHours(endHours, endMinutes); } } else { // Default to 1 hour duration on same day endTime = new Date(startTime); endTime.setHours(startTime.getHours() + 1); } } } } // Create unique ID for this event (to track multi-day events) let eventId = event.title + "_" + event['start-date'] + "_" + (event['start-time'] || 'allday'); // Create event object template let extractedUrl = ""; if (!event.url && event.description) { let urlMatch = event.description.match(/https?:\/\/[^\s]+/); if (urlMatch) { extractedUrl = urlMatch[0]; } } let eventTemplate = { "id": eventId, "title": event.title || "Untitled Event", "start": startTime, "end": endTime, "location": event.location || "", "description": event.description || "", "url": event.url || extractedUrl, "calendar": "", "color": "", "allDay": event['all-day'] === "True", "isMultiDay": startDate.toDateString() !== endDate.toDateString() }; // Add event to each day it spans let currentDate = new Date(startDate); while (currentDate <= endDate) { let dateKey = Qt.formatDate(currentDate, "yyyy-MM-dd"); if (!newEventsByDate[dateKey]) newEventsByDate[dateKey] = []; // Check if this exact event is already added to this date (prevent duplicates) let existingEvent = newEventsByDate[dateKey].find(e => { return e.id === eventId; }); if (existingEvent) { // Move to next day without adding duplicate currentDate.setDate(currentDate.getDate() + 1); continue; } // Create a copy of the event for this date let dayEvent = Object.assign({}, eventTemplate); // For multi-day events, adjust the display time for this specific day if (currentDate.getTime() === startDate.getTime()) { // First day - use original start time dayEvent.start = new Date(startTime); } else { // Subsequent days - start at beginning of day for all-day events dayEvent.start = new Date(currentDate); if (!dayEvent.allDay) dayEvent.start.setHours(0, 0, 0, 0); } if (currentDate.getTime() === endDate.getTime()) { // Last day - use original end time dayEvent.end = new Date(endTime); } else { // Earlier days - end at end of day for all-day events dayEvent.end = new Date(currentDate); if (!dayEvent.allDay) dayEvent.end.setHours(23, 59, 59, 999); } newEventsByDate[dateKey].push(dayEvent); // Move to next day currentDate.setDate(currentDate.getDate() + 1); } } } root.khalEventsByDate = newEventsByDate; root.lastError = ""; } catch (error) { root.lastError = "Failed to parse events JSON: " + error.toString(); root.khalEventsByDate = {}; } // Reset for next run eventsProcess.rawOutput = ""; } stdout: SplitParser { splitMarker: "\n" onRead: data => { eventsProcess.rawOutput += data + "\n"; } } } }