Files
workoutsplus/WorkoutsPlus/Exercise/ExerciseLibrary.swift
T

106 lines
2.4 KiB
Swift

//
// ExerciseLibraryView.swift
// WorkoutsPlus
//
// Created by Felix Förtsch on 10.08.24.
//
import SwiftUI
import SwiftData
struct ExerciseLibrary: View {
@Environment(\.modelContext) private var modelContext
@Query(sort: \Exercise.name) private var exercises: [Exercise]
@State private var newExercise: Exercise = Exercise("")
@State private var newExerciseName: String = ""
@State private var isAddingExercise: Bool = false
@FocusState private var isInputFieldFocused: Bool
@State private var searchText: String = ""
var filteredItems: [Exercise] {
if searchText.isEmpty {
return exercises
} else {
return exercises.filter { $0.name.localizedCaseInsensitiveContains(searchText) }
}
}
// TODO: Add search bar to the top
var body: some View {
Group {
List {
ForEach(filteredItems) { exercise in
NavigationLink {
ExerciseDetail(exercise: exercise)
} label: {
Text(exercise.name)
}
}
.onDelete(perform: deleteExercise)
if isAddingExercise {
TextField("New Exercise", text: $newExerciseName, onCommit: {
newExercise.name = newExerciseName
save(exercise: newExercise)
isAddingExercise = false
})
.textInputAutocapitalization(.words)
.focused($isInputFieldFocused)
}
AddItemButton(label: "Exercise", action: addExercise)
}
.searchable(text: $searchText)
}
.navigationTitle("Exercises")
.toolbar {
ToolbarItem {
EditButton()
}
}
}
private func addExercise() {
withAnimation {
newExercise = Exercise("")
newExerciseName = ""
isAddingExercise = true
isInputFieldFocused = true
}
}
private func save(exercise: Exercise) {
if !exercise.name.isEmpty {
modelContext.insert(exercise)
try? modelContext.save()
}
}
private func deleteExercise(offsets: IndexSet) {
withAnimation {
for index in offsets {
modelContext.delete(exercises[index])
}
try? modelContext.save()
}
}
}
#Preview("With Sample Data") {
NavigationStack {
ExerciseLibrary()
}
.modelContainer(SampleData.shared.modelContainer)
}
#Preview("Empty Database") {
NavigationStack {
ExerciseLibrary()
}
.modelContainer(for: WorkoutItem.self, inMemory: true)
}