feat: US-008 - Frontend - Create habit modal with all options

This commit is contained in:
Echo
2026-02-10 16:36:52 +00:00
parent b99133de79
commit 60bf92a610
2 changed files with 848 additions and 3 deletions

View File

@@ -249,6 +249,279 @@
.priority-low {
background: var(--success);
}
/* Modal overlay */
.modal-overlay {
display: none;
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.7);
z-index: 1000;
align-items: center;
justify-content: center;
padding: var(--space-4);
}
.modal-overlay.active {
display: flex;
}
.modal {
background: var(--bg-surface);
border-radius: var(--radius-lg);
max-width: 600px;
width: 100%;
max-height: 90vh;
overflow-y: auto;
box-shadow: var(--shadow-lg);
}
.modal-header {
display: flex;
justify-content: space-between;
align-items: center;
padding: var(--space-4);
border-bottom: 1px solid var(--border);
}
.modal-title {
font-size: var(--text-lg);
font-weight: 600;
color: var(--text-primary);
}
.modal-close {
background: none;
border: none;
color: var(--text-muted);
cursor: pointer;
padding: var(--space-1);
display: flex;
align-items: center;
justify-content: center;
border-radius: var(--radius-md);
transition: all var(--transition-base);
}
.modal-close:hover {
background: var(--bg-hover);
color: var(--text-primary);
}
.modal-close svg {
width: 20px;
height: 20px;
}
.modal-body {
padding: var(--space-4);
}
.modal-footer {
display: flex;
justify-content: flex-end;
gap: var(--space-2);
padding: var(--space-4);
border-top: 1px solid var(--border);
}
/* Form fields */
.form-field {
margin-bottom: var(--space-4);
}
.form-label {
display: block;
font-size: var(--text-sm);
font-weight: 500;
color: var(--text-primary);
margin-bottom: var(--space-1);
}
.form-label.required::after {
content: '*';
color: var(--error);
margin-left: var(--space-1);
}
.form-input,
.form-select,
.form-textarea {
width: 100%;
padding: var(--space-2);
border: 1px solid var(--border);
border-radius: var(--radius-md);
background: var(--bg-base);
color: var(--text-primary);
font-size: var(--text-sm);
transition: all var(--transition-base);
}
.form-input:focus,
.form-select:focus,
.form-textarea:focus {
outline: none;
border-color: var(--accent);
box-shadow: 0 0 0 3px var(--accent-muted);
}
.form-textarea {
min-height: 80px;
resize: vertical;
font-family: inherit;
}
/* Color picker */
.color-picker-swatches {
display: grid;
grid-template-columns: repeat(8, 1fr);
gap: var(--space-2);
margin-bottom: var(--space-2);
}
.color-swatch {
width: 100%;
aspect-ratio: 1;
border: 2px solid transparent;
border-radius: var(--radius-md);
cursor: pointer;
transition: all var(--transition-base);
}
.color-swatch:hover {
transform: scale(1.1);
}
.color-swatch.selected {
border-color: var(--text-primary);
box-shadow: 0 0 0 2px var(--bg-surface), 0 0 0 4px var(--accent);
}
/* Icon picker */
.icon-picker-grid {
display: grid;
grid-template-columns: repeat(6, 1fr);
gap: var(--space-2);
max-height: 250px;
overflow-y: auto;
padding: var(--space-2);
border: 1px solid var(--border);
border-radius: var(--radius-md);
background: var(--bg-base);
}
.icon-option {
aspect-ratio: 1;
display: flex;
align-items: center;
justify-content: center;
border: 2px solid transparent;
border-radius: var(--radius-md);
cursor: pointer;
transition: all var(--transition-base);
background: var(--bg-surface);
}
.icon-option:hover {
background: var(--bg-hover);
transform: scale(1.1);
}
.icon-option.selected {
border-color: var(--accent);
background: var(--accent-muted);
}
.icon-option svg {
width: 20px;
height: 20px;
color: var(--text-primary);
}
/* Frequency params */
.frequency-params {
margin-top: var(--space-2);
}
.day-checkboxes {
display: grid;
grid-template-columns: repeat(7, 1fr);
gap: var(--space-2);
}
.day-checkbox-label {
display: flex;
flex-direction: column;
align-items: center;
gap: var(--space-1);
padding: var(--space-2);
border: 1px solid var(--border);
border-radius: var(--radius-md);
cursor: pointer;
transition: all var(--transition-base);
font-size: var(--text-xs);
}
.day-checkbox-label:has(input:checked) {
background: var(--accent-muted);
border-color: var(--accent);
}
.day-checkbox-label input {
margin: 0;
}
/* Toast notification */
.toast {
position: fixed;
bottom: var(--space-4);
right: var(--space-4);
padding: var(--space-3) var(--space-4);
border-radius: var(--radius-md);
background: var(--bg-surface);
border: 1px solid var(--border);
box-shadow: var(--shadow-lg);
z-index: 2000;
display: flex;
align-items: center;
gap: var(--space-2);
animation: slideIn 0.3s ease-out;
}
@keyframes slideIn {
from {
transform: translateX(100%);
opacity: 0;
}
to {
transform: translateX(0);
opacity: 1;
}
}
.toast.success {
border-color: var(--success);
}
.toast.error {
border-color: var(--error);
}
.toast svg {
width: 20px;
height: 20px;
}
.toast.success svg {
color: var(--success);
}
.toast.error svg {
color: var(--error);
}
</style>
</head>
<body>
@@ -301,6 +574,91 @@
</div>
</main>
<!-- Add/Edit Habit Modal -->
<div id="habitModal" class="modal-overlay">
<div class="modal">
<div class="modal-header">
<h2 class="modal-title">Add Habit</h2>
<button class="modal-close" onclick="closeHabitModal()">
<i data-lucide="x"></i>
</button>
</div>
<div class="modal-body">
<form id="habitForm">
<!-- Name -->
<div class="form-field">
<label class="form-label required" for="habitName">Name</label>
<input type="text" id="habitName" class="form-input" maxlength="100" required>
</div>
<!-- Category -->
<div class="form-field">
<label class="form-label" for="habitCategory">Category</label>
<select id="habitCategory" class="form-select">
<option value="work">Work</option>
<option value="health">Health</option>
<option value="growth">Growth</option>
<option value="personal">Personal</option>
</select>
</div>
<!-- Color -->
<div class="form-field">
<label class="form-label">Color</label>
<div class="color-picker-swatches" id="colorSwatches"></div>
<input type="text" id="customColor" class="form-input" placeholder="#RRGGBB" pattern="^#[0-9A-Fa-f]{6}$">
</div>
<!-- Icon -->
<div class="form-field">
<label class="form-label">Icon</label>
<div class="icon-picker-grid" id="iconPicker"></div>
</div>
<!-- Priority -->
<div class="form-field">
<label class="form-label" for="habitPriority">Priority (1-100)</label>
<input type="number" id="habitPriority" class="form-input" min="1" max="100" value="50">
</div>
<!-- Notes -->
<div class="form-field">
<label class="form-label" for="habitNotes">Notes</label>
<textarea id="habitNotes" class="form-textarea"></textarea>
</div>
<!-- Frequency Type -->
<div class="form-field">
<label class="form-label" for="frequencyType">Frequency</label>
<select id="frequencyType" class="form-select" onchange="updateFrequencyParams()">
<option value="daily">Daily</option>
<option value="specific_days">Specific Days</option>
<option value="x_per_week">X times per week</option>
<option value="weekly">Weekly</option>
<option value="monthly">Monthly</option>
<option value="custom">Custom</option>
</select>
</div>
<!-- Frequency Params (conditional) -->
<div id="frequencyParams" class="frequency-params"></div>
<!-- Reminder Time -->
<div class="form-field">
<label class="form-label" for="reminderTime">Reminder Time (optional)</label>
<input type="time" id="reminderTime" class="form-input">
</div>
</form>
</div>
<div class="modal-footer">
<button type="button" class="btn" onclick="closeHabitModal()">Cancel</button>
<button type="submit" class="btn btn-primary" id="submitHabitBtn" onclick="submitHabitForm(event)">
<span id="submitBtnText">Create Habit</span>
</button>
</div>
</div>
</div>
<script>
// Theme management
function initTheme() {
@@ -471,9 +829,283 @@
return 'low';
}
// Show add habit modal (placeholder - full modal in next stories)
// Modal state
let selectedColor = '#3B82F6';
let selectedIcon = 'dumbbell';
// Preset colors
const presetColors = [
'#EF4444', '#F97316', '#F59E0B', '#10B981',
'#3B82F6', '#8B5CF6', '#EC4899', '#6B7280'
];
// Common icons
const commonIcons = [
'dumbbell', 'moon', 'book', 'brain', 'heart', 'flame',
'star', 'target', 'trophy', 'coffee', 'music', 'camera',
'zap', 'sun', 'droplet', 'leaf', 'feather', 'pencil',
'smile', 'watch', 'footprints', 'activity', 'battery', 'headphones',
'utensils', 'apple', 'pizza', 'glass-water', 'pill', 'stethoscope',
'briefcase', 'laptop', 'smartphone', 'mail', 'calendar', 'clock'
];
// Show add habit modal
function showAddHabitModal() {
alert('Add Habit modal - coming in next story!');
const modal = document.getElementById('habitModal');
const form = document.getElementById('habitForm');
// Reset form
form.reset();
selectedColor = '#3B82F6';
selectedIcon = 'dumbbell';
// Initialize color picker
initColorPicker();
// Initialize icon picker
initIconPicker();
// Update frequency params for initial selection
updateFrequencyParams();
// Show modal
modal.classList.add('active');
lucide.createIcons();
}
// Close habit modal
function closeHabitModal() {
const modal = document.getElementById('habitModal');
modal.classList.remove('active');
}
// Close modal when clicking outside
document.addEventListener('click', (e) => {
const modal = document.getElementById('habitModal');
if (e.target === modal) {
closeHabitModal();
}
});
// Initialize color picker
function initColorPicker() {
const swatchesContainer = document.getElementById('colorSwatches');
swatchesContainer.innerHTML = presetColors.map(color =>
`<div class="color-swatch ${color === selectedColor ? 'selected' : ''}"
style="background-color: ${color}"
onclick="selectColor('${color}')"></div>`
).join('');
// Handle custom color input
const customColorInput = document.getElementById('customColor');
customColorInput.addEventListener('input', (e) => {
const value = e.target.value;
if (/^#[0-9A-Fa-f]{6}$/.test(value)) {
selectColor(value);
}
});
}
// Select color
function selectColor(color) {
selectedColor = color;
// Update swatches
const swatches = document.querySelectorAll('.color-swatch');
swatches.forEach(swatch => {
if (swatch.style.backgroundColor === color || rgbToHex(swatch.style.backgroundColor) === color) {
swatch.classList.add('selected');
} else {
swatch.classList.remove('selected');
}
});
// Update custom color input if not a preset
if (!presetColors.includes(color)) {
document.getElementById('customColor').value = color;
}
}
// Convert RGB to Hex
function rgbToHex(rgb) {
const match = rgb.match(/^rgb\((\d+),\s*(\d+),\s*(\d+)\)$/);
if (!match) return rgb;
return '#' + [1, 2, 3].map(i => {
const hex = parseInt(match[i]).toString(16);
return hex.length === 1 ? '0' + hex : hex;
}).join('').toUpperCase();
}
// Initialize icon picker
function initIconPicker() {
const iconPickerContainer = document.getElementById('iconPicker');
iconPickerContainer.innerHTML = commonIcons.map(icon =>
`<div class="icon-option ${icon === selectedIcon ? 'selected' : ''}"
onclick="selectIcon('${icon}')">
<i data-lucide="${icon}"></i>
</div>`
).join('');
lucide.createIcons();
}
// Select icon
function selectIcon(icon) {
selectedIcon = icon;
// Update icon options
const iconOptions = document.querySelectorAll('.icon-option');
iconOptions.forEach((option, index) => {
if (commonIcons[index] === icon) {
option.classList.add('selected');
} else {
option.classList.remove('selected');
}
});
}
// Update frequency params based on selected type
function updateFrequencyParams() {
const frequencyType = document.getElementById('frequencyType').value;
const paramsContainer = document.getElementById('frequencyParams');
let html = '';
if (frequencyType === 'specific_days') {
html = `
<label class="form-label">Select Days</label>
<div class="day-checkboxes">
${['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'].map((day, index) => `
<label class="day-checkbox-label">
<input type="checkbox" name="day" value="${index}" checked>
<span>${day}</span>
</label>
`).join('')}
</div>
`;
} else if (frequencyType === 'x_per_week') {
html = `
<label class="form-label" for="xPerWeek">Times per week</label>
<input type="number" id="xPerWeek" class="form-input" min="1" max="7" value="3">
`;
} else if (frequencyType === 'custom') {
html = `
<label class="form-label" for="customInterval">Interval (days)</label>
<input type="number" id="customInterval" class="form-input" min="1" value="7">
`;
}
paramsContainer.innerHTML = html;
}
// Submit habit form
async function submitHabitForm(event) {
event.preventDefault();
const submitBtn = document.getElementById('submitHabitBtn');
const submitBtnText = document.getElementById('submitBtnText');
// Get form values
const name = document.getElementById('habitName').value.trim();
const category = document.getElementById('habitCategory').value;
const priority = parseInt(document.getElementById('habitPriority').value);
const notes = document.getElementById('habitNotes').value.trim();
const frequencyType = document.getElementById('frequencyType').value;
const reminderTime = document.getElementById('reminderTime').value;
// Validate name
if (!name) {
showToast('Please enter a habit name', 'error');
return;
}
if (name.length > 100) {
showToast('Habit name must be 100 characters or less', 'error');
return;
}
// Validate color
if (!/^#[0-9A-Fa-f]{6}$/.test(selectedColor)) {
showToast('Invalid color format', 'error');
return;
}
// Build frequency params
let frequencyParams = {};
if (frequencyType === 'specific_days') {
const checkedDays = Array.from(document.querySelectorAll('input[name="day"]:checked'))
.map(cb => parseInt(cb.value));
frequencyParams = { days: checkedDays };
} else if (frequencyType === 'x_per_week') {
const xPerWeek = parseInt(document.getElementById('xPerWeek').value);
frequencyParams = { count: xPerWeek };
} else if (frequencyType === 'custom') {
const interval = parseInt(document.getElementById('customInterval').value);
frequencyParams = { interval };
}
// Build habit object
const habitData = {
name,
category,
color: selectedColor,
icon: selectedIcon,
priority,
notes: notes || undefined,
frequency: {
type: frequencyType,
...frequencyParams
},
reminderTime: reminderTime || undefined
};
// Show loading state
submitBtn.disabled = true;
submitBtnText.textContent = 'Creating...';
try {
const response = await fetch('/echo/api/habits', {
method: 'POST',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify(habitData)
});
if (!response.ok) {
const error = await response.json();
throw new Error(error.error || `HTTP ${response.status}`);
}
// Success
showToast('Habit created successfully!', 'success');
closeHabitModal();
await loadHabits();
} catch (error) {
console.error('Failed to create habit:', error);
showToast('Failed to create habit: ' + error.message, 'error');
} finally {
submitBtn.disabled = false;
submitBtnText.textContent = 'Create Habit';
}
}
// Show toast notification
function showToast(message, type = 'success') {
const toast = document.createElement('div');
toast.className = `toast ${type}`;
toast.innerHTML = `
<i data-lucide="${type === 'success' ? 'check-circle' : 'alert-circle'}"></i>
<span>${escapeHtml(message)}</span>
`;
document.body.appendChild(toast);
lucide.createIcons();
setTimeout(() => {
toast.remove();
}, 3000);
}
// Show edit habit modal (placeholder)