- Add Skip/Skip Forever links to onboarding screens only - Remove Skip links from questionnaire screens (baseline, behavioral, lifestyle) - Add app icon (1024x1024) and generate all platform sizes - Update About screen with correct URLs (privacy, terms, disclaimer, how it works) - Add "No ads. Ever." section to About screen - Configure Android release signing with upload keystore - Add URL intent query for Android 11+ link launching - Add skipForever preference to LocalStorage Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
230 lines
7.6 KiB
Dart
230 lines
7.6 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:url_launcher/url_launcher.dart';
|
|
import '../theme.dart';
|
|
|
|
class AboutScreen extends StatelessWidget {
|
|
const AboutScreen({super.key});
|
|
|
|
static const String _howItWorksUrl = 'https://addmonths.com/#how-it-works';
|
|
static const String _privacyUrl = 'https://addmonths.com/privacy/';
|
|
static const String _termsUrl = 'https://addmonths.com/terms/';
|
|
static const String _disclaimerUrl = 'https://addmonths.com/disclaimer/';
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Scaffold(
|
|
appBar: AppBar(
|
|
title: const Text('About'),
|
|
leading: IconButton(
|
|
icon: const Icon(Icons.arrow_back),
|
|
onPressed: () => Navigator.pop(context),
|
|
),
|
|
),
|
|
body: SingleChildScrollView(
|
|
padding: const EdgeInsets.all(24),
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
// App name and version
|
|
Center(
|
|
child: Column(
|
|
children: [
|
|
Container(
|
|
width: 80,
|
|
height: 80,
|
|
decoration: BoxDecoration(
|
|
color: AppColors.surfaceVariant,
|
|
borderRadius: BorderRadius.circular(20),
|
|
),
|
|
child: const Icon(
|
|
Icons.timeline,
|
|
size: 40,
|
|
color: AppColors.primary,
|
|
),
|
|
),
|
|
const SizedBox(height: 16),
|
|
Text(
|
|
'Add Months',
|
|
style: Theme.of(context).textTheme.headlineMedium,
|
|
),
|
|
const SizedBox(height: 4),
|
|
Text(
|
|
'Version 1.2',
|
|
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
|
|
color: AppColors.textSecondary,
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
const SizedBox(height: 32),
|
|
|
|
// Description
|
|
Text(
|
|
'What is Add Months?',
|
|
style: Theme.of(context).textTheme.titleMedium,
|
|
),
|
|
const SizedBox(height: 8),
|
|
Text(
|
|
'Add Months uses evidence-based hazard ratios from peer-reviewed '
|
|
'meta-analyses to identify which single lifestyle change could '
|
|
'have the biggest impact on your lifespan.',
|
|
style: Theme.of(context).textTheme.bodyMedium,
|
|
),
|
|
const SizedBox(height: 16),
|
|
Text(
|
|
'Answer simple questions about your demographics and habits, '
|
|
'and the app calculates which modifiable factor offers the '
|
|
'greatest potential benefit.',
|
|
style: Theme.of(context).textTheme.bodyMedium,
|
|
),
|
|
const SizedBox(height: 32),
|
|
|
|
// Privacy note
|
|
Container(
|
|
padding: const EdgeInsets.all(16),
|
|
decoration: BoxDecoration(
|
|
color: AppColors.surfaceVariant,
|
|
borderRadius: BorderRadius.circular(12),
|
|
),
|
|
child: Row(
|
|
children: [
|
|
const Icon(
|
|
Icons.lock_outline,
|
|
color: AppColors.primary,
|
|
),
|
|
const SizedBox(width: 12),
|
|
Expanded(
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Text(
|
|
'Your data stays on your device',
|
|
style: Theme.of(context).textTheme.titleSmall,
|
|
),
|
|
const SizedBox(height: 4),
|
|
Text(
|
|
'No accounts, no cloud sync, no analytics. '
|
|
'Everything is stored locally.',
|
|
style: Theme.of(context).textTheme.bodySmall,
|
|
),
|
|
],
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
const SizedBox(height: 16),
|
|
|
|
// No ads promise
|
|
Container(
|
|
padding: const EdgeInsets.all(16),
|
|
decoration: BoxDecoration(
|
|
color: AppColors.surfaceVariant,
|
|
borderRadius: BorderRadius.circular(12),
|
|
),
|
|
child: Row(
|
|
children: [
|
|
const Icon(
|
|
Icons.block,
|
|
color: AppColors.primary,
|
|
),
|
|
const SizedBox(width: 12),
|
|
Expanded(
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Text(
|
|
'No ads. Ever.',
|
|
style: Theme.of(context).textTheme.titleSmall,
|
|
),
|
|
const SizedBox(height: 4),
|
|
Text(
|
|
'No trackers, no monetization schemes. '
|
|
'Just a simple tool that respects your time.',
|
|
style: Theme.of(context).textTheme.bodySmall,
|
|
),
|
|
],
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
const SizedBox(height: 32),
|
|
|
|
// Links
|
|
_buildLinkButton(
|
|
context,
|
|
icon: Icons.help_outline,
|
|
label: 'How It Works',
|
|
onTap: () => _launchUrl(_howItWorksUrl),
|
|
),
|
|
const SizedBox(height: 12),
|
|
_buildLinkButton(
|
|
context,
|
|
icon: Icons.privacy_tip_outlined,
|
|
label: 'Privacy Policy',
|
|
onTap: () => _launchUrl(_privacyUrl),
|
|
),
|
|
const SizedBox(height: 12),
|
|
_buildLinkButton(
|
|
context,
|
|
icon: Icons.description_outlined,
|
|
label: 'Terms of Service',
|
|
onTap: () => _launchUrl(_termsUrl),
|
|
),
|
|
const SizedBox(height: 12),
|
|
_buildLinkButton(
|
|
context,
|
|
icon: Icons.medical_information_outlined,
|
|
label: 'Medical Disclaimer',
|
|
onTap: () => _launchUrl(_disclaimerUrl),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
Widget _buildLinkButton(
|
|
BuildContext context, {
|
|
required IconData icon,
|
|
required String label,
|
|
required VoidCallback onTap,
|
|
}) {
|
|
return InkWell(
|
|
onTap: onTap,
|
|
borderRadius: BorderRadius.circular(12),
|
|
child: Container(
|
|
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 14),
|
|
decoration: BoxDecoration(
|
|
border: Border.all(color: AppColors.divider),
|
|
borderRadius: BorderRadius.circular(12),
|
|
),
|
|
child: Row(
|
|
children: [
|
|
Icon(icon, color: AppColors.primary),
|
|
const SizedBox(width: 12),
|
|
Text(
|
|
label,
|
|
style: Theme.of(context).textTheme.bodyLarge,
|
|
),
|
|
const Spacer(),
|
|
const Icon(
|
|
Icons.open_in_new,
|
|
size: 18,
|
|
color: AppColors.textSecondary,
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
Future<void> _launchUrl(String url) async {
|
|
final uri = Uri.parse(url);
|
|
if (await canLaunchUrl(uri)) {
|
|
await launchUrl(uri, mode: LaunchMode.externalApplication);
|
|
}
|
|
}
|
|
}
|