BudgetView.swift 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. //
  2. // BudgetView.swift
  3. // ydnab
  4. //
  5. // Created by Andrea Franceschini on 23/09/2020.
  6. //
  7. import SwiftUI
  8. import Combine
  9. /// Shows the budget values for the selected budget, month, and year.
  10. struct BudgetView: View {
  11. @State var budget: BudgetInfo
  12. @Binding var currentBudgetId: UUID?
  13. @State private var month: Int = 0
  14. @State private var monthName: String
  15. @State private var year: Int = Calendar(identifier: .gregorian).component(.year, from: Date())
  16. @State private var showMonthPicker: Bool = false
  17. @State private var showNotImplemented: Bool = false
  18. @State private var showQuickBudget: Bool = false
  19. @State private var showEditCategoryEntry: Bool = false
  20. @State private var editingCategoryEntry: BudgetCategoryEntry? = nil
  21. @AppStorage("lastLoadedBudgetId") private var lastLoadedBudgetId = ""
  22. @Environment(\.managedObjectContext) var managedObjectContext
  23. // private var budgetEntriesRequest: FetchRequest<BudgetCategoryEntry>
  24. // private var budgetEntries: FetchedResults<BudgetCategoryEntry> { budgetEntriesRequest.wrappedValue }
  25. @FetchRequest<BudgetCategoryEntry> private var budgetEntries: FetchedResults<BudgetCategoryEntry>
  26. init(budget: BudgetInfo, currentBudgetId: Binding<UUID?>) {
  27. _budget = .init(initialValue: budget)
  28. _currentBudgetId = currentBudgetId
  29. var cal = Calendar(identifier: .gregorian)
  30. cal.locale = Locale(identifier: self._budget.wrappedValue.localeIdentifier)
  31. let now = Date()
  32. _month = .init(initialValue: cal.component(.month, from: now))
  33. _monthName = .init(initialValue: cal.monthSymbols[self._month.wrappedValue])
  34. _year = .init(initialValue: cal.component(.year, from: now))
  35. _budgetEntries = FetchRequest<BudgetCategoryEntry>(entity: BudgetCategoryEntry.entity(),
  36. sortDescriptors: [],
  37. predicate: NSPredicate(format: "(budgetId == %@) AND (month == %d) AND (year == %d)",
  38. budget.id as CVarArg, _month.wrappedValue, _year.wrappedValue))
  39. // budgetEntriesRequest = FetchRequest<BudgetCategoryEntry>(entity: BudgetCategoryEntry.entity(),
  40. // sortDescriptors: [],
  41. // predicate: NSCompoundPredicate(andPredicateWithSubpredicates: [
  42. // NSPredicate(format: "budgetId == %@", budget.id as CVarArg),
  43. // NSPredicate(format: "month == %i", month),
  44. // NSPredicate(format: "year == %i", year)
  45. // ])
  46. // )
  47. }
  48. func formatAmount(_ amount: NSNumber) -> String? {
  49. let f = NumberFormatter()
  50. f.locale = Locale(identifier: budget.localeIdentifier)
  51. f.numberStyle = .currency
  52. return f.string(from: amount)
  53. }
  54. private func findBudgetCategoryEntry(withId: UUID) -> BudgetCategoryEntry? {
  55. return budgetEntries.first(where: { $0.budgetCategoryId == withId })
  56. }
  57. private func zeroBudgetEntries() -> Void {
  58. for section in budget.sections {
  59. for category in section.categories {
  60. if let entry = findBudgetCategoryEntry(withId: category.id) {
  61. entry.amount = 0
  62. } else {
  63. let newEntry = BudgetCategoryEntry(context: managedObjectContext)
  64. newEntry.budgetId = budget.id
  65. newEntry.budgetCategoryId = category.id
  66. newEntry.month = Int64(month)
  67. newEntry.year = Int64(year)
  68. newEntry.amount = 0
  69. }
  70. do {
  71. try managedObjectContext.save()
  72. } catch {
  73. print(error)
  74. }
  75. }
  76. }
  77. }
  78. var body: some View {
  79. VStack {
  80. HStack {
  81. Button(action: { withAnimation() { showMonthPicker.toggle() } } ) {
  82. Text("\(monthName) \(String(year))")
  83. Image(systemName: showMonthPicker ? "chevron.up" : "chevron.down")
  84. }
  85. Spacer()
  86. Button(action: { showQuickBudget.toggle() }) { // TODO: Implement this
  87. Image(systemName: "bolt.fill")
  88. }
  89. Button(action: { showNotImplemented.toggle() }) { // TODO: Implement this
  90. Image(systemName: "gearshape.fill")
  91. }.padding(.leading, 11)
  92. }
  93. .padding(11)
  94. if showMonthPicker {
  95. YearAndMonthPicker(month: $month,
  96. monthName: $monthName,
  97. year: $year,
  98. locale: Locale(identifier: budget.localeIdentifier)
  99. )
  100. }
  101. BudgetViewSummary(budget: $budget, month: $month, monthName: $monthName, year: $year)
  102. .padding([.leading, .trailing], 16)
  103. List {
  104. ForEach(budget.sections) { section in
  105. if !section.hidden {
  106. Section(header: BudgetSectionCell(section: section,
  107. amountText: "", //formatAmount(0) ?? "--",
  108. color: 0 < 0 ? Color("negativeAmount") : Color("positiveAmount"))
  109. ) {
  110. ForEach(section.categories) { category in
  111. if !category.hidden {
  112. BudgetCategoryCell(category: category,
  113. amountText: formatAmount(budgetEntries.first(where: { $0.budgetCategoryId == category.id })?.amount ?? 0) ?? "--",
  114. color: 0 < 0 ? Color("negativeAmount") : Color("positiveAmount")
  115. )
  116. .onLongPressGesture {
  117. editingCategoryEntry = findBudgetCategoryEntry(withId: category.id)
  118. showEditCategoryEntry.toggle()
  119. }
  120. }
  121. }
  122. .onMove { (indexSet, s) in
  123. print(indexSet, s)
  124. }
  125. }
  126. }
  127. }
  128. .onMove { (indexSet, s) in
  129. print(indexSet, s)
  130. }
  131. }
  132. .listStyle(PlainListStyle())
  133. .navigationBarTitle(budget.name, displayMode: .inline)
  134. .toolbar {
  135. EditButton()
  136. }
  137. .onAppear() {
  138. currentBudgetId = budget.id
  139. lastLoadedBudgetId = budget.id.uuidString
  140. }
  141. .onChange(of: month) { e in
  142. }
  143. .onReceive(budgetEntries.publisher) { o in
  144. print(o.month)
  145. }
  146. .actionSheet(isPresented: $showQuickBudget) {
  147. ActionSheet(
  148. title: Text("Quick budget"),
  149. message: Text("How would you like to set up your budget for \(monthName) \(String(year))?"),
  150. buttons: [
  151. .cancel { print(self.showQuickBudget) },
  152. .default(Text("All categories to zero"), action: zeroBudgetEntries),
  153. .default(Text("Values used the previous month"), action: { print(self) })
  154. ]
  155. )
  156. }
  157. .popover(isPresented: $showEditCategoryEntry) {
  158. // BudgetCategoryEntryEditor(amount: $editingCategoryEntry.wrappedValue?.amount)
  159. }
  160. .alert(isPresented: $showNotImplemented) {
  161. Alert(title: Text("Not Implemented!"))
  162. }
  163. }
  164. }
  165. }
  166. struct BudgetCategoryEntryEditor: View {
  167. @Binding var amount: Decimal?
  168. @Binding var locale: Locale
  169. @State private var amountText = ""
  170. var body: some View {
  171. VStack {
  172. TextField("Amount", text: $amountText)
  173. .keyboardType(.numbersAndPunctuation)
  174. .onReceive(Just(amountText)) { newValue in
  175. let filtered = newValue.filter { "0123456789.,".contains($0) }
  176. if filtered != newValue {
  177. amount = Decimal(string: filtered, locale: locale)
  178. }
  179. }
  180. }
  181. .onAppear {
  182. amountText = "\(amount ?? -69)"
  183. }
  184. }
  185. }
  186. struct BudgetView_Previews: PreviewProvider {
  187. static let budget: [BudgetSection] = [
  188. BudgetSection(name: "Everyday Expenses", categories: [
  189. BudgetCategory(name: "Groceries"),
  190. BudgetCategory(name: "Eating out"),
  191. BudgetCategory(name: "Medical"),
  192. BudgetCategory(name: "Clothing"),
  193. BudgetCategory(name: "Household goods")
  194. ]),
  195. BudgetSection(name: "Travel", categories: [
  196. BudgetCategory(name: "Transport"),
  197. BudgetCategory(name: "Fuel"),
  198. BudgetCategory(name: "Accommodation")
  199. ]),
  200. BudgetSection(name: "Rainy Days", categories: [
  201. BudgetCategory(name: "Emergencies"),
  202. BudgetCategory(name: "Car insurance")
  203. ]),
  204. BudgetSection(name: "Monthly Expenses", categories: [
  205. BudgetCategory(name: "Rent"),
  206. BudgetCategory(name: "Mobile")
  207. ]),
  208. BudgetSection(name: "Savings Goals", categories: [
  209. BudgetCategory(name: "Savings"),
  210. BudgetCategory(name: "Holidays")
  211. ])
  212. ]
  213. @State static var currentBudgetId: UUID?
  214. static var previews: some View {
  215. NavigationView {
  216. BudgetView(budget: BudgetInfo(name: "Default Budget", sections: budget),
  217. currentBudgetId: $currentBudgetId)
  218. }
  219. .preferredColorScheme(.light)
  220. }
  221. }