Add Firebase Analytics (GA4) funnel tracking to client_app: - AnalyticsService typed wrapper (enum-gated, no PII) + analyticsProvider - FirebaseAnalyticsObserver on GoRouter (screen_name via nameExtractor) - user_id = customer UUID, user_type property, set on auth resolve/upgrade - funnel events: curhat_start, curhat_repeat_start, auth_*, onboarding_usp_view, payment_view, payment_method_select, payment_started, pairing_matched/no_bestie - bottom-sheet events: verif_choice_view/select, bestie_choice_view/select, extension_offer_view, chat_extension_requested - payment_started carries app_instance_id + ga_session_id in the /payment-requests body for future server-side stitching (backend ignores) - curhat_mode_pick screen name disambiguates the chat/call mode picker (/payment/method-pick) from the payment-channel picker (/payment/method) - unify both home CTAs to "Aku Mau Curhat" Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
204 lines
6.2 KiB
Dart
204 lines
6.2 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
|
import 'package:go_router/go_router.dart';
|
|
import '../../../core/analytics/analytics_service.dart';
|
|
import '../../../core/theme/halo_tokens.dart';
|
|
import '../../../core/theme/widgets/widgets.dart';
|
|
import '../usp_seen_provider.dart';
|
|
|
|
/// Onboarding step 2 — static value-prop ("USP") cards. One-time gate
|
|
/// (Phase 4, 2026-05-12): on Continue we mark the local `usp_seen` flag and
|
|
/// best-effort persist to DB so this screen never shows again for this user.
|
|
///
|
|
/// `verified` ➞ USP → OTP (`/auth/register`).
|
|
/// `anonymous` ➞ USP → `/payment/method-pick`.
|
|
class UspScreen extends ConsumerStatefulWidget {
|
|
final bool verified;
|
|
|
|
const UspScreen({super.key, required this.verified});
|
|
|
|
@override
|
|
ConsumerState<UspScreen> createState() => _UspScreenState();
|
|
}
|
|
|
|
class _UspScreenState extends ConsumerState<UspScreen> {
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
// Activation funnel step 7 — fire on view (not teardown). One-shot:
|
|
// initState runs once per screen instance.
|
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
|
if (!mounted) return;
|
|
// ignore: discarded_futures
|
|
ref
|
|
.read(analyticsProvider)
|
|
.logOnboardingUspView(verified: widget.verified);
|
|
});
|
|
}
|
|
|
|
static const _cards = [
|
|
_UspCard(
|
|
icon: Icons.bolt_outlined,
|
|
title: 'Langsung Curhat',
|
|
body: 'Nggak perlu janji, nggak perlu nunggu. Buka, ngobrol, plong.',
|
|
),
|
|
_UspCard(
|
|
icon: Icons.shield_outlined,
|
|
title: 'Tetap Anonim',
|
|
body: 'Identitasmu disembunyikan. Cerita apa adanya, tanpa khawatir.',
|
|
),
|
|
_UspCard(
|
|
icon: Icons.favorite_outline,
|
|
title: 'Bestie yang Relate',
|
|
body: 'Diisi orang yang udah dilatih buat dengerin, bukan ngehakimin.',
|
|
),
|
|
_UspCard(
|
|
icon: Icons.payments_outlined,
|
|
title: 'Bayar Sesuai Pakai',
|
|
body: 'Pilih durasi yang pas. Nggak ada langganan, nggak ada jebakan.',
|
|
),
|
|
];
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Scaffold(
|
|
appBar: AppBar(
|
|
title: const Padding(
|
|
padding: EdgeInsets.only(top: HaloSpacing.s4),
|
|
child: HaloStepDots(total: 4, current: 2),
|
|
),
|
|
centerTitle: true,
|
|
),
|
|
body: SafeArea(
|
|
top: false,
|
|
child: Padding(
|
|
padding: const EdgeInsets.fromLTRB(
|
|
HaloSpacing.s24,
|
|
HaloSpacing.s8,
|
|
HaloSpacing.s24,
|
|
HaloSpacing.s24,
|
|
),
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
|
children: [
|
|
const Text(
|
|
'Sebelum mulai',
|
|
style: TextStyle(
|
|
fontFamily: HaloTokens.fontDisplay,
|
|
fontSize: 26,
|
|
height: 30 / 26,
|
|
fontWeight: FontWeight.w700,
|
|
color: HaloTokens.ink,
|
|
),
|
|
),
|
|
const SizedBox(height: HaloSpacing.s8),
|
|
const Text(
|
|
'Hal-hal kecil yang bikin Halo Bestie beda.',
|
|
style: TextStyle(
|
|
fontFamily: HaloTokens.fontBody,
|
|
fontSize: 14,
|
|
height: 20 / 14,
|
|
color: HaloTokens.inkSoft,
|
|
),
|
|
),
|
|
const SizedBox(height: HaloSpacing.s24),
|
|
Expanded(
|
|
child: ListView.separated(
|
|
itemCount: _cards.length,
|
|
separatorBuilder: (_, __) =>
|
|
const SizedBox(height: HaloSpacing.s12),
|
|
itemBuilder: (_, i) => _cards[i],
|
|
),
|
|
),
|
|
const SizedBox(height: HaloSpacing.s16),
|
|
HaloButton(
|
|
label: 'aku ngerti, lanjut',
|
|
fullWidth: true,
|
|
onPressed: () => _onContinue(context),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
Future<void> _onContinue(BuildContext context) async {
|
|
// Persist the local + server flag before leaving — next time the user
|
|
// hits VerifChoice, this screen is skipped.
|
|
await ref.read(uspSeenProvider.notifier).markSeen();
|
|
if (!context.mounted) return;
|
|
if (widget.verified) {
|
|
context.push('/auth/register');
|
|
} else {
|
|
context.push('/payment/method-pick');
|
|
}
|
|
}
|
|
}
|
|
|
|
class _UspCard extends StatelessWidget {
|
|
final IconData icon;
|
|
final String title;
|
|
final String body;
|
|
|
|
const _UspCard({
|
|
required this.icon,
|
|
required this.title,
|
|
required this.body,
|
|
});
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Container(
|
|
padding: const EdgeInsets.all(HaloSpacing.s16),
|
|
decoration: BoxDecoration(
|
|
color: HaloTokens.surface,
|
|
borderRadius: HaloRadius.lg,
|
|
border: Border.all(color: HaloTokens.border),
|
|
),
|
|
child: Row(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Container(
|
|
width: 40,
|
|
height: 40,
|
|
decoration: const BoxDecoration(
|
|
color: HaloTokens.brandSofter,
|
|
shape: BoxShape.circle,
|
|
),
|
|
alignment: Alignment.center,
|
|
child: Icon(icon, color: HaloTokens.brandDark, size: 20),
|
|
),
|
|
const SizedBox(width: HaloSpacing.s12),
|
|
Expanded(
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Text(
|
|
title,
|
|
style: const TextStyle(
|
|
fontFamily: HaloTokens.fontBody,
|
|
fontSize: 16,
|
|
fontWeight: FontWeight.w600,
|
|
color: HaloTokens.ink,
|
|
),
|
|
),
|
|
const SizedBox(height: HaloSpacing.s4),
|
|
Text(
|
|
body,
|
|
style: const TextStyle(
|
|
fontFamily: HaloTokens.fontBody,
|
|
fontSize: 13,
|
|
height: 18 / 13,
|
|
color: HaloTokens.inkSoft,
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
}
|