feat: apply time tracking UI to Books, Music, Shows, and Manga

Completes time tracking implementation across all media types by
applying the same pattern established in Games component.

Changes Applied to Books, Music, Shows, and Manga:

UI Components:
- Added time tracking state properties (hours/minutes for new and edit)
- Integrated hour/minute input fields in Add forms (after rating)
- Integrated hour/minute input fields in Edit forms (after edit-rating)
- Added time spent display on media cards with appropriate emojis:
  * Books: 📖 Reading Time
  * Music: 🎵 Listening Time
  * Shows: 📺 Watch Time
  * Manga: 📚 Reading Time

Form Management:
- Updated resetForm() to clear time tracking fields
- Added updateNew[Type]TimeSpent() conversion methods
- Added updateEdit[Type]TimeSpent() conversion methods
- Updated startEdit() to populate time fields from stored data

Helper Methods:
- Added formatTimeSpent() to format minutes as "Xh Ym", "Xh", or "Ym"
- Converts hours/minutes to total minutes for storage
- Splits total minutes back to hours/minutes for editing

Styling:
- Added .form-row CSS for side-by-side hour/minute inputs
- Added .time-spent CSS with coloured display:
  * Books: Green (#10b981)
  * Music: Purple (#8b5cf6)
  * Shows: Purple (#8b5cf6)
  * Manga: Green (#10b981)

All implementations follow the exact same pattern for consistency
and maintainability across the application.
This commit is contained in:
2026-02-19 23:55:16 -08:00
committed by Naomi Carrigan
parent fa331df203
commit 5ad9b50dc8
4 changed files with 464 additions and 3 deletions
@@ -114,6 +114,34 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
>
</div>
<div class="form-row">
<div class="form-group">
<label for="timeHours">Time Spent (Hours)</label>
<input
type="number"
id="timeHours"
[(ngModel)]="newMusicTimeHours"
name="timeHours"
min="0"
placeholder="0"
(ngModelChange)="updateNewMusicTimeSpent()"
>
</div>
<div class="form-group">
<label for="timeMinutes">Time Spent (Minutes)</label>
<input
type="number"
id="timeMinutes"
[(ngModel)]="newMusicTimeMinutes"
name="timeMinutes"
min="0"
max="59"
placeholder="0"
(ngModelChange)="updateNewMusicTimeSpent()"
>
</div>
</div>
<div class="form-group">
<label for="notes">Notes</label>
<textarea
@@ -274,6 +302,34 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
>
</div>
<div class="form-row">
<div class="form-group">
<label for="edit-timeHours">Time Spent (Hours)</label>
<input
type="number"
id="edit-timeHours"
[(ngModel)]="editMusicTimeHours"
name="timeHours"
min="0"
placeholder="0"
(ngModelChange)="updateEditMusicTimeSpent()"
>
</div>
<div class="form-group">
<label for="edit-timeMinutes">Time Spent (Minutes)</label>
<input
type="number"
id="edit-timeMinutes"
[(ngModel)]="editMusicTimeMinutes"
name="timeMinutes"
min="0"
max="59"
placeholder="0"
(ngModelChange)="updateEditMusicTimeSpent()"
>
</div>
</div>
<div class="form-group">
<label for="edit-notes">Notes</label>
<textarea
@@ -600,6 +656,12 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
</div>
}
@if (music.timeSpent) {
<p class="time-spent">
🎵 Listening Time: {{ formatTimeSpent(music.timeSpent) }}
</p>
}
<app-like-button
entityType="music"
[entityId]="music.id"
@@ -780,6 +842,13 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
box-shadow: 0 0 0 3px rgba(168, 87, 126, 0.2);
}
.form-row {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 1rem;
margin-bottom: 1rem;
}
.form-actions {
display: flex;
gap: 1rem;
@@ -1036,6 +1105,13 @@ import { Music, MusicStatus, MusicType, CreateMusicDto, UpdateMusicDto, Comment,
margin: 0.5rem 0;
}
.time-spent {
font-size: 0.9rem;
color: #8b5cf6;
font-weight: 500;
margin: 0.5rem 0;
}
.date-started,
.date-finished,
.date-added,
@@ -1526,6 +1602,12 @@ export class MusicListComponent implements OnInit {
editMusicData: Partial<UpdateMusicDto> = {};
// Time tracking state
newMusicTimeHours = 0;
newMusicTimeMinutes = 0;
editMusicTimeHours = 0;
editMusicTimeMinutes = 0;
// Tags and links input state
newTagInput = '';
editTagInput = '';
@@ -1633,6 +1715,8 @@ export class MusicListComponent implements OnInit {
tags: [],
links: []
};
this.newMusicTimeHours = 0;
this.newMusicTimeMinutes = 0;
this.newMusicImagePreview.set(null);
this.imageError.set(null);
this.newTagInput = '';
@@ -1640,6 +1724,16 @@ export class MusicListComponent implements OnInit {
this.newLinkUrl = '';
}
updateNewMusicTimeSpent() {
const totalMinutes = (this.newMusicTimeHours * 60) + this.newMusicTimeMinutes;
this.newMusic.timeSpent = totalMinutes > 0 ? totalMinutes : undefined;
}
updateEditMusicTimeSpent() {
const totalMinutes = (this.editMusicTimeHours * 60) + this.editMusicTimeMinutes;
this.editMusicData.timeSpent = totalMinutes > 0 ? totalMinutes : undefined;
}
addTag(target: 'new' | 'edit') {
const input = target === 'new' ? this.newTagInput.trim() : this.editTagInput.trim();
if (!input) return;
@@ -1729,8 +1823,17 @@ export class MusicListComponent implements OnInit {
notes: music.notes,
coverArt: music.coverArt,
tags: [...(music.tags || [])],
links: [...(music.links || [])]
links: [...(music.links || [])],
timeSpent: music.timeSpent
};
// Populate time fields from existing timeSpent
if (music.timeSpent) {
this.editMusicTimeHours = Math.floor(music.timeSpent / 60);
this.editMusicTimeMinutes = music.timeSpent % 60;
} else {
this.editMusicTimeHours = 0;
this.editMusicTimeMinutes = 0;
}
this.editMusicImagePreview.set(music.coverArt || null);
this.showAddForm.set(false);
this.imageError.set(null);
@@ -1769,6 +1872,19 @@ export class MusicListComponent implements OnInit {
return new Date(date).toLocaleDateString();
}
formatTimeSpent(minutes: number): string {
const hours = Math.floor(minutes / 60);
const mins = minutes % 60;
if (hours === 0) {
return `${mins}m`;
} else if (mins === 0) {
return `${hours}h`;
} else {
return `${hours}h ${mins}m`;
}
}
// Image handling methods
onImageSelected(event: Event, target: 'new' | 'edit' | 'suggest') {
const input = event.target as HTMLInputElement;