- Drift 21 tables (8 catalog + 11 user + habit_dose_variants + meta_kv) with R1~R10 CHECK constraints and 19 indexes - 8 hand-crafted seed JSON catalogs in app/assets/seed/ (refs 84, protocols 34, methodologies 21, frame_patterns 30, reward_menu_items 30, break_protocols 8, common_frames 5, diet_patterns 5) - 6 domain functions: recommend_variant, compute_streak, validate_frame_level, active_habit_quota, weekly_minimum_ratio, seed_importer (transactional, idempotent) - 4 vertical-slice Riverpod screens: HabitList, HabitCreate, CheckIn, Streak - 31 unit tests passing; flutter analyze clean - OQ-5 streak semantics: missing entry ≠ explicit blank (missing = end of history; only TrackerValue.blank triggers Never-miss-twice) Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
15 lines
385 B
Dart
15 lines
385 B
Dart
import 'package:ulid/ulid.dart';
|
|
|
|
/// Typed ULID with prefix (e.g. `hb_01J9XYZ...`).
|
|
String generateUlid(String prefix) {
|
|
if (prefix.isEmpty) {
|
|
throw ArgumentError('prefix must be non-empty');
|
|
}
|
|
return '${prefix}_${Ulid().toString()}';
|
|
}
|
|
|
|
final RegExp _ulidRegex =
|
|
RegExp(r'^[A-Za-z]{1,8}_[0-9A-HJKMNP-TV-Z]{26}$');
|
|
|
|
bool isValidUlid(String s) => _ulidRegex.hasMatch(s);
|