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 { | android { | ||||||
|     namespace = "com.example.flutter_empty" |     namespace = "com.hyperling.expense_tracker" | ||||||
|     compileSdk = flutter.compileSdkVersion |     compileSdk = flutter.compileSdkVersion | ||||||
|     ndkVersion = flutter.ndkVersion |     ndkVersion = flutter.ndkVersion | ||||||
|  |  | ||||||
| @@ -21,7 +21,7 @@ android { | |||||||
|  |  | ||||||
|     defaultConfig { |     defaultConfig { | ||||||
|         // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html). |         // 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. |         // You can update the following values to match your application needs. | ||||||
|         // For more information, see: https://flutter.dev/to/review-gradle-config. |         // For more information, see: https://flutter.dev/to/review-gradle-config. | ||||||
|         minSdk = flutter.minSdkVersion |         minSdk = flutter.minSdkVersion | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/android"> | <manifest xmlns:android="http://schemas.android.com/apk/res/android"> | ||||||
|     <application |     <application | ||||||
|         android:label="flutter_empty" |         android:label="Recurring Expense Tracker" | ||||||
|         android:name="${applicationName}" |         android:name="${applicationName}" | ||||||
|         android:icon="@mipmap/ic_launcher"> |         android:icon="@mipmap/ic_launcher"> | ||||||
|         <activity |         <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, | ||||||
|  |   ); | ||||||
|  | } | ||||||
							
								
								
									
										209
									
								
								lib/main.dart
									
									
									
									
									
								
							
							
						
						
									
										209
									
								
								lib/main.dart
									
									
									
									
									
								
							| @@ -1,100 +1,145 @@ | |||||||
| // Helpful guides: | // Flutter | ||||||
| // - https://flutter.dev/docs/cookbook/forms/validation |  | ||||||
|  |  | ||||||
| import 'package:flutter/material.dart'; | 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() { | void main() { | ||||||
|  |   loadDB(); | ||||||
|   runApp(const MainApp()); |   runApp(const MainApp()); | ||||||
| } | } | ||||||
|  |  | ||||||
| class MainApp extends StatefulWidget { | class MainApp extends StatelessWidget { | ||||||
|   const MainApp({super.key}); |   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 |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     const inputWidth = 400.0; |     return MaterialApp( | ||||||
|     const inputHeight = 50.0; |       title: 'Recurring Expense Tracker', | ||||||
|  |       theme: ThemeData( | ||||||
|     const recurrenceValues = <DropdownMenuEntry>[ |         useMaterial3: true, | ||||||
|       DropdownMenuEntry(value: Recurrence.daily, label: "Daily"), |         colorScheme: ColorScheme.fromSeed(seedColor: Colors.green), | ||||||
|       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." |  | ||||||
|                     ), |  | ||||||
|                 )), |  | ||||||
|           ]), |  | ||||||
|       ), |       ), | ||||||
|  |       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 | class HomePage extends StatefulWidget { | ||||||
| enum Recurrence { daily, weekly, biweekly, montly, yearly } |   const HomePage({ | ||||||
|  |     super.key, | ||||||
|  |   }); | ||||||
|  |  | ||||||
| class Expense { |   @override | ||||||
|   String name; |   State<HomePage> createState() => _HomePageState(); | ||||||
|   double cost; | } | ||||||
|   Recurrence recurrence; |  | ||||||
|   String description; | class _HomePageState extends State<HomePage> { | ||||||
|  |   var pageSelected = 0; | ||||||
|   Expense(this.name, this.cost, this.recurrence, this.description); |  | ||||||
|  |   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 name of the executable created for the application. Change this to change | ||||||
| # the on-disk name of your application. | # 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: | # The unique GTK application identifier for this application. See: | ||||||
| # https://wiki.gnome.org/HowDoI/ChooseApplicationID | # 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 | # Explicitly opt in to modern CMake behaviors to avoid warnings with recent | ||||||
| # versions of CMake. | # versions of CMake. | ||||||
|   | |||||||
| @@ -40,11 +40,11 @@ static void my_application_activate(GApplication* application) { | |||||||
|   if (use_header_bar) { |   if (use_header_bar) { | ||||||
|     GtkHeaderBar* header_bar = GTK_HEADER_BAR(gtk_header_bar_new()); |     GtkHeaderBar* header_bar = GTK_HEADER_BAR(gtk_header_bar_new()); | ||||||
|     gtk_widget_show(GTK_WIDGET(header_bar)); |     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_header_bar_set_show_close_button(header_bar, TRUE); | ||||||
|     gtk_window_set_titlebar(window, GTK_WIDGET(header_bar)); |     gtk_window_set_titlebar(window, GTK_WIDGET(header_bar)); | ||||||
|   } else { |   } else { | ||||||
|     gtk_window_set_title(window, "flutter_empty"); |     gtk_window_set_title(window, "Recurring Expense Tracker"); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   gtk_window_set_default_size(window, 1280, 720); |   gtk_window_set_default_size(window, 1280, 720); | ||||||
|   | |||||||
| @@ -5,6 +5,8 @@ | |||||||
| import FlutterMacOS | import FlutterMacOS | ||||||
| import Foundation | import Foundation | ||||||
|  |  | ||||||
|  | import sqflite_darwin | ||||||
|  |  | ||||||
| func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) { | 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" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.9.0" |     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: |   sky_engine: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: flutter |     description: flutter | ||||||
| @@ -144,6 +160,46 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.10.0" |     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: |   stack_trace: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -168,6 +224,14 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.3.0" |     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: |   term_glyph: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -202,4 +266,4 @@ packages: | |||||||
|     version: "14.3.0" |     version: "14.3.0" | ||||||
| sdks: | sdks: | ||||||
|   dart: ">=3.6.1 <4.0.0" |   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 | name: flutter_expense_tracker | ||||||
| description: "A new Flutter project." | description: Track recurring expenses against income and liquid assets. | ||||||
| publish_to: 'none' | publish_to: 'none' | ||||||
| version: 0.1.0 | version: 0.1.0 | ||||||
|  |  | ||||||
| @@ -9,6 +9,7 @@ environment: | |||||||
| dependencies: | dependencies: | ||||||
|   flutter: |   flutter: | ||||||
|     sdk: flutter |     sdk: flutter | ||||||
|  |   sqflite: ^2.4.1 | ||||||
|  |  | ||||||
| dev_dependencies: | dev_dependencies: | ||||||
|   flutter_test: |   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