BudgetView.swift 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  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. @State private var showNotImplemented: Bool = false
  102. var body: some View {
  103. VStack {
  104. HStack {
  105. Button(action: { withAnimation() { showMonthPicker.toggle() } } ) {
  106. Text("\(monthName) \(String(year))")
  107. Image(systemName: showMonthPicker ? "chevron.up" : "chevron.down")
  108. }
  109. Spacer()
  110. Button(action: { showNotImplemented.toggle() }) { // TODO: Implement this
  111. Image(systemName: "bolt")
  112. }
  113. Button(action: { showNotImplemented.toggle() }) { // TODO: Implement this
  114. Image(systemName: "gear")
  115. }.padding(.leading, 11)
  116. }
  117. .padding(.vertical, 11)
  118. if showMonthPicker {
  119. YearAndMonthPicker(month: $month, monthName: $monthName, year: $year, locale: Locale(identifier: budget.localeIdentifier))
  120. }
  121. HStack {
  122. Text("To budget") // Or "overbudgeted"
  123. Spacer()
  124. Text("$ 0.00")
  125. }
  126. .alert(isPresented: $showNotImplemented) {
  127. Alert(title: Text("Not Implemented!"))
  128. }
  129. }
  130. .padding(.vertical, 11)
  131. }
  132. }
  133. struct BudgetView: View {
  134. @State var budget: BudgetInfo
  135. @Binding var currentBudgetId: UUID?
  136. @State private var month: Int = 0
  137. @State private var monthName: String
  138. @State private var year: Int = Calendar(identifier: .gregorian).component(.year, from: Date())
  139. @AppStorage("lastLoadedBudgetId") private var lastLoadedBudgetId = ""
  140. init(budget: BudgetInfo, currentBudgetId: Binding<UUID?>) {
  141. _budget = .init(initialValue: budget)
  142. _currentBudgetId = currentBudgetId
  143. var cal = Calendar(identifier: .gregorian)
  144. cal.locale = Locale(identifier: self._budget.wrappedValue.localeIdentifier)
  145. let now = Date()
  146. _month = .init(initialValue: cal.component(.month, from: now))
  147. _monthName = .init(initialValue: cal.monthSymbols[self._month.wrappedValue])
  148. _year = .init(initialValue: cal.component(.year, from: now))
  149. }
  150. func formatAmount(_ amount: NSNumber) -> String? {
  151. let f = NumberFormatter()
  152. f.locale = Locale(identifier: budget.localeIdentifier)
  153. f.numberStyle = .currency
  154. return f.string(from: amount)
  155. }
  156. var body: some View {
  157. VStack {
  158. BudgetViewSummary(budget: $budget, month: $month, monthName: $monthName, year: $year)
  159. .padding([.leading, .trailing], 16)
  160. List {
  161. ForEach(budget.sections) { section in
  162. if !section.hidden {
  163. Section(header: BudgetSectionCell(section: section,
  164. amountText: formatAmount(0) ?? "--",
  165. color: 0 < 0 ? Color("negativeAmount") : Color("positiveAmount"))
  166. ) {
  167. ForEach(section.categories) { category in
  168. if !category.hidden {
  169. BudgetCategoryCell(category: category,
  170. amountText: formatAmount(0) ?? "--",
  171. color: 0 < 0 ? Color("negativeAmount") : Color("positiveAmount")
  172. )
  173. }
  174. }
  175. .onMove { (indexSet, s) in
  176. print(indexSet, s)
  177. }
  178. }
  179. }
  180. }
  181. .onMove { (indexSet, s) in
  182. print(indexSet, s)
  183. }
  184. }
  185. .listStyle(PlainListStyle())
  186. .navigationBarTitle(budget.name, displayMode: .inline)
  187. .toolbar {
  188. EditButton()
  189. }
  190. .onAppear() {
  191. print("BudgetView Appears")
  192. currentBudgetId = budget.id
  193. lastLoadedBudgetId = budget.id.uuidString
  194. }
  195. }
  196. }
  197. }
  198. struct BudgetView_Previews: PreviewProvider {
  199. static let budget: [BudgetSection] = [
  200. BudgetSection(name: "Everyday Expenses", categories: [
  201. BudgetCategory(name: "Groceries"),
  202. BudgetCategory(name: "Eating out"),
  203. BudgetCategory(name: "Medical"),
  204. BudgetCategory(name: "Clothing"),
  205. BudgetCategory(name: "Household goods")
  206. ]),
  207. BudgetSection(name: "Travel", categories: [
  208. BudgetCategory(name: "Transport"),
  209. BudgetCategory(name: "Fuel"),
  210. BudgetCategory(name: "Accommodation")
  211. ]),
  212. BudgetSection(name: "Rainy Days", categories: [
  213. BudgetCategory(name: "Emergencies"),
  214. BudgetCategory(name: "Car insurance")
  215. ]),
  216. BudgetSection(name: "Monthly Expenses", categories: [
  217. BudgetCategory(name: "Rent"),
  218. BudgetCategory(name: "Mobile")
  219. ]),
  220. BudgetSection(name: "Savings Goals", categories: [
  221. BudgetCategory(name: "Savings"),
  222. BudgetCategory(name: "Holidays")
  223. ])
  224. ]
  225. @State static var currentBudgetId: UUID?
  226. static var previews: some View {
  227. NavigationView {
  228. BudgetView(budget: BudgetInfo(name: "Default Budget", sections: budget),
  229. currentBudgetId: $currentBudgetId)
  230. }
  231. .preferredColorScheme(.light)
  232. //.environment(\.layoutDirection, .rightToLeft)
  233. }
  234. }