kanblendar/kanblendar.js
Kumi 2ad254e241
feat: hide completed tasks from previous days
Added 'updateTaskDisplay' method to hide tasks marked as done if their due date is before the current day. This helps declutter the task view by filtering out outdated completed tasks. The 'updateTaskDisplay' method is invoked during initialization and when tasks are loaded.
2024-09-09 13:37:33 +02:00

565 lines
No EOL
23 KiB
JavaScript

class Kanblendar {
constructor(config = {}) {
this.kanbanSection = document.getElementById('kanblendar');
this.tasks = new Map();
this.notificationTimeouts = new Map();
this.currentTask = null;
// Read configuration from data attributes
const dataConfig = {
columns: this.kanbanSection.dataset.columns ? this.kanbanSection.dataset.columns.split(',') : undefined,
startTime: this.kanbanSection.dataset.startTime,
endTime: this.kanbanSection.dataset.endTime,
interval: this.kanbanSection.dataset.interval ? parseInt(this.kanbanSection.dataset.interval, 10) : undefined
};
// Default configuration options
const defaultConfig = {
columns: ['Backlog', 'In Progress', 'Done'],
timeSlots: null,
startTime: '08:00',
endTime: '18:00',
interval: 60, // in minutes
generateModal: true, // Option to generate modal or not
currentDate: new Date().toISOString().split('T')[0] // Default to today's date
};
// Merge configurations
this.config = { ...defaultConfig, ...config, ...dataConfig };
this.timeSlots = this.config.timeSlots || this.generateTimeSlots(this.config.startTime, this.config.endTime, this.config.interval);
this.columns = this.config.columns;
this.init();
}
init() {
this.kanbanSection = document.getElementById('kanblendar');
if (this.config.generateModal) {
this.createModal();
} else {
this.taskModal = document.getElementById('kanblendar-taskModal');
this.closeModal = document.querySelector('.kanblendar-close');
this.taskForm = document.getElementById('kanblendar-taskForm');
}
this.closeModal.addEventListener('click', () => this.closeModalFunc());
window.addEventListener('click', (event) => {
if (event.target === this.taskModal) {
this.closeModalFunc();
}
});
this.taskForm.addEventListener('submit', (event) => this.saveTask(event));
document.addEventListener('taskMoved', (e) => {
console.log(`Task ${e.detail.taskId} moved to ${e.detail.newParent}`);
this.adjustTimeSlotHeights();
});
this.generateKanbanColumns(); // Generate the kanban columns
this.initDragAndDrop(); // Initialize drag and drop functionality
this.requestNotificationPermission(); // Request permission for notifications
this.highlightCurrentTimeSlot(); // Highlight the current time slot
this.updateTaskDisplay(); // Hide tasks completed before today
setInterval(() => this.highlightCurrentTimeSlot(), 60000); // Update highlight every minute
}
highlightCurrentTimeSlot() {
const now = new Date();
const currentTime = `${now.getHours()}:${now.getMinutes() < 10 ? '0' : ''}${now.getMinutes()}`;
document.querySelectorAll('.kanblendar-time-slot').forEach(slot => {
const startTime = slot.dataset.startTime;
const endTime = this.addMinutes(this.parseTime(startTime), this.config.interval).toTimeString().slice(0, 5);
if (currentTime >= startTime && currentTime < endTime) {
slot.classList.add('kanblendar-current-time');
} else {
slot.classList.remove('kanblendar-current-time');
}
});
}
createModal() {
const modalHTML = `
<div id="kanblendar-taskModal" class="kanblendar-modal">
<div class="kanblendar-modal-content">
<span class="kanblendar-close">&times;</span>
<h2 id="kanblendar-modalTitle">Create Task</h2>
<form id="kanblendar-taskForm">
<label for="kanblendar-taskTitle">Title:</label>
<input type="text" id="kanblendar-taskTitle" name="kanblendar-taskTitle" required>
<label for="kanblendar-taskDescription">Description:</label>
<textarea id="kanblendar-taskDescription" name="kanblendar-taskDescription"></textarea>
<label for="kanblendar-taskDueTime">Due Time:</label>
<input type="datetime-local" id="kanblendar-taskDueTime" name="kanblendar-taskDueTime">
<label for="kanblendar-taskColumn">Column:</label>
<select id="kanblendar-taskColumn" name="kanblendar-taskColumn"></select>
<label for="kanblendar-taskNotify">Notify Before (minutes):</label>
<input type="number" id="kanblendar-taskNotify" name="kanblendar-taskNotify" min="0">
<button type="submit">Save</button>
<button type="button" id="kanblendar-deleteTaskBtn" class="kanblendar-delete-task" style="display: none;">Delete</button>
</form>
</div>
</div>
`;
document.body.insertAdjacentHTML('beforeend', modalHTML);
this.taskModal = document.getElementById('kanblendar-taskModal');
this.closeModal = document.querySelector('.kanblendar-close');
this.taskForm = document.getElementById('kanblendar-taskForm');
this.deleteTaskBtn = document.getElementById('kanblendar-deleteTaskBtn');
// Populate the column dropdown
const taskColumnSelect = document.getElementById('kanblendar-taskColumn');
this.columns.forEach(column => {
const option = document.createElement('option');
option.value = column.toLowerCase().replace(/\s+/g, '-');
option.text = column;
taskColumnSelect.appendChild(option);
});
this.deleteTaskBtn.addEventListener('click', () => {
if (this.currentTask) {
this.deleteTask(this.currentTask);
this.closeModalFunc();
}
});
}
generateTimeSlots(startTime, endTime, interval) {
const timeSlots = [];
let currentTime = this.parseTime(startTime);
const end = this.parseTime(endTime);
while (currentTime < end) {
timeSlots.push({
display: this.formatTime(currentTime),
value: currentTime.toTimeString().slice(0, 5) // HH:MM format
});
currentTime = this.addMinutes(currentTime, interval);
}
return timeSlots;
}
parseTime(time) {
const [hours, minutes] = time.split(':').map(Number);
return new Date(1970, 0, 1, hours, minutes);
}
formatTime(date) {
return new Intl.DateTimeFormat(navigator.language, {
hour: 'numeric',
minute: 'numeric'
}).format(date);
}
addMinutes(date, minutes) {
return new Date(date.getTime() + minutes * 60000);
}
generateKanbanColumns() {
this.columns.forEach(column => {
const columnElement = document.createElement('div');
columnElement.classList.add('kanblendar-column');
columnElement.id = column.toLowerCase().replace(/\s+/g, '-');
columnElement.innerHTML = `<h2>${column}</h2><div class="kanblendar-non-timed-tasks" id="${column.toLowerCase().replace(/\s+/g, '-')}-tasks"></div>`;
this.timeSlots.forEach(timeSlot => {
const timeSlotElement = document.createElement('div');
timeSlotElement.classList.add('kanblendar-time-slot');
timeSlotElement.innerText = timeSlot.display;
timeSlotElement.dataset.startTime = timeSlot.value;
columnElement.appendChild(timeSlotElement);
});
this.kanbanSection.appendChild(columnElement);
});
this.adjustTimeSlotHeights(); // Ensure time slots have equal heights initially
}
initDragAndDrop() {
const timeSlots = document.querySelectorAll('.kanblendar-time-slot, .kanblendar-non-timed-tasks');
timeSlots.forEach(slot => {
slot.addEventListener('dragover', (e) => this.dragOver(e));
slot.addEventListener('drop', (e) => this.drop(e));
slot.addEventListener('dragenter', (e) => this.dragEnter(e));
slot.addEventListener('dragleave', (e) => this.dragLeave(e));
});
const tasks = document.querySelectorAll('.kanblendar-task');
tasks.forEach(task => {
task.addEventListener('dragstart', (e) => this.dragStart(e));
task.addEventListener('click', (e) => this.openModal(task)); // Add click event to open modal for editing
});
}
dragStart(e) {
e.dataTransfer.setData('text/plain', e.target.id);
e.dataTransfer.effectAllowed = 'move';
}
dragOver(e) {
e.preventDefault();
e.dataTransfer.dropEffect = 'move';
}
dragEnter(e) {
e.preventDefault();
if (e.target.classList.contains('kanblendar-task')) {
e.target.parentElement.classList.add('kanblendar-drag-over');
} else if (e.target.parentElement.classList.contains('kanblendar-task')) {
e.target.parentElement.parentElement.classList.add('kanblendar-drag-over');
} else {
e.target.classList.add('kanblendar-drag-over');
}
}
dragLeave(e) {
if (e.target.classList.contains('kanblendar-task')) {
e.target.parentElement.classList.remove('kanblendar-drag-over');
} else if (e.target.parentElement.classList.contains('kanblendar-task')) {
e.target.parentElement.parentElement.classList.remove('kanblendar-drag-over');
} else {
e.target.classList.remove('kanblendar-drag-over');
}
}
drop(e) {
e.preventDefault();
if (e.target.classList.contains('kanblendar-task')) {
e.target.parentElement.classList.remove('kanblendar-drag-over');
} else if (e.target.parentElement.classList.contains('kanblendar-task')) {
e.target.parentElement.parentElement.classList.remove('kanblendar-drag-over');
} else {
e.target.classList.remove('kanblendar-drag-over');
}
const id = e.dataTransfer.getData('text/plain');
const task = document.getElementById(id);
// Check if the drop target is a valid drop zone (time slot or non-timed tasks area)
let dropTarget = e.target;
if (dropTarget.classList.contains('kanblendar-task')) {
dropTarget = dropTarget.parentElement;
} else if (dropTarget.parentElement.classList.contains('kanblendar-task')) {
dropTarget = dropTarget.parentElement.parentElement;
}
if (dropTarget.classList.contains('kanblendar-time-slot') || dropTarget.classList.contains('kanblendar-non-timed-tasks')) {
dropTarget.appendChild(task);
}
// Update the task's due time if dropped in a time slot and the current due time is not valid for that slot
if (dropTarget.classList.contains('kanblendar-time-slot')) {
const startTime = dropTarget.dataset.startTime;
const parentColumn = dropTarget.closest('.kanblendar-column').id;
const slotStartTime = new Date(`${this.config.currentDate}T${startTime}:00`);
const slotEndTime = this.addMinutes(slotStartTime, this.config.interval);
if (startTime) {
const dueTime = new Date(task.dataset.dueTime);
if (!(dueTime >= slotStartTime && dueTime <= slotEndTime)) {
task.dataset.dueTime = slotStartTime.toISOString();
}
} else {
task.dataset.dueTime = slotStartTime.toISOString();
}
// Update the task's column attribute
task.dataset.column = parentColumn.replace('-tasks', '');
} else if (dropTarget.classList.contains('kanblendar-non-timed-tasks')) {
const parentColumn = dropTarget.closest('.kanblendar-column').id;
task.dataset.column = parentColumn.replace('-tasks', '');
task.dataset.dueTime = null;
}
// Update the task in the tasks map
this.tasks.get(task.id).column = task.dataset.column;
this.tasks.get(task.id).dueTime = task.dataset.dueTime;
this.adjustTimeSlotHeights(); // Adjust heights after dropping a task
this.emitTaskChangedEvent('move', task);
}
openModal(task = null) {
this.currentTask = task;
if (task) {
document.getElementById('kanblendar-modalTitle').innerText = 'Edit Task';
document.getElementById('kanblendar-taskTitle').value = task.querySelector('.kanblendar-task-title').innerText;
document.getElementById('kanblendar-taskDescription').value = task.querySelector('.kanblendar-task-desc').innerText;
// Convert task's dueTime to a valid datetime-local format
const dueTime = task.dataset.dueTime ? new Date(task.dataset.dueTime).toISOString().slice(0, 16) : '';
document.getElementById('kanblendar-taskDueTime').value = dueTime;
document.getElementById('kanblendar-taskColumn').value = task.dataset.column || '';
document.getElementById('kanblendar-taskNotify').value = task.dataset.notifyBefore || '';
this.deleteTaskBtn.style.display = 'block'; // Show delete button when editing
} else {
document.getElementById('kanblendar-modalTitle').innerText = 'Create Task';
this.taskForm.reset();
this.deleteTaskBtn.style.display = 'none'; // Hide delete button when creating
}
this.taskModal.style.display = 'flex';
}
closeModalFunc() {
this.taskModal.style.display = 'none';
}
saveTask(event) {
event.preventDefault();
const title = document.getElementById('kanblendar-taskTitle').value;
const description = document.getElementById('kanblendar-taskDescription').value;
const dueTime = document.getElementById('kanblendar-taskDueTime').value || null;
const column = document.getElementById('kanblendar-taskColumn').value;
const notifyBefore = parseInt(document.getElementById('kanblendar-taskNotify').value, 10);
let newTask = null;
if (this.currentTask) {
this.currentTask.querySelector('.kanblendar-task-title').innerText = title;
this.currentTask.querySelector('.kanblendar-task-desc').innerText = description;
this.currentTask.dataset.dueTime = dueTime;
this.currentTask.dataset.column = column;
this.currentTask.dataset.notifyBefore = notifyBefore;
this.moveTaskToColumn(this.currentTask, column);
if (dueTime) {
this.updateTaskLocation(this.currentTask, dueTime);
} else {
this.moveTaskToNonTimedSection(this.currentTask, column);
}
this.cancelNotification(this.currentTask.id);
this.emitTaskChangedEvent('edit', this.currentTask);
} else {
newTask = this.createTaskElement(title, description, dueTime, column, notifyBefore);
this.moveTaskToColumn(newTask, column);
if (dueTime) {
this.updateTaskLocation(newTask, dueTime);
} else {
this.moveTaskToNonTimedSection(newTask, column);
}
this.emitTaskChangedEvent('create', newTask);
}
if (dueTime && notifyBefore >= 0) {
this.scheduleNotification(title, description, new Date(dueTime), notifyBefore, this.currentTask ? this.currentTask.id : newTask.id);
}
this.adjustTimeSlotHeights(); // Adjust heights after saving a task
this.closeModalFunc();
}
moveTaskToNonTimedSection(task, column) {
const columnTasks = document.getElementById(`${column}-tasks`);
if (columnTasks) {
columnTasks.appendChild(task);
}
}
createTaskElement(title, description, dueTime, column, notifyBefore, id = null) {
const taskId = id || `task-${Date.now()}`;
const newTask = document.createElement('div');
newTask.classList.add('kanblendar-task');
newTask.setAttribute('draggable', 'true');
newTask.setAttribute('id', taskId);
newTask.dataset.dueTime = dueTime;
newTask.dataset.column = column;
newTask.dataset.notifyBefore = notifyBefore;
newTask.innerHTML = `
<div class="kanblendar-task-title">${title}</div>
<div class="kanblendar-task-desc">${description}</div>
`;
newTask.addEventListener('dragstart', (e) => this.dragStart(e));
newTask.addEventListener('click', (e) => this.openModal(newTask)); // Add click event to open modal for editing
this.tasks.set(taskId, { title, description, dueTime, column, notifyBefore });
return newTask;
}
moveTaskToColumn(task, column) {
const columnTasks = document.getElementById(`${column}-tasks`);
if (columnTasks) {
columnTasks.appendChild(task);
}
}
updateTaskLocation(task, dueTime) {
const taskDate = new Date(dueTime).toISOString().split('T')[0];
if (taskDate === this.config.currentDate) {
const taskTime = new Date(dueTime).toTimeString().slice(0, 5); // HH:MM format
let placedInTimeSlot = false;
document.querySelectorAll(`#${task.dataset.column} .kanblendar-time-slot`).forEach(timeSlotElement => {
const startTime = timeSlotElement.dataset.startTime;
const endTime = this.addMinutes(this.parseTime(startTime), this.config.interval).toTimeString().slice(0, 5);
if (taskTime >= startTime && taskTime < endTime) {
timeSlotElement.appendChild(task);
placedInTimeSlot = true;
}
});
if (!placedInTimeSlot) {
const columnTasks = document.getElementById(`${task.dataset.column}-tasks`);
columnTasks.appendChild(task);
}
} else {
const columnTasks = document.getElementById(`${task.dataset.column}-tasks`);
columnTasks.appendChild(task);
}
}
deleteTask(task) {
this.cancelNotification(task.id);
task.remove();
this.tasks.delete(task.id);
this.adjustTimeSlotHeights(); // Adjust heights after deleting a task
this.emitTaskChangedEvent('delete', task);
}
scheduleNotification(title, description, dueTime, notifyBefore, taskId) {
const notifyTime = new Date(dueTime.getTime() - notifyBefore * 60000);
const now = new Date();
if (notifyTime > now) {
const timeout = notifyTime.getTime() - now.getTime();
const timeoutId = setTimeout(() => {
this.showNotification(title, description);
this.notificationTimeouts.delete(taskId);
}, timeout);
this.notificationTimeouts.set(taskId, timeoutId);
}
}
cancelNotification(taskId) {
if (this.notificationTimeouts.has(taskId)) {
clearTimeout(this.notificationTimeouts.get(taskId));
this.notificationTimeouts.delete(taskId);
}
}
showNotification(title, description) {
if (Notification.permission === 'granted') {
new Notification(title, { body: description });
} else {
alert(`Reminder: ${title}\n${description}`);
}
}
requestNotificationPermission() {
if ('Notification' in window) {
Notification.requestPermission().then(permission => {
if (permission === 'granted') {
console.log('Notification permission granted.');
} else {
console.log('Notification permission denied.');
}
});
}
}
adjustTimeSlotHeights() {
const timeSlotMap = new Map();
let maxNonTimedHeight = 0;
// Collect heights of each time slot
document.querySelectorAll('.kanblendar-time-slot').forEach(slot => {
const startTime = slot.dataset.startTime;
slot.style.height = 'auto'; // Remove any explicitly set height to get the actual height
const height = slot.offsetHeight;
if (!timeSlotMap.has(startTime)) {
timeSlotMap.set(startTime, height);
} else {
const maxHeight = Math.max(timeSlotMap.get(startTime), height);
timeSlotMap.set(startTime, maxHeight);
}
});
// Collect heights of non-timed tasks sections
document.querySelectorAll('.kanblendar-non-timed-tasks').forEach(section => {
section.style.height = 'auto'; // Remove any explicitly set height to get the actual height
const height = section.offsetHeight;
maxNonTimedHeight = Math.max(maxNonTimedHeight, height);
});
// Apply the maximum height to all corresponding time slots
document.querySelectorAll('.kanblendar-time-slot').forEach(slot => {
const startTime = slot.dataset.startTime;
slot.style.height = `${timeSlotMap.get(startTime)}px`;
});
// Apply the maximum height to all non-timed tasks sections
document.querySelectorAll('.kanblendar-non-timed-tasks').forEach(section => {
section.style.height = `${maxNonTimedHeight}px`;
});
}
serialize() {
const tasksArray = Array.from(this.tasks.entries()).map(([id, task]) => ({
id,
title: task.title,
description: task.description,
dueTime: task.dueTime,
column: task.column,
notifyBefore: task.notifyBefore
}));
return JSON.stringify(tasksArray);
}
deserialize(serializedData) {
const tasksArray = JSON.parse(serializedData);
tasksArray.forEach(taskData => {
const { id, title, description, dueTime, column, notifyBefore } = taskData;
const taskElement = this.createTaskElement(title, description, dueTime, column, notifyBefore, id);
this.tasks.set(id, { title, description, dueTime, column, notifyBefore });
this.moveTaskToColumn(taskElement, column);
if (dueTime) {
this.updateTaskLocation(taskElement, dueTime);
} else {
this.moveTaskToNonTimedSection(taskElement, column);
}
});
this.updateTaskDisplay();
this.adjustTimeSlotHeights();
}
emitTaskChangedEvent(action, task) {
const event = new CustomEvent('taskChanged', {
detail: {
action,
taskId: task.id
}
});
document.dispatchEvent(event);
}
updateTaskDisplay() {
const now = new Date();
now.setHours(0, 0, 0, 0);
// Iterate over each task and check its column and due date
this.tasks.forEach((task, taskId) => {
const taskElement = document.getElementById(taskId);
// Parse the task's due time
const taskDueDate = task.dueTime ? new Date(task.dueTime) : null;
if (task.column === 'done' && taskDueDate && taskDueDate < now) {
// Hide tasks that are done and before today's date
taskElement.style.display = 'none';
} else {
// Show all other tasks
taskElement.style.display = 'block';
}
});
}
}