yayacal/COMMENTS.md
xfy 6c3e773a38 Add commenting conventions document for the project
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-14 15:10:26 +08:00

6.0 KiB
Raw Permalink Blame History

YaYa 注释规范

基于 KDoc 规范与 Compose 语义化约定,结合项目实际情况制定。

核心原则

  1. 注释解释「为什么」,而非「做什么」——代码本身应能说明做什么,注释补充意图、约束和决策原因
  2. 宁可没有注释,也不要废话注释——当前项目风格是自描述代码 + 极少注释,这是好的基线
  3. 公共 API 必须有 KDoc内部实现按需注释

何时必须写注释

1. Public Composable 函数——KDoc 必需

所有 public@Composable 函数必须写 KDoc说明用途、参数含义和回调触发时机。

/**
 * 月度日历视图,支持周/月切换和折叠动画。
 *
 * @param selectedDate 当前选中日期
 * @param today 今天的日期,用于高亮标记
 * @param onDateClick 日期点击回调
 * @param collapseProgress 折叠进度0f=展开1f=折叠
 * @param modifier 外部布局修饰符
 */
@Composable
fun CalendarMonthPage(
    year: Int,
    month: Int,
    selectedDate: LocalDate,
    today: LocalDate,
    onDateClick: (LocalDate) -> Unit,
    collapseProgress: Float,
    modifier: Modifier = Modifier
)

约定:

  • Modifier 参数放最后KDoc 中注明"外部布局修饰符"
  • 回调参数用 on 前缀命名(onDateClick 而非 dateClickKDoc 说明触发时机
  • Float 进度/比例参数注明取值范围和含义

2. 非显而易见的算法和计算

// 6行×7列=42格覆盖跨月首尾周保证网格完整
return (0 until 42).map { i -> ... }

// 折叠时选中行上方行上移、下方行下移,模拟"挤压"效果
val offsetY = when {
    isAboveSelected -> -progress * 200f
    isBelowSelected -> progress * 200f
    else -> 0f
}

3. Magic Number

用命名常量 + 注释,而非裸数字:

// 好
val MaxPagerPages = Int.MAX_VALUE  // 无限分页,中心页为起始月
val CenterPage = MaxPagerPages / 2

// 坏
HorizontalPager(pageCount = 2147483647) { ... }

4. Workaround / Hack / 平台限制

必须注明原因和追踪信息:

@Suppress("DEPRECATION")  // monthNumber 无替代 APIkotlinx-datetime 尚未提供新接口
currentMonth = weekMonday.monthNumber

5. 状态与副作用的业务意图

rememberLaunchedEffectDisposableEffect 等解释业务意图,而非技术实现:

// 仅在首次展开时记录完整日历高度,折叠后不再覆盖
if (!viewModel.isCollapsed && viewModel.collapseProgress < 0.01f) {
    expandedCalendarHeightPx = size.height
}

6. 动画参数与业务状态的映射

// collapseProgress: 0f=月视图(6行), 1f=周视图(1行)
// 折叠偏移量 = 进度 × 展开高度的5/6保留1行可见
val collapseOffsetPx = -(viewModel.collapseProgress * expandedCalendarHeightPx * 5f / 6f).toInt()

何时不需要注释

1. 代码本身已足够清晰

// 坏——废话注释
// 设置水平 padding 为 16dp
modifier = Modifier.padding(horizontal = 16.dp)

// 好——无需注释
modifier = Modifier.padding(horizontal = 16.dp)

2. 函数名已说明用途

// 坏
// 获取 ISO 周号
fun getIsoWeekNumber(date: LocalDate): Int

// 好——函数名已足够
fun getIsoWeekNumber(date: LocalDate): Int

3. 简单的属性赋值和数据类

// 不需要注释
data class DayData(
    val date: LocalDate,
    val isCurrentMonth: Boolean
)

KDoc 格式规范

Composable 函数模板

/**
 * 一句话描述组件用途。
 *
 * 可选:补充说明重组行为、副作用等。
 *
 * @param param1 参数说明
 * @param param2 参数说明
 * @param modifier 外部布局修饰符
 */
@Composable
fun MyComponent(
    param1: Type,
    param2: Type,
    modifier: Modifier = Modifier
) { ... }

ViewModel / 工具函数模板

/**
 * 一句话描述功能。
 *
 * @param input 输入说明
 * @return 返回值说明
 */
fun calculateSomething(input: Int): Int { ... }

文件级注释

仅在文件包含多个紧密关联的组件且关系不直观时使用,大多数文件不需要:

/**
 * 日历分页组件,包含月视图和周视图的 HorizontalPager 实现。
 */

项目特定约定

场景 规范
@Suppress("DEPRECATION") 必须附带行内注释说明原因(当前主要用于 monthNumber
Pager 页码映射 pageToYearMonth() / yearMonthToPage() 等算术映射需注释公式逻辑
折叠动画参数 注释 collapseProgress 的取值范围和物理含义
尺寸测量px 注释为何需要 px 而非 dp动画偏移量需要像素精度
remember key 当 key 列表不直观时,注释为何选择这些 key
Int.MAX_VALUE 分页 注释无限分页的设计意图和中心页计算方式

Preview 注释

Preview 函数不需要 KDoc@Preview 注解应提供有意义的 name

@Preview(name = "Month View - Expanded", showBackground = true)
@Preview(name = "Month View - Collapsed", showBackground = true)
@Composable
private fun CalendarMonthViewPreview() { ... }

反模式清单

// ❌ 翻译代码
// 遍历每一周
weeks.forEachIndexed { ... }

// ❌ 过时注释
// TODO: 后续优化(已完成但未删除)

// ❌ 注释掉的代码
// val oldImplementation = ...

// ❌ 用 // 替代 KDoc
// 这个函数渲染日历
@Composable fun Calendar() { ... }

// ❌ 无原因的 Suppress
@Suppress("DEPRECATION")
fun foo() { ... }

检查清单

  • Public Composable 是否有 KDoc
  • KDoc 是否说明了参数含义和回调触发时机?
  • Modifier 参数是否在最后?
  • 非显而易见的计算是否有行内注释?
  • Magic Number 是否有命名常量 + 注释?
  • @Suppress 是否有原因注释?
  • 动画/进度参数是否注明了取值范围?
  • 是否存在废话注释、过时注释或注释掉的代码?