Update: 2026-06-10 02:44:54

This commit is contained in:
Hamza-Ayed
2026-06-10 02:44:55 +03:00
parent 9bc7a31c94
commit a0473a8b0f
134 changed files with 1706 additions and 544 deletions

View File

@@ -7,9 +7,11 @@ import '../../../constant/colors.dart';
import '../../../controller/auth/captin/invit_controller.dart';
import '../../../controller/functions/encrypt_decrypt.dart';
import '../../../main.dart';
import '../../../controller/home/invites_rewards_controller.dart';
class InviteScreen extends StatelessWidget {
final InviteController controller = Get.put(InviteController());
final InvitesRewardsController rewardsController = Get.put(InvitesRewardsController());
@override
Widget build(BuildContext context) {
@@ -111,6 +113,8 @@ class InviteScreen extends StatelessWidget {
_buildActionButtons(),
const SizedBox(height: 20),
_buildInvitationsList(context),
const SizedBox(height: 20),
_buildUnifiedRewardsList(isDriver: true),
],
);
}
@@ -146,6 +150,8 @@ class InviteScreen extends StatelessWidget {
const SizedBox(height: 20),
const SizedBox(height: 20),
_buildInvitationsListPassengers(context),
const SizedBox(height: 20),
_buildUnifiedRewardsList(isDriver: false),
],
);
}
@@ -324,9 +330,12 @@ class InviteScreen extends StatelessWidget {
}
Widget _buildInvitationsList(BuildContext context) {
return SizedBox(
height: Get.height * .4,
child: controller.driverInvitationData.isEmpty
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text("Invitations Sent".tr, style: const TextStyle(fontSize: 17, fontWeight: FontWeight.bold)),
const SizedBox(height: 10),
controller.driverInvitationData.isEmpty
? Center(
child: Text(
"No invitation found yet!".tr,
@@ -337,18 +346,24 @@ class InviteScreen extends StatelessWidget {
),
)
: ListView.builder(
shrinkWrap: true,
physics: const NeverScrollableScrollPhysics(),
itemCount: controller.driverInvitationData.length,
itemBuilder: (context, index) {
return _buildInvitationItem(context, index);
},
),
],
);
}
Widget _buildInvitationsListPassengers(BuildContext context) {
return SizedBox(
height: Get.height * .4,
child: controller.driverInvitationDataToPassengers.isEmpty
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text("Invitations Sent".tr, style: const TextStyle(fontSize: 17, fontWeight: FontWeight.bold)),
const SizedBox(height: 10),
controller.driverInvitationDataToPassengers.isEmpty
? Center(
child: Text(
"No invitation found yet!".tr,
@@ -359,11 +374,14 @@ class InviteScreen extends StatelessWidget {
),
)
: ListView.builder(
shrinkWrap: true,
physics: const NeverScrollableScrollPhysics(),
itemCount: controller.driverInvitationDataToPassengers.length,
itemBuilder: (context, index) {
return _buildInvitationItemPassengers(context, index);
},
),
],
);
}
@@ -636,4 +654,76 @@ class InviteScreen extends StatelessWidget {
),
);
}
Widget _buildUnifiedRewardsList({required bool isDriver}) {
return GetBuilder<InvitesRewardsController>(
builder: (rewardsController) {
if (rewardsController.isLoading) {
return const Center(child: CupertinoActivityIndicator());
}
var filteredList = rewardsController.referrals.where((ref) =>
isDriver ? ref['invited_user_type'] == 'driver' : ref['invited_user_type'] == 'passenger'
).toList();
if (filteredList.isEmpty) {
return const SizedBox(); // Hide if no valid rewards
}
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text("Reward Status".tr, style: const TextStyle(fontSize: 17, fontWeight: FontWeight.bold)),
const SizedBox(height: 10),
ListView.builder(
shrinkWrap: true,
physics: const NeverScrollableScrollPhysics(),
itemCount: filteredList.length,
itemBuilder: (context, index) {
var ref = filteredList[index];
int trips = ref['trip_count'] ?? 0;
int target = ref['target_trips'] ?? 1;
bool canClaim = ref['can_claim'] ?? false;
bool isClaimed = ref['is_reward_claimed'] == 1;
return Container(
margin: const EdgeInsets.only(bottom: 12),
padding: const EdgeInsets.all(16),
decoration: BoxDecoration(
color: CupertinoColors.systemGrey6,
borderRadius: BorderRadius.circular(12),
),
child: Row(
children: [
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(isDriver ? "Driver Referral".tr : "Passenger Referral".tr, style: const TextStyle(fontWeight: FontWeight.bold)),
const SizedBox(height: 4),
Text("Trips: $trips / $target".tr, style: const TextStyle(color: CupertinoColors.secondaryLabel, fontSize: 13)),
if (isClaimed)
Text("Reward Claimed".tr, style: const TextStyle(color: CupertinoColors.activeGreen, fontSize: 12, fontWeight: FontWeight.bold))
else if (!canClaim)
Text("Waiting for trips".tr, style: const TextStyle(color: CupertinoColors.systemOrange, fontSize: 12))
],
),
),
if (canClaim && !isClaimed)
CupertinoButton(
color: AppColor.blueColor,
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
onPressed: () => rewardsController.claimUnifiedReward(ref['id']),
child: Text("Claim".tr, style: const TextStyle(color: Colors.white, fontSize: 14)),
)
],
),
);
},
),
],
);
},
);
}
}

View File

@@ -1,5 +1,6 @@
import 'package:flutter/material.dart';
import 'package:get/get.dart';
import 'package:qr_flutter/qr_flutter.dart';
import 'package:siro_driver/views/widgets/error_snakbar.dart';
import 'package:siro_driver/views/widgets/my_textField.dart';
import 'package:siro_driver/views/widgets/elevated_btn.dart'; // Checked import
@@ -7,6 +8,7 @@ import 'package:flutter_font_icons/flutter_font_icons.dart';
import '../../../../constant/box_name.dart';
import '../../../../constant/colors.dart';
import '../../../../constant/links.dart';
import '../../../../constant/style.dart';
import '../../../../controller/firebase/notification_service.dart';
import '../../../../controller/functions/launch.dart';
@@ -50,6 +52,16 @@ class SosConnect extends StatelessWidget {
isPulsing: true,
onTap: () => _handleSosCall(controller),
),
const SizedBox(height: 8),
// === Quick Invite Button ===
_buildModernActionButton(
icon: Icons.qr_code_rounded,
color: Colors.white,
bgColor: AppColor.blueColor,
tooltip: 'Quick Invite',
isPulsing: false,
onTap: () => _showQuickInviteDialog(controller),
),
],
),
);
@@ -128,4 +140,70 @@ class SosConnect extends StatelessWidget {
launchCommunication('phone', box.read(BoxName.sosPhoneDriver), '');
}
}
void _showQuickInviteDialog(MapDriverController controller) {
// In a real scenario, this code would be fetched from the backend (ReferralController)
// For now we will use a generated code or driverId. We should use the one from ReferralController
// But since we are accessing it globally, we can just use the driverID + 123 for now until it's linked
String driverId = box.read(BoxName.driverID).toString();
String inviteCode = "SR$driverId"; // Placeholder code
String deepLink = "https://${AppLink.appDomain}/invite?ref=$inviteCode";
Get.defaultDialog(
title: "Quick Invite".tr,
titleStyle: AppStyle.title.copyWith(fontWeight: FontWeight.bold),
content: Column(
children: [
Text(
"Let the passenger scan this code to sign up".tr,
textAlign: TextAlign.center,
style: const TextStyle(fontSize: 14, color: Colors.grey),
),
const SizedBox(height: 20),
Container(
padding: const EdgeInsets.all(16),
decoration: BoxDecoration(
color: Colors.white,
borderRadius: BorderRadius.circular(16),
boxShadow: [
BoxShadow(
color: Colors.black.withOpacity(0.05),
blurRadius: 10,
offset: const Offset(0, 4),
),
],
),
child: QrImageView(
data: deepLink,
version: QrVersions.auto,
size: 200.0,
backgroundColor: Colors.white,
),
),
const SizedBox(height: 20),
Container(
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 12),
decoration: BoxDecoration(
color: AppColor.blueColor.withOpacity(0.1),
borderRadius: BorderRadius.circular(12),
border: Border.all(color: AppColor.blueColor.withOpacity(0.3)),
),
child: Text(
inviteCode,
style: TextStyle(
fontSize: 24,
fontWeight: FontWeight.w900,
color: AppColor.blueColor,
letterSpacing: 4,
),
),
),
],
),
confirm: MyElevatedButton(
title: 'Done'.tr,
onPressed: () => Get.back(),
),
);
}
}