it works.
This commit is contained in:
parent
a4ecd54f1e
commit
534ed4718f
|
@ -296,12 +296,14 @@
|
||||||
ENABLE_PREVIEWS = YES;
|
ENABLE_PREVIEWS = YES;
|
||||||
GENERATE_INFOPLIST_FILE = YES;
|
GENERATE_INFOPLIST_FILE = YES;
|
||||||
INFOPLIST_FILE = "Gas Man/Info.plist";
|
INFOPLIST_FILE = "Gas Man/Info.plist";
|
||||||
|
INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.travel";
|
||||||
"INFOPLIST_KEY_UIApplicationSceneManifest_Generation[sdk=iphoneos*]" = YES;
|
"INFOPLIST_KEY_UIApplicationSceneManifest_Generation[sdk=iphoneos*]" = YES;
|
||||||
"INFOPLIST_KEY_UIApplicationSceneManifest_Generation[sdk=iphonesimulator*]" = YES;
|
"INFOPLIST_KEY_UIApplicationSceneManifest_Generation[sdk=iphonesimulator*]" = YES;
|
||||||
"INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents[sdk=iphoneos*]" = YES;
|
"INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents[sdk=iphoneos*]" = YES;
|
||||||
"INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents[sdk=iphonesimulator*]" = YES;
|
"INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents[sdk=iphonesimulator*]" = YES;
|
||||||
"INFOPLIST_KEY_UILaunchScreen_Generation[sdk=iphoneos*]" = YES;
|
"INFOPLIST_KEY_UILaunchScreen_Generation[sdk=iphoneos*]" = YES;
|
||||||
"INFOPLIST_KEY_UILaunchScreen_Generation[sdk=iphonesimulator*]" = YES;
|
"INFOPLIST_KEY_UILaunchScreen_Generation[sdk=iphonesimulator*]" = YES;
|
||||||
|
INFOPLIST_KEY_UIStatusBarStyle = UIStatusBarStyleDarkContent;
|
||||||
"INFOPLIST_KEY_UIStatusBarStyle[sdk=iphoneos*]" = UIStatusBarStyleDefault;
|
"INFOPLIST_KEY_UIStatusBarStyle[sdk=iphoneos*]" = UIStatusBarStyleDefault;
|
||||||
"INFOPLIST_KEY_UIStatusBarStyle[sdk=iphonesimulator*]" = UIStatusBarStyleDefault;
|
"INFOPLIST_KEY_UIStatusBarStyle[sdk=iphonesimulator*]" = UIStatusBarStyleDefault;
|
||||||
INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight";
|
INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight";
|
||||||
|
@ -309,15 +311,16 @@
|
||||||
IPHONEOS_DEPLOYMENT_TARGET = 18.1;
|
IPHONEOS_DEPLOYMENT_TARGET = 18.1;
|
||||||
LD_RUNPATH_SEARCH_PATHS = "@executable_path/Frameworks";
|
LD_RUNPATH_SEARCH_PATHS = "@executable_path/Frameworks";
|
||||||
"LD_RUNPATH_SEARCH_PATHS[sdk=macosx*]" = "@executable_path/../Frameworks";
|
"LD_RUNPATH_SEARCH_PATHS[sdk=macosx*]" = "@executable_path/../Frameworks";
|
||||||
MACOSX_DEPLOYMENT_TARGET = 14.7;
|
MACOSX_DEPLOYMENT_TARGET = 14.6;
|
||||||
MARKETING_VERSION = 1.0;
|
MARKETING_VERSION = 1.0;
|
||||||
PRODUCT_BUNDLE_IDENTIFIER = "pro.thelinux.Gas-Man";
|
PRODUCT_BUNDLE_IDENTIFIER = "pro.thelinux.Gas-Man";
|
||||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||||
SDKROOT = auto;
|
SDKROOT = auto;
|
||||||
SUPPORTED_PLATFORMS = "iphoneos iphonesimulator macosx xros xrsimulator";
|
SUPPORTED_PLATFORMS = "iphoneos iphonesimulator macosx";
|
||||||
|
SUPPORTS_MACCATALYST = NO;
|
||||||
SWIFT_EMIT_LOC_STRINGS = YES;
|
SWIFT_EMIT_LOC_STRINGS = YES;
|
||||||
SWIFT_VERSION = 5.0;
|
SWIFT_VERSION = 5.0;
|
||||||
TARGETED_DEVICE_FAMILY = "1,2,7";
|
TARGETED_DEVICE_FAMILY = "1,2";
|
||||||
XROS_DEPLOYMENT_TARGET = 2.1;
|
XROS_DEPLOYMENT_TARGET = 2.1;
|
||||||
};
|
};
|
||||||
name = Debug;
|
name = Debug;
|
||||||
|
@ -336,12 +339,14 @@
|
||||||
ENABLE_PREVIEWS = YES;
|
ENABLE_PREVIEWS = YES;
|
||||||
GENERATE_INFOPLIST_FILE = YES;
|
GENERATE_INFOPLIST_FILE = YES;
|
||||||
INFOPLIST_FILE = "Gas Man/Info.plist";
|
INFOPLIST_FILE = "Gas Man/Info.plist";
|
||||||
|
INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.travel";
|
||||||
"INFOPLIST_KEY_UIApplicationSceneManifest_Generation[sdk=iphoneos*]" = YES;
|
"INFOPLIST_KEY_UIApplicationSceneManifest_Generation[sdk=iphoneos*]" = YES;
|
||||||
"INFOPLIST_KEY_UIApplicationSceneManifest_Generation[sdk=iphonesimulator*]" = YES;
|
"INFOPLIST_KEY_UIApplicationSceneManifest_Generation[sdk=iphonesimulator*]" = YES;
|
||||||
"INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents[sdk=iphoneos*]" = YES;
|
"INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents[sdk=iphoneos*]" = YES;
|
||||||
"INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents[sdk=iphonesimulator*]" = YES;
|
"INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents[sdk=iphonesimulator*]" = YES;
|
||||||
"INFOPLIST_KEY_UILaunchScreen_Generation[sdk=iphoneos*]" = YES;
|
"INFOPLIST_KEY_UILaunchScreen_Generation[sdk=iphoneos*]" = YES;
|
||||||
"INFOPLIST_KEY_UILaunchScreen_Generation[sdk=iphonesimulator*]" = YES;
|
"INFOPLIST_KEY_UILaunchScreen_Generation[sdk=iphonesimulator*]" = YES;
|
||||||
|
INFOPLIST_KEY_UIStatusBarStyle = UIStatusBarStyleDarkContent;
|
||||||
"INFOPLIST_KEY_UIStatusBarStyle[sdk=iphoneos*]" = UIStatusBarStyleDefault;
|
"INFOPLIST_KEY_UIStatusBarStyle[sdk=iphoneos*]" = UIStatusBarStyleDefault;
|
||||||
"INFOPLIST_KEY_UIStatusBarStyle[sdk=iphonesimulator*]" = UIStatusBarStyleDefault;
|
"INFOPLIST_KEY_UIStatusBarStyle[sdk=iphonesimulator*]" = UIStatusBarStyleDefault;
|
||||||
INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight";
|
INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight";
|
||||||
|
@ -349,15 +354,16 @@
|
||||||
IPHONEOS_DEPLOYMENT_TARGET = 18.1;
|
IPHONEOS_DEPLOYMENT_TARGET = 18.1;
|
||||||
LD_RUNPATH_SEARCH_PATHS = "@executable_path/Frameworks";
|
LD_RUNPATH_SEARCH_PATHS = "@executable_path/Frameworks";
|
||||||
"LD_RUNPATH_SEARCH_PATHS[sdk=macosx*]" = "@executable_path/../Frameworks";
|
"LD_RUNPATH_SEARCH_PATHS[sdk=macosx*]" = "@executable_path/../Frameworks";
|
||||||
MACOSX_DEPLOYMENT_TARGET = 14.7;
|
MACOSX_DEPLOYMENT_TARGET = 14.6;
|
||||||
MARKETING_VERSION = 1.0;
|
MARKETING_VERSION = 1.0;
|
||||||
PRODUCT_BUNDLE_IDENTIFIER = "pro.thelinux.Gas-Man";
|
PRODUCT_BUNDLE_IDENTIFIER = "pro.thelinux.Gas-Man";
|
||||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||||
SDKROOT = auto;
|
SDKROOT = auto;
|
||||||
SUPPORTED_PLATFORMS = "iphoneos iphonesimulator macosx xros xrsimulator";
|
SUPPORTED_PLATFORMS = "iphoneos iphonesimulator macosx";
|
||||||
|
SUPPORTS_MACCATALYST = NO;
|
||||||
SWIFT_EMIT_LOC_STRINGS = YES;
|
SWIFT_EMIT_LOC_STRINGS = YES;
|
||||||
SWIFT_VERSION = 5.0;
|
SWIFT_VERSION = 5.0;
|
||||||
TARGETED_DEVICE_FAMILY = "1,2,7";
|
TARGETED_DEVICE_FAMILY = "1,2";
|
||||||
XROS_DEPLOYMENT_TARGET = 2.1;
|
XROS_DEPLOYMENT_TARGET = 2.1;
|
||||||
};
|
};
|
||||||
name = Release;
|
name = Release;
|
||||||
|
|
|
@ -0,0 +1,5 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict/>
|
||||||
|
</plist>
|
|
@ -0,0 +1,14 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>BuildLocationStyle</key>
|
||||||
|
<string>UseAppPreferences</string>
|
||||||
|
<key>CustomBuildLocationType</key>
|
||||||
|
<string>RelativeToDerivedData</string>
|
||||||
|
<key>DerivedDataLocationStyle</key>
|
||||||
|
<string>Default</string>
|
||||||
|
<key>ShowSharedSchemesAutomaticallyEnabled</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
|
@ -0,0 +1,102 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<Scheme
|
||||||
|
LastUpgradeVersion = "1610"
|
||||||
|
version = "1.7">
|
||||||
|
<BuildAction
|
||||||
|
parallelizeBuildables = "YES"
|
||||||
|
buildImplicitDependencies = "YES"
|
||||||
|
buildArchitectures = "Automatic">
|
||||||
|
<BuildActionEntries>
|
||||||
|
<BuildActionEntry
|
||||||
|
buildForTesting = "YES"
|
||||||
|
buildForRunning = "YES"
|
||||||
|
buildForProfiling = "YES"
|
||||||
|
buildForArchiving = "YES"
|
||||||
|
buildForAnalyzing = "YES">
|
||||||
|
<BuildableReference
|
||||||
|
BuildableIdentifier = "primary"
|
||||||
|
BlueprintIdentifier = "3D34444F2D889F7D00AA3172"
|
||||||
|
BuildableName = "Gas Man.app"
|
||||||
|
BlueprintName = "Gas Man"
|
||||||
|
ReferencedContainer = "container:Gas Man.xcodeproj">
|
||||||
|
</BuildableReference>
|
||||||
|
</BuildActionEntry>
|
||||||
|
</BuildActionEntries>
|
||||||
|
</BuildAction>
|
||||||
|
<TestAction
|
||||||
|
buildConfiguration = "Debug"
|
||||||
|
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
|
||||||
|
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
|
||||||
|
shouldUseLaunchSchemeArgsEnv = "YES"
|
||||||
|
shouldAutocreateTestPlan = "YES">
|
||||||
|
<Testables>
|
||||||
|
<TestableReference
|
||||||
|
skipped = "NO"
|
||||||
|
parallelizable = "YES">
|
||||||
|
<BuildableReference
|
||||||
|
BuildableIdentifier = "primary"
|
||||||
|
BlueprintIdentifier = "3D3444662D889F7F00AA3172"
|
||||||
|
BuildableName = "Gas ManTests.xctest"
|
||||||
|
BlueprintName = "Gas ManTests"
|
||||||
|
ReferencedContainer = "container:Gas Man.xcodeproj">
|
||||||
|
</BuildableReference>
|
||||||
|
</TestableReference>
|
||||||
|
<TestableReference
|
||||||
|
skipped = "NO"
|
||||||
|
parallelizable = "YES">
|
||||||
|
<BuildableReference
|
||||||
|
BuildableIdentifier = "primary"
|
||||||
|
BlueprintIdentifier = "3D3444702D889F8000AA3172"
|
||||||
|
BuildableName = "Gas ManUITests.xctest"
|
||||||
|
BlueprintName = "Gas ManUITests"
|
||||||
|
ReferencedContainer = "container:Gas Man.xcodeproj">
|
||||||
|
</BuildableReference>
|
||||||
|
</TestableReference>
|
||||||
|
</Testables>
|
||||||
|
</TestAction>
|
||||||
|
<LaunchAction
|
||||||
|
buildConfiguration = "Debug"
|
||||||
|
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
|
||||||
|
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
|
||||||
|
launchStyle = "0"
|
||||||
|
useCustomWorkingDirectory = "NO"
|
||||||
|
ignoresPersistentStateOnLaunch = "NO"
|
||||||
|
debugDocumentVersioning = "YES"
|
||||||
|
debugServiceExtension = "internal"
|
||||||
|
allowLocationSimulation = "YES">
|
||||||
|
<BuildableProductRunnable
|
||||||
|
runnableDebuggingMode = "0">
|
||||||
|
<BuildableReference
|
||||||
|
BuildableIdentifier = "primary"
|
||||||
|
BlueprintIdentifier = "3D34444F2D889F7D00AA3172"
|
||||||
|
BuildableName = "Gas Man.app"
|
||||||
|
BlueprintName = "Gas Man"
|
||||||
|
ReferencedContainer = "container:Gas Man.xcodeproj">
|
||||||
|
</BuildableReference>
|
||||||
|
</BuildableProductRunnable>
|
||||||
|
</LaunchAction>
|
||||||
|
<ProfileAction
|
||||||
|
buildConfiguration = "Release"
|
||||||
|
shouldUseLaunchSchemeArgsEnv = "YES"
|
||||||
|
savedToolIdentifier = ""
|
||||||
|
useCustomWorkingDirectory = "NO"
|
||||||
|
debugDocumentVersioning = "YES">
|
||||||
|
<BuildableProductRunnable
|
||||||
|
runnableDebuggingMode = "0">
|
||||||
|
<BuildableReference
|
||||||
|
BuildableIdentifier = "primary"
|
||||||
|
BlueprintIdentifier = "3D34444F2D889F7D00AA3172"
|
||||||
|
BuildableName = "Gas Man.app"
|
||||||
|
BlueprintName = "Gas Man"
|
||||||
|
ReferencedContainer = "container:Gas Man.xcodeproj">
|
||||||
|
</BuildableReference>
|
||||||
|
</BuildableProductRunnable>
|
||||||
|
</ProfileAction>
|
||||||
|
<AnalyzeAction
|
||||||
|
buildConfiguration = "Debug">
|
||||||
|
</AnalyzeAction>
|
||||||
|
<ArchiveAction
|
||||||
|
buildConfiguration = "Release"
|
||||||
|
revealArchiveInOrganizer = "YES">
|
||||||
|
</ArchiveAction>
|
||||||
|
</Scheme>
|
|
@ -10,5 +10,23 @@
|
||||||
<integer>0</integer>
|
<integer>0</integer>
|
||||||
</dict>
|
</dict>
|
||||||
</dict>
|
</dict>
|
||||||
|
<key>SuppressBuildableAutocreation</key>
|
||||||
|
<dict>
|
||||||
|
<key>3D34444F2D889F7D00AA3172</key>
|
||||||
|
<dict>
|
||||||
|
<key>primary</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>3D3444662D889F7F00AA3172</key>
|
||||||
|
<dict>
|
||||||
|
<key>primary</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>3D3444702D889F8000AA3172</key>
|
||||||
|
<dict>
|
||||||
|
<key>primary</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
|
|
|
@ -0,0 +1,249 @@
|
||||||
|
//
|
||||||
|
// AddFuelLogView.swift
|
||||||
|
// Gas Man
|
||||||
|
//
|
||||||
|
// Created by Kameron Kenny on 3/17/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
import SwiftUI
|
||||||
|
import CoreData
|
||||||
|
|
||||||
|
struct AddFuelLogView: View {
|
||||||
|
@Environment(\.managedObjectContext) private var viewContext
|
||||||
|
@Environment(\.dismiss) var dismiss
|
||||||
|
|
||||||
|
// Form fields
|
||||||
|
@State private var date = Date()
|
||||||
|
@State private var odometer = ""
|
||||||
|
@State private var fuelVolume = ""
|
||||||
|
@State private var cost = ""
|
||||||
|
@State private var locationCoordinates = ""
|
||||||
|
@State private var locationName = ""
|
||||||
|
@State private var selectedOctane: Int = 87 // Default or from previous record
|
||||||
|
@State private var pricePerGalon = ""
|
||||||
|
|
||||||
|
// Allowed octane options
|
||||||
|
let octaneOptions = [87, 89, 91, 92, 93, 95]
|
||||||
|
|
||||||
|
// Location manager for automatic location updates
|
||||||
|
@StateObject private var locationManager = LocationManager()
|
||||||
|
|
||||||
|
// For tracking the previous odometer reading
|
||||||
|
@State private var previousOdometer: Double? = nil
|
||||||
|
@State private var showOdometerAlert = false
|
||||||
|
|
||||||
|
// Flag to avoid update loops in our calculation logic
|
||||||
|
@State private var isUpdatingCalculation = false
|
||||||
|
|
||||||
|
// Computed property for formatted previous odometer value
|
||||||
|
private var previousOdometerString: String {
|
||||||
|
previousOdometer.map { String(format: "%.0f", $0) } ?? "N/A"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Computed property for the odometer alert
|
||||||
|
var odometerAlert: Alert {
|
||||||
|
let messageString = "Odometer reading must be greater than the previous record (\(previousOdometerString))."
|
||||||
|
return Alert(
|
||||||
|
title: Text("Odometer Reading Error"),
|
||||||
|
message: Text(messageString),
|
||||||
|
dismissButton: .default(Text("OK"))
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
NavigationView {
|
||||||
|
Form {
|
||||||
|
fuelLogDetailsSection
|
||||||
|
}
|
||||||
|
.navigationTitle("Add Fuel Log")
|
||||||
|
.toolbar {
|
||||||
|
ToolbarItem(placement: .navigationBarTrailing) {
|
||||||
|
Button("Save") { saveFuelLog() }
|
||||||
|
}
|
||||||
|
ToolbarItem(placement: .navigationBarLeading) {
|
||||||
|
Button("Cancel") { dismiss() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.onChange(of: locationManager.location) { newLocation in
|
||||||
|
if let loc = newLocation {
|
||||||
|
locationCoordinates = "\(loc.coordinate.latitude), \(loc.coordinate.longitude)"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.onChange(of: locationManager.placemark) { newPlacemark in
|
||||||
|
if let placemark = newPlacemark {
|
||||||
|
locationName = placemark.name ?? ""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.onAppear(perform: loadPreviousData)
|
||||||
|
.alert(isPresented: $showOdometerAlert) { odometerAlert }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: - Subviews
|
||||||
|
|
||||||
|
private var fuelLogDetailsSection: some View {
|
||||||
|
Section(header: Text("Fuel Log Details")) {
|
||||||
|
DatePicker("Date", selection: $date, displayedComponents: [.date, .hourAndMinute])
|
||||||
|
|
||||||
|
// Odometer field
|
||||||
|
HStack(spacing: 4) {
|
||||||
|
Text("Odometer: ")
|
||||||
|
.font(.caption)
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
TextField("", text: $odometer)
|
||||||
|
.keyboardType(.decimalPad)
|
||||||
|
.textFieldStyle(RoundedBorderTextFieldStyle())
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fuel Volume field
|
||||||
|
HStack(spacing: 4) {
|
||||||
|
Text("Gallons: ")
|
||||||
|
.font(.caption)
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
TextField("", text: $fuelVolume)
|
||||||
|
.keyboardType(.decimalPad)
|
||||||
|
.textFieldStyle(RoundedBorderTextFieldStyle())
|
||||||
|
.onChange(of: fuelVolume) { _ in updateCalculatedValues() }
|
||||||
|
}
|
||||||
|
|
||||||
|
// Price per Gallon field
|
||||||
|
HStack(spacing: 4) {
|
||||||
|
Text("Price/Gal: $")
|
||||||
|
.font(.caption)
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
TextField("", text: $pricePerGalon)
|
||||||
|
.keyboardType(.decimalPad)
|
||||||
|
.textFieldStyle(RoundedBorderTextFieldStyle())
|
||||||
|
.onChange(of: pricePerGalon) { _ in updateCalculatedValues() }
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cost field
|
||||||
|
HStack(spacing: 4) {
|
||||||
|
Text("Cost: $")
|
||||||
|
.font(.caption)
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
TextField("", text: $cost)
|
||||||
|
.keyboardType(.decimalPad)
|
||||||
|
.textFieldStyle(RoundedBorderTextFieldStyle())
|
||||||
|
.onChange(of: cost) { _ in updateCalculatedValues() }
|
||||||
|
}
|
||||||
|
|
||||||
|
Button("Get Current Location") {
|
||||||
|
locationManager.requestLocation()
|
||||||
|
}
|
||||||
|
|
||||||
|
if !locationCoordinates.isEmpty {
|
||||||
|
Text("Coordinates: \(locationCoordinates)")
|
||||||
|
.font(.caption)
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
}
|
||||||
|
|
||||||
|
TextField("Location Coordinates", text: $locationCoordinates)
|
||||||
|
|
||||||
|
HStack(spacing: 4) {
|
||||||
|
Text("Location Name:")
|
||||||
|
.font(.caption)
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
TextField("", text: $locationName)
|
||||||
|
}
|
||||||
|
|
||||||
|
Picker("Octane", selection: $selectedOctane) {
|
||||||
|
ForEach(octaneOptions, id: \.self) { option in
|
||||||
|
Text("\(option)").tag(option)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.pickerStyle(MenuPickerStyle())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: - Helper Methods
|
||||||
|
|
||||||
|
private func loadPreviousData() {
|
||||||
|
let fetchRequest: NSFetchRequest<FuelLog> = FuelLog.fetchRequest()
|
||||||
|
fetchRequest.sortDescriptors = [NSSortDescriptor(key: "date", ascending: false)]
|
||||||
|
fetchRequest.fetchLimit = 1
|
||||||
|
if let lastFuelLog = try? viewContext.fetch(fetchRequest).first {
|
||||||
|
selectedOctane = Int(lastFuelLog.octane)
|
||||||
|
previousOdometer = lastFuelLog.odometer
|
||||||
|
odometer = String(format: "%.0f", lastFuelLog.odometer)
|
||||||
|
} else {
|
||||||
|
selectedOctane = 87
|
||||||
|
odometer = ""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func roundToThree(_ value: Double) -> Double {
|
||||||
|
return (value * 1000).rounded() / 1000
|
||||||
|
}
|
||||||
|
|
||||||
|
private func roundToTwo(_ value: Double) -> Double {
|
||||||
|
return (value * 100).rounded() / 100
|
||||||
|
}
|
||||||
|
|
||||||
|
private func updateCalculatedValues() {
|
||||||
|
// Prevent recursive updates.
|
||||||
|
guard !isUpdatingCalculation else { return }
|
||||||
|
isUpdatingCalculation = true
|
||||||
|
|
||||||
|
let fuel = Double(fuelVolume)
|
||||||
|
let costVal = Double(cost)
|
||||||
|
let price = Double(pricePerGalon)
|
||||||
|
|
||||||
|
// 1. If fuelVolume and pricePerGalon are provided, calculate cost.
|
||||||
|
if let f = fuel, let p = price, f > 0 {
|
||||||
|
let computedCost = roundToTwo(f * p)
|
||||||
|
let computedCostStr = String(format: "%.2f", computedCost)
|
||||||
|
if cost != computedCostStr {
|
||||||
|
cost = computedCostStr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 2. If fuelVolume and cost are provided, and pricePerGalon is empty, calculate pricePerGalon.
|
||||||
|
else if let f = fuel, let c = costVal, f > 0, pricePerGalon.trimmingCharacters(in: .whitespaces).isEmpty {
|
||||||
|
let computedPrice = roundToThree(c / f)
|
||||||
|
let computedPriceStr = String(format: "%.3f", computedPrice)
|
||||||
|
if pricePerGalon != computedPriceStr {
|
||||||
|
pricePerGalon = computedPriceStr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 3. If pricePerGalon and cost are provided, and fuelVolume is empty, calculate fuelVolume.
|
||||||
|
else if let p = price, let c = costVal, p > 0, fuelVolume.trimmingCharacters(in: .whitespaces).isEmpty {
|
||||||
|
let computedFuel = roundToThree(c / p)
|
||||||
|
let computedFuelStr = String(format: "%.3f", computedFuel)
|
||||||
|
if fuelVolume != computedFuelStr {
|
||||||
|
fuelVolume = computedFuelStr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
isUpdatingCalculation = false
|
||||||
|
}
|
||||||
|
|
||||||
|
private func saveFuelLog() {
|
||||||
|
guard let newOdometer = Double(odometer) else {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// Validate that the new odometer reading is greater than the previous record
|
||||||
|
if let previous = previousOdometer, newOdometer <= previous {
|
||||||
|
showOdometerAlert = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
let newLog = FuelLog(context: viewContext)
|
||||||
|
newLog.id = UUID()
|
||||||
|
newLog.date = date
|
||||||
|
newLog.odometer = newOdometer
|
||||||
|
newLog.fuelVolume = Double(fuelVolume) ?? 0
|
||||||
|
newLog.cost = Double(cost) ?? 0
|
||||||
|
newLog.locationCoordinates = locationCoordinates
|
||||||
|
newLog.locationName = locationName
|
||||||
|
newLog.octane = Int16(selectedOctane)
|
||||||
|
newLog.pricePerGalon = Double(pricePerGalon) ?? 0
|
||||||
|
|
||||||
|
do {
|
||||||
|
try viewContext.save()
|
||||||
|
dismiss()
|
||||||
|
} catch {
|
||||||
|
let nsError = error as NSError
|
||||||
|
fatalError("Unresolved error \(nsError), \(nsError.userInfo)")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,73 @@
|
||||||
|
//
|
||||||
|
// AddMaintenanceView.swift
|
||||||
|
// Gas Man
|
||||||
|
//
|
||||||
|
// Created by Kameron Kenny on 3/17/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
|
||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
struct AddMaintenanceView: View {
|
||||||
|
@Environment(\.managedObjectContext) private var viewContext
|
||||||
|
@Environment(\.dismiss) var dismiss
|
||||||
|
|
||||||
|
@State private var date = Date()
|
||||||
|
@State private var odometer = ""
|
||||||
|
@State private var eventType = ""
|
||||||
|
@State private var cost = ""
|
||||||
|
@State private var notes = ""
|
||||||
|
@State private var locationCoordinates = ""
|
||||||
|
@State private var locationName = ""
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
NavigationView {
|
||||||
|
Form {
|
||||||
|
Section(header: Text("Maintenance Details")) {
|
||||||
|
DatePicker("Date", selection: $date, displayedComponents: [.date, .hourAndMinute])
|
||||||
|
TextField("Odometer Reading", text: $odometer)
|
||||||
|
.keyboardType(.decimalPad)
|
||||||
|
TextField("Service Type", text: $eventType)
|
||||||
|
TextField("Cost ($)", text: $cost)
|
||||||
|
.keyboardType(.decimalPad)
|
||||||
|
TextField("Notes", text: $notes)
|
||||||
|
TextField("Location Coordinates", text: $locationCoordinates)
|
||||||
|
TextField("Location Name", text: $locationName)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.navigationTitle("Add Maintenance")
|
||||||
|
.toolbar {
|
||||||
|
ToolbarItem(placement: .navigationBarTrailing) {
|
||||||
|
Button("Save") {
|
||||||
|
saveMaintenance()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ToolbarItem(placement: .navigationBarLeading) {
|
||||||
|
Button("Cancel") {
|
||||||
|
dismiss()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func saveMaintenance() {
|
||||||
|
let newEvent = MaintenanceEvent(context: viewContext)
|
||||||
|
newEvent.id = UUID()
|
||||||
|
newEvent.date = date
|
||||||
|
newEvent.odometer = Double(odometer) ?? 0
|
||||||
|
newEvent.eventType = eventType
|
||||||
|
newEvent.cost = Double(cost) ?? 0
|
||||||
|
newEvent.notes = notes
|
||||||
|
newEvent.locationCoordinates = locationCoordinates
|
||||||
|
newEvent.locationName = locationName
|
||||||
|
|
||||||
|
do {
|
||||||
|
try viewContext.save()
|
||||||
|
dismiss()
|
||||||
|
} catch {
|
||||||
|
let nsError = error as NSError
|
||||||
|
fatalError("Unresolved error \(nsError), \(nsError.userInfo)")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -6,83 +6,83 @@
|
||||||
//
|
//
|
||||||
|
|
||||||
import SwiftUI
|
import SwiftUI
|
||||||
import CoreData
|
//import CoreData
|
||||||
|
//
|
||||||
|
//struct ContentView: View {
|
||||||
|
// @Environment(\.managedObjectContext) private var viewContext
|
||||||
|
//
|
||||||
|
// @FetchRequest(
|
||||||
|
// sortDescriptors: [NSSortDescriptor(keyPath: \Item.timestamp, ascending: true)],
|
||||||
|
// animation: .default)
|
||||||
|
// private var items: FetchedResults<Item>
|
||||||
|
//
|
||||||
|
// var body: some View {
|
||||||
|
// NavigationView {
|
||||||
|
// List {
|
||||||
|
// ForEach(items) { item in
|
||||||
|
// NavigationLink {
|
||||||
|
// Text("Item at \(item.timestamp!, formatter: itemFormatter)")
|
||||||
|
// } label: {
|
||||||
|
// Text(item.timestamp!, formatter: itemFormatter)
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// .onDelete(perform: deleteItems)
|
||||||
|
// }
|
||||||
|
// .toolbar {
|
||||||
|
//#if os(iOS)
|
||||||
|
// ToolbarItem(placement: .navigationBarTrailing) {
|
||||||
|
// EditButton()
|
||||||
|
// }
|
||||||
|
//#endif
|
||||||
|
// ToolbarItem {
|
||||||
|
// Button(action: addItem) {
|
||||||
|
// Label("Add Item", systemImage: "plus")
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// Text("Select an item")
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// private func addItem() {
|
||||||
|
// withAnimation {
|
||||||
|
// let newItem = Item(context: viewContext)
|
||||||
|
// newItem.timestamp = Date()
|
||||||
|
//
|
||||||
|
// do {
|
||||||
|
// try viewContext.save()
|
||||||
|
// } catch {
|
||||||
|
// // Replace this implementation with code to handle the error appropriately.
|
||||||
|
// // fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
|
||||||
|
// let nsError = error as NSError
|
||||||
|
// fatalError("Unresolved error \(nsError), \(nsError.userInfo)")
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// private func deleteItems(offsets: IndexSet) {
|
||||||
|
// withAnimation {
|
||||||
|
// offsets.map { items[$0] }.forEach(viewContext.delete)
|
||||||
|
//
|
||||||
|
// do {
|
||||||
|
// try viewContext.save()
|
||||||
|
// } catch {
|
||||||
|
// // Replace this implementation with code to handle the error appropriately.
|
||||||
|
// // fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
|
||||||
|
// let nsError = error as NSError
|
||||||
|
// fatalError("Unresolved error \(nsError), \(nsError.userInfo)")
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//}
|
||||||
|
//
|
||||||
|
//private let itemFormatter: DateFormatter = {
|
||||||
|
// let formatter = DateFormatter()
|
||||||
|
// formatter.dateStyle = .short
|
||||||
|
// formatter.timeStyle = .medium
|
||||||
|
// return formatter
|
||||||
|
//}()
|
||||||
|
|
||||||
struct ContentView: View {
|
//#Preview {
|
||||||
@Environment(\.managedObjectContext) private var viewContext
|
// ContentView().environment(\.managedObjectContext, PersistenceController.preview.container.viewContext)
|
||||||
|
//}
|
||||||
@FetchRequest(
|
|
||||||
sortDescriptors: [NSSortDescriptor(keyPath: \Item.timestamp, ascending: true)],
|
|
||||||
animation: .default)
|
|
||||||
private var items: FetchedResults<Item>
|
|
||||||
|
|
||||||
var body: some View {
|
|
||||||
NavigationView {
|
|
||||||
List {
|
|
||||||
ForEach(items) { item in
|
|
||||||
NavigationLink {
|
|
||||||
Text("Item at \(item.timestamp!, formatter: itemFormatter)")
|
|
||||||
} label: {
|
|
||||||
Text(item.timestamp!, formatter: itemFormatter)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.onDelete(perform: deleteItems)
|
|
||||||
}
|
|
||||||
.toolbar {
|
|
||||||
#if os(iOS)
|
|
||||||
ToolbarItem(placement: .navigationBarTrailing) {
|
|
||||||
EditButton()
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
ToolbarItem {
|
|
||||||
Button(action: addItem) {
|
|
||||||
Label("Add Item", systemImage: "plus")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Text("Select an item")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private func addItem() {
|
|
||||||
withAnimation {
|
|
||||||
let newItem = Item(context: viewContext)
|
|
||||||
newItem.timestamp = Date()
|
|
||||||
|
|
||||||
do {
|
|
||||||
try viewContext.save()
|
|
||||||
} catch {
|
|
||||||
// Replace this implementation with code to handle the error appropriately.
|
|
||||||
// fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
|
|
||||||
let nsError = error as NSError
|
|
||||||
fatalError("Unresolved error \(nsError), \(nsError.userInfo)")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private func deleteItems(offsets: IndexSet) {
|
|
||||||
withAnimation {
|
|
||||||
offsets.map { items[$0] }.forEach(viewContext.delete)
|
|
||||||
|
|
||||||
do {
|
|
||||||
try viewContext.save()
|
|
||||||
} catch {
|
|
||||||
// Replace this implementation with code to handle the error appropriately.
|
|
||||||
// fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
|
|
||||||
let nsError = error as NSError
|
|
||||||
fatalError("Unresolved error \(nsError), \(nsError.userInfo)")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private let itemFormatter: DateFormatter = {
|
|
||||||
let formatter = DateFormatter()
|
|
||||||
formatter.dateStyle = .short
|
|
||||||
formatter.timeStyle = .medium
|
|
||||||
return formatter
|
|
||||||
}()
|
|
||||||
|
|
||||||
#Preview {
|
|
||||||
ContentView().environment(\.managedObjectContext, PersistenceController.preview.container.viewContext)
|
|
||||||
}
|
|
||||||
|
|
|
@ -0,0 +1,74 @@
|
||||||
|
//
|
||||||
|
// FuelLogDetailView.swift
|
||||||
|
// Gas Man
|
||||||
|
//
|
||||||
|
// Created by Kameron Kenny on 3/17/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
private let dateFormatter: DateFormatter = {
|
||||||
|
let formatter = DateFormatter()
|
||||||
|
formatter.dateStyle = .short
|
||||||
|
formatter.timeStyle = .short
|
||||||
|
return formatter
|
||||||
|
}()
|
||||||
|
|
||||||
|
|
||||||
|
struct FuelLogDetailView: View {
|
||||||
|
var fuelLog: FuelLog
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
Form {
|
||||||
|
Section(header: Text("Basic Information")) {
|
||||||
|
HStack {
|
||||||
|
Text("Date:")
|
||||||
|
Spacer()
|
||||||
|
Text(fuelLog.date ?? Date(), formatter: dateFormatter)
|
||||||
|
}
|
||||||
|
HStack {
|
||||||
|
Text("Odometer:")
|
||||||
|
Spacer()
|
||||||
|
Text("\(fuelLog.odometer, specifier: "%.0f") miles")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Section(header: Text("Fuel Information")) {
|
||||||
|
HStack {
|
||||||
|
Text("Fuel Volume:")
|
||||||
|
Spacer()
|
||||||
|
Text("\(fuelLog.fuelVolume, specifier: "%.3f") gallons")
|
||||||
|
}
|
||||||
|
HStack {
|
||||||
|
Text("Cost:")
|
||||||
|
Spacer()
|
||||||
|
Text("$\(fuelLog.cost, specifier: "%.2f")")
|
||||||
|
}
|
||||||
|
HStack {
|
||||||
|
Text("Price per Gallon:")
|
||||||
|
Spacer()
|
||||||
|
Text("$\(fuelLog.pricePerGalon, specifier: "%.3f")")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Section(header: Text("Location")) {
|
||||||
|
HStack {
|
||||||
|
Text("Coordinates:")
|
||||||
|
Spacer()
|
||||||
|
Text(fuelLog.locationCoordinates ?? "N/A")
|
||||||
|
}
|
||||||
|
HStack {
|
||||||
|
Text("Location Name:")
|
||||||
|
Spacer()
|
||||||
|
Text(fuelLog.locationName ?? "N/A")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Section(header: Text("Fuel Details")) {
|
||||||
|
HStack {
|
||||||
|
Text("Octane:")
|
||||||
|
Spacer()
|
||||||
|
Text("\(fuelLog.octane)")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.navigationTitle("Fuel Log Detail")
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,85 @@
|
||||||
|
//
|
||||||
|
// FuelLogListView.swift
|
||||||
|
// Gas Man
|
||||||
|
//
|
||||||
|
// Created by Kameron Kenny on 3/17/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
|
||||||
|
import SwiftUI
|
||||||
|
import CoreData
|
||||||
|
|
||||||
|
struct FuelLogListView: View {
|
||||||
|
@Environment(\.managedObjectContext) private var viewContext
|
||||||
|
@FetchRequest(
|
||||||
|
sortDescriptors: [NSSortDescriptor(keyPath: \FuelLog.date, ascending: false)],
|
||||||
|
animation: .default)
|
||||||
|
private var fuelLogs: FetchedResults<FuelLog>
|
||||||
|
|
||||||
|
@State private var showingAddFuelLog = false // Controls sheet presentation
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
NavigationView {
|
||||||
|
List {
|
||||||
|
ForEach(Array(fuelLogs.enumerated()), id: \.element) { index, log in
|
||||||
|
// Compute distance since the previous record:
|
||||||
|
let distance: Double? = {
|
||||||
|
if index < fuelLogs.count - 1 {
|
||||||
|
let previousLog = fuelLogs[index + 1]
|
||||||
|
let milesDriven = log.odometer - previousLog.odometer
|
||||||
|
return milesDriven > 0 ? milesDriven : nil
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}()
|
||||||
|
|
||||||
|
// Compute MPG (if possible)
|
||||||
|
let mpg: Double? = {
|
||||||
|
if index < fuelLogs.count - 1 {
|
||||||
|
let previousLog = fuelLogs[index + 1]
|
||||||
|
let milesDriven = log.odometer - previousLog.odometer
|
||||||
|
if log.fuelVolume > 0 && milesDriven > 0 {
|
||||||
|
return milesDriven / log.fuelVolume
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}()
|
||||||
|
|
||||||
|
NavigationLink(destination: FuelLogDetailView(fuelLog: log)) {
|
||||||
|
FuelLogSummaryView(log: log, mpg: mpg, distanceSincePrevious: distance)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.onDelete(perform: deleteFuelLogs)
|
||||||
|
}
|
||||||
|
.listStyle(InsetGroupedListStyle())
|
||||||
|
.navigationTitle("Fuel Logs")
|
||||||
|
.toolbar {
|
||||||
|
ToolbarItem(placement: .navigationBarTrailing) {
|
||||||
|
Button(action: {
|
||||||
|
showingAddFuelLog = true // Present AddFuelLogView
|
||||||
|
}) {
|
||||||
|
Label("Add Fuel Log", systemImage: "plus")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ToolbarItem(placement: .navigationBarLeading) {
|
||||||
|
EditButton()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Sheet presentation for AddFuelLogView
|
||||||
|
.sheet(isPresented: $showingAddFuelLog) {
|
||||||
|
AddFuelLogView().environment(\.managedObjectContext, viewContext)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func deleteFuelLogs(offsets: IndexSet) {
|
||||||
|
withAnimation {
|
||||||
|
offsets.map { fuelLogs[$0] }.forEach(viewContext.delete)
|
||||||
|
do {
|
||||||
|
try viewContext.save()
|
||||||
|
} catch {
|
||||||
|
let nsError = error as NSError
|
||||||
|
fatalError("Unresolved error \(nsError), \(nsError.userInfo)")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,84 @@
|
||||||
|
//
|
||||||
|
// FuelLogSummaryView.swift
|
||||||
|
// Gas Man
|
||||||
|
//
|
||||||
|
// Created by Kameron Kenny on 3/17/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
|
||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
private let dateFormatter: DateFormatter = {
|
||||||
|
let formatter = DateFormatter()
|
||||||
|
formatter.dateStyle = .short
|
||||||
|
formatter.timeStyle = .short
|
||||||
|
return formatter
|
||||||
|
}()
|
||||||
|
|
||||||
|
struct FuelLogSummaryView: View {
|
||||||
|
var log: FuelLog
|
||||||
|
var mpg: Double?
|
||||||
|
var distanceSincePrevious: Double?
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
VStack(alignment: .leading, spacing: 8) {
|
||||||
|
// Row 1: Date and MPG (if available)
|
||||||
|
HStack {
|
||||||
|
Text(log.date ?? Date(), formatter: dateFormatter)
|
||||||
|
.font(.subheadline)
|
||||||
|
.foregroundColor(.primary)
|
||||||
|
Spacer()
|
||||||
|
if let mpg = mpg {
|
||||||
|
Text("MPG: \(mpg, specifier: "%.3f")")
|
||||||
|
.font(.subheadline)
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Divider()
|
||||||
|
// Row 2: Distance (instead of odometer) and Fuel Volume
|
||||||
|
HStack {
|
||||||
|
VStack(alignment: .leading) {
|
||||||
|
Text("Distance")
|
||||||
|
.font(.caption)
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
if let distance = distanceSincePrevious {
|
||||||
|
Text("\(distance, specifier: "%.0f") miles")
|
||||||
|
.bold()
|
||||||
|
} else {
|
||||||
|
Text("N/A")
|
||||||
|
.bold()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Spacer()
|
||||||
|
VStack(alignment: .leading) {
|
||||||
|
Text("Fuel")
|
||||||
|
.font(.caption)
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
Text("\(log.fuelVolume, specifier: "%.3f") gal")
|
||||||
|
.bold()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Row 3: Cost and Price per Gallon
|
||||||
|
HStack {
|
||||||
|
VStack(alignment: .leading) {
|
||||||
|
Text("Cost")
|
||||||
|
.font(.caption)
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
Text("$\(log.cost, specifier: "%.2f")")
|
||||||
|
.bold()
|
||||||
|
}
|
||||||
|
Spacer()
|
||||||
|
VStack(alignment: .leading) {
|
||||||
|
Text("Price/Gal")
|
||||||
|
.font(.caption)
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
Text("$\(log.pricePerGalon, specifier: "%.3f")")
|
||||||
|
.bold()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.padding(8)
|
||||||
|
.background(Color(.secondarySystemBackground))
|
||||||
|
.cornerRadius(8)
|
||||||
|
}
|
||||||
|
}
|
|
@ -14,7 +14,9 @@
|
||||||
</array>
|
</array>
|
||||||
<key>com.apple.security.app-sandbox</key>
|
<key>com.apple.security.app-sandbox</key>
|
||||||
<true/>
|
<true/>
|
||||||
<key>com.apple.security.files.user-selected.read-only</key>
|
<key>com.apple.security.files.user-selected.read-write</key>
|
||||||
|
<true/>
|
||||||
|
<key>com.apple.security.personal-information.location</key>
|
||||||
<true/>
|
<true/>
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
|
|
|
@ -1,9 +1,25 @@
|
||||||
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
|
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
|
||||||
<model type="com.apple.IDECoreDataModeler.DataModel" documentVersion="1.0" lastSavedToolsVersion="1" systemVersion="11A491" minimumToolsVersion="Automatic" sourceLanguage="Swift" usedWithCloudKit="true" userDefinedModelVersionIdentifier="">
|
<model type="com.apple.IDECoreDataModeler.DataModel" documentVersion="1.0" lastSavedToolsVersion="23507" systemVersion="23H222" minimumToolsVersion="Automatic" sourceLanguage="Swift" usedWithCloudKit="YES" userDefinedModelVersionIdentifier="">
|
||||||
<entity name="Item" representedClassName="Item" syncable="YES" codeGenerationType="class">
|
<entity name="FuelLog" representedClassName="FuelLog" isAbstract="YES" syncable="YES" codeGenerationType="class">
|
||||||
<attribute name="timestamp" optional="YES" attributeType="Date" usesScalarValueType="NO"/>
|
<attribute name="cost" optional="YES" attributeType="Double" defaultValueString="0.0" usesScalarValueType="YES"/>
|
||||||
|
<attribute name="date" optional="YES" attributeType="Date" usesScalarValueType="NO"/>
|
||||||
|
<attribute name="fuelVolume" optional="YES" attributeType="Double" defaultValueString="0.0" usesScalarValueType="YES"/>
|
||||||
|
<attribute name="id" optional="YES" attributeType="UUID" usesScalarValueType="NO"/>
|
||||||
|
<attribute name="locationCoordinates" optional="YES" attributeType="String"/>
|
||||||
|
<attribute name="locationName" optional="YES" attributeType="String"/>
|
||||||
|
<attribute name="octane" optional="YES" attributeType="Integer 16" defaultValueString="0" usesScalarValueType="YES"/>
|
||||||
|
<attribute name="odometer" optional="YES" attributeType="Double" defaultValueString="0.0" usesScalarValueType="YES"/>
|
||||||
|
<attribute name="pricePerGalon" optional="YES" attributeType="Double" defaultValueString="0.0" usesScalarValueType="YES"/>
|
||||||
|
</entity>
|
||||||
|
<entity name="Item" representedClassName="Item" syncable="YES" codeGenerationType="class"/>
|
||||||
|
<entity name="MaintenanceEvent" representedClassName="MaintenanceEvent" isAbstract="YES" syncable="YES" codeGenerationType="class">
|
||||||
|
<attribute name="cost" optional="YES" attributeType="Double" defaultValueString="0.0" usesScalarValueType="YES"/>
|
||||||
|
<attribute name="date" optional="YES" attributeType="Date" usesScalarValueType="NO"/>
|
||||||
|
<attribute name="eventType" optional="YES" attributeType="String"/>
|
||||||
|
<attribute name="id" optional="YES" attributeType="UUID" usesScalarValueType="NO"/>
|
||||||
|
<attribute name="locationCoordinates" optional="YES" attributeType="String"/>
|
||||||
|
<attribute name="locationName" optional="YES" attributeType="String"/>
|
||||||
|
<attribute name="notes" optional="YES" attributeType="String"/>
|
||||||
|
<attribute name="odometer" optional="YES" attributeType="Double" defaultValueString="0.0" usesScalarValueType="YES"/>
|
||||||
</entity>
|
</entity>
|
||||||
<elements>
|
|
||||||
<element name="Item" positionX="-63" positionY="-18" width="128" height="44"/>
|
|
||||||
</elements>
|
|
||||||
</model>
|
</model>
|
|
@ -13,8 +13,9 @@ struct Gas_ManApp: App {
|
||||||
|
|
||||||
var body: some Scene {
|
var body: some Scene {
|
||||||
WindowGroup {
|
WindowGroup {
|
||||||
ContentView()
|
MainTabView() // Replace with your actual view name.
|
||||||
.environment(\.managedObjectContext, persistenceController.container.viewContext)
|
.environment(\.managedObjectContext, persistenceController.container.viewContext)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2,6 +2,8 @@
|
||||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
<plist version="1.0">
|
<plist version="1.0">
|
||||||
<dict>
|
<dict>
|
||||||
|
<key>NSLocationWhenInUseUsageDescription</key>
|
||||||
|
<string>This app needs your location to automatically fill in location details for fuel and maintenance logs</string>
|
||||||
<key>UIBackgroundModes</key>
|
<key>UIBackgroundModes</key>
|
||||||
<array>
|
<array>
|
||||||
<string>remote-notification</string>
|
<string>remote-notification</string>
|
||||||
|
|
|
@ -0,0 +1,55 @@
|
||||||
|
//
|
||||||
|
// LocationManager.swift
|
||||||
|
// Gas Man
|
||||||
|
//
|
||||||
|
// Created by Kameron Kenny on 3/17/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
|
||||||
|
import Foundation
|
||||||
|
import CoreLocation
|
||||||
|
import Combine
|
||||||
|
|
||||||
|
class LocationManager: NSObject, ObservableObject, CLLocationManagerDelegate {
|
||||||
|
private let manager = CLLocationManager()
|
||||||
|
|
||||||
|
@Published var location: CLLocation?
|
||||||
|
@Published var placemark: CLPlacemark?
|
||||||
|
|
||||||
|
override init() {
|
||||||
|
super.init()
|
||||||
|
manager.delegate = self
|
||||||
|
manager.desiredAccuracy = kCLLocationAccuracyBest
|
||||||
|
}
|
||||||
|
|
||||||
|
func requestLocation() {
|
||||||
|
manager.requestWhenInUseAuthorization()
|
||||||
|
manager.requestLocation()
|
||||||
|
}
|
||||||
|
|
||||||
|
// CLLocationManagerDelegate
|
||||||
|
func locationManager(_ manager: CLLocationManager, didUpdateLocations locations: [CLLocation]) {
|
||||||
|
if let loc = locations.first {
|
||||||
|
DispatchQueue.main.async {
|
||||||
|
self.location = loc
|
||||||
|
}
|
||||||
|
// Reverse geocode
|
||||||
|
let geocoder = CLGeocoder()
|
||||||
|
geocoder.reverseGeocodeLocation(loc) { [weak self] placemarks, error in
|
||||||
|
if let error = error {
|
||||||
|
print("Reverse geocode error: \(error)")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if let firstPlacemark = placemarks?.first {
|
||||||
|
DispatchQueue.main.async {
|
||||||
|
self?.placemark = firstPlacemark
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func locationManager(_ manager: CLLocationManager, didFailWithError error: Error) {
|
||||||
|
print("Failed to get user location: \(error)")
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,23 @@
|
||||||
|
//
|
||||||
|
// MainTabView.swift
|
||||||
|
// Gas Man
|
||||||
|
//
|
||||||
|
// Created by Kameron Kenny on 3/17/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
struct MainTabView: View {
|
||||||
|
var body: some View {
|
||||||
|
TabView {
|
||||||
|
FuelLogListView()
|
||||||
|
.tabItem {
|
||||||
|
Label("Fuel", systemImage: "fuelpump.fill")
|
||||||
|
}
|
||||||
|
MaintenanceListView()
|
||||||
|
.tabItem {
|
||||||
|
Label("Maintenance", systemImage: "wrench.fill")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,78 @@
|
||||||
|
//
|
||||||
|
// MaintenanceListView.swift
|
||||||
|
// Gas Man
|
||||||
|
//
|
||||||
|
// Created by Kameron Kenny on 3/17/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
|
||||||
|
import SwiftUI
|
||||||
|
import CoreData
|
||||||
|
|
||||||
|
struct MaintenanceListView: View {
|
||||||
|
@Environment(\.managedObjectContext) private var viewContext
|
||||||
|
@FetchRequest(
|
||||||
|
sortDescriptors: [NSSortDescriptor(keyPath: \MaintenanceEvent.date, ascending: false)],
|
||||||
|
animation: .default)
|
||||||
|
private var maintenanceEvents: FetchedResults<MaintenanceEvent>
|
||||||
|
|
||||||
|
@State private var showingAddMaintenance = false
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
NavigationView {
|
||||||
|
List {
|
||||||
|
ForEach(maintenanceEvents) { event in
|
||||||
|
VStack(alignment: .leading, spacing: 4) {
|
||||||
|
Text("Date: \(event.date ?? Date(), formatter: dateFormatter)")
|
||||||
|
Text("Odometer: \(event.odometer, specifier: "%.0f") miles")
|
||||||
|
Text("Type: \(event.eventType)")
|
||||||
|
Text("Cost: $\(event.cost, specifier: "%.2f")")
|
||||||
|
if let notes = event.notes, !notes.isEmpty {
|
||||||
|
Text("Notes: \(notes)")
|
||||||
|
}
|
||||||
|
if let locationName = event.locationName, !locationName.isEmpty {
|
||||||
|
Text("Location: \(locationName)")
|
||||||
|
}
|
||||||
|
if let locationCoordinates = event.locationCoordinates, !locationCoordinates.isEmpty {
|
||||||
|
Text("Coordinates: \(locationCoordinates)")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.onDelete(perform: deleteMaintenance)
|
||||||
|
}
|
||||||
|
.navigationTitle("Maintenance")
|
||||||
|
.toolbar {
|
||||||
|
ToolbarItem(placement: .navigationBarTrailing) {
|
||||||
|
Button(action: { showingAddMaintenance.toggle() }) {
|
||||||
|
Label("Add Maintenance", systemImage: "plus")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ToolbarItem(placement: .navigationBarLeading) {
|
||||||
|
EditButton()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.sheet(isPresented: $showingAddMaintenance) {
|
||||||
|
AddMaintenanceView().environment(\.managedObjectContext, viewContext)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func deleteMaintenance(offsets: IndexSet) {
|
||||||
|
withAnimation {
|
||||||
|
offsets.map { maintenanceEvents[$0] }.forEach(viewContext.delete)
|
||||||
|
do {
|
||||||
|
try viewContext.save()
|
||||||
|
} catch {
|
||||||
|
let nsError = error as NSError
|
||||||
|
fatalError("Unresolved error \(nsError), \(nsError.userInfo)")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private let dateFormatter: DateFormatter = {
|
||||||
|
let formatter = DateFormatter()
|
||||||
|
formatter.dateStyle = .short
|
||||||
|
formatter.timeStyle = .short
|
||||||
|
return formatter
|
||||||
|
}()
|
|
@ -9,49 +9,31 @@ import CoreData
|
||||||
|
|
||||||
struct PersistenceController {
|
struct PersistenceController {
|
||||||
static let shared = PersistenceController()
|
static let shared = PersistenceController()
|
||||||
|
|
||||||
@MainActor
|
|
||||||
static let preview: PersistenceController = {
|
|
||||||
let result = PersistenceController(inMemory: true)
|
|
||||||
let viewContext = result.container.viewContext
|
|
||||||
for _ in 0..<10 {
|
|
||||||
let newItem = Item(context: viewContext)
|
|
||||||
newItem.timestamp = Date()
|
|
||||||
}
|
|
||||||
do {
|
|
||||||
try viewContext.save()
|
|
||||||
} catch {
|
|
||||||
// Replace this implementation with code to handle the error appropriately.
|
|
||||||
// fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
|
|
||||||
let nsError = error as NSError
|
|
||||||
fatalError("Unresolved error \(nsError), \(nsError.userInfo)")
|
|
||||||
}
|
|
||||||
return result
|
|
||||||
}()
|
|
||||||
|
|
||||||
let container: NSPersistentCloudKitContainer
|
let container: NSPersistentCloudKitContainer
|
||||||
|
|
||||||
init(inMemory: Bool = false) {
|
init(inMemory: Bool = false) {
|
||||||
|
// "GasMan" must match the name of your .xcdatamodeld file.
|
||||||
container = NSPersistentCloudKitContainer(name: "Gas_Man")
|
container = NSPersistentCloudKitContainer(name: "Gas_Man")
|
||||||
|
|
||||||
if inMemory {
|
if inMemory {
|
||||||
container.persistentStoreDescriptions.first!.url = URL(fileURLWithPath: "/dev/null")
|
container.persistentStoreDescriptions.first?.url = URL(fileURLWithPath: "/dev/null")
|
||||||
}
|
}
|
||||||
container.loadPersistentStores(completionHandler: { (storeDescription, error) in
|
|
||||||
|
// Enable CloudKit options if necessary
|
||||||
|
guard let description = container.persistentStoreDescriptions.first else {
|
||||||
|
fatalError("No persistent store description found.")
|
||||||
|
}
|
||||||
|
description.setOption(true as NSNumber, forKey: NSPersistentStoreRemoteChangeNotificationPostOptionKey)
|
||||||
|
description.cloudKitContainerOptions = NSPersistentCloudKitContainerOptions(containerIdentifier: "iCloud.com.pro.thelinux.GasMan") // Update with your container ID
|
||||||
|
|
||||||
|
container.loadPersistentStores { storeDescription, error in
|
||||||
if let error = error as NSError? {
|
if let error = error as NSError? {
|
||||||
// Replace this implementation with code to handle the error appropriately.
|
|
||||||
// fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
|
|
||||||
|
|
||||||
/*
|
|
||||||
Typical reasons for an error here include:
|
|
||||||
* The parent directory does not exist, cannot be created, or disallows writing.
|
|
||||||
* The persistent store is not accessible, due to permissions or data protection when the device is locked.
|
|
||||||
* The device is out of space.
|
|
||||||
* The store could not be migrated to the current model version.
|
|
||||||
Check the error message to determine what the actual problem was.
|
|
||||||
*/
|
|
||||||
fatalError("Unresolved error \(error), \(error.userInfo)")
|
fatalError("Unresolved error \(error), \(error.userInfo)")
|
||||||
}
|
}
|
||||||
})
|
}
|
||||||
|
|
||||||
|
// Automatically merge changes from iCloud
|
||||||
container.viewContext.automaticallyMergesChangesFromParent = true
|
container.viewContext.automaticallyMergesChangesFromParent = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue