671 lines
23 KiB
Dart
671 lines
23 KiB
Dart
import 'package:auto_route/auto_route.dart';
|
|
import 'package:dio/dio.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:gap/gap.dart';
|
|
import 'package:google_fonts/google_fonts.dart';
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
import 'package:island/models/wallet.dart';
|
|
import 'package:island/pods/network.dart';
|
|
import 'package:island/pods/userinfo.dart';
|
|
import 'package:island/services/responsive.dart';
|
|
import 'package:island/services/time.dart';
|
|
import 'package:island/widgets/account/leveling_progress.dart';
|
|
import 'package:island/widgets/alert.dart';
|
|
import 'package:island/widgets/app_scaffold.dart';
|
|
import 'package:island/widgets/payment/payment_overlay.dart';
|
|
import 'package:easy_localization/easy_localization.dart';
|
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
|
|
|
part 'leveling.g.dart';
|
|
|
|
@riverpod
|
|
Future<SnWalletSubscription?> accountStellarSubscription(Ref ref) async {
|
|
try {
|
|
final client = ref.watch(apiClientProvider);
|
|
final resp = await client.get('/subscriptions/fuzzy/solian.stellar');
|
|
return SnWalletSubscription.fromJson(resp.data);
|
|
} catch (err) {
|
|
if (err is DioException && err.response?.statusCode == 404) return null;
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
@RoutePage()
|
|
class LevelingScreen extends HookConsumerWidget {
|
|
const LevelingScreen({super.key});
|
|
|
|
@override
|
|
Widget build(BuildContext context, WidgetRef ref) {
|
|
final user = ref.watch(userInfoProvider);
|
|
final stellarSubscription = ref.watch(accountStellarSubscriptionProvider);
|
|
|
|
if (user.value == null) {
|
|
return AppScaffold(
|
|
appBar: AppBar(title: Text('levelingProgress'.tr())),
|
|
body: const Center(child: CircularProgressIndicator()),
|
|
);
|
|
}
|
|
|
|
final currentLevel = user.value!.profile.level;
|
|
final currentExp = user.value!.profile.experience;
|
|
final progress = user.value!.profile.levelingProgress;
|
|
|
|
return AppScaffold(
|
|
appBar: AppBar(title: Text('levelingProgress'.tr())),
|
|
body: SingleChildScrollView(
|
|
padding: getTabbedPadding(context, horizontal: 20, vertical: 20),
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
|
children: [
|
|
// Current Progress Card
|
|
LevelingProgressCard(
|
|
level: currentLevel,
|
|
experience: currentExp,
|
|
progress: progress,
|
|
),
|
|
const Gap(24),
|
|
|
|
// Level Stairs Graph
|
|
Text(
|
|
'levelProgress'.tr(),
|
|
style: Theme.of(
|
|
context,
|
|
).textTheme.headlineSmall?.copyWith(fontWeight: FontWeight.bold),
|
|
),
|
|
const Gap(16),
|
|
|
|
// Stairs visualization with fixed height and horizontal scroll
|
|
_buildLevelStairs(context, currentLevel),
|
|
|
|
const Gap(24),
|
|
|
|
// Membership section
|
|
_buildMembershipSection(context, ref, stellarSubscription),
|
|
const Gap(16),
|
|
|
|
// Unlocked features section
|
|
Container(
|
|
width: double.infinity,
|
|
padding: const EdgeInsets.all(16),
|
|
decoration: BoxDecoration(
|
|
color: Theme.of(context).colorScheme.surfaceContainerHigh,
|
|
borderRadius: BorderRadius.circular(12),
|
|
),
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Text(
|
|
'unlockedFeatures'.tr(),
|
|
style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold),
|
|
),
|
|
const Gap(8),
|
|
Text(
|
|
'unlockedFeaturesDescription'.tr(),
|
|
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
|
|
color: Theme.of(context).colorScheme.onSurfaceVariant,
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
Widget _buildLevelStairs(BuildContext context, int currentLevel) {
|
|
const totalLevels = 14;
|
|
const stairHeight = 20.0;
|
|
const stairWidth = 50.0;
|
|
const containerHeight = 280.0;
|
|
|
|
return Container(
|
|
height: containerHeight,
|
|
decoration: BoxDecoration(
|
|
borderRadius: BorderRadius.circular(12),
|
|
border: Border.all(
|
|
color: Theme.of(context).colorScheme.outline.withOpacity(0.2),
|
|
),
|
|
),
|
|
child: SingleChildScrollView(
|
|
scrollDirection: Axis.horizontal,
|
|
padding: const EdgeInsets.symmetric(horizontal: 16),
|
|
child: SizedBox(
|
|
width: (totalLevels * (stairWidth + 8)) + 40,
|
|
height: containerHeight,
|
|
child: CustomPaint(
|
|
painter: LevelStairsPainter(
|
|
currentLevel: currentLevel,
|
|
totalLevels: totalLevels,
|
|
primaryColor: Theme.of(context).colorScheme.primary,
|
|
surfaceColor: Theme.of(context).colorScheme.surfaceContainerHigh,
|
|
onSurfaceColor: Theme.of(context).colorScheme.onSurface,
|
|
stairHeight: stairHeight,
|
|
stairWidth: stairWidth,
|
|
),
|
|
child: Stack(
|
|
children: List.generate(totalLevels, (index) {
|
|
final level = index + 1;
|
|
final isCompleted = level <= currentLevel;
|
|
final isCurrent = level == currentLevel;
|
|
|
|
// Calculate position from bottom
|
|
final bottomPosition = 0.0;
|
|
final leftPosition = 20.0 + (index * (stairWidth + 8));
|
|
|
|
// Make higher levels progressively taller
|
|
final progressiveHeight =
|
|
40.0 + (index * 15.0); // Base height + progressive increase
|
|
|
|
return Positioned(
|
|
left: leftPosition,
|
|
bottom: bottomPosition,
|
|
child: Container(
|
|
width: stairWidth,
|
|
height: progressiveHeight,
|
|
decoration: BoxDecoration(
|
|
color:
|
|
isCompleted
|
|
? Theme.of(context).colorScheme.primary
|
|
: Theme.of(
|
|
context,
|
|
).colorScheme.surfaceContainerHigh,
|
|
borderRadius: const BorderRadius.only(
|
|
topLeft: Radius.circular(6),
|
|
topRight: Radius.circular(6),
|
|
),
|
|
border:
|
|
isCurrent
|
|
? Border.all(
|
|
color: Theme.of(context).colorScheme.primary,
|
|
width: 2,
|
|
)
|
|
: null,
|
|
boxShadow:
|
|
isCurrent
|
|
? [
|
|
BoxShadow(
|
|
color: Theme.of(
|
|
context,
|
|
).colorScheme.primary.withOpacity(0.3),
|
|
blurRadius: 6,
|
|
spreadRadius: 1,
|
|
),
|
|
]
|
|
: null,
|
|
),
|
|
child: Padding(
|
|
padding: const EdgeInsets.only(top: 8),
|
|
child: Column(
|
|
children: [
|
|
Text(
|
|
level.toString(),
|
|
style: GoogleFonts.robotoMono(
|
|
fontSize: 14,
|
|
fontWeight: FontWeight.bold,
|
|
color:
|
|
isCompleted
|
|
? Theme.of(context).colorScheme.onPrimary
|
|
: Theme.of(context).colorScheme.onSurface,
|
|
),
|
|
),
|
|
if (isCurrent) ...[
|
|
const Gap(4),
|
|
Container(
|
|
width: 4,
|
|
height: 4,
|
|
decoration: BoxDecoration(
|
|
color: Theme.of(context).colorScheme.onPrimary,
|
|
shape: BoxShape.circle,
|
|
),
|
|
),
|
|
],
|
|
],
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}),
|
|
),
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
Widget _buildMembershipSection(
|
|
BuildContext context,
|
|
WidgetRef ref,
|
|
AsyncValue<SnWalletSubscription?> stellarSubscriptionAsync,
|
|
) {
|
|
return stellarSubscriptionAsync.when(
|
|
data: (membership) => _buildMembershipContent(context, ref, membership),
|
|
loading:
|
|
() => Container(
|
|
width: double.infinity,
|
|
padding: const EdgeInsets.all(16),
|
|
decoration: BoxDecoration(
|
|
gradient: LinearGradient(
|
|
colors: [
|
|
Theme.of(context).colorScheme.primaryContainer,
|
|
Theme.of(context).colorScheme.secondaryContainer,
|
|
],
|
|
begin: Alignment.topLeft,
|
|
end: Alignment.bottomRight,
|
|
),
|
|
borderRadius: BorderRadius.circular(12),
|
|
),
|
|
child: const Center(child: CircularProgressIndicator()),
|
|
),
|
|
error:
|
|
(error, stack) => Container(
|
|
width: double.infinity,
|
|
padding: const EdgeInsets.all(16),
|
|
decoration: BoxDecoration(
|
|
gradient: LinearGradient(
|
|
colors: [
|
|
Theme.of(context).colorScheme.primaryContainer,
|
|
Theme.of(context).colorScheme.secondaryContainer,
|
|
],
|
|
begin: Alignment.topLeft,
|
|
end: Alignment.bottomRight,
|
|
),
|
|
borderRadius: BorderRadius.circular(12),
|
|
),
|
|
child: Text('Error loading membership: $error'),
|
|
),
|
|
);
|
|
}
|
|
|
|
Widget _buildMembershipContent(
|
|
BuildContext context,
|
|
WidgetRef ref,
|
|
SnWalletSubscription? membership,
|
|
) {
|
|
final isActive = membership?.isActive ?? false;
|
|
|
|
return Container(
|
|
width: double.infinity,
|
|
padding: const EdgeInsets.all(16),
|
|
decoration: BoxDecoration(
|
|
gradient: LinearGradient(
|
|
colors: [
|
|
Theme.of(context).colorScheme.primaryContainer,
|
|
Theme.of(context).colorScheme.secondaryContainer,
|
|
],
|
|
begin: Alignment.topLeft,
|
|
end: Alignment.bottomRight,
|
|
),
|
|
borderRadius: BorderRadius.circular(12),
|
|
),
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Row(
|
|
children: [
|
|
Icon(
|
|
isActive ? Icons.star : Icons.star_border,
|
|
color: Theme.of(context).colorScheme.primary,
|
|
size: 24,
|
|
),
|
|
const Gap(8),
|
|
Text(
|
|
'stellarMembership'.tr(),
|
|
style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold),
|
|
),
|
|
],
|
|
),
|
|
const Gap(12),
|
|
|
|
if (isActive) ...[
|
|
_buildCurrentMembershipCard(context, membership!),
|
|
const Gap(16),
|
|
],
|
|
|
|
Text(
|
|
isActive ? 'upgradeYourPlan'.tr() : 'chooseYourPlan'.tr(),
|
|
style: TextStyle(fontSize: 16, fontWeight: FontWeight.w600),
|
|
),
|
|
const Gap(12),
|
|
|
|
_buildMembershipTiers(context, ref, membership),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
|
|
Widget _buildCurrentMembershipCard(
|
|
BuildContext context,
|
|
SnWalletSubscription membership,
|
|
) {
|
|
final tierName = _getMembershipTierName(membership.identifier);
|
|
final tierColor = _getMembershipTierColor(context, membership.identifier);
|
|
|
|
return Container(
|
|
padding: const EdgeInsets.all(12),
|
|
decoration: BoxDecoration(
|
|
color: tierColor.withOpacity(0.1),
|
|
borderRadius: BorderRadius.circular(8),
|
|
border: Border.all(color: tierColor, width: 1),
|
|
),
|
|
child: Row(
|
|
children: [
|
|
Icon(Icons.verified, color: tierColor, size: 20),
|
|
const Gap(8),
|
|
Expanded(
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Text(
|
|
'currentMembership'.tr(args: [tierName]),
|
|
style: TextStyle(
|
|
fontSize: 14,
|
|
fontWeight: FontWeight.w600,
|
|
color: tierColor,
|
|
),
|
|
),
|
|
if (membership.endedAt != null)
|
|
Text(
|
|
'membershipExpires'.tr(
|
|
args: [membership.endedAt!.formatSystem()],
|
|
),
|
|
style: Theme.of(context).textTheme.bodySmall?.copyWith(
|
|
color: Theme.of(context).colorScheme.onSurfaceVariant,
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
|
|
Widget _buildMembershipTiers(
|
|
BuildContext context,
|
|
WidgetRef ref,
|
|
SnWalletSubscription? currentMembership,
|
|
) {
|
|
final tiers = [
|
|
{
|
|
'id': 'solian.stellar.primary',
|
|
'name': 'membershipTierStellar'.tr(),
|
|
'price': 'membershipPriceStellar'.tr(),
|
|
'features': [
|
|
'membershipFeatureBasic'.tr(),
|
|
'membershipFeaturePrioritySupport'.tr(),
|
|
'membershipFeatureAdFree'.tr(),
|
|
],
|
|
'color': Colors.blue,
|
|
},
|
|
{
|
|
'id': 'solian.stellar.nova',
|
|
'name': 'membershipTierNova'.tr(),
|
|
'price': 'membershipPriceNova'.tr(),
|
|
'features': [
|
|
'membershipFeatureAllPrimary'.tr(),
|
|
'membershipFeatureAdvancedCustomization'.tr(),
|
|
'membershipFeatureEarlyAccess'.tr(),
|
|
],
|
|
'color': Colors.purple,
|
|
},
|
|
{
|
|
'id': 'solian.stellar.supernova',
|
|
'name': 'membershipTierSupernova'.tr(),
|
|
'price': 'membershipPriceSupernova'.tr(),
|
|
'features': [
|
|
'membershipFeatureAllNova'.tr(),
|
|
'membershipFeatureExclusiveContent'.tr(),
|
|
'membershipFeatureVipSupport'.tr(),
|
|
],
|
|
'color': Colors.orange,
|
|
},
|
|
];
|
|
|
|
return Column(
|
|
children:
|
|
tiers.map((tier) {
|
|
final isCurrentTier = currentMembership?.identifier == tier['id'];
|
|
final tierColor = tier['color'] as Color;
|
|
|
|
return Container(
|
|
margin: const EdgeInsets.only(bottom: 8),
|
|
child: Material(
|
|
color: Colors.transparent,
|
|
child: InkWell(
|
|
onTap:
|
|
isCurrentTier
|
|
? null
|
|
: () => _purchaseMembership(
|
|
context,
|
|
ref,
|
|
tier['id'] as String,
|
|
),
|
|
borderRadius: BorderRadius.circular(8),
|
|
child: Container(
|
|
padding: const EdgeInsets.all(12),
|
|
decoration: BoxDecoration(
|
|
color:
|
|
isCurrentTier
|
|
? tierColor.withOpacity(0.1)
|
|
: Theme.of(context).colorScheme.surface,
|
|
borderRadius: BorderRadius.circular(8),
|
|
border: Border.all(
|
|
color:
|
|
isCurrentTier
|
|
? tierColor
|
|
: Theme.of(
|
|
context,
|
|
).colorScheme.outline.withOpacity(0.2),
|
|
width: isCurrentTier ? 2 : 1,
|
|
),
|
|
),
|
|
child: Row(
|
|
children: [
|
|
Container(
|
|
width: 4,
|
|
height: 40,
|
|
decoration: BoxDecoration(
|
|
color: tierColor,
|
|
borderRadius: BorderRadius.circular(2),
|
|
),
|
|
),
|
|
const Gap(12),
|
|
Expanded(
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Row(
|
|
children: [
|
|
Text(
|
|
tier['name'] as String,
|
|
style: TextStyle(
|
|
fontSize: 16,
|
|
fontWeight: FontWeight.w600,
|
|
color: isCurrentTier ? tierColor : null,
|
|
),
|
|
),
|
|
const Gap(8),
|
|
if (isCurrentTier)
|
|
Container(
|
|
padding: const EdgeInsets.symmetric(
|
|
horizontal: 6,
|
|
vertical: 2,
|
|
),
|
|
decoration: BoxDecoration(
|
|
color: tierColor,
|
|
borderRadius: BorderRadius.circular(4),
|
|
),
|
|
child: Text(
|
|
'membershipCurrentBadge'.tr(),
|
|
style: TextStyle(
|
|
fontSize: 10,
|
|
fontWeight: FontWeight.bold,
|
|
color: Colors.white,
|
|
),
|
|
),
|
|
),
|
|
],
|
|
),
|
|
Text(
|
|
tier['price'] as String,
|
|
style: Theme.of(
|
|
context,
|
|
).textTheme.bodyMedium?.copyWith(
|
|
color:
|
|
Theme.of(
|
|
context,
|
|
).colorScheme.onSurfaceVariant,
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
if (!isCurrentTier)
|
|
Icon(
|
|
Icons.arrow_forward_ios,
|
|
size: 16,
|
|
color:
|
|
Theme.of(context).colorScheme.onSurfaceVariant,
|
|
),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}).toList(),
|
|
);
|
|
}
|
|
|
|
String _getMembershipTierName(String identifier) {
|
|
switch (identifier) {
|
|
case 'solian.stellar.primary':
|
|
return 'membershipTierStellar'.tr();
|
|
case 'solian.stellar.nova':
|
|
return 'membershipTierNova'.tr();
|
|
case 'solian.stellar.supernova':
|
|
return 'membershipTierSupernova'.tr();
|
|
default:
|
|
return 'membershipTierUnknown'.tr();
|
|
}
|
|
}
|
|
|
|
Color _getMembershipTierColor(BuildContext context, String identifier) {
|
|
switch (identifier) {
|
|
case 'solian.stellar.primary':
|
|
return Colors.blue;
|
|
case 'solian.stellar.nova':
|
|
return Colors.purple;
|
|
case 'solian.stellar.supernova':
|
|
return Colors.orange;
|
|
default:
|
|
return Theme.of(context).colorScheme.primary;
|
|
}
|
|
}
|
|
|
|
Future<void> _purchaseMembership(
|
|
BuildContext context,
|
|
WidgetRef ref,
|
|
String tierId,
|
|
) async {
|
|
final client = ref.watch(apiClientProvider);
|
|
try {
|
|
showLoadingModal(context);
|
|
final resp = await client.post(
|
|
'/subscriptions',
|
|
data: {
|
|
'identifier': tierId,
|
|
'payment_method': 'solian.wallet',
|
|
'payment_details': {'currency': 'golds'},
|
|
'cycle_duration_days': 30,
|
|
},
|
|
options: Options(headers: {'X-Noop': true}),
|
|
);
|
|
final subscription = SnWalletSubscription.fromJson(resp.data);
|
|
if (subscription.status == 1) return;
|
|
final orderResp = await client.post(
|
|
'/subscriptions/${subscription.identifier}/order',
|
|
);
|
|
final order = SnWalletOrder.fromJson(orderResp.data);
|
|
|
|
if (context.mounted) hideLoadingModal(context);
|
|
|
|
// Show payment overlay to complete the payment
|
|
if (!context.mounted) return;
|
|
final paidOrder = await PaymentOverlay.show(
|
|
context: context,
|
|
order: order,
|
|
enableBiometric: true,
|
|
);
|
|
|
|
if (context.mounted) showLoadingModal(context);
|
|
|
|
if (paidOrder != null) {
|
|
await client.post(
|
|
'/subscriptions/order/handle',
|
|
data: {'order_id': paidOrder.id},
|
|
);
|
|
|
|
ref.invalidate(accountStellarSubscriptionProvider);
|
|
ref.read(userInfoProvider.notifier).fetchUser();
|
|
if (context.mounted) {
|
|
showSnackBar(context, 'membershipPurchaseSuccess'.tr());
|
|
}
|
|
}
|
|
} catch (err) {
|
|
showErrorAlert(err);
|
|
} finally {
|
|
if (context.mounted) hideLoadingModal(context);
|
|
}
|
|
}
|
|
}
|
|
|
|
class LevelStairsPainter extends CustomPainter {
|
|
final int currentLevel;
|
|
final int totalLevels;
|
|
final Color primaryColor;
|
|
final Color surfaceColor;
|
|
final Color onSurfaceColor;
|
|
final double stairHeight;
|
|
final double stairWidth;
|
|
|
|
LevelStairsPainter({
|
|
required this.currentLevel,
|
|
required this.totalLevels,
|
|
required this.primaryColor,
|
|
required this.surfaceColor,
|
|
required this.onSurfaceColor,
|
|
required this.stairHeight,
|
|
required this.stairWidth,
|
|
});
|
|
|
|
@override
|
|
void paint(Canvas canvas, Size size) {
|
|
final paint =
|
|
Paint()
|
|
..color = surfaceColor.withOpacity(0.2)
|
|
..strokeWidth = 1.5
|
|
..style = PaintingStyle.stroke;
|
|
|
|
// Draw connecting lines between stairs
|
|
for (int i = 0; i < totalLevels - 1; i++) {
|
|
final startX = 20.0 + (i * (stairWidth + 8)) + stairWidth;
|
|
final startHeight =
|
|
40.0 + (i * 15.0); // Progressive height for current stair
|
|
final startY = size.height - (20.0 + startHeight);
|
|
|
|
final endX = 20.0 + ((i + 1) * (stairWidth + 8));
|
|
final endHeight =
|
|
40.0 + ((i + 1) * 15.0); // Progressive height for next stair
|
|
final endY = size.height - (20.0 + endHeight);
|
|
|
|
canvas.drawLine(Offset(startX, startY), Offset(endX, endY), paint);
|
|
}
|
|
}
|
|
|
|
@override
|
|
bool shouldRepaint(covariant CustomPainter oldDelegate) => false;
|
|
}
|