Compare commits
	
		
			34 Commits
		
	
	
		
			45b5e33491
			...
			last_versi
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 5425b22ba2 | |||
| 305012ffd4 | |||
| 8c31d868b9 | |||
| 9d478b9cbf | |||
| 452eb73773 | |||
| 631555af59 | |||
| 0f65166123 | |||
| 595aaefedc | |||
| 9d8a5e6685 | |||
| 2acabf4d3b | |||
| 15fa4aadbd | |||
| 78a407d0ec | |||
| 54cd86c34b | |||
| 66fd966de8 | |||
| 5561f50736 | |||
| 6b25e6e552 | |||
| 360a36f024 | |||
| ecbac615e9 | |||
| 87392cc73c | |||
| 1b95feb5d4 | |||
| 99b1ec82e6 | |||
| 7ead0e5ebb | |||
| bcae40e0e2 | |||
| aa3c2f9304 | |||
| 96811f5bfd | |||
| c44d4553c3 | |||
| d722172cb3 | |||
| 43dd151cb4 | |||
| c2cc71eae0 | |||
| 0c9b365f7f | |||
| f820265dba | |||
| 85b7a0a3e6 | |||
| 861e8bf904 | |||
| aaa95bd3a6 | 
| @@ -6,7 +6,7 @@ plugins { | ||||
| } | ||||
|  | ||||
| android { | ||||
|     namespace = "com.example.flutter_empty" | ||||
|     namespace = "com.hyperling.expense_tracker" | ||||
|     compileSdk = flutter.compileSdkVersion | ||||
|     ndkVersion = flutter.ndkVersion | ||||
|  | ||||
| @@ -21,7 +21,7 @@ android { | ||||
|  | ||||
|     defaultConfig { | ||||
|         // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html). | ||||
|         applicationId = "com.example.flutter_empty" | ||||
|         applicationId = "com.hyperling.expense_tracker" | ||||
|         // You can update the following values to match your application needs. | ||||
|         // For more information, see: https://flutter.dev/to/review-gradle-config. | ||||
|         minSdk = flutter.minSdkVersion | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <application | ||||
|         android:label="flutter_empty" | ||||
|         android:label="Recurring Expense Tracker" | ||||
|         android:name="${applicationName}" | ||||
|         android:icon="@mipmap/ic_launcher"> | ||||
|         <activity | ||||
|   | ||||
							
								
								
									
										45
									
								
								lib/db.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								lib/db.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| // https://docs.flutter.dev/cookbook/persistence/sqlite | ||||
|  | ||||
| // SQLite | ||||
| import 'dart:async'; | ||||
| import 'package:flutter/widgets.dart'; | ||||
| import 'package:flutter_expense_tracker/models/frequency.dart'; | ||||
| import 'package:path/path.dart'; | ||||
| import 'package:sqflite/sqflite.dart'; | ||||
|  | ||||
| // Local | ||||
| import '/models/expense.dart'; | ||||
|  | ||||
| void loadDB() async { | ||||
|   // Avoid errors caused by flutter upgrade. | ||||
|   WidgetsFlutterBinding.ensureInitialized(); | ||||
|  | ||||
|   final String frequencies = | ||||
|       "'${Frequency.values.map((freq) => freq.title).join("','")}'"; | ||||
|   print(frequencies); | ||||
|  | ||||
|   // Open the database and store the reference. | ||||
|   final database = openDatabase( | ||||
|     // Set the path to the database. Note: Using the `join` function from the | ||||
|     // `path` package is best practice to ensure the path is correctly | ||||
|     // constructed for each platform. | ||||
|     join(await getDatabasesPath(), 'expense_tracker.db'), | ||||
|  | ||||
|     onCreate: (db, version) { | ||||
|       // Run the CREATE TABLE statement on the database. | ||||
|       return db.execute( | ||||
|         """ | ||||
|         CREATE TABLE expense | ||||
|           ( id INTEGER PRIMARY KEY | ||||
|           , name TEXT | ||||
|           , cost DOUBLE | ||||
|           , frequency TEXT CHECK(frequency IN ($frequencies) ) | ||||
|           , description TEXT | ||||
|         )""", | ||||
|       ); | ||||
|     }, | ||||
|     // Set the version. This executes the onCreate function and provides a | ||||
|     // path to perform database upgrades and downgrades. | ||||
|     version: 1, | ||||
|   ); | ||||
| } | ||||
							
								
								
									
										211
									
								
								lib/main.dart
									
									
									
									
									
								
							
							
						
						
									
										211
									
								
								lib/main.dart
									
									
									
									
									
								
							| @@ -1,100 +1,145 @@ | ||||
| // Helpful guides: | ||||
| // - https://flutter.dev/docs/cookbook/forms/validation | ||||
|  | ||||
| // Flutter | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| // Local | ||||
| import '/pages/expense.dart'; | ||||
| import '/pages/income.dart'; | ||||
| import '/pages/asset.dart'; | ||||
| import '/pages/report.dart'; | ||||
| import '/pages/settings.dart'; | ||||
| import '/pages/help.dart'; | ||||
| import '/db.dart'; | ||||
|  | ||||
| void main() { | ||||
|   loadDB(); | ||||
|   runApp(const MainApp()); | ||||
| } | ||||
|  | ||||
| class MainApp extends StatefulWidget { | ||||
| class MainApp extends StatelessWidget { | ||||
|   const MainApp({super.key}); | ||||
|  | ||||
|   @override | ||||
|   State<MainApp> createState() => _MainAppState(); | ||||
| } | ||||
|  | ||||
| class _MainAppState extends State<MainApp> { | ||||
|   final nameFieldController = TextEditingController(); | ||||
|  | ||||
|   @override | ||||
|   void dispose() { | ||||
|     // Clean up the controller when the widget is disposed. | ||||
|     nameFieldController.dispose(); | ||||
|     super.dispose(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     const inputWidth = 400.0; | ||||
|     const inputHeight = 50.0; | ||||
|  | ||||
|     const recurrenceValues = <DropdownMenuEntry>[ | ||||
|       DropdownMenuEntry(value: Recurrence.daily, label: "Daily"), | ||||
|       DropdownMenuEntry(value: Recurrence.weekly, label: "Weekly"), | ||||
|       DropdownMenuEntry(value: Recurrence.biweekly, label: "Biweekly"), | ||||
|       DropdownMenuEntry(value: Recurrence.montly, label: "Monthly"), | ||||
|       DropdownMenuEntry(value: Recurrence.yearly, label: "Yearly"), | ||||
|     ]; | ||||
|  | ||||
|     return const MaterialApp( | ||||
|       home: Scaffold( | ||||
|         body: Center( | ||||
|           child: Column(mainAxisSize: MainAxisSize.min, spacing: 10, children: [ | ||||
|             Text('Input an expense below!'), | ||||
|             SizedBox( | ||||
|                 width: inputWidth, | ||||
|                 height: inputHeight, | ||||
|                 child: TextField( | ||||
|                   keyboardType: TextInputType.text, | ||||
|                   decoration: InputDecoration( | ||||
|                     labelText: "Name", | ||||
|                     hintText: "Example: Red Pocket Phone Bill", | ||||
|                   ), | ||||
|                   // https://docs.flutter.dev/cookbook/forms/retrieve-input | ||||
|                   //controller: nameFieldController, | ||||
|                 )), | ||||
|             SizedBox( | ||||
|               width: inputWidth, | ||||
|               height: inputHeight, | ||||
|               child: TextField( | ||||
|                 keyboardType: TextInputType.numberWithOptions(decimal: true), | ||||
|                 decoration: InputDecoration( | ||||
|                     labelText: "Cost", hintText: "Example: 10.00"), | ||||
|               ), | ||||
|             ), | ||||
|             DropdownMenu( | ||||
|               dropdownMenuEntries: recurrenceValues, | ||||
|               width: inputWidth, | ||||
|               label: Text("Recurrence"), | ||||
|               hintText: "Example: Monthly", | ||||
|             ), | ||||
|             SizedBox( | ||||
|                 width: inputWidth, | ||||
|                 height: inputHeight, | ||||
|                 child: TextField( | ||||
|                   keyboardType: TextInputType.text, | ||||
|                   decoration: InputDecoration( | ||||
|                       labelText: "Description", | ||||
|                       hintText: | ||||
|                           "Example: 1GB data with unlimited talk & text." | ||||
|                     ), | ||||
|                 )), | ||||
|           ]), | ||||
|         ), | ||||
|     return MaterialApp( | ||||
|       title: 'Recurring Expense Tracker', | ||||
|       theme: ThemeData( | ||||
|         useMaterial3: true, | ||||
|         colorScheme: ColorScheme.fromSeed(seedColor: Colors.green), | ||||
|       ), | ||||
|       darkTheme: ThemeData( | ||||
|         useMaterial3: true, | ||||
|         brightness: Brightness.dark, | ||||
|         colorSchemeSeed: Colors.green, | ||||
|       ), | ||||
|       themeMode: ThemeMode.system, | ||||
|       home: HomePage(), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| // https://www.tutorialspoint.com/dart_programming/dart_programming_enumeration.htm | ||||
| enum Recurrence { daily, weekly, biweekly, montly, yearly } | ||||
| class HomePage extends StatefulWidget { | ||||
|   const HomePage({ | ||||
|     super.key, | ||||
|   }); | ||||
|  | ||||
| class Expense { | ||||
|   String name; | ||||
|   double cost; | ||||
|   Recurrence recurrence; | ||||
|   String description; | ||||
|  | ||||
|   Expense(this.name, this.cost, this.recurrence, this.description); | ||||
|   @override | ||||
|   State<HomePage> createState() => _HomePageState(); | ||||
| } | ||||
|  | ||||
| class _HomePageState extends State<HomePage> { | ||||
|   var pageSelected = 0; | ||||
|  | ||||
|   refresh() { | ||||
|     setState(() {}); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     Widget page; | ||||
|     Widget? dialog; | ||||
|     switch (pageSelected) { | ||||
|       case 0: | ||||
|         page = ExpensePage(); | ||||
|         dialog = ExpenseInputDialog( | ||||
|           notifyParent: refresh, | ||||
|         ); | ||||
|       case 1: | ||||
|         page = IncomePage(); | ||||
|       case 2: | ||||
|         page = AssetPage(); | ||||
|       case 3: | ||||
|         page = ProjectionPage(); | ||||
|       case 4: | ||||
|         page = SettingsPage(); | ||||
|       case 5: | ||||
|         page = HelpPage(); | ||||
|       default: | ||||
|         throw UnimplementedError('No widget for page $pageSelected yet!'); | ||||
|     } | ||||
|  | ||||
|     Future<void> addNewValue(BuildContext context) { | ||||
|       return showDialog( | ||||
|         context: context, | ||||
|         builder: (_) => AlertDialog(content: dialog), | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     Widget? floatingButton; | ||||
|     if (dialog != null) { | ||||
|       floatingButton = IconButton( | ||||
|         onPressed: () { | ||||
|           addNewValue(context); | ||||
|         }, | ||||
|         icon: Icon(Icons.add), | ||||
|         color: Theme.of(context).colorScheme.onSurface, | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return LayoutBuilder(builder: (context, constraints) { | ||||
|       return Scaffold( | ||||
|         appBar: AppBar(title: Text("Expense Tracker")), | ||||
|         drawer: NavigationRail( | ||||
|           extended: true, | ||||
|           destinations: [ | ||||
|             NavigationRailDestination( | ||||
|               icon: Icon(Icons.payment), | ||||
|               label: Text('Expenses'), | ||||
|             ), | ||||
|             NavigationRailDestination( | ||||
|               icon: Icon(Icons.account_balance), | ||||
|               label: Text('Income'), | ||||
|             ), | ||||
|             NavigationRailDestination( | ||||
|               icon: Icon(Icons.attach_money), | ||||
|               label: Text('Liquid Assets'), | ||||
|             ), | ||||
|             NavigationRailDestination( | ||||
|               icon: Icon(Icons.bar_chart), | ||||
|               label: Text('Reports'), | ||||
|             ), | ||||
|             NavigationRailDestination( | ||||
|               icon: Icon(Icons.settings), | ||||
|               label: Text('Settings'), | ||||
|             ), | ||||
|             NavigationRailDestination( | ||||
|               icon: Icon(Icons.help), | ||||
|               label: Text('Help'), | ||||
|             ), | ||||
|           ], | ||||
|           selectedIndex: pageSelected, | ||||
|           onDestinationSelected: (value) { | ||||
|             setState(() { | ||||
|               pageSelected = value; | ||||
|               Navigator.pop(context); | ||||
|             }); | ||||
|           }, | ||||
|         ), | ||||
|         body: Container( | ||||
|           color: Theme.of(context).colorScheme.primaryContainer, | ||||
|           child: Center(child: page), | ||||
|         ), | ||||
|         floatingActionButton: floatingButton, | ||||
|       ); | ||||
|     }); | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										27
									
								
								lib/models/expense.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								lib/models/expense.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| import '/models/frequency.dart'; | ||||
|  | ||||
| class Expense { | ||||
|   final String name; | ||||
|   final double cost; | ||||
|   final Frequency frequency; | ||||
|   final String description; | ||||
|  | ||||
|   const Expense( | ||||
|       {required this.name, | ||||
|       required this.cost, | ||||
|       required this.frequency, | ||||
|       required this.description}); | ||||
|  | ||||
|   @override | ||||
|   String toString() { | ||||
|     return "$name, $cost, ${frequency.title}, $description"; | ||||
|   } | ||||
|  | ||||
|   double calcComparableCost() { | ||||
|     return cost * frequency.timesPerYear; | ||||
|   } | ||||
|  | ||||
|   double calcComparableCostDaily() { | ||||
|     return cost / frequency.numDays; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										63
									
								
								lib/models/frequency.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								lib/models/frequency.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| // https://www.tutorialspoint.com/dart_programming/dart_programming_enumeration.htm | ||||
| enum Frequency { | ||||
|   daily( | ||||
|     title: "Daily", | ||||
|     hint: "Once Per Day", | ||||
|     timesPerYear: 364.25, | ||||
|     numDays: 1, | ||||
|   ), | ||||
|   weekly( | ||||
|     title: "Weekly", | ||||
|     hint: "Once Per Week", | ||||
|     timesPerYear: (364.25 / 7), | ||||
|     numDays: 7, | ||||
|   ), | ||||
|   biweekly( | ||||
|     title: "Biweekly", | ||||
|     hint: "Every Other Week", | ||||
|     timesPerYear: (364.25 / 14), | ||||
|     numDays: 14, | ||||
|   ), | ||||
|   bimonthly( | ||||
|     title: "Bimonthly", | ||||
|     hint: "Twice Per Month", | ||||
|     timesPerYear: 24, | ||||
|     numDays: (364.25 / 24), | ||||
|   ), | ||||
|   monthly( | ||||
|     title: "Monthly", | ||||
|     hint: "Once Per Month", | ||||
|     timesPerYear: 12, | ||||
|     numDays: (364.25 / 12), | ||||
|   ), | ||||
|   quarterly( | ||||
|     title: "Quarterly", | ||||
|     hint: "Every Three Months", | ||||
|     timesPerYear: 4, | ||||
|     numDays: (364.25 / 4), | ||||
|   ), | ||||
|   biannual( | ||||
|     title: "Biannual", | ||||
|     hint: "Twice Per Year", | ||||
|     timesPerYear: 2, | ||||
|     numDays: (364.25 / 2), | ||||
|   ), | ||||
|   yearly( | ||||
|     title: "Yearly", | ||||
|     hint: "Once Per Year", | ||||
|     timesPerYear: 1, | ||||
|     numDays: 364.25, | ||||
|   ); | ||||
|  | ||||
|   const Frequency({ | ||||
|     required this.title, | ||||
|     required this.hint, | ||||
|     required this.timesPerYear, | ||||
|     required this.numDays, | ||||
|   }); | ||||
|  | ||||
|   final String title; | ||||
|   final String hint; | ||||
|   final double timesPerYear; | ||||
|   final double numDays; | ||||
| } | ||||
							
								
								
									
										12
									
								
								lib/pages/asset.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								lib/pages/asset.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| class AssetPage extends StatelessWidget { | ||||
|   const AssetPage({ | ||||
|     super.key, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Placeholder(); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										365
									
								
								lib/pages/expense.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										365
									
								
								lib/pages/expense.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,365 @@ | ||||
| // Flutter | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| // Local | ||||
| import '/models/expense.dart'; | ||||
| import '/models/frequency.dart'; | ||||
|  | ||||
| List<Expense> expenses = []; | ||||
|  | ||||
| class ExpensePage extends StatefulWidget { | ||||
|   const ExpensePage({ | ||||
|     super.key, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
|   State<ExpensePage> createState() => _ExpensePageState(); | ||||
| } | ||||
|  | ||||
| class _ExpensePageState extends State<ExpensePage> { | ||||
|   refresh() { | ||||
|     setState(() {}); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final theme = Theme.of(context); | ||||
|  | ||||
|     expenses.sort( | ||||
|       (a, b) => (b.calcComparableCost() - a.calcComparableCost()).toInt(), | ||||
|     ); | ||||
|  | ||||
|     return expenses.isEmpty | ||||
|         ? Text("Add expenses to get started!") | ||||
|         : ListView.builder( | ||||
|             itemCount: expenses.length, | ||||
|             itemBuilder: (_, index) { | ||||
|               final Expense curr = expenses[index]; | ||||
|               final String estimateSymbolYearly = curr.frequency.timesPerYear | ||||
|                           .toStringAsFixed(2) | ||||
|                           .endsWith(".00") && | ||||
|                       curr.calcComparableCost().toStringAsFixed(3).endsWith("0") | ||||
|                   ? "" | ||||
|                   : "~"; | ||||
|               final String estimateSymbolDaily = | ||||
|                   curr.frequency.numDays.toStringAsFixed(2).endsWith(".00") && | ||||
|                           curr | ||||
|                               .calcComparableCostDaily() | ||||
|                               .toStringAsFixed(3) | ||||
|                               .endsWith("0") | ||||
|                       ? "" | ||||
|                       : "~"; | ||||
|               return Padding( | ||||
|                 padding: const EdgeInsets.all(4.0), | ||||
|                 child: Dismissible( | ||||
|                   key: Key(curr.toString()), | ||||
|                   background: Container( | ||||
|                     color: Colors.red, | ||||
|                     child: Row( | ||||
|                       mainAxisAlignment: MainAxisAlignment.start, | ||||
|                       children: [ | ||||
|                         Icon(Icons.delete), | ||||
|                         Text("Delete"), | ||||
|                       ], | ||||
|                     ), | ||||
|                   ), | ||||
|                   secondaryBackground: Container( | ||||
|                     color: Colors.orange, | ||||
|                     child: Row( | ||||
|                       mainAxisAlignment: MainAxisAlignment.end, | ||||
|                       children: [ | ||||
|                         Text("Edit"), | ||||
|                         Icon(Icons.edit), | ||||
|                       ], | ||||
|                     ), | ||||
|                   ), | ||||
|                   onDismissed: (direction) { | ||||
|                     setState(() { | ||||
|                       expenses.remove(curr); | ||||
|                     }); | ||||
|                     switch (direction) { | ||||
|                       case DismissDirection.startToEnd: | ||||
|                         // Only remove the item from the list. | ||||
|                         break; | ||||
|                       case DismissDirection.endToStart: | ||||
|                         // Open an edit dialog, then remove the item from the list. | ||||
|                         showDialog( | ||||
|                           context: context, | ||||
|                           builder: (_) => AlertDialog( | ||||
|                             content: ExpenseInputDialog( | ||||
|                               notifyParent: refresh, | ||||
|                               expense: curr, | ||||
|                             ), | ||||
|                           ), | ||||
|                         ); | ||||
|                         break; | ||||
|                       default: | ||||
|                         UnimplementedError( | ||||
|                           "Direction ${direction.toString()} not recognized.", | ||||
|                         ); | ||||
|                     } | ||||
|                   }, | ||||
|                   child: Container( | ||||
|                     decoration: BoxDecoration( | ||||
|                       borderRadius: BorderRadius.circular(4), | ||||
|                       color: theme.colorScheme.onPrimary, | ||||
|                     ), | ||||
|                     child: Padding( | ||||
|                       padding: const EdgeInsets.all(4.0), | ||||
|                       child: Row( | ||||
|                         mainAxisSize: MainAxisSize.max, | ||||
|                         children: [ | ||||
|                           Column( | ||||
|                             mainAxisSize: MainAxisSize.min, | ||||
|                             crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                             children: [ | ||||
|                               Text( | ||||
|                                 curr.name, | ||||
|                                 style: TextStyle(fontSize: 20.0), | ||||
|                               ), | ||||
|                               Text( | ||||
|                                 "${curr.cost.toStringAsFixed(2)} ${curr.frequency.title}", | ||||
|                                 style: TextStyle(fontSize: 12.0), | ||||
|                               ), | ||||
|                             ], | ||||
|                           ), | ||||
|                           Expanded( | ||||
|                             child: Center( | ||||
|                               child: Text( | ||||
|                                 curr.description, | ||||
|                                 style: TextStyle( | ||||
|                                   fontSize: 12.0, | ||||
|                                 ), | ||||
|                                 softWrap: true, | ||||
|                                 textAlign: TextAlign.center, | ||||
|                               ), | ||||
|                             ), | ||||
|                           ), | ||||
|                           Column( | ||||
|                             crossAxisAlignment: CrossAxisAlignment.end, | ||||
|                             children: [ | ||||
|                               //if (curr.frequency != Frequency.daily) | ||||
|                               Text( | ||||
|                                 "$estimateSymbolDaily${curr.calcComparableCostDaily().toStringAsFixed(2)} ${Frequency.daily.title}", | ||||
|                                 style: TextStyle(fontSize: 12.0), | ||||
|                               ), | ||||
|                               //if (curr.frequency != Frequency.yearly) | ||||
|                               Text( | ||||
|                                 "$estimateSymbolYearly${curr.calcComparableCost().toStringAsFixed(2)} ${Frequency.yearly.title}", | ||||
|                                 style: TextStyle(fontSize: 12.0), | ||||
|                               ), | ||||
|                             ], | ||||
|                           ), | ||||
|                         ], | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ), | ||||
|               ); | ||||
|             }, | ||||
|           ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class ExpenseInputDialog extends StatefulWidget { | ||||
|   final Function() notifyParent; | ||||
|   final Expense? expense; | ||||
|  | ||||
|   const ExpenseInputDialog({ | ||||
|     super.key, | ||||
|     required this.notifyParent, | ||||
|     this.expense, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
|   State<ExpenseInputDialog> createState() => _ExpenseInputDialogState(); | ||||
| } | ||||
|  | ||||
| class _ExpenseInputDialogState extends State<ExpenseInputDialog> { | ||||
|   final _expenseFormKey = GlobalKey<FormState>(); | ||||
|  | ||||
|   String _name = ""; | ||||
|   double _cost = 0; | ||||
|   Frequency _freq = Frequency.monthly; | ||||
|   String _desc = ""; | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     if (widget.expense != null) { | ||||
|       _name = widget.expense!.name; | ||||
|       _cost = widget.expense!.cost; | ||||
|       _freq = widget.expense!.frequency; | ||||
|       _desc = widget.expense!.description; | ||||
|     } | ||||
|  | ||||
|     return Column( | ||||
|       // prevent AlertDialog from taking full vertical height. | ||||
|       mainAxisSize: MainAxisSize.min, | ||||
|       children: [ | ||||
|         Container( | ||||
|           alignment: FractionalOffset.topRight, | ||||
|           child: IconButton( | ||||
|             onPressed: () { | ||||
|               if (widget.expense != null) { | ||||
|                 setState(() { | ||||
|                   expenses.add(widget.expense!); | ||||
|                   widget.notifyParent(); | ||||
|                 }); | ||||
|               } | ||||
|               Navigator.of(context).pop(); | ||||
|             }, | ||||
|             icon: Icon(Icons.clear), | ||||
|           ), | ||||
|         ), | ||||
|         AlertDialog( | ||||
|           insetPadding: EdgeInsets.all(0), | ||||
|           title: Center( | ||||
|             child: widget.expense == null | ||||
|                 ? Text("New Expense") | ||||
|                 : Text("Edit Expense"), | ||||
|           ), | ||||
|           content: Form( | ||||
|             key: _expenseFormKey, | ||||
|             child: Column( | ||||
|               mainAxisSize: MainAxisSize.min, | ||||
|               children: [ | ||||
|                 TextFormField( | ||||
|                   keyboardType: TextInputType.text, | ||||
|                   textCapitalization: TextCapitalization.words, | ||||
|                   decoration: InputDecoration( | ||||
|                     labelText: "Name", | ||||
|                     hintText: "Example: Red Pocket", | ||||
|                     hintStyle: TextStyle(fontSize: 10.0), | ||||
|                     errorStyle: TextStyle(fontSize: 10.0), | ||||
|                   ), | ||||
|                   initialValue: _name, | ||||
|                   validator: (value) { | ||||
|                     if (value!.isEmpty) { | ||||
|                       return "Name must be provided."; | ||||
|                     } | ||||
|                     if (!expenses.every((expense) => expense.name != value)) { | ||||
|                       return "Name must be unique, already in use."; | ||||
|                     } | ||||
|                     return null; | ||||
|                   }, | ||||
|                   onSaved: (value) { | ||||
|                     _name = value!; | ||||
|                   }, | ||||
|                 ), | ||||
|                 TextFormField( | ||||
|                   keyboardType: TextInputType.numberWithOptions(decimal: true), | ||||
|                   decoration: InputDecoration( | ||||
|                     labelText: "Cost", | ||||
|                     hintText: "Example: 10.00", | ||||
|                     hintStyle: TextStyle(fontSize: 10.0), | ||||
|                     errorStyle: TextStyle(fontSize: 10.0), | ||||
|                   ), | ||||
|                   initialValue: _cost != 0 ? _cost.toString() : "", | ||||
|                   validator: (value) { | ||||
|                     if (value == null || value.isEmpty) { | ||||
|                       return "Cost must be provided."; | ||||
|                     } | ||||
|                     if (double.parse(value) < 0) { | ||||
|                       return "Please use the Income page rather than having negative expenses."; | ||||
|                     } | ||||
|                     if (double.parse(value) < 0.01) { | ||||
|                       return "Cost must be one hundreth (0.01) or higher."; | ||||
|                     } | ||||
|                     if (double.tryParse(value) == null) { | ||||
|                       return "Cost must be a valid number."; | ||||
|                     } | ||||
|                     return null; | ||||
|                   }, | ||||
|                   onSaved: (value) { | ||||
|                     _cost = double.parse(value!); | ||||
|                   }, | ||||
|                 ), | ||||
|                 DropdownButtonFormField( | ||||
|                   items: Frequency.values | ||||
|                       .map( | ||||
|                         (freq) => DropdownMenuItem( | ||||
|                           value: freq, | ||||
|                           child: Row( | ||||
|                             children: [ | ||||
|                               Text( | ||||
|                                 freq.title, | ||||
|                               ), | ||||
|                               Padding( | ||||
|                                 padding: EdgeInsets.all(1.0), | ||||
|                                 child: Text( | ||||
|                                   " (${freq.hint})", | ||||
|                                   style: TextStyle(fontSize: 10.0), | ||||
|                                 ), | ||||
|                               ), | ||||
|                             ], | ||||
|                           ), | ||||
|                         ), | ||||
|                       ) | ||||
|                       .toList(), | ||||
|                   value: _freq, | ||||
|                   decoration: InputDecoration( | ||||
|                     labelText: "Frequency", | ||||
|                     errorStyle: TextStyle(fontSize: 10.0), | ||||
|                   ), | ||||
|                   validator: (value) { | ||||
|                     if (value == null) { | ||||
|                       return "Frequency must be provided."; | ||||
|                     } | ||||
|                     if (!Frequency.values.contains(value)) { | ||||
|                       return "Value not valid."; | ||||
|                     } | ||||
|                     return null; | ||||
|                   }, | ||||
|                   onChanged: (value) { | ||||
|                     _freq = value!; | ||||
|                   }, | ||||
|                 ), | ||||
|                 TextFormField( | ||||
|                   keyboardType: TextInputType.text, | ||||
|                   textCapitalization: TextCapitalization.sentences, | ||||
|                   decoration: InputDecoration( | ||||
|                     labelText: "Description", | ||||
|                     hintText: "Example: 1GB data with unlimited talk & text.", | ||||
|                     hintStyle: TextStyle(fontSize: 8.0), | ||||
|                     errorStyle: TextStyle(fontSize: 10.0), | ||||
|                   ), | ||||
|                   initialValue: _desc, | ||||
|                   validator: (value) { | ||||
|                     return null; | ||||
|                   }, | ||||
|                   onSaved: (value) { | ||||
|                     _desc = value!; | ||||
|                   }, | ||||
|                 ), | ||||
|               ], | ||||
|             ), | ||||
|           ), | ||||
|           actions: [ | ||||
|             Center( | ||||
|               child: ElevatedButton.icon( | ||||
|                 onPressed: () { | ||||
|                   if (_expenseFormKey.currentState!.validate()) { | ||||
|                     _expenseFormKey.currentState!.save(); | ||||
|                     setState(() { | ||||
|                       expenses.add( | ||||
|                         Expense( | ||||
|                             name: _name, | ||||
|                             cost: _cost, | ||||
|                             frequency: _freq, | ||||
|                             description: _desc), | ||||
|                       ); | ||||
|                     }); | ||||
|                     widget.notifyParent(); | ||||
|                     Navigator.of(context).pop(); | ||||
|                   } | ||||
|                 }, | ||||
|                 icon: Icon(Icons.save), | ||||
|                 label: Text('Submit'), | ||||
|               ), | ||||
|             ) | ||||
|           ], | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										72
									
								
								lib/pages/help.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								lib/pages/help.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,72 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| class HelpPage extends StatelessWidget { | ||||
|   const HelpPage({ | ||||
|     super.key, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final theme = Theme.of(context); | ||||
|  | ||||
|     return Column( | ||||
|       children: [ | ||||
|         Expanded( | ||||
|           child: Padding( | ||||
|             padding: const EdgeInsets.all(8.0), | ||||
|             child: Container( | ||||
|               decoration: BoxDecoration( | ||||
|                 borderRadius: BorderRadius.circular(4), | ||||
|                 color: theme.colorScheme.onPrimary, | ||||
|               ), | ||||
|               child: Column( | ||||
|                 children: [ | ||||
|                   Text("This app is meant to be a simple budgeting tool," | ||||
|                       " allowing you to view your income and expenses at a high level" | ||||
|                       " without micro managing specific budget items or adding receipts."), | ||||
|                   //Text("Another paragraph.") | ||||
|                 ], | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|         Row( | ||||
|           children: [ | ||||
|             Expanded( | ||||
|               child: Padding( | ||||
|                 padding: const EdgeInsets.all(8.0), | ||||
|                 child: Container( | ||||
|                   decoration: BoxDecoration( | ||||
|                     borderRadius: BorderRadius.circular(4), | ||||
|                     color: theme.colorScheme.onPrimary, | ||||
|                   ), | ||||
|                   child: TextButton.icon( | ||||
|                     onPressed: () {}, | ||||
|                     icon: Icon(Icons.code), | ||||
|                     label: Text("Code Repository"), | ||||
|                   ), | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|             Expanded( | ||||
|               child: Padding( | ||||
|                 padding: const EdgeInsets.all(8.0), | ||||
|                 child: Container( | ||||
|                   decoration: BoxDecoration( | ||||
|                     borderRadius: BorderRadius.circular(4), | ||||
|                     color: theme.colorScheme.onPrimary, | ||||
|                   ), | ||||
|                   child: TextButton.icon( | ||||
|                     onPressed: () {}, | ||||
|                     icon: Icon(Icons.web_asset), | ||||
|                     label: Text("Personal Website"), | ||||
|                   ), | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|           ], | ||||
|         ) | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										12
									
								
								lib/pages/income.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								lib/pages/income.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| class IncomePage extends StatelessWidget { | ||||
|   const IncomePage({ | ||||
|     super.key, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Placeholder(); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										12
									
								
								lib/pages/report.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								lib/pages/report.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| class ProjectionPage extends StatelessWidget { | ||||
|   const ProjectionPage({ | ||||
|     super.key, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Placeholder(); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										12
									
								
								lib/pages/settings.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								lib/pages/settings.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| class SettingsPage extends StatelessWidget { | ||||
|   const SettingsPage({ | ||||
|     super.key, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Placeholder(); | ||||
|   } | ||||
| } | ||||
| @@ -4,10 +4,10 @@ project(runner LANGUAGES CXX) | ||||
|  | ||||
| # The name of the executable created for the application. Change this to change | ||||
| # the on-disk name of your application. | ||||
| set(BINARY_NAME "flutter_empty") | ||||
| set(BINARY_NAME "expense_tracker") | ||||
| # The unique GTK application identifier for this application. See: | ||||
| # https://wiki.gnome.org/HowDoI/ChooseApplicationID | ||||
| set(APPLICATION_ID "com.example.flutter_empty") | ||||
| set(APPLICATION_ID "com.hyperling.expense_tracker") | ||||
|  | ||||
| # Explicitly opt in to modern CMake behaviors to avoid warnings with recent | ||||
| # versions of CMake. | ||||
|   | ||||
| @@ -40,11 +40,11 @@ static void my_application_activate(GApplication* application) { | ||||
|   if (use_header_bar) { | ||||
|     GtkHeaderBar* header_bar = GTK_HEADER_BAR(gtk_header_bar_new()); | ||||
|     gtk_widget_show(GTK_WIDGET(header_bar)); | ||||
|     gtk_header_bar_set_title(header_bar, "flutter_empty"); | ||||
|     gtk_header_bar_set_title(header_bar, "Recurring Expense Tracker"); | ||||
|     gtk_header_bar_set_show_close_button(header_bar, TRUE); | ||||
|     gtk_window_set_titlebar(window, GTK_WIDGET(header_bar)); | ||||
|   } else { | ||||
|     gtk_window_set_title(window, "flutter_empty"); | ||||
|     gtk_window_set_title(window, "Recurring Expense Tracker"); | ||||
|   } | ||||
|  | ||||
|   gtk_window_set_default_size(window, 1280, 720); | ||||
|   | ||||
| @@ -5,6 +5,8 @@ | ||||
| import FlutterMacOS | ||||
| import Foundation | ||||
|  | ||||
| import sqflite_darwin | ||||
|  | ||||
| func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) { | ||||
|   SqflitePlugin.register(with: registry.registrar(forPlugin: "SqflitePlugin")) | ||||
| } | ||||
|   | ||||
							
								
								
									
										66
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
									
										66
									
								
								pubspec.lock
									
									
									
									
									
								
							| @@ -131,6 +131,22 @@ packages: | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.9.0" | ||||
|   platform: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: platform | ||||
|       sha256: "5d6b1b0036a5f331ebc77c850ebc8506cbc1e9416c27e59b439f917a902a4984" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.1.6" | ||||
|   plugin_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: plugin_platform_interface | ||||
|       sha256: "4820fbfdb9478b1ebae27888254d445073732dae3d6ea81f0b7e06d5dedc3f02" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.1.8" | ||||
|   sky_engine: | ||||
|     dependency: transitive | ||||
|     description: flutter | ||||
| @@ -144,6 +160,46 @@ packages: | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.10.0" | ||||
|   sqflite: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: sqflite | ||||
|       sha256: "2d7299468485dca85efeeadf5d38986909c5eb0cd71fd3db2c2f000e6c9454bb" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.4.1" | ||||
|   sqflite_android: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: sqflite_android | ||||
|       sha256: "78f489aab276260cdd26676d2169446c7ecd3484bbd5fead4ca14f3ed4dd9ee3" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.4.0" | ||||
|   sqflite_common: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: sqflite_common | ||||
|       sha256: "761b9740ecbd4d3e66b8916d784e581861fd3c3553eda85e167bc49fdb68f709" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.5.4+6" | ||||
|   sqflite_darwin: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: sqflite_darwin | ||||
|       sha256: "22adfd9a2c7d634041e96d6241e6e1c8138ca6817018afc5d443fef91dcefa9c" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.4.1+1" | ||||
|   sqflite_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: sqflite_platform_interface | ||||
|       sha256: "8dd4515c7bdcae0a785b0062859336de775e8c65db81ae33dd5445f35be61920" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.4.0" | ||||
|   stack_trace: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -168,6 +224,14 @@ packages: | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.3.0" | ||||
|   synchronized: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: synchronized | ||||
|       sha256: "69fe30f3a8b04a0be0c15ae6490fc859a78ef4c43ae2dd5e8a623d45bfcf9225" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.3.0+3" | ||||
|   term_glyph: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -202,4 +266,4 @@ packages: | ||||
|     version: "14.3.0" | ||||
| sdks: | ||||
|   dart: ">=3.6.1 <4.0.0" | ||||
|   flutter: ">=3.18.0-18.0.pre.54" | ||||
|   flutter: ">=3.24.0" | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| name: flutter_empty | ||||
| description: "A new Flutter project." | ||||
| name: flutter_expense_tracker | ||||
| description: Track recurring expenses against income and liquid assets. | ||||
| publish_to: 'none' | ||||
| version: 0.1.0 | ||||
|  | ||||
| @@ -9,6 +9,7 @@ environment: | ||||
| dependencies: | ||||
|   flutter: | ||||
|     sdk: flutter | ||||
|   sqflite: ^2.4.1 | ||||
|  | ||||
| dev_dependencies: | ||||
|   flutter_test: | ||||
|   | ||||
							
								
								
									
										3
									
								
								run_offline.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										3
									
								
								run_offline.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| flutter run --no-pub | ||||
		Reference in New Issue
	
	Block a user