Update: 2026-05-08 14:11:53

This commit is contained in:
Hamza-Ayed
2026-05-08 14:11:53 +03:00
parent 155c2d0fc0
commit be0571648a
3 changed files with 98 additions and 27 deletions

View File

@@ -306,7 +306,6 @@ class CompaniesManagementView extends StatelessWidget {
void _showLinkJoFotaraDialog(BuildContext context, Map<String, dynamic> company, CompaniesManagementController controller) {
final clientIdC = TextEditingController();
final secretKeyC = TextEditingController();
final sequenceC = TextEditingController();
Get.dialog(
AlertDialog(
@@ -319,7 +318,6 @@ class CompaniesManagementView extends StatelessWidget {
const SizedBox(height: 16),
_editField('Client ID', clientIdC, Icons.vpn_key),
_editField('Secret Key', secretKeyC, Icons.lock),
_editField('Income Source Sequence (اختياري)', sequenceC, Icons.format_list_numbered),
],
),
),
@@ -332,7 +330,7 @@ class CompaniesManagementView extends StatelessWidget {
return;
}
Get.back();
controller.connectJoFotara(company['id'], clientIdC.text, secretKeyC.text, sequenceC.text);
controller.connectJoFotara(company['id'], clientIdC.text, secretKeyC.text, '');
},
style: ElevatedButton.styleFrom(backgroundColor: const Color(0xFF6366F1)),
child: const Text('ربط الآن', style: TextStyle(color: Colors.white)),

View File

@@ -391,15 +391,12 @@ class DashboardView extends GetView<DashboardController> {
}
Widget _buildGamificationCard(Map gamification, bool isDark) {
final points = gamification['points'] ?? 0;
final points = gamification['total_points'] ?? 0;
final level = gamification['level'] ?? 1;
final levelName = gamification['level_name'] ?? 'مبتدئ';
final currentLevelThreshold = gamification['current_level_threshold'] ?? 0;
final nextLevelThreshold = gamification['next_level_threshold'] ?? 1000;
final progress = (points - currentLevelThreshold) /
((nextLevelThreshold - currentLevelThreshold) > 0
? (nextLevelThreshold - currentLevelThreshold)
: 1);
final progressPercent = gamification['progress_percent'] ?? 0;
final badgesCount = gamification['badges_count'] ?? 0;
final availableBadges = gamification['available_badges'] ?? 9;
return Container(
padding: const EdgeInsets.all(16),
@@ -447,15 +444,27 @@ class DashboardView extends GetView<DashboardController> {
],
),
),
Container(
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 6),
decoration: BoxDecoration(
color: Colors.white24,
borderRadius: BorderRadius.circular(20),
),
child: const Text(
'المكافآت',
style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold),
InkWell(
onTap: () => _showBadgesDialog(gamification),
child: Container(
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 6),
decoration: BoxDecoration(
color: Colors.white24,
borderRadius: BorderRadius.circular(20),
),
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
const Text(
'المكافآت ',
style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold),
),
Text(
'$badgesCount/$availableBadges',
style: const TextStyle(color: Colors.white70, fontSize: 12),
),
],
),
),
),
],
@@ -464,7 +473,7 @@ class DashboardView extends GetView<DashboardController> {
ClipRRect(
borderRadius: BorderRadius.circular(6),
child: LinearProgressIndicator(
value: progress.clamp(0.0, 1.0).toDouble(),
value: progressPercent / 100.0,
backgroundColor: Colors.white24,
color: Colors.amber,
minHeight: 8,
@@ -472,7 +481,7 @@ class DashboardView extends GetView<DashboardController> {
),
const SizedBox(height: 8),
Text(
'باقي ${nextLevelThreshold - points} نقطة للمستوى القادم',
'باقي ${100 - progressPercent} نقطة للمستوى القادم',
style: const TextStyle(color: Colors.white70, fontSize: 12),
),
],
@@ -480,6 +489,36 @@ class DashboardView extends GetView<DashboardController> {
);
}
void _showBadgesDialog(Map gamification) {
final badges = gamification['badges'] as List? ?? [];
Get.dialog(
AlertDialog(
title: const Text('شاراتك ومكافآتك', textAlign: TextAlign.center, style: TextStyle(color: Color(0xFF0F4C81))),
content: SizedBox(
width: double.maxFinite,
child: badges.isEmpty
? const Text('لم تحصل على أي شارات بعد. قم برفع الفواتير لتبدأ!', textAlign: TextAlign.center)
: ListView.builder(
shrinkWrap: true,
itemCount: badges.length,
itemBuilder: (context, index) {
final b = badges[index];
return ListTile(
leading: Text(b['badge_icon'] ?? '🌟', style: const TextStyle(fontSize: 24)),
title: Text(b['badge_name'] ?? '', style: const TextStyle(fontWeight: FontWeight.bold)),
subtitle: Text('تم الحصول عليها: ${(b['earned_at'] ?? '').toString().split(' ')[0]}'),
);
},
),
),
actions: [
TextButton(onPressed: () => Get.back(), child: const Text('إغلاق')),
],
),
);
}
Widget _buildQuotaMeter(Map subscription, bool isDark) {
int limit = subscription['limit'] ?? 100;
int used = subscription['used'] ?? 0;

View File

@@ -1572,7 +1572,7 @@
</template>
</ul>
<button x-show="subscription?.plan_id !== p.id" class="btn-primary" style="margin-top:auto;"
@click="alert('يرجى التواصل مع الدعم الفني لترقية باقتك إلى ' + p.name_ar)">
@click="upgradePlan(p)">
ترقية الباقة الآن
</button>
</div>
@@ -1868,11 +1868,7 @@
<input type="password" x-model="connectData.secret_key" placeholder="أدخل Secret Key"
class="form-input mono" required>
</div>
<div class="form-group">
<label class="form-label">تسلسل مصدر الدخل <span class="form-label-sub">(مثال: 1)</span></label>
<input type="text" x-model="connectData.income_source_sequence" placeholder="1"
class="form-input mono">
</div>
</div>
<div class="modal-divider"></div>
<div class="modal-footer">
@@ -2250,6 +2246,42 @@
</div>
</div>
<!-- ── UPLOAD PAYMENT RECEIPT MODAL ─────────────────────────── -->
<div x-show="showPaymentModal" x-cloak class="modal-backdrop">
<div class="modal-box">
<div class="modal-head">
<div class="modal-head-icon gold">💳</div>
<div style="flex:1;">
<div class="modal-title">تأكيد دفع الاشتراك</div>
<div class="modal-subtitle">يرجى تحويل المبلغ ثم رفع وصل الدفع</div>
</div>
<button @click="showPaymentModal = false" class="modal-close-btn"></button>
</div>
<div class="modal-divider"></div>
<div class="modal-body" style="display:flex; flex-direction:column; gap:14px;">
<div style="background:var(--teal-subtle); border:1px solid rgba(4,120,87,0.2); border-radius:10px; padding:12px; text-align:center;">
<h4 style="color:var(--navy); font-weight:bold; margin-bottom:8px;">تفاصيل الدفع (CliQ)</h4>
<p style="margin-bottom:4px; font-size:14px;">يرجى تحويل مبلغ <strong style="color:var(--teal);" x-text="paymentData.amount + ' دينار'"></strong></p>
<p style="margin-bottom:8px; font-size:14px;">إلى حساب CliQ التالي:</p>
<div style="background:white; padding:8px; border-radius:6px; font-family:monospace; font-size:18px; font-weight:bold; color:var(--navy); display:inline-block; letter-spacing:2px;" x-text="paymentData.cliq_alias"></div>
</div>
<div class="form-group" style="margin-top:10px;">
<label class="form-label">صورة وصل التحويل</label>
<input type="file" @change="selectedFile = $event.target.files[0]" class="form-input" style="padding:8px;" accept="image/*,application/pdf" required>
<p style="font-size:12px; color:var(--text-3); margin-top:6px;">يرجى التأكد من وضوح رقم الحوالة والتاريخ.</p>
</div>
</div>
<div class="modal-divider"></div>
<div class="modal-footer">
<button @click="uploadReceipt" class="btn btn-navy" :disabled="isBusy" style="flex:1;">
<span x-show="!isBusy">📤 إرسال الوصل للتدقيق</span>
<span x-show="isBusy"> جاري الإرسال...</span>
</button>
<button type="button" @click="showPaymentModal = false" class="btn btn-ghost">إلغاء</button>
</div>
</div>
</div>
<!-- ════════════════════════════════════════════════════════
ALPINE.JS LOGIC UNCHANGED
@@ -2260,6 +2292,8 @@
user: JSON.parse(localStorage.getItem('user')),
page: 'dashboard',
users: [], companies: [], invoices: [], tenants: [], subscription: null, plans: [],
showPaymentModal: false,
paymentData: { cliq_alias: '', amount: 0, plan_name: '', request_id: '' },
stats: { total: 0, pending: 0, approved: 0 },
showAddUserModal: false, showAddCompanyModal: false, showConnectModal: false,