BudgetView.swift 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. //
  2. // BudgetView.swift
  3. // ydnab
  4. //
  5. // Created by Andrea Franceschini on 23/09/2020.
  6. //
  7. import SwiftUI
  8. struct BudgetCategoryCell: View {
  9. @State var category: BudgetCategory
  10. @State var amountText: String
  11. @State var color: Color
  12. var body: some View {
  13. HStack {
  14. Text(category.name)
  15. Spacer()
  16. Text(amountText)
  17. .bold()
  18. .foregroundColor(color)
  19. }
  20. }
  21. }
  22. struct BudgetSectionCell: View {
  23. @State var section: BudgetSection
  24. @State var amountText: String
  25. @State var color: Color
  26. @Environment(\.editMode) var editMode
  27. var body: some View {
  28. HStack {
  29. Text(section.name)
  30. .textCase(.none)
  31. Spacer()
  32. if editMode?.wrappedValue == .inactive {
  33. Text(amountText)
  34. .foregroundColor(color)
  35. }
  36. }
  37. .padding(.vertical, 11)
  38. }
  39. }
  40. struct YearAndMonthPicker: View {
  41. @Binding var month: Int
  42. @Binding var monthName: String
  43. @Binding var year: Int
  44. @State var locale: Locale
  45. private var monthNames: [String] {
  46. var c = Calendar(identifier: .gregorian)
  47. c.locale = locale
  48. return c.monthSymbols
  49. }
  50. var columns: [GridItem] = [
  51. GridItem(spacing: 8),
  52. GridItem(spacing: 8),
  53. GridItem(spacing: 8)
  54. ]
  55. var body: some View {
  56. VStack {
  57. HStack {
  58. Button(action: { year -= 1 }) {
  59. Image(systemName: "chevron.backward.square")
  60. .imageScale(.large)
  61. }
  62. Spacer()
  63. Text(String(year)).bold()
  64. Spacer()
  65. Button(action: { year += 1 }) {
  66. Image(systemName: "chevron.forward.square")
  67. .imageScale(.large)
  68. }
  69. }
  70. GeometryReader { geometry in
  71. LazyVGrid(columns: columns, spacing: 4) {
  72. ForEach(monthNames.indices) { i in
  73. Button(action: {
  74. month = i
  75. monthName = monthNames[i]
  76. }, label: {
  77. Text(monthNames[i])
  78. .frame(minWidth: geometry.size.width / 3,
  79. idealWidth: geometry.size.width / 3,
  80. maxWidth: geometry.size.width / 3,
  81. minHeight: geometry.size.height / 4,
  82. idealHeight: geometry.size.height / 4,
  83. maxHeight: geometry.size.height / 4,
  84. alignment: .center)
  85. })
  86. .background(month == i ? Color("bgActive") : Color("bgInactive"))
  87. .foregroundColor(month == i ? Color.white : Color("AccentColor"))
  88. }
  89. }
  90. }
  91. }
  92. .padding(.bottom, 22)
  93. }
  94. }
  95. struct BudgetViewSummary: View {
  96. @Binding var budget: BudgetInfo
  97. @Binding var month: Int
  98. @Binding var monthName: String
  99. @Binding var year: Int
  100. @State private var showMonthPicker: Bool = false
  101. var body: some View {
  102. VStack {
  103. HStack {
  104. Button("\(monthName) \(String(year))", action: { withAnimation() { showMonthPicker.toggle() } } )
  105. Spacer()
  106. }
  107. .padding(.vertical, 11)
  108. if showMonthPicker {
  109. YearAndMonthPicker(month: $month, monthName: $monthName, year: $year, locale: Locale(identifier: budget.localeIdentifier))
  110. }
  111. HStack {
  112. Text("To budget") // Or "overbudgeted"
  113. Spacer()
  114. Text("$ 0.00")
  115. }
  116. }
  117. .padding(.vertical, 11)
  118. }
  119. }
  120. struct BudgetView: View {
  121. @State var budget: BudgetInfo
  122. @Binding var currentBudgetId: UUID?
  123. @State private var month: Int = 0
  124. @State private var monthName: String
  125. @State private var year: Int = Calendar(identifier: .gregorian).component(.year, from: Date())
  126. @AppStorage("lastLoadedBudgetId") private var lastLoadedBudgetId = ""
  127. init(budget: BudgetInfo, currentBudgetId: Binding<UUID?>) {
  128. _budget = .init(initialValue: budget)
  129. _currentBudgetId = currentBudgetId
  130. var cal = Calendar(identifier: .gregorian)
  131. cal.locale = Locale(identifier: self._budget.wrappedValue.localeIdentifier)
  132. let now = Date()
  133. _month = .init(initialValue: cal.component(.month, from: now))
  134. _monthName = .init(initialValue: cal.monthSymbols[self._month.wrappedValue])
  135. _year = .init(initialValue: cal.component(.year, from: now))
  136. }
  137. func formatAmount(_ amount: NSNumber) -> String? {
  138. let f = NumberFormatter()
  139. f.locale = Locale(identifier: budget.localeIdentifier)
  140. f.numberStyle = .currency
  141. return f.string(from: amount)
  142. }
  143. var body: some View {
  144. VStack {
  145. BudgetViewSummary(budget: $budget, month: $month, monthName: $monthName, year: $year)
  146. .padding([.leading, .trailing], 16)
  147. List {
  148. ForEach(budget.sections) { section in
  149. if !section.hidden {
  150. Section(header: BudgetSectionCell(section: section,
  151. amountText: formatAmount(0) ?? "--",
  152. color: 0 < 0 ? Color("negativeAmount") : Color("positiveAmount"))
  153. ) {
  154. ForEach(section.categories) { category in
  155. if !category.hidden {
  156. BudgetCategoryCell(category: category,
  157. amountText: formatAmount(0) ?? "--",
  158. color: 0 < 0 ? Color("negativeAmount") : Color("positiveAmount")
  159. )
  160. }
  161. }
  162. .onMove { (indexSet, s) in
  163. print(indexSet, s)
  164. }
  165. }
  166. }
  167. }
  168. .onMove { (indexSet, s) in
  169. print(indexSet, s)
  170. }
  171. }
  172. .listStyle(PlainListStyle())
  173. .navigationBarTitle(budget.name, displayMode: .inline)
  174. .toolbar {
  175. EditButton()
  176. }
  177. .onAppear() {
  178. print("BudgetView Appears")
  179. currentBudgetId = budget.id
  180. lastLoadedBudgetId = budget.id.uuidString
  181. }
  182. }
  183. }
  184. }
  185. struct BudgetView_Previews: PreviewProvider {
  186. static let budget: [BudgetSection] = [
  187. BudgetSection(name: "Everyday Expenses", categories: [
  188. BudgetCategory(name: "Groceries"),
  189. BudgetCategory(name: "Eating out"),
  190. BudgetCategory(name: "Medical"),
  191. BudgetCategory(name: "Clothing"),
  192. BudgetCategory(name: "Household goods")
  193. ]),
  194. BudgetSection(name: "Travel", categories: [
  195. BudgetCategory(name: "Transport"),
  196. BudgetCategory(name: "Fuel"),
  197. BudgetCategory(name: "Accommodation")
  198. ]),
  199. BudgetSection(name: "Rainy Days", categories: [
  200. BudgetCategory(name: "Emergencies"),
  201. BudgetCategory(name: "Car insurance")
  202. ]),
  203. BudgetSection(name: "Monthly Expenses", categories: [
  204. BudgetCategory(name: "Rent"),
  205. BudgetCategory(name: "Mobile")
  206. ]),
  207. BudgetSection(name: "Savings Goals", categories: [
  208. BudgetCategory(name: "Savings"),
  209. BudgetCategory(name: "Holidays")
  210. ])
  211. ]
  212. @State static var currentBudgetId: UUID?
  213. static var previews: some View {
  214. NavigationView {
  215. BudgetView(budget: BudgetInfo(name: "Default Budget", sections: budget),
  216. currentBudgetId: $currentBudgetId)
  217. }
  218. .preferredColorScheme(.light)
  219. //.environment(\.layoutDirection, .rightToLeft)
  220. }
  221. }