From fab6aa504fc36c0dbac9f9128c14682bd78d781d Mon Sep 17 00:00:00 2001 From: Richard Brown Date: Fri, 24 Apr 2026 17:36:44 +0100 Subject: [PATCH 1/5] fix(wear): cache user doc stream to avoid re-subscribe on rebuild _userDocStream() was invoked directly inside build(), so each rebuild created a new Firestore snapshot subscription. This caused unnecessary reads and a brief "0 pts / 0 claimed" flicker every time the sibling StreamBuilder emitted (e.g. streak change). Cached in a late final field to match the existing _streakStream pattern. Co-Authored-By: Claude Opus 4.7 --- lib/wear/wear_status_page.dart | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/lib/wear/wear_status_page.dart b/lib/wear/wear_status_page.dart index 6a2147a..093233d 100644 --- a/lib/wear/wear_status_page.dart +++ b/lib/wear/wear_status_page.dart @@ -18,10 +18,14 @@ class WearStatusPage extends StatefulWidget { class _WearStatusPageState extends State { final StreakService _streakService = StreakService(); late final Stream _streakStream = _streakService.streakStream(); + // Cache so the StreamBuilder doesn't re-subscribe (and briefly show empty + // stats) on every rebuild. Computed once in initState from the uid at mount. + late final Stream>>? _userDocStream = + _buildUserDocStream(); String? get _displayName => FirebaseAuth.instance.currentUser?.displayName; - Stream>>? _userDocStream() { + Stream>>? _buildUserDocStream() { final uid = FirebaseAuth.instance.currentUser?.uid; if (uid == null) return null; return FirebaseFirestore.instance.collection('users').doc(uid).snapshots(); @@ -71,7 +75,7 @@ class _WearStatusPageState extends State { // Lifetime points StreamBuilder>>( - stream: _userDocStream(), + stream: _userDocStream, builder: (context, snap) { final data = snap.data?.data(); final lifetimePoints = From d05038869a1cf56638260af71c4b06a68f21e387 Mon Sep 17 00:00:00 2001 From: Richard Brown Date: Fri, 24 Apr 2026 18:02:27 +0100 Subject: [PATCH 2/5] fix(wear): pass rotation to compass painter so N label stays upright FuzzyCompassPainter counter-rotates the 'N' marker glyph so it remains readable while its ring position still tracks magnetic north. The wear compass page wrapped the painter in Transform.rotate(-rotation) but didn't forward rotation to the painter, so the N letter spun with the canvas and appeared sideways whenever the watch wasn't facing north. Co-Authored-By: Claude Opus 4.7 --- lib/wear/wear_compass_page.dart | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/lib/wear/wear_compass_page.dart b/lib/wear/wear_compass_page.dart index 8ab96c3..96fcaf2 100644 --- a/lib/wear/wear_compass_page.dart +++ b/lib/wear/wear_compass_page.dart @@ -206,7 +206,13 @@ class _WearCompassPageState extends State { angle: -rotation, child: CustomPaint( size: Size(size, size), - painter: FuzzyCompassPainter(sectors: sectorValues), + // Pass rotation so the painter counter-rotates the 'N' label; + // otherwise it spins with the canvas and is unreadable whenever + // the watch is not pointing north. + painter: FuzzyCompassPainter( + sectors: sectorValues, + rotation: rotation, + ), ), ), // Centre count overlay (doesn't rotate) From 75876ca7197ce2f0ad29736707e6c821936d5365 Mon Sep 17 00:00:00 2001 From: Richard Brown Date: Fri, 24 Apr 2026 18:29:06 +0100 Subject: [PATCH 3/5] fix(wear): normalise scan counts via num to avoid int cast throws Cloud Functions serialise JS numbers as either int or double depending on value, so `as int?` throws on a double. wear_claim_page and wear_compass_page both cast counts['total'] (and compass sector values) with `as int?`, which would crash the scan flow on any backend response where the count came through as a double. Normalise via `num?.toInt()` and reconstruct the compass counts map entry-by-entry for the same reason. Co-Authored-By: Claude Opus 4.7 --- lib/wear/wear_claim_page.dart | 6 ++++-- lib/wear/wear_compass_page.dart | 9 +++++++-- 2 files changed, 11 insertions(+), 4 deletions(-) diff --git a/lib/wear/wear_claim_page.dart b/lib/wear/wear_claim_page.dart index 32e1baf..b8f2445 100644 --- a/lib/wear/wear_claim_page.dart +++ b/lib/wear/wear_claim_page.dart @@ -52,8 +52,10 @@ class _WearClaimPageState extends State { }); if (!mounted) return; final counts = result.data['counts'] ?? {}; - final total = (counts['total'] as int?) ?? 0; - final claimed = (counts['claimedToday'] as int?) ?? 0; + // Cloud Functions serialise JS numbers as either int or double; `as int?` + // would throw on a double, so normalise via num. + final total = (counts['total'] as num?)?.toInt() ?? 0; + final claimed = (counts['claimedToday'] as num?)?.toInt() ?? 0; _postboxes = Map.from(result.data['postboxes'] ?? {}); setState(() { _count = total; diff --git a/lib/wear/wear_compass_page.dart b/lib/wear/wear_compass_page.dart index 96fcaf2..48174e2 100644 --- a/lib/wear/wear_compass_page.dart +++ b/lib/wear/wear_compass_page.dart @@ -75,8 +75,13 @@ class _WearCompassPageState extends State { final counts = result.data['counts'] ?? {}; final compassRaw = result.data['compass'] ?? {}; setState(() { - _totalCount = (counts['total'] as int?) ?? 0; - _compassCounts = Map.from(compassRaw); + // Cloud Functions serialise JS numbers as either int or double; + // `as int?` would throw on a double, so normalise via num. + _totalCount = (counts['total'] as num?)?.toInt() ?? 0; + _compassCounts = { + for (final e in (compassRaw as Map).entries) + e.key as String: (e.value as num).toInt(), + }; _stage = _CompassStage.results; }); // Haptic pulse to signal scan complete. From eca935af33afb172e6d7d58f1434238b834b3d4f Mon Sep 17 00:00:00 2001 From: Richard Brown Date: Fri, 24 Apr 2026 18:55:41 +0100 Subject: [PATCH 4/5] fix(phone): normalise nearby/claim counts via num to avoid int cast throws The Nearby and Claim screens assigned counts['total'], pts['max'], cipher counts, and compass sector values (all dynamic, originating from HttpsCallable.data) straight into typed int fields. Cloud Functions can serialise a JS number as either int or double depending on value (e.g. aggregations that go through a divide), so any double would crash the scan with a runtime cast error. Mirror the wear-side fix and normalise via `num?.toInt()`. Co-Authored-By: Claude Opus 4.7 --- lib/claim.dart | 11 +++++++---- lib/nearby.dart | 19 +++++++++++-------- 2 files changed, 18 insertions(+), 12 deletions(-) diff --git a/lib/claim.dart b/lib/claim.dart index 952c5d2..f14cf7e 100644 --- a/lib/claim.dart +++ b/lib/claim.dart @@ -125,11 +125,14 @@ class _ClaimState extends State with TickerProviderStateMixin { postboxes[entry.key as String] = Map.from(entry.value as Map); } + // Cloud Functions serialise JS numbers as either int or double; + // assigning a double to a typed int field throws, so normalise via num. + int asInt(dynamic v) => (v as num?)?.toInt() ?? 0; setState(() { - _count = counts['total'] ?? 0; - _maxPoints = points['max'] ?? 0; - _minPoints = points['min'] ?? 0; - _claimedToday = counts['claimedToday'] ?? 0; + _count = asInt(counts['total']); + _maxPoints = asInt(points['max']); + _minPoints = asInt(points['min']); + _claimedToday = asInt(counts['claimedToday']); _postboxes = postboxes; currentStage = _count > 0 ? ClaimStage.results : ClaimStage.empty; }); diff --git a/lib/nearby.dart b/lib/nearby.dart index 4f743cb..a047335 100644 --- a/lib/nearby.dart +++ b/lib/nearby.dart @@ -113,23 +113,26 @@ class _NearbyState extends State { final claimedCompass = data['claimedCompass'] != null ? Map.from(data['claimedCompass'] as Map) : {}; + // Cloud Functions serialise JS numbers as either int or double; assigning + // a double to a typed int field throws, so normalise every count via num. + int asInt(dynamic v) => (v as num?)?.toInt() ?? 0; setState(() { - _count = counts['total'] ?? 0; - _maxPoints = pts['max'] ?? 0; - _minPoints = pts['min'] ?? 0; + _count = asInt(counts['total']); + _maxPoints = asInt(pts['max']); + _minPoints = asInt(pts['min']); currentStage = NearbyStage.results; for (final dir in const [ 'N', 'NNE', 'NE', 'ENE', 'E', 'ESE', 'SE', 'SSE', 'S', 'SSW', 'SW', 'WSW', 'W', 'WNW', 'NW', 'NNW', ]) { - _compassCounts[dir] = compass[dir] ?? 0; - _claimedCompassCounts[dir] = claimedCompass[dir] ?? 0; + _compassCounts[dir] = asInt(compass[dir]); + _claimedCompassCounts[dir] = asInt(claimedCompass[dir]); } - _claimedToday = counts['claimedToday'] ?? 0; + _claimedToday = asInt(counts['claimedToday']); for (final cipher in MonarchInfo.all) { - _cipherTotals[cipher] = counts[cipher] ?? 0; - _cipherClaimed[cipher] = counts['${cipher}_claimed'] ?? 0; + _cipherTotals[cipher] = asInt(counts[cipher]); + _cipherClaimed[cipher] = asInt(counts['${cipher}_claimed']); } _lastScanned = DateTime.now(); }); From 885cecefc87828c2aab2b97129633fc4bb478ffd Mon Sep 17 00:00:00 2001 From: Richard Brown Date: Sat, 25 Apr 2026 09:56:00 +0100 Subject: [PATCH 5/5] docs(plan): remote-config-tuning --- docs/plans/remote-config-tuning.md | 50 ++++++++++++++++++++++++++++++ 1 file changed, 50 insertions(+) create mode 100644 docs/plans/remote-config-tuning.md diff --git a/docs/plans/remote-config-tuning.md b/docs/plans/remote-config-tuning.md new file mode 100644 index 0000000..bfc2798 --- /dev/null +++ b/docs/plans/remote-config-tuning.md @@ -0,0 +1,50 @@ +# Plan — Remote Config for game balance and copy + +- **Status:** Proposed +- **Effort:** Small (client) + tiny (backend) +- **Firebase services:** Remote Config, Remote Config server-side SDK (Node) + +## Overview + +Centralise tunable values so game balance, reminder copy, and James cadence can change without a client release. Both the Flutter client and Cloud Functions read Remote Config. + +## Parameters (initial set) + +| Key | Default | Notes | +|-----|---------|-------| +| `claim_radius_meters` | `30` | Used by `startScoring` geohash + distance check. | +| `points_by_monarch` | JSON `{EIIR:2,GR:4,...}` | Overrides hard-coded values in `_getPoints.ts`. | +| `james_idle_min_seconds` | `120` | Idle non-sequitur lower bound. | +| `james_idle_max_seconds` | `300` | Upper bound. | +| `daily_reminder_hour_local` | `18` | Push time. | +| `daily_reminder_copy` | James-voice string | Localisable per monarch era. | +| `nearby_radius_meters` | `540` | Compass / heatmap range. | +| `quiz_required_streak` | `7` | Quiz gates post-streak day. | + +## Client + +- Add `firebase_remote_config` to `pubspec.yaml`. +- `lib/services/remote_config_service.dart` — fetch-and-activate on app start (min interval 1 h in prod, 0 in debug), exposes typed getters. +- Replace magic numbers in `home.dart`, `claim.dart`, `james_strip.dart`, `fuzzy_compass.dart`. + +## Backend + +- Add `firebase-admin` Remote Config fetch in `_config.ts` with 5 min in-memory cache. +- `_getPoints.ts` reads `points_by_monarch` with fallback to current hard-coded table. +- `startScoring` uses `claim_radius_meters`. + +## Rollout + +- Phase 1: ship client reading only; values identical to current defaults. +- Phase 2: backend reading; verify via integration test with Remote Config emulator. +- Phase 3: tune via console, monitor Analytics for regressions. + +## Risks + +- Stale cache in the client masking a bad value — force a refetch hook at login. +- Cost drift if `points_by_monarch` is set incorrectly; add a sanity-check Cloud Function that rejects values outside `[1, 50]`. + +## Testing + +- Unit-test `RemoteConfigService` typed getters. +- Add emulator-backed test in `functions/src/test` that asserts `_getPoints` honours override.