Compare commits

..

No commits in common. "d0bd40421af7afd57fc4e695abb663c796db8549" and "f59fda3de78c9a950773792582c38a6a8e3eac13" have entirely different histories.

17 changed files with 846 additions and 1108 deletions

View File

@ -4,14 +4,6 @@
<selectionStates> <selectionStates>
<SelectionState runConfigName="app"> <SelectionState runConfigName="app">
<option name="selectionMode" value="DROPDOWN" /> <option name="selectionMode" value="DROPDOWN" />
<DropdownSelection timestamp="2024-11-27T02:15:16.043756Z">
<Target type="DEFAULT_BOOT">
<handle>
<DeviceId pluginId="LocalEmulator" identifier="path=/Users/yovinchen/.android/avd/Pixel_7a_API_34.avd" />
</handle>
</Target>
</DropdownSelection>
<DialogSelection />
</SelectionState> </SelectionState>
</selectionStates> </selectionStates>
</component> </component>

View File

@ -1,6 +1,5 @@
<?xml version="1.0" encoding="UTF-8"?> <?xml version="1.0" encoding="UTF-8"?>
<project version="4"> <project version="4">
<component name="GradleMigrationSettings" migrationVersion="1" />
<component name="GradleSettings"> <component name="GradleSettings">
<option name="linkedExternalProjectsSettings"> <option name="linkedExternalProjectsSettings">
<GradleProjectSettings> <GradleProjectSettings>

View File

@ -1,6 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="StudioBotProjectSettings">
<option name="shareContext" value="OptedIn" />
</component>
</project>

128
README.md
View File

@ -1,59 +1,103 @@
# 轻记账 (Lightweight Bookkeeping) # Bookkeeping App
一个轻量级的个人记账应用,专注于隐私和离线使用。 一个基于 Jetpack Compose 开发的现代化记账应用。
## 🌟 特点 ## 项目概述
- 🔒 完全离线运行,无需网络连接 本项目是一个使用 Kotlin 和 Jetpack Compose 开发的 Android 记账应用,采用 MVVM 架构,提供简洁直观的用户界面和丰富的记账功能。
- 📱 极简权限要求,仅使用必要的系统权限
- 💰 支持收入和支出记录
- 👥 支持多人记账
- 📊 按日期和类别统计
- 🎨 Material You 设计风格
## 🛠 技术栈 ## 主要特性
- 语言Kotlin - 💰 收入/支出记录管理
- UI框架Jetpack Compose - 👥 成员管理系统
- 数据库Room - 📊 分类管理系统
- 架构MVVM - 📅 自定义日期选择器
- 📈 月度统计视图
- 🎨 Material 3 设计风格
## 📱 功能 ## 技术栈
### 记账管理 - 开发语言Kotlin
- 收入和支出记录 - UI 框架Jetpack Compose
- 自定义分类管理 - 架构模式MVVM
- 日期和时间选择 - 数据存储Room Database
- 备注说明 - 依赖注入Hilt
- 异步处理Kotlin Coroutines
### 成员管理 ## 开发计划
- 多人记账支持
- 成员关联记录
- 按成员筛选统计
### 数据统计 ### 0. 基础功能 (已完成)
- 月度收支统计 - [x] 收入/支出记录管理
- 分类统计 - [x] 分类管理系统
- 每日收支明细 - [x] 默认分类
- [x] 自定义分类
- [x] 分类编辑/删除
- [x] 自定义日期选择器
- [x] Material 3 设计界面
- [x] 深色/浅色主题切换
- [x] 主题色自定义
## 🔒 隐私保护 ### 1. 成员管理功能 (feature/member)
- [ ] 成员添加/编辑/删除
- [ ] 记账时选择相关成员
- [ ] 成员消费统计
- [ ] 成员间账单分摊
- 完全离线运行,数据存储在本地 ### 2. 数据统计与可视化 (feature/statistics)
- 无需任何网络权限 - [ ] 支出/收入趋势图表
- 最小化系统权限要求 - [ ] 分类占比饼图
- [ ] 月度/年度报表
## 📝 系统要求 ### 3. 数据导出与备份 (feature/backup)
- [ ] 导出 CSV/Excel 功能
- [ ] 云端备份支持
- [ ] 数据迁移工具
- Android 5.0 (API 21) 或更高版本 ### 4. 预算管理 (feature/budget)
- 存储权限(用于数据备份,可选) - [ ] 月度预算设置
- [ ] 预算超支提醒
- [ ] 分类预算管理
## 🔜 未来计划 ### 5. 用户体验优化 (feature/ux-enhancement)
- [x] 深色模式支持
- [ ] 手势操作优化
- [ ] 快速记账小组件
- [ ] 多语言支持
- [ ] 数据导出和备份 ### 6. 性能优化 (feature/performance)
- [ ] 预算管理 - [ ] 大数据量处理优化
- [ ] 更多统计图表 - [ ] 启动速度优化
- [ ] 自定义主题 - [ ] 内存使用优化
## 📄 许可证 ## 分支管理
[MIT License](LICENSE) - `master`: 稳定主分支
- `develop`: 主开发分支
- `feature/*`: 功能开发分支
- `release/*`: 版本发布分支
## 版本历史
### v1.0.0
- ✨ 基础记账功能
- 收入/支出记录
- 金额、日期、分类、备注管理
- 🎨 Material 3 设计界面
- 深色/浅色主题切换
- 主题色自定义
- 📊 分类管理
- 默认分类预设
- 自定义分类支持
- 分类编辑与删除
- 📅 月度统计
- 月度收支总览
- 月份快速切换
- 🗓️ 自定义日期选择器
## 贡献指南
欢迎提交 Issue 和 Pull Request 来帮助改进项目。
## 许可证
本项目采用 MIT 许可证。

View File

@ -12,42 +12,50 @@ interface BookkeepingDao {
@Query("SELECT * FROM bookkeeping_records ORDER BY date DESC") @Query("SELECT * FROM bookkeeping_records ORDER BY date DESC")
fun getAllRecords(): Flow<List<BookkeepingRecord>> fun getAllRecords(): Flow<List<BookkeepingRecord>>
@Query("SELECT * FROM bookkeeping_records WHERE memberId = :memberId OR memberId IS NULL ORDER BY date DESC")
fun getRecordsByMember(memberId: Int): Flow<List<BookkeepingRecord>>
@Query("SELECT * FROM bookkeeping_records WHERE date BETWEEN :startDate AND :endDate ORDER BY date DESC")
fun getRecordsByDateRange(startDate: Date, endDate: Date): Flow<List<BookkeepingRecord>>
@Query("SELECT * FROM bookkeeping_records WHERE (memberId = :memberId OR memberId IS NULL) AND date BETWEEN :startDate AND :endDate ORDER BY date DESC")
fun getRecordsByMemberAndDateRange(memberId: Int, startDate: Date, endDate: Date): Flow<List<BookkeepingRecord>>
@Query("SELECT * FROM bookkeeping_records WHERE type = :type ORDER BY date DESC")
fun getRecordsByType(type: TransactionType): Flow<List<BookkeepingRecord>>
@Query("SELECT SUM(amount) FROM bookkeeping_records WHERE type = :type AND (memberId = :memberId OR memberId IS NULL)")
fun getTotalAmountByType(type: TransactionType, memberId: Int? = null): Flow<Double?>
@Insert @Insert
suspend fun insertRecord(record: BookkeepingRecord): Long suspend fun insertRecord(record: BookkeepingRecord)
@Update
suspend fun updateRecord(record: BookkeepingRecord)
@Delete @Delete
suspend fun deleteRecord(record: BookkeepingRecord) suspend fun deleteRecord(record: BookkeepingRecord)
@Update
suspend fun updateRecord(record: BookkeepingRecord)
@Query("SELECT * FROM bookkeeping_records WHERE type = 'INCOME'")
fun getAllIncome(): Flow<List<BookkeepingRecord>>
@Query("SELECT * FROM bookkeeping_records WHERE type = 'EXPENSE'")
fun getAllExpense(): Flow<List<BookkeepingRecord>>
// 按日期查询
@Query("SELECT * FROM bookkeeping_records WHERE date >= :startOfDay AND date < :endOfDay ORDER BY date DESC")
fun getRecordsByDate(startOfDay: Date, endOfDay: Date): Flow<List<BookkeepingRecord>>
// 按日期范围查询
@Query("SELECT * FROM bookkeeping_records WHERE date BETWEEN :startDate AND :endDate ORDER BY date DESC")
fun getRecordsByDateRange(startDate: Date, endDate: Date): Flow<List<BookkeepingRecord>>
// 按类别查询
@Query("SELECT * FROM bookkeeping_records WHERE category = :category ORDER BY date DESC")
fun getRecordsByCategory(category: String): Flow<List<BookkeepingRecord>>
// 按类型查询
@Query("SELECT * FROM bookkeeping_records WHERE type = :type ORDER BY date DESC")
fun getRecordsByType(type: TransactionType): Flow<List<BookkeepingRecord>>
// Category related queries
@Query("SELECT * FROM categories WHERE type = :type ORDER BY name ASC") @Query("SELECT * FROM categories WHERE type = :type ORDER BY name ASC")
fun getCategoriesByType(type: TransactionType): Flow<List<Category>> fun getCategoriesByType(type: TransactionType): Flow<List<Category>>
@Insert @Insert
suspend fun insertCategory(category: Category): Long suspend fun insertCategory(category: Category)
@Update
suspend fun updateCategory(category: Category)
@Delete @Delete
suspend fun deleteCategory(category: Category) suspend fun deleteCategory(category: Category)
@Update
suspend fun updateCategory(category: Category)
@Query("SELECT EXISTS(SELECT 1 FROM bookkeeping_records WHERE category = :categoryName LIMIT 1)") @Query("SELECT EXISTS(SELECT 1 FROM bookkeeping_records WHERE category = :categoryName LIMIT 1)")
suspend fun isCategoryInUse(categoryName: String): Boolean suspend fun isCategoryInUse(categoryName: String): Boolean

View File

@ -11,164 +11,159 @@ import androidx.sqlite.db.SupportSQLiteDatabase
import com.yovinchen.bookkeeping.model.BookkeepingRecord import com.yovinchen.bookkeeping.model.BookkeepingRecord
import com.yovinchen.bookkeeping.model.Category import com.yovinchen.bookkeeping.model.Category
import com.yovinchen.bookkeeping.model.Converters import com.yovinchen.bookkeeping.model.Converters
import com.yovinchen.bookkeeping.model.Member
import com.yovinchen.bookkeeping.model.TransactionType import com.yovinchen.bookkeeping.model.TransactionType
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
@Database( @Database(entities = [BookkeepingRecord::class, Category::class], version = 2, exportSchema = false)
entities = [BookkeepingRecord::class, Category::class, Member::class],
version = 3,
exportSchema = false
)
@TypeConverters(Converters::class) @TypeConverters(Converters::class)
abstract class BookkeepingDatabase : RoomDatabase() { abstract class BookkeepingDatabase : RoomDatabase() {
abstract fun bookkeepingDao(): BookkeepingDao abstract fun bookkeepingDao(): BookkeepingDao
abstract fun categoryDao(): CategoryDao
abstract fun memberDao(): MemberDao
companion object { companion object {
private const val TAG = "BookkeepingDatabase" private const val TAG = "BookkeepingDatabase"
private val MIGRATION_1_2 = object : Migration(1, 2) {
override fun migrate(database: SupportSQLiteDatabase) {
// 创建成员表
database.execSQL("""
CREATE TABLE IF NOT EXISTS members (
id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
name TEXT NOT NULL,
description TEXT NOT NULL DEFAULT ''
)
""")
// 插入默认成员
database.execSQL("""
INSERT INTO members (name, description)
VALUES ('自己', '默认成员')
""")
// 修改记账记录表添加成员ID字段
database.execSQL("""
ALTER TABLE bookkeeping_records
ADD COLUMN memberId INTEGER DEFAULT NULL
REFERENCES members(id) ON DELETE SET NULL
""")
// 更新现有记录,将其关联到默认成员
database.execSQL("""
UPDATE bookkeeping_records
SET memberId = (SELECT id FROM members WHERE name = '我自己')
""")
}
}
private val MIGRATION_2_3 = object : Migration(2, 3) {
override fun migrate(database: SupportSQLiteDatabase) {
// 重新创建记账记录表
database.execSQL("""
CREATE TABLE IF NOT EXISTS bookkeeping_records_new (
id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
amount REAL NOT NULL,
type TEXT NOT NULL,
category TEXT NOT NULL,
description TEXT NOT NULL,
date INTEGER NOT NULL,
memberId INTEGER,
FOREIGN KEY(memberId) REFERENCES members(id) ON DELETE SET NULL
)
""")
// 复制数据
database.execSQL("""
INSERT INTO bookkeeping_records_new (id, amount, type, category, description, date, memberId)
SELECT id, amount, type, category, description, date, memberId FROM bookkeeping_records
""")
// 删除旧表
database.execSQL("DROP TABLE bookkeeping_records")
// 重命名新表
database.execSQL("ALTER TABLE bookkeeping_records_new RENAME TO bookkeeping_records")
// 重新创建分类表
database.execSQL("""
CREATE TABLE IF NOT EXISTS categories_new (
id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
name TEXT NOT NULL,
type TEXT NOT NULL
)
""")
// 复制分类数据
database.execSQL("""
INSERT INTO categories_new (id, name, type)
SELECT id, name, type FROM categories
""")
// 删除旧表
database.execSQL("DROP TABLE categories")
// 重命名新表
database.execSQL("ALTER TABLE categories_new RENAME TO categories")
}
}
@Volatile @Volatile
private var INSTANCE: BookkeepingDatabase? = null private var Instance: BookkeepingDatabase? = null
private val MIGRATION_1_2 = object : Migration(1, 2) {
override fun migrate(db: SupportSQLiteDatabase) {
try {
Log.d(TAG, "Starting migration from version 1 to 2")
// 检查表是否存在
val cursor = db.query("SELECT name FROM sqlite_master WHERE type='table' AND name='categories'")
val tableExists = cursor.moveToFirst()
cursor.close()
if (tableExists) {
// 如果表存在,执行迁移
Log.d(TAG, "Categories table exists, performing migration")
db.execSQL("ALTER TABLE categories RENAME TO categories_old")
db.execSQL("""
CREATE TABLE categories (
id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
name TEXT NOT NULL,
type TEXT NOT NULL
)
""")
db.execSQL("""
INSERT INTO categories (name, type)
SELECT name, type FROM categories_old
""")
db.execSQL("DROP TABLE categories_old")
} else {
// 如果表不存在,直接创建新表
Log.d(TAG, "Categories table does not exist, creating new table")
db.execSQL("""
CREATE TABLE IF NOT EXISTS categories (
id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
name TEXT NOT NULL,
type TEXT NOT NULL
)
""")
}
// 确保 bookkeeping_records 表存在
db.execSQL("""
CREATE TABLE IF NOT EXISTS bookkeeping_records (
id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
type TEXT NOT NULL,
amount REAL NOT NULL,
category TEXT NOT NULL,
description TEXT NOT NULL,
date INTEGER NOT NULL
)
""")
Log.d(TAG, "Migration completed successfully")
} catch (e: Exception) {
Log.e(TAG, "Error during migration", e)
throw e
}
}
}
private suspend fun populateDefaultCategories(dao: BookkeepingDao) {
try {
Log.d(TAG, "Starting to populate default categories")
// 支出类别
listOf(
"餐饮",
"交通",
"购物",
"娱乐",
"医疗",
"住房",
"其他支出"
).forEach { name ->
try {
dao.insertCategory(Category(name = name, type = TransactionType.EXPENSE))
Log.d(TAG, "Added expense category: $name")
} catch (e: Exception) {
Log.e(TAG, "Error adding expense category: $name", e)
}
}
// 收入类别
listOf(
"工资",
"奖金",
"投资",
"其他收入"
).forEach { name ->
try {
dao.insertCategory(Category(name = name, type = TransactionType.INCOME))
Log.d(TAG, "Added income category: $name")
} catch (e: Exception) {
Log.e(TAG, "Error adding income category: $name", e)
}
}
Log.d(TAG, "Finished populating default categories")
} catch (e: Exception) {
Log.e(TAG, "Error during category population", e)
}
}
fun getDatabase(context: Context): BookkeepingDatabase { fun getDatabase(context: Context): BookkeepingDatabase {
return INSTANCE ?: synchronized(this) { return Instance ?: synchronized(this) {
val instance = Room.databaseBuilder( try {
context.applicationContext, Log.d(TAG, "Creating new database instance")
BookkeepingDatabase::class.java, val instance = Room.databaseBuilder(
"bookkeeping_database" context.applicationContext,
) BookkeepingDatabase::class.java,
.addMigrations(MIGRATION_1_2, MIGRATION_2_3) "bookkeeping_database"
)
.addCallback(object : Callback() { .addCallback(object : Callback() {
override fun onCreate(db: SupportSQLiteDatabase) { override fun onCreate(db: SupportSQLiteDatabase) {
super.onCreate(db) super.onCreate(db)
Log.d(TAG, "Database created, initializing default data") Log.d(TAG, "Database created, initializing default categories")
CoroutineScope(Dispatchers.IO).launch { CoroutineScope(Dispatchers.IO).launch {
try { try {
val database = getDatabase(context) Instance?.let { database ->
populateDefaultCategories(database.bookkeepingDao())
// 初始化默认成员
database.memberDao().apply {
if (getMemberCount() == 0) {
insertMember(Member(name = "自己", description = "默认成员"))
}
} }
// 初始化默认分类
database.categoryDao().apply {
// 支出分类
insertCategory(Category(name = "餐饮", type = TransactionType.EXPENSE))
insertCategory(Category(name = "交通", type = TransactionType.EXPENSE))
insertCategory(Category(name = "购物", type = TransactionType.EXPENSE))
insertCategory(Category(name = "娱乐", type = TransactionType.EXPENSE))
insertCategory(Category(name = "居住", type = TransactionType.EXPENSE))
insertCategory(Category(name = "医疗", type = TransactionType.EXPENSE))
insertCategory(Category(name = "教育", type = TransactionType.EXPENSE))
insertCategory(Category(name = "其他支出", type = TransactionType.EXPENSE))
// 收入分类
insertCategory(Category(name = "工资", type = TransactionType.INCOME))
insertCategory(Category(name = "奖金", type = TransactionType.INCOME))
insertCategory(Category(name = "投资", type = TransactionType.INCOME))
insertCategory(Category(name = "其他收入", type = TransactionType.INCOME))
}
Log.d(TAG, "Default data initialized successfully")
} catch (e: Exception) { } catch (e: Exception) {
Log.e(TAG, "Error initializing default data", e) Log.e(TAG, "Error in onCreate callback", e)
} }
} }
} }
}) })
.addMigrations(MIGRATION_1_2)
.fallbackToDestructiveMigration() // 如果迁移失败,允许重建数据库
.build() .build()
INSTANCE = instance
instance Instance = instance
Log.d(TAG, "Database instance created successfully")
instance
} catch (e: Exception) {
Log.e(TAG, "Error creating database", e)
throw e
}
} }
} }
} }

View File

@ -1,30 +0,0 @@
package com.yovinchen.bookkeeping.data
import androidx.room.*
import com.yovinchen.bookkeeping.model.Category
import com.yovinchen.bookkeeping.model.TransactionType
import kotlinx.coroutines.flow.Flow
@Dao
interface CategoryDao {
@Query("SELECT * FROM categories WHERE type = :type ORDER BY name ASC")
fun getCategoriesByType(type: TransactionType): Flow<List<Category>>
@Query("SELECT * FROM categories ORDER BY type ASC, name ASC")
fun getAllCategories(): Flow<List<Category>>
@Insert
suspend fun insertCategory(category: Category): Long
@Update
suspend fun updateCategory(category: Category)
@Delete
suspend fun deleteCategory(category: Category)
@Query("SELECT EXISTS(SELECT 1 FROM bookkeeping_records WHERE category = :categoryName LIMIT 1)")
suspend fun isCategoryInUse(categoryName: String): Boolean
@Query("SELECT COUNT(*) FROM categories WHERE type = :type")
suspend fun getCategoryCountByType(type: TransactionType): Int
}

View File

@ -1,26 +0,0 @@
package com.yovinchen.bookkeeping.data
import androidx.room.*
import com.yovinchen.bookkeeping.model.Member
import kotlinx.coroutines.flow.Flow
@Dao
interface MemberDao {
@Query("SELECT * FROM members ORDER BY name ASC")
fun getAllMembers(): Flow<List<Member>>
@Query("SELECT * FROM members WHERE id = :memberId")
suspend fun getMemberById(memberId: Int): Member?
@Insert
suspend fun insertMember(member: Member): Long
@Update
suspend fun updateMember(member: Member)
@Delete
suspend fun deleteMember(member: Member)
@Query("SELECT COUNT(*) FROM members")
suspend fun getMemberCount(): Int
}

View File

@ -1,11 +1,9 @@
package com.yovinchen.bookkeeping.model package com.yovinchen.bookkeeping.model
import androidx.room.Entity import androidx.room.Entity
import androidx.room.ForeignKey
import androidx.room.PrimaryKey import androidx.room.PrimaryKey
import androidx.room.TypeConverter import androidx.room.TypeConverter
import androidx.room.TypeConverters import androidx.room.TypeConverters
import com.yovinchen.bookkeeping.model.Member
import java.util.Date import java.util.Date
enum class TransactionType { enum class TransactionType {
@ -34,17 +32,7 @@ class Converters {
} }
} }
@Entity( @Entity(tableName = "bookkeeping_records")
tableName = "bookkeeping_records",
foreignKeys = [
ForeignKey(
entity = Member::class,
parentColumns = ["id"],
childColumns = ["memberId"],
onDelete = ForeignKey.SET_NULL
)
]
)
@TypeConverters(Converters::class) @TypeConverters(Converters::class)
data class BookkeepingRecord( data class BookkeepingRecord(
@PrimaryKey(autoGenerate = true) @PrimaryKey(autoGenerate = true)
@ -53,6 +41,5 @@ data class BookkeepingRecord(
val type: TransactionType, val type: TransactionType,
val category: String, val category: String,
val description: String, val description: String,
val date: Date, val date: Date
val memberId: Int? = null // 可为空,表示未指定成员
) )

View File

@ -1,250 +0,0 @@
package com.yovinchen.bookkeeping.ui.components
import androidx.compose.foundation.background
import androidx.compose.foundation.clickable
import androidx.compose.foundation.layout.*
import androidx.compose.foundation.lazy.grid.GridCells
import androidx.compose.foundation.lazy.grid.LazyVerticalGrid
import androidx.compose.foundation.lazy.grid.items
import androidx.compose.foundation.shape.RoundedCornerShape
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.automirrored.filled.KeyboardArrowLeft
import androidx.compose.material.icons.automirrored.filled.KeyboardArrowRight
import androidx.compose.material3.*
import androidx.compose.runtime.*
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.unit.dp
import androidx.compose.ui.window.Dialog
import com.yovinchen.bookkeeping.model.TransactionType
import java.time.YearMonth
@Composable
fun MonthYearPickerDialog(
selectedMonth: YearMonth,
onMonthSelected: (YearMonth) -> Unit,
onDismiss: () -> Unit
) {
var currentYearMonth by remember { mutableStateOf(selectedMonth) }
Dialog(onDismissRequest = onDismiss) {
Surface(
modifier = Modifier
.fillMaxWidth()
.wrapContentHeight(),
shape = MaterialTheme.shapes.extraLarge,
tonalElevation = 6.dp
) {
Column(
modifier = Modifier.padding(16.dp)
) {
Text(
text = "选择年月",
style = MaterialTheme.typography.titleLarge,
modifier = Modifier.padding(bottom = 16.dp)
)
// 年份选择
Row(
modifier = Modifier.fillMaxWidth(),
horizontalArrangement = Arrangement.SpaceBetween,
verticalAlignment = Alignment.CenterVertically
) {
IconButton(onClick = {
currentYearMonth = currentYearMonth.minusYears(1)
}) {
Text("<")
}
Text(
text = "${currentYearMonth.year}",
style = MaterialTheme.typography.titleMedium
)
IconButton(onClick = {
currentYearMonth = currentYearMonth.plusYears(1)
}) {
Text(">")
}
}
Spacer(modifier = Modifier.height(16.dp))
// 月份网格
LazyVerticalGrid(
columns = GridCells.Fixed(3),
modifier = Modifier.height(200.dp)
) {
items(12) { index ->
val month = index + 1
val isSelected = month == currentYearMonth.monthValue
Surface(
modifier = Modifier
.padding(4.dp)
.aspectRatio(1.5f)
.clickable {
currentYearMonth = YearMonth.of(currentYearMonth.year, month)
},
shape = MaterialTheme.shapes.small,
color = if (isSelected) MaterialTheme.colorScheme.primary
else MaterialTheme.colorScheme.surface
) {
Box(
contentAlignment = Alignment.Center,
modifier = Modifier.fillMaxSize()
) {
Text(
text = "${month}",
color = if (isSelected) MaterialTheme.colorScheme.onPrimary
else MaterialTheme.colorScheme.onSurface
)
}
}
}
}
// 按钮行
Row(
modifier = Modifier
.fillMaxWidth()
.padding(top = 16.dp),
horizontalArrangement = Arrangement.End
) {
TextButton(onClick = onDismiss) {
Text("取消")
}
Spacer(modifier = Modifier.width(8.dp))
Button(onClick = {
onMonthSelected(currentYearMonth)
onDismiss()
}) {
Text("确定")
}
}
}
}
}
}
@Composable
fun MonthlyStatistics(
totalIncome: Double,
totalExpense: Double,
onIncomeClick: () -> Unit,
onExpenseClick: () -> Unit,
selectedType: TransactionType?,
onClearFilter: () -> Unit,
selectedMonth: YearMonth,
onPreviousMonth: () -> Unit,
onNextMonth: () -> Unit,
onMonthSelected: (YearMonth) -> Unit,
modifier: Modifier = Modifier
) {
var showMonthPicker by remember { mutableStateOf(false) }
Card(
modifier = modifier
.fillMaxWidth()
.padding(16.dp),
elevation = CardDefaults.cardElevation(defaultElevation = 4.dp)
) {
Column(
modifier = Modifier
.fillMaxWidth()
.padding(16.dp)
) {
// 月份选择器
Row(
modifier = Modifier.fillMaxWidth(),
horizontalArrangement = Arrangement.SpaceBetween,
verticalAlignment = Alignment.CenterVertically
) {
IconButton(onClick = onPreviousMonth) {
Icon(Icons.AutoMirrored.Filled.KeyboardArrowLeft, "上个月")
}
Text(
text = "${selectedMonth.year}${selectedMonth.monthValue}",
style = MaterialTheme.typography.titleLarge,
modifier = Modifier.clickable { showMonthPicker = true }
)
IconButton(onClick = onNextMonth) {
Icon(Icons.AutoMirrored.Filled.KeyboardArrowRight, "下个月")
}
}
Spacer(modifier = Modifier.height(16.dp))
Row(
modifier = Modifier.fillMaxWidth(),
horizontalArrangement = Arrangement.SpaceBetween
) {
// 收入统计
Column(
modifier = Modifier
.weight(1f)
.clickable { onIncomeClick() }
.background(
if (selectedType == TransactionType.INCOME) MaterialTheme.colorScheme.primaryContainer
else Color.Transparent,
RoundedCornerShape(8.dp)
)
.padding(8.dp)
) {
Text(
text = "收入",
style = MaterialTheme.typography.titleMedium
)
Text(
text = "¥${String.format("%.2f", totalIncome)}",
style = MaterialTheme.typography.bodyLarge,
color = MaterialTheme.colorScheme.primary
)
}
Spacer(modifier = Modifier.width(16.dp))
// 支出统计
Column(
modifier = Modifier
.weight(1f)
.clickable { onExpenseClick() }
.background(
if (selectedType == TransactionType.EXPENSE) MaterialTheme.colorScheme.primaryContainer
else Color.Transparent,
RoundedCornerShape(8.dp)
)
.padding(8.dp)
) {
Text(
text = "支出",
style = MaterialTheme.typography.titleMedium
)
Text(
text = "¥${String.format("%.2f", totalExpense)}",
style = MaterialTheme.typography.bodyLarge,
color = MaterialTheme.colorScheme.error
)
}
}
if (selectedType != null) {
TextButton(
onClick = onClearFilter,
modifier = Modifier.align(Alignment.End)
) {
Text("清除筛选")
}
}
}
}
if (showMonthPicker) {
MonthYearPickerDialog(
selectedMonth = selectedMonth,
onMonthSelected = onMonthSelected,
onDismiss = { showMonthPicker = false }
)
}
}

View File

@ -1,103 +0,0 @@
package com.yovinchen.bookkeeping.ui.components
import androidx.compose.foundation.clickable
import androidx.compose.foundation.layout.*
import androidx.compose.material3.*
import androidx.compose.runtime.*
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.unit.dp
import com.yovinchen.bookkeeping.model.BookkeepingRecord
import com.yovinchen.bookkeeping.model.Member
import com.yovinchen.bookkeeping.model.TransactionType
import java.text.SimpleDateFormat
import java.util.*
@Composable
fun RecordItem(
record: BookkeepingRecord,
onClick: () -> Unit = {},
onDelete: () -> Unit = {},
modifier: Modifier = Modifier,
members: List<Member> = emptyList()
) {
var showDeleteDialog by remember { mutableStateOf(false) }
val timeFormat = remember { SimpleDateFormat("HH:mm", Locale.getDefault()) }
val member = members.find { it.id == record.memberId }
Card(
modifier = modifier
.fillMaxWidth()
.padding(horizontal = 16.dp, vertical = 4.dp)
.clickable(onClick = onClick),
elevation = CardDefaults.cardElevation(defaultElevation = 2.dp)
) {
Row(
modifier = Modifier
.fillMaxWidth()
.padding(16.dp),
horizontalArrangement = Arrangement.SpaceBetween,
verticalAlignment = Alignment.CenterVertically
) {
Column(
modifier = Modifier.weight(1f)
) {
// 第一行:分类
Text(
text = record.category,
style = MaterialTheme.typography.bodyLarge
)
// 第二行:时间 | 成员 | 详情
Text(
text = buildString {
append(timeFormat.format(record.date))
if (member != null && member.name != "自己") {
append(" | ")
append(member.name)
}
if (record.description.isNotEmpty()) {
append(" | ")
append(record.description)
}
},
style = MaterialTheme.typography.bodySmall,
color = MaterialTheme.colorScheme.onSurfaceVariant
)
}
// 金额显示
Text(
text = String.format("%.2f", record.amount),
style = MaterialTheme.typography.titleMedium,
color = if (record.type == TransactionType.EXPENSE)
MaterialTheme.colorScheme.error
else
MaterialTheme.colorScheme.primary
)
}
}
if (showDeleteDialog) {
AlertDialog(
onDismissRequest = { showDeleteDialog = false },
title = { Text("确认删除") },
text = { Text("确定要删除这条记录吗?") },
confirmButton = {
TextButton(
onClick = {
onDelete()
showDeleteDialog = false
}
) {
Text("删除")
}
},
dismissButton = {
TextButton(onClick = { showDeleteDialog = false }) {
Text("取消")
}
}
)
}
}

View File

@ -1,57 +1,36 @@
@file:OptIn(ExperimentalMaterial3Api::class)
package com.yovinchen.bookkeeping.ui.dialog package com.yovinchen.bookkeeping.ui.dialog
import androidx.compose.foundation.layout.* import androidx.compose.foundation.layout.*
import androidx.compose.foundation.text.KeyboardOptions
import androidx.compose.material3.* import androidx.compose.material3.*
import androidx.compose.runtime.* import androidx.compose.runtime.*
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.compose.ui.text.input.KeyboardType
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
import androidx.compose.ui.window.Dialog import androidx.compose.ui.window.Dialog
import com.yovinchen.bookkeeping.model.Category import com.yovinchen.bookkeeping.model.Category
import com.yovinchen.bookkeeping.model.Member
import com.yovinchen.bookkeeping.model.TransactionType import com.yovinchen.bookkeeping.model.TransactionType
import com.yovinchen.bookkeeping.ui.components.DateTimePicker import com.yovinchen.bookkeeping.ui.components.DateTimePicker
import java.time.LocalDateTime import java.time.LocalDateTime
import java.time.ZoneId
import java.util.Date
@OptIn(ExperimentalMaterial3Api::class)
@Composable @Composable
fun AddRecordDialog( fun AddRecordDialog(
categories: List<Category>,
members: List<Member>,
onDismiss: () -> Unit, onDismiss: () -> Unit,
onConfirm: (amount: Double, category: String, description: String, date: Date, type: TransactionType, memberId: Int?) -> Unit onConfirm: (TransactionType, Double, String, String) -> Unit,
categories: List<Category>,
selectedType: TransactionType,
onTypeChange: (TransactionType) -> Unit,
selectedDateTime: LocalDateTime,
onDateTimeSelected: (LocalDateTime) -> Unit
) { ) {
var amount by remember { mutableStateOf("") } var amount by remember { mutableStateOf("") }
var expanded by remember { mutableStateOf(false) } var selectedCategory by remember { mutableStateOf<Category?>(null) }
var memberExpanded by remember { mutableStateOf(false) }
var description by remember { mutableStateOf("") } var description by remember { mutableStateOf("") }
var selectedType by remember { mutableStateOf(TransactionType.EXPENSE) } var expanded by remember { mutableStateOf(false) }
// 找到默认成员("自己" // 根据当前选择的类型过滤类别
val defaultMember = remember(members) { val filteredCategories = categories.filter { it.type == selectedType }
members.find { it.name == "自己" }
}
var currentSelectedMember by remember(defaultMember) {
mutableStateOf(defaultMember)
}
// 设置默认分类为"餐饮"
var selectedCategory by remember {
mutableStateOf(categories.find { it.type == selectedType && it.name == "餐饮" }?.name ?: categories.firstOrNull { it.type == selectedType }?.name ?: "")
}
var selectedDateTime by remember {
mutableStateOf(LocalDateTime.now())
}
// 当类型改变时更新分类
LaunchedEffect(selectedType) {
selectedCategory = categories.find { it.type == selectedType && it.name == "餐饮" }?.name
?: categories.firstOrNull { it.type == selectedType }?.name
?: ""
}
Dialog(onDismissRequest = onDismiss) { Dialog(onDismissRequest = onDismiss) {
Card( Card(
@ -72,59 +51,74 @@ fun AddRecordDialog(
Spacer(modifier = Modifier.height(16.dp)) Spacer(modifier = Modifier.height(16.dp))
// 收入/支出选择 // 类型选择
Row( Row(
modifier = Modifier.fillMaxWidth(), modifier = Modifier.fillMaxWidth(),
horizontalArrangement = Arrangement.SpaceEvenly horizontalArrangement = Arrangement.SpaceEvenly
) { ) {
FilterChip( FilterChip(
selected = selectedType == TransactionType.EXPENSE, selected = selectedType == TransactionType.EXPENSE,
onClick = { selectedType = TransactionType.EXPENSE }, onClick = {
onTypeChange(TransactionType.EXPENSE)
selectedCategory = null
},
label = { Text("支出") } label = { Text("支出") }
) )
FilterChip( FilterChip(
selected = selectedType == TransactionType.INCOME, selected = selectedType == TransactionType.INCOME,
onClick = { selectedType = TransactionType.INCOME }, onClick = {
onTypeChange(TransactionType.INCOME)
selectedCategory = null
},
label = { Text("收入") } label = { Text("收入") }
) )
} }
Spacer(modifier = Modifier.height(16.dp)) Spacer(modifier = Modifier.height(16.dp))
OutlinedTextField( // 日期时间选择
value = amount, DateTimePicker(
onValueChange = { amount = it }, selectedDateTime = selectedDateTime,
label = { Text("金额") }, onDateTimeSelected = onDateTimeSelected,
modifier = Modifier.fillMaxWidth(), modifier = Modifier.fillMaxWidth()
singleLine = true
) )
Spacer(modifier = Modifier.height(16.dp)) Spacer(modifier = Modifier.height(16.dp))
// 金额输入
OutlinedTextField(
value = amount,
onValueChange = { amount = it },
label = { Text("金额") },
keyboardOptions = KeyboardOptions(keyboardType = KeyboardType.Number),
modifier = Modifier.fillMaxWidth()
)
Spacer(modifier = Modifier.height(8.dp))
// 类别选择
ExposedDropdownMenuBox( ExposedDropdownMenuBox(
expanded = expanded, expanded = expanded,
onExpandedChange = { expanded = it } onExpandedChange = { expanded = it }
) { ) {
OutlinedTextField( OutlinedTextField(
value = selectedCategory, value = selectedCategory?.name ?: "",
onValueChange = {}, onValueChange = {},
readOnly = true, readOnly = true,
label = { Text("类别") }, label = { Text("类别") },
trailingIcon = { ExposedDropdownMenuDefaults.TrailingIcon(expanded = expanded) },
modifier = Modifier modifier = Modifier
.fillMaxWidth() .fillMaxWidth()
.menuAnchor() .menuAnchor()
) )
ExposedDropdownMenu( ExposedDropdownMenu(
expanded = expanded, expanded = expanded,
onDismissRequest = { expanded = false } onDismissRequest = { expanded = false }
) { ) {
categories.filter { it.type == selectedType }.forEach { category -> filteredCategories.forEach { category ->
DropdownMenuItem( DropdownMenuItem(
text = { Text(category.name) }, text = { Text(category.name) },
onClick = { onClick = {
selectedCategory = category.name selectedCategory = category
expanded = false expanded = false
} }
) )
@ -132,59 +126,19 @@ fun AddRecordDialog(
} }
} }
Spacer(modifier = Modifier.height(16.dp)) Spacer(modifier = Modifier.height(8.dp))
ExposedDropdownMenuBox( // 描述输入
expanded = memberExpanded, OutlinedTextField(
onExpandedChange = { memberExpanded = it } value = description,
) { onValueChange = { description = it },
OutlinedTextField( label = { Text("描述") },
value = currentSelectedMember?.name ?: "选择成员",
onValueChange = {},
readOnly = true,
label = { Text("成员") },
trailingIcon = { ExposedDropdownMenuDefaults.TrailingIcon(expanded = memberExpanded) },
modifier = Modifier
.fillMaxWidth()
.menuAnchor()
)
ExposedDropdownMenu(
expanded = memberExpanded,
onDismissRequest = { memberExpanded = false }
) {
members.forEach { member ->
DropdownMenuItem(
text = { Text(member.name) },
onClick = {
currentSelectedMember = member
memberExpanded = false
}
)
}
}
}
Spacer(modifier = Modifier.height(16.dp))
DateTimePicker(
selectedDateTime = selectedDateTime,
onDateTimeSelected = { selectedDateTime = it },
modifier = Modifier.fillMaxWidth() modifier = Modifier.fillMaxWidth()
) )
Spacer(modifier = Modifier.height(16.dp)) Spacer(modifier = Modifier.height(16.dp))
OutlinedTextField( // 按钮
value = description,
onValueChange = { description = it },
label = { Text("备注") },
modifier = Modifier.fillMaxWidth(),
singleLine = true
)
Spacer(modifier = Modifier.height(24.dp))
Row( Row(
modifier = Modifier.fillMaxWidth(), modifier = Modifier.fillMaxWidth(),
horizontalArrangement = Arrangement.End horizontalArrangement = Arrangement.End
@ -195,21 +149,13 @@ fun AddRecordDialog(
Spacer(modifier = Modifier.width(8.dp)) Spacer(modifier = Modifier.width(8.dp))
Button( Button(
onClick = { onClick = {
val amountValue = amount.toDoubleOrNull() val amountValue = amount.toDoubleOrNull() ?: 0.0
if (amountValue != null) { selectedCategory?.let { category ->
onConfirm( onConfirm(selectedType, amountValue, category.name, description)
amountValue, onDismiss()
selectedCategory,
description,
Date.from(
selectedDateTime.atZone(ZoneId.systemDefault()).toInstant()
),
selectedType,
currentSelectedMember?.id
)
} }
}, },
enabled = amount.isNotEmpty() && selectedCategory.isNotEmpty() enabled = amount.isNotEmpty() && selectedCategory != null
) { ) {
Text("确定") Text("确定")
} }

View File

@ -8,33 +8,25 @@ import androidx.compose.runtime.*
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
import androidx.compose.ui.window.Dialog import androidx.compose.ui.window.Dialog
import androidx.lifecycle.viewmodel.compose.viewModel
import com.yovinchen.bookkeeping.model.BookkeepingRecord import com.yovinchen.bookkeeping.model.BookkeepingRecord
import com.yovinchen.bookkeeping.model.Category import com.yovinchen.bookkeeping.model.Category
import com.yovinchen.bookkeeping.model.Member
import com.yovinchen.bookkeeping.ui.components.DateTimePicker import com.yovinchen.bookkeeping.ui.components.DateTimePicker
import com.yovinchen.bookkeeping.viewmodel.HomeViewModel
import java.time.Instant import java.time.Instant
import java.time.LocalDateTime import java.time.LocalDateTime
import java.time.ZoneId import java.time.ZoneId
import java.util.Date import java.util.Date
import kotlinx.coroutines.launch
@Composable @Composable
fun RecordEditDialog( fun RecordEditDialog(
record: BookkeepingRecord, record: BookkeepingRecord,
categories: List<Category>, categories: List<Category>,
members: List<Member>,
onDismiss: () -> Unit, onDismiss: () -> Unit,
onConfirm: (BookkeepingRecord) -> Unit, onConfirm: (BookkeepingRecord) -> Unit
viewModel: HomeViewModel = viewModel()
) { ) {
var amount by remember { mutableStateOf(record.amount.toString()) } var amount by remember { mutableStateOf(record.amount.toString()) }
var selectedCategory by remember { mutableStateOf(record.category) } var selectedCategory by remember { mutableStateOf(record.category) }
var description by remember { mutableStateOf(record.description) } var description by remember { mutableStateOf(record.description) }
var expanded by remember { mutableStateOf(false) } var expanded by remember { mutableStateOf(false) }
var memberExpanded by remember { mutableStateOf(false) }
var currentSelectedMember by remember { mutableStateOf<Member?>(null) }
var selectedDateTime by remember { var selectedDateTime by remember {
mutableStateOf( mutableStateOf(
LocalDateTime.ofInstant( LocalDateTime.ofInstant(
@ -44,16 +36,6 @@ fun RecordEditDialog(
) )
} }
// 加载原关联成员
val coroutineScope = rememberCoroutineScope()
LaunchedEffect(record.memberId) {
if (record.memberId != null) {
coroutineScope.launch {
currentSelectedMember = viewModel.getMemberById(record.memberId)
}
}
}
Dialog(onDismissRequest = onDismiss) { Dialog(onDismissRequest = onDismiss) {
Card( Card(
modifier = Modifier modifier = Modifier
@ -73,16 +55,24 @@ fun RecordEditDialog(
Spacer(modifier = Modifier.height(16.dp)) Spacer(modifier = Modifier.height(16.dp))
// 日期时间选择
DateTimePicker(
selectedDateTime = selectedDateTime,
onDateTimeSelected = { selectedDateTime = it },
modifier = Modifier.fillMaxWidth()
)
Spacer(modifier = Modifier.height(16.dp))
// 金额输入 // 金额输入
OutlinedTextField( OutlinedTextField(
value = amount, value = amount,
onValueChange = { amount = it }, onValueChange = { amount = it },
label = { Text("金额") }, label = { Text("金额") },
modifier = Modifier.fillMaxWidth(), modifier = Modifier.fillMaxWidth()
singleLine = true
) )
Spacer(modifier = Modifier.height(16.dp)) Spacer(modifier = Modifier.height(8.dp))
// 类别选择 // 类别选择
ExposedDropdownMenuBox( ExposedDropdownMenuBox(
@ -94,12 +84,10 @@ fun RecordEditDialog(
onValueChange = {}, onValueChange = {},
readOnly = true, readOnly = true,
label = { Text("类别") }, label = { Text("类别") },
trailingIcon = { ExposedDropdownMenuDefaults.TrailingIcon(expanded = expanded) },
modifier = Modifier modifier = Modifier
.fillMaxWidth() .fillMaxWidth()
.menuAnchor() .menuAnchor()
) )
ExposedDropdownMenu( ExposedDropdownMenu(
expanded = expanded, expanded = expanded,
onDismissRequest = { expanded = false } onDismissRequest = { expanded = false }
@ -116,72 +104,19 @@ fun RecordEditDialog(
} }
} }
Spacer(modifier = Modifier.height(16.dp)) Spacer(modifier = Modifier.height(8.dp))
// 成员选择 // 描述输入
ExposedDropdownMenuBox( OutlinedTextField(
expanded = memberExpanded, value = description,
onExpandedChange = { memberExpanded = it } onValueChange = { description = it },
) { label = { Text("描述") },
OutlinedTextField(
value = currentSelectedMember?.name ?: "选择成员",
onValueChange = {},
readOnly = true,
label = { Text("成员") },
trailingIcon = { ExposedDropdownMenuDefaults.TrailingIcon(expanded = memberExpanded) },
modifier = Modifier
.fillMaxWidth()
.menuAnchor()
)
ExposedDropdownMenu(
expanded = memberExpanded,
onDismissRequest = { memberExpanded = false }
) {
// 添加一个"清除选择"选项
DropdownMenuItem(
text = { Text("清除选择") },
onClick = {
currentSelectedMember = null
memberExpanded = false
}
)
members.forEach { member ->
DropdownMenuItem(
text = { Text(member.name) },
onClick = {
currentSelectedMember = member
memberExpanded = false
}
)
}
}
}
Spacer(modifier = Modifier.height(16.dp))
// 日期时间选择
DateTimePicker(
selectedDateTime = selectedDateTime,
onDateTimeSelected = { selectedDateTime = it },
modifier = Modifier.fillMaxWidth() modifier = Modifier.fillMaxWidth()
) )
Spacer(modifier = Modifier.height(16.dp)) Spacer(modifier = Modifier.height(16.dp))
// 备注输入 // 按钮
OutlinedTextField(
value = description,
onValueChange = { description = it },
label = { Text("备注") },
modifier = Modifier.fillMaxWidth(),
singleLine = true
)
Spacer(modifier = Modifier.height(24.dp))
// 按钮行
Row( Row(
modifier = Modifier.fillMaxWidth(), modifier = Modifier.fillMaxWidth(),
horizontalArrangement = Arrangement.End horizontalArrangement = Arrangement.End
@ -192,22 +127,15 @@ fun RecordEditDialog(
Spacer(modifier = Modifier.width(8.dp)) Spacer(modifier = Modifier.width(8.dp))
Button( Button(
onClick = { onClick = {
val amountValue = amount.toDoubleOrNull() val updatedRecord = record.copy(
if (amountValue != null) { amount = amount.toDoubleOrNull() ?: record.amount,
onConfirm( category = selectedCategory,
record.copy( description = description,
amount = amountValue, date = Date.from(selectedDateTime.atZone(ZoneId.systemDefault()).toInstant())
category = selectedCategory, )
description = description, onConfirm(updatedRecord)
date = Date.from( onDismiss()
selectedDateTime.atZone(ZoneId.systemDefault()).toInstant() }
),
memberId = currentSelectedMember?.id
)
)
}
},
enabled = amount.isNotEmpty()
) { ) {
Text("确定") Text("确定")
} }

View File

@ -1,73 +1,60 @@
package com.yovinchen.bookkeeping.ui.screen package com.yovinchen.bookkeeping.ui.screen
import androidx.compose.foundation.background import androidx.compose.foundation.background
import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.clickable
import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.*
import androidx.compose.foundation.layout.PaddingValues
import androidx.compose.foundation.layout.Spacer
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.fillMaxWidth
import androidx.compose.foundation.layout.height
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.lazy.LazyColumn import androidx.compose.foundation.lazy.LazyColumn
import androidx.compose.foundation.lazy.items
import androidx.compose.foundation.lazy.grid.LazyVerticalGrid
import androidx.compose.foundation.lazy.grid.GridCells
import androidx.compose.foundation.lazy.grid.items
import androidx.compose.foundation.shape.RoundedCornerShape
import androidx.compose.material.icons.Icons import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.automirrored.filled.KeyboardArrowLeft
import androidx.compose.material.icons.automirrored.filled.KeyboardArrowRight
import androidx.compose.material.icons.filled.Add import androidx.compose.material.icons.filled.Add
import androidx.compose.material3.Card import androidx.compose.material.icons.filled.Delete
import androidx.compose.material3.CardDefaults import androidx.compose.material3.*
import androidx.compose.material3.ExperimentalMaterial3Api import androidx.compose.runtime.*
import androidx.compose.material3.ExtendedFloatingActionButton import androidx.compose.ui.Alignment
import androidx.compose.material3.HorizontalDivider
import androidx.compose.material3.Icon
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Scaffold
import androidx.compose.material3.Surface
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.runtime.collectAsState
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.setValue
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.text.style.TextAlign
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
import androidx.compose.ui.unit.sp
import androidx.compose.ui.window.Dialog
import androidx.lifecycle.viewmodel.compose.viewModel import androidx.lifecycle.viewmodel.compose.viewModel
import com.yovinchen.bookkeeping.model.BookkeepingRecord import com.yovinchen.bookkeeping.model.BookkeepingRecord
import com.yovinchen.bookkeeping.model.TransactionType import com.yovinchen.bookkeeping.model.TransactionType
import com.yovinchen.bookkeeping.ui.components.MonthlyStatistics
import com.yovinchen.bookkeeping.ui.components.RecordItem
import com.yovinchen.bookkeeping.ui.dialog.AddRecordDialog import com.yovinchen.bookkeeping.ui.dialog.AddRecordDialog
import com.yovinchen.bookkeeping.ui.dialog.RecordEditDialog import com.yovinchen.bookkeeping.ui.dialog.RecordEditDialog
import com.yovinchen.bookkeeping.viewmodel.HomeViewModel import com.yovinchen.bookkeeping.viewmodel.HomeViewModel
import java.time.YearMonth
import java.text.SimpleDateFormat import java.text.SimpleDateFormat
import java.util.Locale import java.util.*
@OptIn(ExperimentalMaterial3Api::class) @OptIn(ExperimentalMaterial3Api::class)
@Composable @Composable
fun HomeScreen( fun HomeScreen(
modifier: Modifier = Modifier, modifier: Modifier = Modifier, viewModel: HomeViewModel = viewModel()
viewModel: HomeViewModel = viewModel()
) { ) {
val filteredRecords by viewModel.filteredRecords.collectAsState()
val totalIncome by viewModel.totalIncome.collectAsState()
val totalExpense by viewModel.totalExpense.collectAsState()
val categories by viewModel.categories.collectAsState()
val selectedRecordType by viewModel.selectedRecordType.collectAsState()
val selectedMonth by viewModel.selectedMonth.collectAsState()
var showAddDialog by remember { mutableStateOf(false) } var showAddDialog by remember { mutableStateOf(false) }
var selectedRecord by remember { mutableStateOf<BookkeepingRecord?>(null) } var selectedRecord by remember { mutableStateOf<BookkeepingRecord?>(null) }
val selectedMonth by viewModel.selectedMonth.collectAsState() Scaffold(modifier = modifier.fillMaxSize(), floatingActionButton = {
val filteredRecords by viewModel.filteredRecords.collectAsState() FloatingActionButton(onClick = { showAddDialog = true }) {
val categories by viewModel.categories.collectAsState(initial = emptyList()) Icon(Icons.Default.Add, contentDescription = "添加记录")
val members by viewModel.members.collectAsState(initial = emptyList())
val selectedMember by viewModel.selectedMember.collectAsState()
val totalIncome by viewModel.totalIncome.collectAsState()
val totalExpense by viewModel.totalExpense.collectAsState()
Scaffold(
modifier = modifier.fillMaxSize(),
floatingActionButton = {
ExtendedFloatingActionButton(
onClick = { showAddDialog = true },
icon = { Icon(Icons.Default.Add, contentDescription = null) },
text = { Text("记一笔") }
)
} }
) { padding -> }, floatingActionButtonPosition = FabPosition.End, topBar = {
TopAppBar(title = { Text("记账本") })
}) { padding ->
Column( Column(
modifier = Modifier modifier = Modifier
.fillMaxSize() .fillMaxSize()
@ -75,103 +62,408 @@ fun HomeScreen(
.background(MaterialTheme.colorScheme.background) .background(MaterialTheme.colorScheme.background)
) { ) {
// 顶部统计信息 // 顶部统计信息
MonthlyStatistics( MonthlyStatistics(totalIncome = totalIncome,
totalIncome = totalIncome,
totalExpense = totalExpense, totalExpense = totalExpense,
selectedType = null,
onIncomeClick = { viewModel.setSelectedRecordType(TransactionType.INCOME) }, onIncomeClick = { viewModel.setSelectedRecordType(TransactionType.INCOME) },
onExpenseClick = { viewModel.setSelectedRecordType(TransactionType.EXPENSE) }, onExpenseClick = { viewModel.setSelectedRecordType(TransactionType.EXPENSE) },
selectedType = selectedRecordType,
onClearFilter = { viewModel.setSelectedRecordType(null) }, onClearFilter = { viewModel.setSelectedRecordType(null) },
selectedMonth = selectedMonth, selectedMonth = selectedMonth,
onPreviousMonth = { viewModel.moveMonth(false) }, onPreviousMonth = { viewModel.setSelectedMonth(selectedMonth.minusMonths(1)) },
onNextMonth = { viewModel.moveMonth(true) }, onNextMonth = { viewModel.setSelectedMonth(selectedMonth.plusMonths(1)) },
onMonthSelected = { viewModel.setSelectedMonth(it) } onMonthSelected = { viewModel.setSelectedMonth(it) })
)
// 记录列表 // 记录列表
LazyColumn( LazyColumn(
modifier = Modifier.fillMaxSize(), modifier = Modifier.fillMaxSize(),
contentPadding = PaddingValues(16.dp), contentPadding = PaddingValues(16.dp),
verticalArrangement = Arrangement.spacedBy(8.dp) verticalArrangement = Arrangement.spacedBy(12.dp)
) { ) {
items(filteredRecords.size) { index -> filteredRecords.forEach { (date, records) ->
val (date, dayRecords) = filteredRecords.toList()[index] item {
Card( Surface(
modifier = Modifier
.fillMaxWidth()
.padding(vertical = 4.dp),
elevation = CardDefaults.cardElevation(defaultElevation = 2.dp)
) {
Column(
modifier = Modifier modifier = Modifier
.fillMaxWidth() .fillMaxWidth()
.padding(16.dp) .padding(vertical = 4.dp),
color = MaterialTheme.colorScheme.surfaceVariant.copy(alpha = 0.7f),
shape = RoundedCornerShape(12.dp),
tonalElevation = 2.dp
) { ) {
// 日期标签
Text(
text = SimpleDateFormat(
"yyyy年MM月dd日 E",
Locale.CHINESE
).format(date),
style = MaterialTheme.typography.titleMedium,
color = MaterialTheme.colorScheme.onSurfaceVariant
)
Spacer(modifier = Modifier.height(8.dp))
// 当天的记录
Column( Column(
modifier = Modifier.fillMaxWidth(), modifier = Modifier.padding(horizontal = 16.dp, vertical = 12.dp)
verticalArrangement = Arrangement.spacedBy(4.dp)
) { ) {
dayRecords.forEachIndexed { recordIndex, record -> // 日期标签
RecordItem( Text(
record = record, text = SimpleDateFormat(
onClick = { selectedRecord = record }, "yyyy年MM月dd日 E", Locale.CHINESE
onDelete = { viewModel.deleteRecord(record) }, ).format(date),
members = members style = MaterialTheme.typography.titleMedium,
) color = MaterialTheme.colorScheme.onSurfaceVariant
)
if (recordIndex < dayRecords.size - 1) { Spacer(modifier = Modifier.height(12.dp))
// 当天的记录
records.forEachIndexed { index, record ->
RecordItem(record = record,
onClick = { selectedRecord = record },
onDelete = { viewModel.deleteRecord(record) })
if (index < records.size - 1) {
HorizontalDivider( HorizontalDivider(
modifier = Modifier.padding(vertical = 4.dp), modifier = Modifier.padding(vertical = 8.dp),
color = MaterialTheme.colorScheme.surfaceVariant, color = MaterialTheme.colorScheme.surfaceVariant,
thickness = 0.5.dp thickness = 0.5.dp
) )
} }
} }
Spacer(modifier = Modifier.height(8.dp))
// 当天统计
HorizontalDivider(
color = MaterialTheme.colorScheme.surfaceVariant,
thickness = 0.5.dp
)
val dayIncome = records.filter { it.type == TransactionType.INCOME }
.sumOf { it.amount }
val dayExpense =
records.filter { it.type == TransactionType.EXPENSE }
.sumOf { it.amount }
Row(
modifier = Modifier
.fillMaxWidth()
.padding(top = 12.dp),
horizontalArrangement = Arrangement.SpaceBetween
) {
Text(
text = "收入: ¥%.2f".format(dayIncome),
style = MaterialTheme.typography.bodyMedium,
color = MaterialTheme.colorScheme.primary
)
Text(
text = "支出: ¥%.2f".format(dayExpense),
style = MaterialTheme.typography.bodyMedium,
color = MaterialTheme.colorScheme.error
)
}
} }
} }
} }
} }
} }
} }
}
// 添加记录对话框 // 添加记录对话框
if (showAddDialog) { if (showAddDialog) {
AddRecordDialog( val selectedDateTime by viewModel.selectedDateTime.collectAsState()
categories = categories, val selectedCategoryType by viewModel.selectedCategoryType.collectAsState()
members = members, AddRecordDialog(onDismiss = {
onDismiss = { showAddDialog = false },
onConfirm = { amount, category, description, date, type, memberId ->
viewModel.addRecord(amount, category, description, date, type, memberId)
showAddDialog = false showAddDialog = false
} viewModel.resetSelectedDateTime()
) },
} onConfirm = { type, amount, category, description ->
viewModel.addRecord(type, amount, category, description)
showAddDialog = false
},
categories = categories,
selectedType = selectedCategoryType,
onTypeChange = viewModel::setSelectedCategoryType,
selectedDateTime = selectedDateTime,
onDateTimeSelected = viewModel::setSelectedDateTime
)
}
// 编辑记录对话框 // 编辑记录对话框
selectedRecord?.let { record -> selectedRecord?.let { record ->
RecordEditDialog( RecordEditDialog(record = record,
record = record, categories = categories,
categories = categories, onDismiss = { selectedRecord = null },
members = members, onConfirm = { updatedRecord ->
onDismiss = { selectedRecord = null }, viewModel.updateRecord(updatedRecord)
onConfirm = { updatedRecord -> selectedRecord = null
viewModel.updateRecord(updatedRecord) })
selectedRecord = null }
} }
) }
@Composable
fun MonthYearPickerDialog(
selectedMonth: YearMonth, onMonthSelected: (YearMonth) -> Unit, onDismiss: () -> Unit
) {
var currentYearMonth by remember { mutableStateOf(selectedMonth) }
Dialog(onDismissRequest = onDismiss) {
Surface(
modifier = Modifier
.fillMaxWidth()
.wrapContentHeight(),
shape = MaterialTheme.shapes.extraLarge,
tonalElevation = 6.dp
) {
Column(
modifier = Modifier.padding(16.dp)
) {
Text(
text = "选择年月",
style = MaterialTheme.typography.titleLarge,
modifier = Modifier.padding(bottom = 16.dp)
)
// 年份选择
Row(
modifier = Modifier.fillMaxWidth(),
horizontalArrangement = Arrangement.SpaceBetween,
verticalAlignment = Alignment.CenterVertically
) {
IconButton(onClick = {
currentYearMonth = currentYearMonth.minusYears(1)
}) {
Text("<")
}
Text(
text = "${currentYearMonth.year}",
style = MaterialTheme.typography.titleMedium
)
IconButton(onClick = {
currentYearMonth = currentYearMonth.plusYears(1)
}) {
Text(">")
}
}
Spacer(modifier = Modifier.height(16.dp))
// 月份网格
LazyVerticalGrid(
columns = GridCells.Fixed(3), modifier = Modifier.height(200.dp)
) {
items(12) { index ->
val month = index + 1
val isSelected = month == currentYearMonth.monthValue
Surface(
modifier = Modifier
.padding(4.dp)
.aspectRatio(1.5f)
.clickable {
currentYearMonth = YearMonth.of(currentYearMonth.year, month)
},
shape = MaterialTheme.shapes.small,
color = if (isSelected) MaterialTheme.colorScheme.primary
else MaterialTheme.colorScheme.surface
) {
Box(
contentAlignment = Alignment.Center,
modifier = Modifier.fillMaxSize()
) {
Text(
text = "${month}",
color = if (isSelected) MaterialTheme.colorScheme.onPrimary
else MaterialTheme.colorScheme.onSurface
)
}
}
}
}
// 按钮行
Row(
modifier = Modifier
.fillMaxWidth()
.padding(top = 16.dp),
horizontalArrangement = Arrangement.End
) {
TextButton(onClick = onDismiss) {
Text("取消")
}
Spacer(modifier = Modifier.width(8.dp))
Button(onClick = {
onMonthSelected(currentYearMonth)
onDismiss()
}) {
Text("确定")
}
}
}
}
}
}
@Composable
fun MonthlyStatistics(
totalIncome: Double,
totalExpense: Double,
onIncomeClick: () -> Unit,
onExpenseClick: () -> Unit,
selectedType: TransactionType?,
onClearFilter: () -> Unit,
selectedMonth: YearMonth,
onPreviousMonth: () -> Unit,
onNextMonth: () -> Unit,
onMonthSelected: (YearMonth) -> Unit,
modifier: Modifier = Modifier
) {
var showMonthPicker by remember { mutableStateOf(false) }
Card(
modifier = modifier
.fillMaxWidth()
.padding(16.dp),
elevation = CardDefaults.cardElevation(defaultElevation = 4.dp)
) {
Column(
modifier = Modifier
.fillMaxWidth()
.padding(16.dp)
) {
// 月份选择器
Row(
modifier = Modifier.fillMaxWidth(),
horizontalArrangement = Arrangement.SpaceBetween,
verticalAlignment = Alignment.CenterVertically
) {
IconButton(onClick = onPreviousMonth) {
Icon(Icons.AutoMirrored.Filled.KeyboardArrowLeft, "上个月")
}
Text(text = "${selectedMonth.year}${selectedMonth.monthValue}",
style = MaterialTheme.typography.titleLarge,
modifier = Modifier.clickable { showMonthPicker = true })
IconButton(onClick = onNextMonth) {
Icon(Icons.AutoMirrored.Filled.KeyboardArrowRight, "下个月")
}
}
Spacer(modifier = Modifier.height(16.dp))
Row(
modifier = Modifier.fillMaxWidth(), horizontalArrangement = Arrangement.SpaceBetween
) {
// 收入统计
Column(modifier = Modifier
.weight(1f)
.clickable { onIncomeClick() }
.background(
if (selectedType == TransactionType.INCOME) MaterialTheme.colorScheme.primaryContainer
else Color.Transparent, RoundedCornerShape(8.dp)
)
.padding(8.dp)) {
Text(
text = "收入", style = MaterialTheme.typography.titleMedium
)
Text(
text = "¥${String.format("%.2f", totalIncome)}",
style = MaterialTheme.typography.bodyLarge,
color = MaterialTheme.colorScheme.primary
)
}
Spacer(modifier = Modifier.width(16.dp))
// 支出统计
Column(modifier = Modifier
.weight(1f)
.clickable { onExpenseClick() }
.background(
if (selectedType == TransactionType.EXPENSE) MaterialTheme.colorScheme.primaryContainer
else Color.Transparent, RoundedCornerShape(8.dp)
)
.padding(8.dp)) {
Text(
text = "支出", style = MaterialTheme.typography.titleMedium
)
Text(
text = "¥${String.format("%.2f", totalExpense)}",
style = MaterialTheme.typography.bodyLarge,
color = MaterialTheme.colorScheme.error
)
}
}
if (selectedType != null) {
TextButton(
onClick = onClearFilter, modifier = Modifier.align(Alignment.End)
) {
Text("清除筛选")
}
}
}
}
if (showMonthPicker) {
MonthYearPickerDialog(selectedMonth = selectedMonth,
onMonthSelected = onMonthSelected,
onDismiss = { showMonthPicker = false })
}
}
@Composable
fun RecordItem(
record: BookkeepingRecord,
onClick: () -> Unit = {},
onDelete: () -> Unit = {},
modifier: Modifier = Modifier
) {
Card(
modifier = modifier
.fillMaxWidth()
.clickable(onClick = onClick),
elevation = CardDefaults.cardElevation(defaultElevation = 2.dp)
) {
Row(
modifier = Modifier
.fillMaxWidth()
.padding(16.dp),
horizontalArrangement = Arrangement.SpaceBetween,
verticalAlignment = Alignment.CenterVertically
) {
Column(modifier = Modifier.weight(1f)) {
Text(
text = record.category, style = MaterialTheme.typography.titleMedium
)
if (record.description.isNotEmpty()) {
Text(
text = record.description,
style = MaterialTheme.typography.bodyMedium,
color = MaterialTheme.colorScheme.onSurfaceVariant
)
}
Text(
text = SimpleDateFormat(
"yyyy-MM-dd HH:mm", Locale.getDefault()
).format(record.date),
style = MaterialTheme.typography.bodySmall,
color = MaterialTheme.colorScheme.onSurfaceVariant
)
}
Row(
verticalAlignment = Alignment.CenterVertically
) {
Text(
text = if (record.type == TransactionType.EXPENSE) "-" else "+",
color = if (record.type == TransactionType.EXPENSE) MaterialTheme.colorScheme.error
else MaterialTheme.colorScheme.primary,
style = MaterialTheme.typography.titleMedium
)
Text(
text = String.format("%.2f", record.amount),
color = if (record.type == TransactionType.EXPENSE) MaterialTheme.colorScheme.error
else MaterialTheme.colorScheme.primary,
style = MaterialTheme.typography.titleMedium,
modifier = Modifier.padding(end = 8.dp)
)
IconButton(onClick = onDelete) {
Icon(
Icons.Default.Delete,
contentDescription = "删除",
tint = MaterialTheme.colorScheme.error
)
}
}
}
} }
} }

View File

@ -28,8 +28,6 @@ import com.yovinchen.bookkeeping.model.ThemeMode
import com.yovinchen.bookkeeping.ui.components.ColorPicker import com.yovinchen.bookkeeping.ui.components.ColorPicker
import com.yovinchen.bookkeeping.ui.components.predefinedColors import com.yovinchen.bookkeeping.ui.components.predefinedColors
import com.yovinchen.bookkeeping.ui.dialog.CategoryManagementDialog import com.yovinchen.bookkeeping.ui.dialog.CategoryManagementDialog
import com.yovinchen.bookkeeping.ui.dialog.MemberManagementDialog
import com.yovinchen.bookkeeping.viewmodel.MemberViewModel
import com.yovinchen.bookkeeping.viewmodel.SettingsViewModel import com.yovinchen.bookkeeping.viewmodel.SettingsViewModel
@OptIn(ExperimentalMaterial3Api::class) @OptIn(ExperimentalMaterial3Api::class)
@ -37,27 +35,15 @@ import com.yovinchen.bookkeeping.viewmodel.SettingsViewModel
fun SettingsScreen( fun SettingsScreen(
currentTheme: ThemeMode, currentTheme: ThemeMode,
onThemeChange: (ThemeMode) -> Unit, onThemeChange: (ThemeMode) -> Unit,
viewModel: SettingsViewModel = viewModel(), viewModel: SettingsViewModel = viewModel()
memberViewModel: MemberViewModel = viewModel()
) { ) {
var showThemeDialog by remember { mutableStateOf(false) } var showThemeDialog by remember { mutableStateOf(false) }
var showCategoryDialog by remember { mutableStateOf(false) } var showCategoryDialog by remember { mutableStateOf(false) }
var showMemberDialog by remember { mutableStateOf(false) }
val categories by viewModel.categories.collectAsState() val categories by viewModel.categories.collectAsState()
val selectedType by viewModel.selectedCategoryType.collectAsState() val selectedType by viewModel.selectedCategoryType.collectAsState()
val members by memberViewModel.allMembers.collectAsState(initial = emptyList())
Column(modifier = Modifier.fillMaxSize()) { Column(modifier = Modifier.fillMaxSize()) {
// 成员管理设置项
ListItem(
headlineContent = { Text("成员管理") },
supportingContent = { Text("管理账本成员") },
modifier = Modifier.clickable { showMemberDialog = true }
)
Divider()
// 类别管理设置项 // 类别管理设置项
ListItem( ListItem(
headlineContent = { Text("类别管理") }, headlineContent = { Text("类别管理") },
@ -159,19 +145,6 @@ fun SettingsScreen(
onTypeChange = viewModel::setSelectedCategoryType onTypeChange = viewModel::setSelectedCategoryType
) )
} }
// 成员管理对话框
if (showMemberDialog) {
MemberManagementDialog(
onDismiss = { showMemberDialog = false },
members = members,
onAddMember = memberViewModel::addMember,
onDeleteMember = memberViewModel::deleteMember,
onUpdateMember = { member, name, description ->
memberViewModel.updateMember(member.copy(name = name, description = description))
}
)
}
} }
@Composable @Composable

View File

@ -7,7 +7,6 @@ import androidx.lifecycle.viewModelScope
import com.yovinchen.bookkeeping.data.BookkeepingDatabase import com.yovinchen.bookkeeping.data.BookkeepingDatabase
import com.yovinchen.bookkeeping.model.BookkeepingRecord import com.yovinchen.bookkeeping.model.BookkeepingRecord
import com.yovinchen.bookkeeping.model.Category import com.yovinchen.bookkeeping.model.Category
import com.yovinchen.bookkeeping.model.Member
import com.yovinchen.bookkeeping.model.TransactionType import com.yovinchen.bookkeeping.model.TransactionType
import kotlinx.coroutines.ExperimentalCoroutinesApi import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.flow.* import kotlinx.coroutines.flow.*
@ -15,39 +14,37 @@ import kotlinx.coroutines.launch
import java.time.LocalDateTime import java.time.LocalDateTime
import java.time.ZoneId import java.time.ZoneId
import java.time.YearMonth import java.time.YearMonth
import java.util.* import java.util.Date
import java.util.Calendar
@OptIn(ExperimentalCoroutinesApi::class) @OptIn(ExperimentalCoroutinesApi::class)
class HomeViewModel(application: Application) : AndroidViewModel(application) { class HomeViewModel(application: Application) : AndroidViewModel(application) {
private val TAG = "HomeViewModel" private val TAG = "HomeViewModel"
private val bookkeepingDao = BookkeepingDatabase.getDatabase(application).bookkeepingDao() private val dao = BookkeepingDatabase.getDatabase(application).bookkeepingDao()
private val memberDao = BookkeepingDatabase.getDatabase(application).memberDao()
private val categoryDao = BookkeepingDatabase.getDatabase(application).categoryDao()
private val _selectedRecordType = MutableStateFlow<TransactionType?>(null) private val _selectedRecordType = MutableStateFlow<TransactionType?>(null)
val selectedRecordType: StateFlow<TransactionType?> = _selectedRecordType.asStateFlow() val selectedRecordType: StateFlow<TransactionType?> = _selectedRecordType.asStateFlow()
private val _selectedDateTime = MutableStateFlow(LocalDateTime.now())
val selectedDateTime: StateFlow<LocalDateTime> = _selectedDateTime.asStateFlow()
private val _selectedCategoryType = MutableStateFlow(TransactionType.EXPENSE)
val selectedCategoryType: StateFlow<TransactionType> = _selectedCategoryType.asStateFlow()
private val _selectedMonth = MutableStateFlow(YearMonth.now()) private val _selectedMonth = MutableStateFlow(YearMonth.now())
val selectedMonth: StateFlow<YearMonth> = _selectedMonth.asStateFlow() val selectedMonth: StateFlow<YearMonth> = _selectedMonth.asStateFlow()
private val _selectedMember = MutableStateFlow<Member?>(null) private val records = dao.getAllRecords()
val selectedMember: StateFlow<Member?> = _selectedMember.asStateFlow()
val members = memberDao.getAllMembers()
.stateIn( .stateIn(
scope = viewModelScope, scope = viewModelScope,
started = SharingStarted.WhileSubscribed(5000), started = SharingStarted.WhileSubscribed(5000),
initialValue = emptyList() initialValue = emptyList()
) )
val categories = categoryDao.getAllCategories() val categories: StateFlow<List<Category>> = _selectedCategoryType
.stateIn( .flatMapLatest { type ->
scope = viewModelScope, dao.getCategoriesByType(type)
started = SharingStarted.WhileSubscribed(5000), }
initialValue = emptyList()
)
private val allRecords = bookkeepingDao.getAllRecords()
.stateIn( .stateIn(
scope = viewModelScope, scope = viewModelScope,
started = SharingStarted.WhileSubscribed(5000), started = SharingStarted.WhileSubscribed(5000),
@ -55,11 +52,10 @@ class HomeViewModel(application: Application) : AndroidViewModel(application) {
) )
val filteredRecords = combine( val filteredRecords = combine(
allRecords, records,
_selectedRecordType, _selectedRecordType,
_selectedMonth, _selectedMonth
_selectedMember ) { records, selectedType, selectedMonth ->
) { records, selectedType, selectedMonth, selectedMember ->
records records
.filter { record -> .filter { record ->
val recordDate = record.date.toInstant() val recordDate = record.date.toInstant()
@ -69,14 +65,13 @@ class HomeViewModel(application: Application) : AndroidViewModel(application) {
val typeMatches = selectedType?.let { record.type == it } ?: true val typeMatches = selectedType?.let { record.type == it } ?: true
val monthMatches = recordYearMonth == selectedMonth val monthMatches = recordYearMonth == selectedMonth
val memberMatches = selectedMember?.let { record.memberId == it.id } ?: true
monthMatches && memberMatches && typeMatches typeMatches && monthMatches
} }
.sortedByDescending { it.date } .sortedByDescending { it.date }
.groupBy { record -> .groupBy { record ->
Calendar.getInstance().apply { val calendar = Calendar.getInstance().apply { time = record.date }
time = record.date calendar.apply {
set(Calendar.HOUR_OF_DAY, 0) set(Calendar.HOUR_OF_DAY, 0)
set(Calendar.MINUTE, 0) set(Calendar.MINUTE, 0)
set(Calendar.SECOND, 0) set(Calendar.SECOND, 0)
@ -84,16 +79,15 @@ class HomeViewModel(application: Application) : AndroidViewModel(application) {
}.time }.time
} }
}.stateIn( }.stateIn(
scope = viewModelScope, viewModelScope,
started = SharingStarted.WhileSubscribed(5000), SharingStarted.WhileSubscribed(5000),
initialValue = emptyMap() emptyMap()
) )
val totalIncome = combine( val totalIncome = combine(
allRecords, records,
_selectedMonth, _selectedMonth
_selectedMember ) { records, selectedMonth ->
) { records, selectedMonth, selectedMember ->
records records
.filter { record -> .filter { record ->
val recordDate = record.date.toInstant() val recordDate = record.date.toInstant()
@ -101,24 +95,19 @@ class HomeViewModel(application: Application) : AndroidViewModel(application) {
.toLocalDate() .toLocalDate()
val recordYearMonth = YearMonth.from(recordDate) val recordYearMonth = YearMonth.from(recordDate)
val monthMatches = recordYearMonth == selectedMonth record.type == TransactionType.INCOME && recordYearMonth == selectedMonth
val memberMatches = selectedMember?.let { record.memberId == it.id } ?: true
val typeMatches = record.type == TransactionType.INCOME
monthMatches && memberMatches && typeMatches
} }
.sumOf { it.amount } .sumOf { it.amount }
}.stateIn( }.stateIn(
scope = viewModelScope, viewModelScope,
started = SharingStarted.WhileSubscribed(5000), SharingStarted.WhileSubscribed(5000),
initialValue = 0.0 0.0
) )
val totalExpense = combine( val totalExpense = combine(
allRecords, records,
_selectedMonth, _selectedMonth
_selectedMember ) { records, selectedMonth ->
) { records, selectedMonth, selectedMember ->
records records
.filter { record -> .filter { record ->
val recordDate = record.date.toInstant() val recordDate = record.date.toInstant()
@ -126,73 +115,111 @@ class HomeViewModel(application: Application) : AndroidViewModel(application) {
.toLocalDate() .toLocalDate()
val recordYearMonth = YearMonth.from(recordDate) val recordYearMonth = YearMonth.from(recordDate)
val monthMatches = recordYearMonth == selectedMonth record.type == TransactionType.EXPENSE && recordYearMonth == selectedMonth
val memberMatches = selectedMember?.let { record.memberId == it.id } ?: true
val typeMatches = record.type == TransactionType.EXPENSE
monthMatches && memberMatches && typeMatches
} }
.sumOf { it.amount } .sumOf { it.amount }
}.stateIn( }.stateIn(
scope = viewModelScope, viewModelScope,
started = SharingStarted.WhileSubscribed(5000), SharingStarted.WhileSubscribed(5000),
initialValue = 0.0 0.0
) )
fun setSelectedMonth(yearMonth: YearMonth) { private fun updateTotals() {
_selectedMonth.value = yearMonth // 移除未使用的参数
} }
fun setSelectedMember(member: Member?) { init {
_selectedMember.value = member viewModelScope.launch {
} records.collect {
updateTotals()
fun moveMonth(forward: Boolean) { }
_selectedMonth.value = if (forward) {
_selectedMonth.value.plusMonths(1)
} else {
_selectedMonth.value.minusMonths(1)
} }
} }
suspend fun getMemberById(memberId: Int): Member? { fun addRecord(type: TransactionType, amount: Double, category: String, description: String) {
return memberDao.getMemberById(memberId)
}
fun addRecord(
amount: Double,
category: String,
description: String,
date: Date,
type: TransactionType,
memberId: Int?
) {
viewModelScope.launch { viewModelScope.launch {
val record = BookkeepingRecord( val record = BookkeepingRecord(
type = type,
amount = amount, amount = amount,
type = type,
category = category, category = category,
description = description, description = description,
date = date, date = Date.from(_selectedDateTime.value.atZone(ZoneId.systemDefault()).toInstant())
memberId = memberId
) )
bookkeepingDao.insertRecord(record) dao.insertRecord(record)
resetSelectedDateTime()
} }
} }
fun updateRecord(record: BookkeepingRecord) { fun setSelectedDateTime(dateTime: LocalDateTime) {
viewModelScope.launch { _selectedDateTime.value = dateTime
bookkeepingDao.updateRecord(record)
}
}
fun deleteRecord(record: BookkeepingRecord) {
viewModelScope.launch {
bookkeepingDao.deleteRecord(record)
}
} }
fun setSelectedRecordType(type: TransactionType?) { fun setSelectedRecordType(type: TransactionType?) {
_selectedRecordType.value = type _selectedRecordType.value = type
} }
fun setSelectedCategoryType(type: TransactionType) {
_selectedCategoryType.value = type
}
fun setSelectedMonth(yearMonth: YearMonth) {
_selectedMonth.value = yearMonth
}
fun moveMonth(forward: Boolean) {
val current = _selectedMonth.value
_selectedMonth.value = if (forward) {
current.plusMonths(1)
} else {
current.minusMonths(1)
}
}
fun resetSelectedDateTime() {
_selectedDateTime.value = LocalDateTime.now()
}
fun updateRecord(record: BookkeepingRecord) {
viewModelScope.launch {
dao.updateRecord(record)
}
}
fun deleteRecord(record: BookkeepingRecord) {
viewModelScope.launch {
dao.deleteRecord(record)
}
}
// 获取指定日期的记录
fun getRecordsByDate(date: LocalDateTime): Flow<List<BookkeepingRecord>> {
val calendar = Calendar.getInstance().apply {
time = Date.from(date.atZone(ZoneId.systemDefault()).toInstant())
set(Calendar.HOUR_OF_DAY, 0)
set(Calendar.MINUTE, 0)
set(Calendar.SECOND, 0)
set(Calendar.MILLISECOND, 0)
}
val startOfDay = calendar.time
calendar.add(Calendar.DAY_OF_MONTH, 1)
val endOfDay = calendar.time
return dao.getRecordsByDateRange(startOfDay, endOfDay)
}
// 获取指定日期范围的记录
fun getRecordsByDateRange(startDate: LocalDateTime, endDate: LocalDateTime): Flow<List<BookkeepingRecord>> {
val start = Date.from(startDate.atZone(ZoneId.systemDefault()).toInstant())
val end = Date.from(endDate.atZone(ZoneId.systemDefault()).toInstant())
return dao.getRecordsByDateRange(start, end)
}
// 获取指定类型的记录
fun getRecordsByType(type: TransactionType): Flow<List<BookkeepingRecord>> {
return dao.getRecordsByType(type)
}
} }
data class UiState(
val isAddingRecord: Boolean = false,
val isManagingCategories: Boolean = false
)

View File

@ -1,38 +0,0 @@
package com.yovinchen.bookkeeping.viewmodel
import android.app.Application
import androidx.lifecycle.AndroidViewModel
import androidx.lifecycle.viewModelScope
import com.yovinchen.bookkeeping.data.BookkeepingDatabase
import com.yovinchen.bookkeeping.model.Member
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.launch
class MemberViewModel(application: Application) : AndroidViewModel(application) {
private val memberDao = BookkeepingDatabase.getDatabase(application).memberDao()
val allMembers: Flow<List<Member>> = memberDao.getAllMembers()
fun addMember(name: String, description: String = "") {
viewModelScope.launch {
val member = Member(name = name, description = description)
memberDao.insertMember(member)
}
}
fun updateMember(member: Member) {
viewModelScope.launch {
memberDao.updateMember(member)
}
}
fun deleteMember(member: Member) {
viewModelScope.launch {
memberDao.deleteMember(member)
}
}
suspend fun getMemberCount(): Int {
return memberDao.getMemberCount()
}
}