import 'dart:io'; import 'package:drift/drift.dart'; import 'package:drift/native.dart'; import 'package:path_provider/path_provider.dart'; import 'package:path/path.dart' as p; import '../../core/utils/result.dart'; import 'tables.dart'; part 'app_database.g.dart'; @DriftDatabase(tables: [Transactions, Categories, Budgets, ExchangeRates]) class AppDatabase extends _$AppDatabase { AppDatabase() : super(_openConnection()); @override int get schemaVersion => 1; // ============================================================================ // TRANSACTIONS // ============================================================================ /// Get all transactions ordered by date descending Future> getAllTransactions() { return (select( transactions, )..orderBy([(t) => OrderingTerm.desc(t.date)])).get(); } /// Get transactions by date range Future> getTransactionsByDateRange( DateTime start, DateTime end, ) { return (select(transactions) ..where((t) => t.date.isBiggerOrEqualValue(start)) ..where((t) => t.date.isSmallerOrEqualValue(end)) ..orderBy([(t) => OrderingTerm.desc(t.date)])) .get(); } /// Get transactions by type Future> getTransactionsByType(String type) { return (select(transactions) ..where((t) => t.type.equals(type)) ..orderBy([(t) => OrderingTerm.desc(t.date)])) .get(); } /// Get transactions by category Future> getTransactionsByCategory(String category) { return (select(transactions) ..where((t) => t.category.equals(category)) ..orderBy([(t) => OrderingTerm.desc(t.date)])) .get(); } /// Search transactions by note or category Future> searchTransactions(String query) { final lowerQuery = query.toLowerCase(); return (select(transactions) ..where( (t) => t.category.lower().like('%$lowerQuery%') | t.note.lower().like('%$lowerQuery%'), ) ..orderBy([(t) => OrderingTerm.desc(t.date)])) .get(); } /// Get transaction by ID Future getTransactionById(String id) { return (select( transactions, )..where((t) => t.id.equals(id))).getSingleOrNull(); } /// Insert transaction Future> insertTransaction( TransactionsCompanion transaction, ) async { return asyncResultOf(() async { await into(transactions).insert(transaction); }); } /// Update transaction Future> updateTransaction(dynamic transaction) async { return asyncResultOf(() async { final companion = transaction as TransactionsCompanion; final updated = await (update( transactions, )..where((t) => t.id.equals(companion.id.value))).write(companion); if (updated == 0) { throw Exception('Transaction not found'); } }); } /// Delete transaction Future> deleteTransaction(String id) async { return asyncResultOf(() async { final deleted = await (delete( transactions, )..where((t) => t.id.equals(id))).go(); if (deleted == 0) { throw Exception('Transaction not found'); } }); } /// Delete all transactions Future deleteAllTransactions() { return delete(transactions).go(); } /// Get recurring transactions that need processing Future> getRecurringTransactions() { return (select( transactions, )..where((t) => t.recurrence.equals('none').not())).get(); } // ============================================================================ // CATEGORIES // ============================================================================ /// Get all categories Future> getAllCategories() { return select(categories).get(); } /// Get categories by type Future> getCategoriesByType(String type) { return (select(categories)..where((c) => c.type.equals(type))).get(); } /// Insert category Future insertCategory(CategoriesCompanion category) { return into(categories).insert(category); } /// Update category Future updateCategory(Category category) { return update(categories).replace(category); } /// Delete category Future deleteCategory(int id) { return (delete(categories)..where((c) => c.id.equals(id))).go(); } // ============================================================================ // BUDGETS // ============================================================================ /// Get budget for month/year Future getBudget(int month, int year) { return (select(budgets) ..where((b) => b.month.equals(month) & b.year.equals(year))) .getSingleOrNull(); } /// Insert or update budget Future upsertBudget(BudgetsCompanion budget) { return into(budgets).insertOnConflictUpdate(budget); } /// Delete budget Future deleteBudget(int id) { return (delete(budgets)..where((b) => b.id.equals(id))).go(); } // ============================================================================ // EXCHANGE RATES // ============================================================================ /// Get exchange rate Future getExchangeRate(String from, String to) { return (select(exchangeRates) ..where((r) => r.fromCurrency.equals(from) & r.toCurrency.equals(to))) .getSingleOrNull(); } /// Insert or update exchange rate Future upsertExchangeRate(ExchangeRatesCompanion rate) { return into(exchangeRates).insertOnConflictUpdate(rate); } /// Get all exchange rates Future> getAllExchangeRates() { return select(exchangeRates).get(); } /// Delete old exchange rates (older than 24 hours) Future deleteOldExchangeRates() { final yesterday = DateTime.now().subtract(const Duration(hours: 24)); return (delete( exchangeRates, )..where((r) => r.updatedAt.isSmallerThanValue(yesterday))).go(); } // ============================================================================ // STATISTICS & AGGREGATIONS // ============================================================================ /// Get total balance Future getTotalBalance() async { final txs = await getAllTransactions(); return txs.fold(0.0, (sum, tx) { return tx.type == 'income' ? sum + tx.amount : sum - tx.amount; }); } /// Get total income for date range Future getTotalIncome(DateTime start, DateTime end) async { final txs = await getTransactionsByDateRange(start, end); return txs .where((tx) => tx.type == 'income') .fold(0.0, (sum, tx) => sum + tx.amount); } /// Get total expense for date range Future getTotalExpense(DateTime start, DateTime end) async { final txs = await getTransactionsByDateRange(start, end); return txs .where((tx) => tx.type == 'expense') .fold(0.0, (sum, tx) => sum + tx.amount); } /// Get category totals for date range Future> getCategoryTotals( DateTime start, DateTime end, String type, ) async { final txs = await getTransactionsByDateRange(start, end); final filtered = txs.where((tx) => tx.type == type); final Map totals = {}; for (final tx in filtered) { totals[tx.category] = (totals[tx.category] ?? 0) + tx.amount; } return totals; } } LazyDatabase _openConnection() { return LazyDatabase(() async { final dbFolder = await getApplicationDocumentsDirectory(); final file = File(p.join(dbFolder.path, 'casha.db')); return NativeDatabase(file); }); }