django-cfg 1.2.29__py3-none-any.whl → 1.2.31__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (126) hide show
  1. django_cfg/__init__.py +1 -1
  2. django_cfg/apps/payments/admin/__init__.py +3 -2
  3. django_cfg/apps/payments/admin/balance_admin.py +18 -18
  4. django_cfg/apps/payments/admin/currencies_admin.py +319 -131
  5. django_cfg/apps/payments/admin/payments_admin.py +15 -4
  6. django_cfg/apps/payments/config/module.py +2 -2
  7. django_cfg/apps/payments/config/utils.py +2 -2
  8. django_cfg/apps/payments/decorators.py +2 -2
  9. django_cfg/apps/payments/management/commands/README.md +95 -127
  10. django_cfg/apps/payments/management/commands/currency_stats.py +5 -24
  11. django_cfg/apps/payments/management/commands/manage_currencies.py +229 -0
  12. django_cfg/apps/payments/management/commands/manage_providers.py +235 -0
  13. django_cfg/apps/payments/managers/__init__.py +3 -2
  14. django_cfg/apps/payments/managers/balance_manager.py +2 -2
  15. django_cfg/apps/payments/managers/currency_manager.py +272 -49
  16. django_cfg/apps/payments/managers/payment_manager.py +161 -13
  17. django_cfg/apps/payments/middleware/api_access.py +2 -2
  18. django_cfg/apps/payments/middleware/rate_limiting.py +8 -18
  19. django_cfg/apps/payments/middleware/usage_tracking.py +20 -17
  20. django_cfg/apps/payments/migrations/0002_network_providercurrency_and_more.py +241 -0
  21. django_cfg/apps/payments/migrations/0003_add_usd_rate_cache.py +30 -0
  22. django_cfg/apps/payments/models/__init__.py +3 -2
  23. django_cfg/apps/payments/models/currencies.py +187 -71
  24. django_cfg/apps/payments/models/payments.py +3 -2
  25. django_cfg/apps/payments/serializers/__init__.py +3 -2
  26. django_cfg/apps/payments/serializers/currencies.py +20 -12
  27. django_cfg/apps/payments/services/cache/simple_cache.py +2 -2
  28. django_cfg/apps/payments/services/core/balance_service.py +2 -2
  29. django_cfg/apps/payments/services/core/fallback_service.py +2 -2
  30. django_cfg/apps/payments/services/core/payment_service.py +3 -6
  31. django_cfg/apps/payments/services/core/subscription_service.py +4 -7
  32. django_cfg/apps/payments/services/internal_types.py +171 -7
  33. django_cfg/apps/payments/services/monitoring/api_schemas.py +58 -204
  34. django_cfg/apps/payments/services/monitoring/provider_health.py +2 -2
  35. django_cfg/apps/payments/services/providers/base.py +144 -43
  36. django_cfg/apps/payments/services/providers/cryptapi/__init__.py +4 -0
  37. django_cfg/apps/payments/services/providers/cryptapi/config.py +8 -0
  38. django_cfg/apps/payments/services/providers/cryptapi/models.py +192 -0
  39. django_cfg/apps/payments/services/providers/cryptapi/provider.py +439 -0
  40. django_cfg/apps/payments/services/providers/cryptomus/__init__.py +4 -0
  41. django_cfg/apps/payments/services/providers/cryptomus/models.py +176 -0
  42. django_cfg/apps/payments/services/providers/cryptomus/provider.py +429 -0
  43. django_cfg/apps/payments/services/providers/cryptomus/provider_v2.py +564 -0
  44. django_cfg/apps/payments/services/providers/models/__init__.py +34 -0
  45. django_cfg/apps/payments/services/providers/models/currencies.py +190 -0
  46. django_cfg/apps/payments/services/providers/nowpayments/__init__.py +4 -0
  47. django_cfg/apps/payments/services/providers/nowpayments/models.py +196 -0
  48. django_cfg/apps/payments/services/providers/nowpayments/provider.py +380 -0
  49. django_cfg/apps/payments/services/providers/registry.py +294 -11
  50. django_cfg/apps/payments/services/providers/stripe/__init__.py +4 -0
  51. django_cfg/apps/payments/services/providers/stripe/models.py +184 -0
  52. django_cfg/apps/payments/services/providers/stripe/provider.py +109 -0
  53. django_cfg/apps/payments/services/security/error_handler.py +6 -8
  54. django_cfg/apps/payments/services/security/payment_notifications.py +2 -2
  55. django_cfg/apps/payments/services/security/webhook_validator.py +3 -4
  56. django_cfg/apps/payments/signals/api_key_signals.py +2 -2
  57. django_cfg/apps/payments/signals/payment_signals.py +11 -5
  58. django_cfg/apps/payments/signals/subscription_signals.py +2 -2
  59. django_cfg/apps/payments/tasks/webhook_processing.py +2 -2
  60. django_cfg/apps/payments/templates/admin/payments/currency/change_list.html +50 -0
  61. django_cfg/apps/payments/templates/payments/base.html +4 -4
  62. django_cfg/apps/payments/templates/payments/components/payment_card.html +6 -6
  63. django_cfg/apps/payments/templates/payments/components/payment_qr_code.html +4 -4
  64. django_cfg/apps/payments/templates/payments/components/progress_bar.html +14 -7
  65. django_cfg/apps/payments/templates/payments/components/provider_stats.html +2 -2
  66. django_cfg/apps/payments/templates/payments/components/status_badge.html +8 -1
  67. django_cfg/apps/payments/templates/payments/components/status_overview.html +34 -30
  68. django_cfg/apps/payments/templates/payments/dashboard.html +202 -290
  69. django_cfg/apps/payments/templates/payments/dashboard_simple_test.html +35 -0
  70. django_cfg/apps/payments/templates/payments/payment_create.html +579 -0
  71. django_cfg/apps/payments/templates/payments/payment_detail.html +373 -0
  72. django_cfg/apps/payments/templates/payments/payment_list.html +354 -0
  73. django_cfg/apps/payments/templates/payments/stats.html +261 -0
  74. django_cfg/apps/payments/templates/payments/test.html +213 -0
  75. django_cfg/apps/payments/urls.py +3 -1
  76. django_cfg/apps/payments/{urls_templates.py → urls_admin.py} +6 -0
  77. django_cfg/apps/payments/utils/__init__.py +1 -3
  78. django_cfg/apps/payments/utils/billing_utils.py +2 -2
  79. django_cfg/apps/payments/utils/config_utils.py +2 -8
  80. django_cfg/apps/payments/utils/validation_utils.py +2 -2
  81. django_cfg/apps/payments/views/__init__.py +3 -2
  82. django_cfg/apps/payments/views/currency_views.py +31 -20
  83. django_cfg/apps/payments/views/payment_views.py +2 -2
  84. django_cfg/apps/payments/views/templates/ajax.py +141 -2
  85. django_cfg/apps/payments/views/templates/base.py +21 -13
  86. django_cfg/apps/payments/views/templates/payment_detail.py +1 -1
  87. django_cfg/apps/payments/views/templates/payment_management.py +34 -40
  88. django_cfg/apps/payments/views/templates/stats.py +8 -4
  89. django_cfg/apps/payments/views/webhook_views.py +2 -2
  90. django_cfg/apps/payments/viewsets.py +3 -2
  91. django_cfg/apps/tasks/urls.py +0 -2
  92. django_cfg/apps/tasks/urls_admin.py +14 -0
  93. django_cfg/apps/urls.py +4 -4
  94. django_cfg/core/config.py +35 -0
  95. django_cfg/models/payments.py +2 -8
  96. django_cfg/modules/django_currency/__init__.py +16 -11
  97. django_cfg/modules/django_currency/clients/__init__.py +4 -4
  98. django_cfg/modules/django_currency/clients/coinpaprika_client.py +289 -0
  99. django_cfg/modules/django_currency/clients/yahoo_client.py +157 -0
  100. django_cfg/modules/django_currency/core/__init__.py +1 -7
  101. django_cfg/modules/django_currency/core/converter.py +18 -23
  102. django_cfg/modules/django_currency/core/models.py +122 -11
  103. django_cfg/modules/django_currency/database/__init__.py +4 -4
  104. django_cfg/modules/django_currency/database/database_loader.py +190 -309
  105. django_cfg/modules/django_unfold/dashboard.py +7 -2
  106. django_cfg/template_archive/django_sample.zip +0 -0
  107. django_cfg/templates/admin/components/action_grid.html +9 -9
  108. django_cfg/templates/admin/components/metric_card.html +5 -5
  109. django_cfg/templates/admin/components/status_badge.html +2 -2
  110. django_cfg/templates/admin/layouts/dashboard_with_tabs.html +152 -24
  111. django_cfg/templates/admin/snippets/components/quick_actions.html +3 -3
  112. django_cfg/templates/admin/snippets/components/system_health.html +1 -1
  113. django_cfg/templates/admin/snippets/tabs/overview_tab.html +49 -52
  114. {django_cfg-1.2.29.dist-info → django_cfg-1.2.31.dist-info}/METADATA +2 -4
  115. {django_cfg-1.2.29.dist-info → django_cfg-1.2.31.dist-info}/RECORD +118 -96
  116. django_cfg/apps/payments/management/commands/populate_currencies.py +0 -246
  117. django_cfg/apps/payments/management/commands/update_currencies.py +0 -336
  118. django_cfg/apps/payments/services/providers/cryptapi.py +0 -273
  119. django_cfg/apps/payments/services/providers/cryptomus.py +0 -311
  120. django_cfg/apps/payments/services/providers/nowpayments.py +0 -293
  121. django_cfg/apps/payments/services/validators/__init__.py +0 -8
  122. django_cfg/modules/django_currency/clients/coingecko_client.py +0 -257
  123. django_cfg/modules/django_currency/clients/yfinance_client.py +0 -246
  124. {django_cfg-1.2.29.dist-info → django_cfg-1.2.31.dist-info}/WHEEL +0 -0
  125. {django_cfg-1.2.29.dist-info → django_cfg-1.2.31.dist-info}/entry_points.txt +0 -0
  126. {django_cfg-1.2.29.dist-info → django_cfg-1.2.31.dist-info}/licenses/LICENSE +0 -0
@@ -7,178 +7,179 @@
7
7
  {% block header_title %}Payment Dashboard{% endblock %}
8
8
  {% block header_subtitle %}Monitor and manage payment transactions in real-time{% endblock %}
9
9
 
10
- {% block extra_head %}
11
- <!-- Dashboard-specific styles -->
12
- <style>
13
- .dashboard-grid {
14
- display: grid;
15
- grid-template-columns: 1fr;
16
- gap: 1.5rem;
17
- }
18
-
19
- @media (min-width: 1024px) {
20
- .dashboard-grid {
21
- grid-template-columns: 2fr 1fr;
22
- }
23
- }
24
- </style>
25
- {% endblock %}
26
-
27
10
  {% block content %}
28
- <div class="payment-dashboard">
11
+ <div class="space-y-6">
12
+ <!-- Debug Info (Collapsible) -->
13
+ <div class="bg-yellow-50 dark:bg-yellow-900/20 border border-yellow-200 dark:border-yellow-800 rounded-lg mb-6">
14
+ <button onclick="toggleDebug()" class="w-full px-4 py-3 text-left flex items-center justify-between text-yellow-800 dark:text-yellow-200 hover:bg-yellow-100 dark:hover:bg-yellow-900/30 transition-colors duration-200">
15
+ <span class="font-medium">🐛 Debug Information</span>
16
+ <span id="debug-icon" class="material-icons text-sm transform transition-transform duration-200">expand_more</span>
17
+ </button>
18
+ <div id="debug-content" class="hidden px-4 pb-3 text-yellow-700 dark:text-yellow-300 text-sm">
19
+ <div class="space-y-1">
20
+ <p><strong>Payments:</strong> {{ payments|length }}</p>
21
+ <p><strong>Payment Stats:</strong> {{ payment_stats }}</p>
22
+ <p><strong>Provider Stats:</strong> {{ provider_stats|length }}</p>
23
+ <p><strong>Context Keys:</strong> {{ view|default:"no view" }}</p>
24
+ </div>
25
+ </div>
26
+ </div>
27
+
29
28
  <!-- Status Overview -->
30
29
  {% include 'payments/components/status_overview.html' %}
31
30
 
32
31
  <!-- Main Dashboard Grid -->
33
- <div class="dashboard-grid">
34
- <!-- Left Column: Recent Payments -->
35
- <div class="space-y-6">
36
- <!-- Filter Controls -->
37
- <div class="payment-filters">
38
- <div class="flex items-center justify-between mb-4">
39
- <h3 class="text-lg font-medium text-gray-900 dark:text-white">Recent Payments</h3>
40
- <div class="flex items-center space-x-2">
41
- <select class="filter-select" id="status-filter" onchange="filterPayments()">
42
- <option value="">All Statuses</option>
43
- <option value="pending">Pending</option>
44
- <option value="confirming">Confirming</option>
45
- <option value="completed">Completed</option>
46
- <option value="failed">Failed</option>
47
- </select>
48
- <select class="filter-select" id="provider-filter" onchange="filterPayments()">
49
- <option value="">All Providers</option>
50
- <option value="nowpayments">NowPayments</option>
51
- <option value="cryptapi">CryptAPI</option>
52
- <option value="cryptomus">Cryptomus</option>
53
- <option value="stripe">Stripe</option>
54
- </select>
32
+ <div class="grid grid-cols-1 lg:grid-cols-3 gap-6">
33
+ <!-- Recent Payments -->
34
+ <div class="lg:col-span-2">
35
+ <div class="bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm">
36
+ <div class="p-6 border-b border-gray-200 dark:border-gray-700">
37
+ <div class="flex items-center justify-between">
38
+ <h3 class="text-lg font-semibold text-gray-900 dark:text-white">Recent Payments</h3>
39
+ <a href="/admin/django_cfg_payments/universalpayment/"
40
+ class="text-blue-600 dark:text-blue-400 hover:text-blue-700 dark:hover:text-blue-300 text-sm font-medium">
41
+ View All
42
+ </a>
55
43
  </div>
56
44
  </div>
57
45
 
58
- <div class="filter-group">
59
- <input type="text"
60
- class="filter-input"
61
- placeholder="Search by payment ID or amount..."
62
- id="search-input"
63
- onkeyup="searchPayments(event)">
64
- <input type="date"
65
- class="filter-input"
66
- id="date-filter"
67
- onchange="filterPayments()">
68
- <button class="btn btn-outline btn-sm" onclick="clearFilters()">
69
- <span class="material-icons text-sm mr-1">clear</span>
70
- Clear
71
- </button>
72
- </div>
73
- </div>
74
-
75
- <!-- Payment Cards Grid -->
76
- <div id="payments-grid" class="grid grid-cols-1 lg:grid-cols-2 gap-6">
77
- {% if payments %}
78
- {% for payment in payments %}
79
- {% payment_card payment %}
80
- {% endfor %}
81
- {% else %}
82
- <!-- Empty State -->
83
- <div class="col-span-full">
84
- <div class="text-center py-12">
85
- <span class="material-icons text-gray-400 text-6xl mb-4">payment</span>
86
- <h3 class="text-lg font-medium text-gray-900 dark:text-white mb-2">No payments found</h3>
87
- <p class="text-gray-500 dark:text-gray-400 mb-4">Get started by creating your first payment.</p>
88
- <button class="btn btn-primary" onclick="createNewPayment()">
89
- <span class="material-icons text-sm mr-2">add</span>
90
- Create Payment
91
- </button>
92
- </div>
46
+ <div class="p-6">
47
+ <!-- Debug: Payments Section -->
48
+ <div class="bg-blue-50 dark:bg-blue-900/20 border border-blue-200 dark:border-blue-800 text-blue-800 dark:text-blue-200 px-4 py-3 rounded mb-4">
49
+ <strong>Payments Debug:</strong> Count={{ payments|length }}
50
+ {% if payments %}
51
+ <br><strong>First payment:</strong> {{ payments.0.user.email }} - ${{ payments.0.amount_usd }}
52
+ {% endif %}
93
53
  </div>
94
- {% endif %}
95
- </div>
96
-
97
- <!-- Load More Button -->
98
- {% if has_more_payments %}
99
- <div class="text-center">
100
- <button class="btn btn-outline" id="load-more-btn" onclick="loadMorePayments()">
101
- <span class="material-icons text-sm mr-2">expand_more</span>
102
- Load More Payments
103
- </button>
54
+
55
+ {% if payments %}
56
+ <div class="space-y-4">
57
+ {% for payment in payments %}
58
+ <div class="flex items-center justify-between p-4 bg-gray-50 dark:bg-gray-800 rounded-lg border border-gray-200 dark:border-gray-700">
59
+ <div class="flex items-center space-x-4">
60
+ <div class="w-3 h-3 rounded-full
61
+ {% if payment.provider == 'cryptapi' %}bg-orange-500
62
+ {% elif payment.provider == 'cryptomus' %}bg-blue-500
63
+ {% elif payment.provider == 'stripe' %}bg-purple-500
64
+ {% elif payment.provider == 'nowpayments' %}bg-green-500
65
+ {% else %}bg-gray-500{% endif %}">
66
+ </div>
67
+ <div>
68
+ <p class="text-sm font-medium text-gray-900 dark:text-white">
69
+ ${{ payment.amount_usd|floatformat:2 }}
70
+ </p>
71
+ <p class="text-xs text-gray-500 dark:text-gray-400">
72
+ {{ payment.user.email }} • {{ payment.provider|capfirst }}
73
+ </p>
74
+ </div>
75
+ </div>
76
+ <div class="flex items-center space-x-3">
77
+ <span class="px-2 py-1 text-xs rounded-full bg-warning-100 text-warning-800 dark:bg-warning-900 dark:text-warning-200">
78
+ {{ payment.status|capfirst }}
79
+ </span>
80
+ <span class="text-xs text-gray-500 dark:text-gray-400">
81
+ {{ payment.created_at|timesince }} ago
82
+ </span>
83
+ </div>
84
+ </div>
85
+ {% endfor %}
86
+ </div>
87
+ {% else %}
88
+ <div class="text-center py-8">
89
+ <div class="text-gray-400 mb-2">💳</div>
90
+ <p class="text-gray-600 dark:text-gray-400">No payments found</p>
91
+ <a href="/admin/django_cfg_payments/universalpayment/add/"
92
+ class="text-blue-600 dark:text-blue-400 hover:text-blue-700 dark:hover:text-blue-300 text-sm font-medium">
93
+ Create first payment
94
+ </a>
95
+ </div>
96
+ {% endif %}
97
+ </div>
104
98
  </div>
105
- {% endif %}
106
99
  </div>
107
100
 
108
- <!-- Right Column: Analytics & Quick Info -->
101
+ <!-- Sidebar Stats -->
109
102
  <div class="space-y-6">
110
- <!-- Provider Statistics -->
111
- <div class="bg-white dark:bg-gray-800 rounded-xl shadow-sm border border-gray-200 dark:border-gray-700 p-6">
112
- <h3 class="text-lg font-medium text-gray-900 dark:text-white mb-4">Provider Performance</h3>
113
- {% provider_statistics %}
103
+ <!-- Provider Performance -->
104
+ <div class="bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm">
105
+ <div class="p-6 border-b border-gray-200 dark:border-gray-700">
106
+ <h4 class="text-lg font-semibold text-gray-900 dark:text-white">Provider Status</h4>
107
+ </div>
108
+ <div class="p-6">
109
+ {% if provider_stats %}
110
+ <div class="space-y-3">
111
+ {% for provider in provider_stats %}
112
+ <div class="flex items-center justify-between">
113
+ <div class="flex items-center space-x-2">
114
+ <div class="w-2 h-2 rounded-full
115
+ {% if provider.provider == 'cryptapi' %}bg-orange-500
116
+ {% elif provider.provider == 'cryptomus' %}bg-blue-500
117
+ {% elif provider.provider == 'stripe' %}bg-purple-500
118
+ {% elif provider.provider == 'nowpayments' %}bg-green-500
119
+ {% else %}bg-gray-500{% endif %}">
120
+ </div>
121
+ <span class="text-sm text-gray-700 dark:text-gray-300 capitalize">{{ provider.provider }}</span>
122
+ </div>
123
+ <span class="text-xs text-gray-500 dark:text-gray-400">
124
+ {{ provider.count|default:"0" }} payments
125
+ </span>
126
+ </div>
127
+ {% endfor %}
128
+ </div>
129
+ {% else %}
130
+ <p class="text-sm text-gray-500 dark:text-gray-400 text-center">No provider data</p>
131
+ {% endif %}
132
+ </div>
114
133
  </div>
115
134
 
116
- <!-- Recent Activity -->
117
- <div class="bg-white dark:bg-gray-800 rounded-xl shadow-sm border border-gray-200 dark:border-gray-700 p-6">
118
- <h3 class="text-lg font-medium text-gray-900 dark:text-white mb-4">Recent Activity</h3>
119
- <div class="space-y-3">
120
- {% for event in recent_events|slice:":5" %}
121
- <div class="flex items-center space-x-3 p-2 rounded-lg hover:bg-gray-50 dark:hover:bg-gray-700">
122
- <div class="flex-shrink-0">
123
- {% if event.event_type == 'created' %}
124
- <span class="material-icons text-blue-500">add_circle</span>
125
- {% elif event.event_type == 'completed' %}
126
- <span class="material-icons text-green-500">check_circle</span>
127
- {% elif event.event_type == 'failed' %}
128
- <span class="material-icons text-red-500">error</span>
129
- {% else %}
130
- <span class="material-icons text-gray-500">circle</span>
131
- {% endif %}
135
+ <!-- System Status -->
136
+ <div class="bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm">
137
+ <div class="p-6 border-b border-gray-200 dark:border-gray-700">
138
+ <h4 class="text-lg font-semibold text-gray-900 dark:text-white">System Status</h4>
139
+ </div>
140
+ <div class="p-6">
141
+ <div class="space-y-3">
142
+ <div class="flex items-center justify-between">
143
+ <span class="text-sm text-gray-700 dark:text-gray-300">API Status</span>
144
+ <span class="inline-flex items-center px-2 py-1 rounded-full text-xs font-medium bg-green-100 text-green-800 dark:bg-green-900 dark:text-green-200">
145
+ Online
146
+ </span>
132
147
  </div>
133
- <div class="flex-1 min-w-0">
134
- <p class="text-sm font-medium text-gray-900 dark:text-white">
135
- Payment {{ event.event_type }}
136
- </p>
137
- <p class="text-xs text-gray-500 dark:text-gray-400">
138
- #{{ event.payment.internal_payment_id|default:event.payment.id|truncatechars:8 }}
139
- • {{ event.created_at|timesince }} ago
140
- </p>
148
+ <div class="flex items-center justify-between">
149
+ <span class="text-sm text-gray-700 dark:text-gray-300">Webhook Queue</span>
150
+ <span class="inline-flex items-center px-2 py-1 rounded-full text-xs font-medium bg-blue-100 text-blue-800 dark:bg-blue-900 dark:text-blue-200">
151
+ Processing
152
+ </span>
141
153
  </div>
142
- <div class="text-sm font-medium text-gray-900 dark:text-white">
143
- ${{ event.payment.amount_usd|floatformat:2 }}
154
+ <div class="flex items-center justify-between">
155
+ <span class="text-sm text-gray-700 dark:text-gray-300">Database</span>
156
+ <span class="inline-flex items-center px-2 py-1 rounded-full text-xs font-medium bg-green-100 text-green-800 dark:bg-green-900 dark:text-green-200">
157
+ Connected
158
+ </span>
144
159
  </div>
145
160
  </div>
146
- {% empty %}
147
- <p class="text-sm text-gray-500 dark:text-gray-400 text-center py-4">
148
- No recent activity
149
- </p>
150
- {% endfor %}
151
161
  </div>
152
162
  </div>
153
163
 
154
- <!-- System Status -->
155
- <div class="bg-white dark:bg-gray-800 rounded-xl shadow-sm border border-gray-200 dark:border-gray-700 p-6">
156
- <h3 class="text-lg font-medium text-gray-900 dark:text-white mb-4">System Status</h3>
157
- <div class="space-y-3">
158
- <div class="flex items-center justify-between">
159
- <span class="text-sm text-gray-500 dark:text-gray-400">API Status</span>
160
- <span class="inline-flex items-center px-2 py-1 rounded-full text-xs font-medium bg-green-100 text-green-800">
161
- <span class="w-2 h-2 bg-green-500 rounded-full mr-1"></span>
162
- Online
163
- </span>
164
- </div>
165
- <div class="flex items-center justify-between">
166
- <span class="text-sm text-gray-500 dark:text-gray-400">Webhook Queue</span>
167
- <span class="inline-flex items-center px-2 py-1 rounded-full text-xs font-medium bg-green-100 text-green-800">
168
- Processing
169
- </span>
170
- </div>
171
- <div class="flex items-center justify-between">
172
- <span class="text-sm text-gray-500 dark:text-gray-400">Database</span>
173
- <span class="inline-flex items-center px-2 py-1 rounded-full text-xs font-medium bg-green-100 text-green-800">
174
- Connected
175
- </span>
176
- </div>
177
- <div class="flex items-center justify-between">
178
- <span class="text-sm text-gray-500 dark:text-gray-400">Redis Cache</span>
179
- <span class="inline-flex items-center px-2 py-1 rounded-full text-xs font-medium bg-green-100 text-green-800">
180
- Active
181
- </span>
164
+ <!-- Quick Actions -->
165
+ <div class="bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm">
166
+ <div class="p-6 border-b border-gray-200 dark:border-gray-700">
167
+ <h4 class="text-lg font-semibold text-gray-900 dark:text-white">Quick Actions</h4>
168
+ </div>
169
+ <div class="p-6">
170
+ <div class="space-y-3">
171
+ <a href="/cfg/admin/django_cfg_payments/admin/stats/"
172
+ class="block w-full text-center px-3 py-2 bg-blue-600 hover:bg-blue-700 dark:bg-blue-500 dark:hover:bg-blue-600 text-white rounded-md text-sm font-medium transition-colors duration-200">
173
+ View Analytics
174
+ </a>
175
+ <a href="/cfg/admin/django_cfg_payments/admin/test/"
176
+ class="block w-full text-center px-3 py-2 bg-gray-600 hover:bg-gray-700 dark:bg-gray-500 dark:hover:bg-gray-600 text-white rounded-md text-sm font-medium transition-colors duration-200">
177
+ Test Providers
178
+ </a>
179
+ <a href="/admin/django_cfg_payments/universalpayment/add/"
180
+ class="block w-full text-center px-3 py-2 border border-gray-300 dark:border-gray-500 text-gray-700 dark:text-gray-200 bg-white dark:bg-gray-700 hover:bg-gray-50 dark:hover:bg-gray-600 rounded-md text-sm font-medium transition-colors duration-200">
181
+ Create Payment
182
+ </a>
182
183
  </div>
183
184
  </div>
184
185
  </div>
@@ -189,158 +190,69 @@
189
190
 
190
191
  {% block extra_js %}
191
192
  <script>
192
- // Dashboard JavaScript functionality
193
- let currentFilters = {
194
- status: '',
195
- provider: '',
196
- search: '',
197
- date: ''
198
- };
199
-
200
- let currentPage = 1;
201
- const PAYMENTS_PER_PAGE = 10;
202
-
203
- // Filter functions
204
- function filterPayments() {
205
- currentFilters.status = document.getElementById('status-filter').value;
206
- currentFilters.provider = document.getElementById('provider-filter').value;
207
- currentFilters.date = document.getElementById('date-filter').value;
193
+ // Simplified dashboard JavaScript
194
+ document.addEventListener('DOMContentLoaded', function() {
195
+ console.log('Payment Dashboard loaded');
208
196
 
209
- currentPage = 1;
210
- loadPayments();
211
- }
212
-
213
- function searchPayments(event) {
214
- // Debounce search input
215
- clearTimeout(window.searchTimeout);
216
- window.searchTimeout = setTimeout(() => {
217
- currentFilters.search = event.target.value;
218
- currentPage = 1;
219
- loadPayments();
220
- }, 500);
221
- }
197
+ // Initialize any tooltips or interactive elements
198
+ initializeDashboard();
199
+ });
222
200
 
223
- function clearFilters() {
224
- document.getElementById('status-filter').value = '';
225
- document.getElementById('provider-filter').value = '';
226
- document.getElementById('search-input').value = '';
227
- document.getElementById('date-filter').value = '';
201
+ function initializeDashboard() {
202
+ // Add click handlers for any interactive elements
203
+ const copyButtons = document.querySelectorAll('[data-copy]');
204
+ copyButtons.forEach(button => {
205
+ button.addEventListener('click', function() {
206
+ const text = this.getAttribute('data-copy');
207
+ if (window.paymentUtils) {
208
+ window.paymentUtils.copyToClipboard(text);
209
+ }
210
+ });
211
+ });
228
212
 
229
- currentFilters = { status: '', provider: '', search: '', date: '' };
230
- currentPage = 1;
231
- loadPayments();
213
+ // Auto-refresh timestamp every minute
214
+ setInterval(updateTimestamps, 60000);
232
215
  }
233
216
 
234
- function loadPayments() {
235
- const grid = document.getElementById('payments-grid');
236
-
237
- // Show loading state
238
- if (currentPage === 1) {
239
- grid.innerHTML = '<div class="col-span-full text-center py-8"><div class="skeleton h-32 w-full"></div></div>';
240
- }
241
-
242
- // Build query parameters
243
- const params = new URLSearchParams({
244
- page: currentPage,
245
- page_size: PAYMENTS_PER_PAGE,
246
- ...currentFilters
247
- });
248
-
249
- // Remove empty parameters
250
- for (const [key, value] of params.entries()) {
251
- if (!value) {
252
- params.delete(key);
217
+ function updateTimestamps() {
218
+ const timestampElements = document.querySelectorAll('[data-timestamp]');
219
+ timestampElements.forEach(element => {
220
+ const timestamp = element.getAttribute('data-timestamp');
221
+ if (timestamp && window.paymentUtils) {
222
+ element.textContent = window.paymentUtils.timeAgo(timestamp);
253
223
  }
254
- }
255
-
256
- fetch(`/api/payments/list/?${params}`)
257
- .then(response => response.json())
258
- .then(data => {
259
- if (currentPage === 1) {
260
- grid.innerHTML = '';
261
- }
262
-
263
- if (data.results && data.results.length > 0) {
264
- data.results.forEach(payment => {
265
- const cardHtml = renderPaymentCard(payment);
266
- grid.insertAdjacentHTML('beforeend', cardHtml);
267
- });
268
-
269
- // Update load more button
270
- const loadMoreBtn = document.getElementById('load-more-btn');
271
- if (loadMoreBtn) {
272
- loadMoreBtn.style.display = data.has_next ? 'block' : 'none';
273
- }
274
- } else if (currentPage === 1) {
275
- // Show empty state
276
- grid.innerHTML = `
277
- <div class="col-span-full text-center py-12">
278
- <span class="material-icons text-gray-400 text-6xl mb-4">search_off</span>
279
- <h3 class="text-lg font-medium text-gray-900 dark:text-white mb-2">No payments found</h3>
280
- <p class="text-gray-500 dark:text-gray-400">Try adjusting your filters.</p>
281
- </div>
282
- `;
283
- }
284
- })
285
- .catch(error => {
286
- console.error('Failed to load payments:', error);
287
- if (window.notificationManager) {
288
- window.notificationManager.error('Failed to load payments');
289
- }
290
- });
224
+ });
291
225
  }
292
226
 
293
- function loadMorePayments() {
294
- currentPage++;
295
- loadPayments();
227
+ // Quick action functions (from status_overview.html)
228
+ function createNewPayment() {
229
+ window.location.href = '/admin/django_cfg_payments/universalpayment/add/';
296
230
  }
297
231
 
298
- function renderPaymentCard(payment) {
299
- // This would be a simplified version of the payment card template
300
- // In a real implementation, you'd want to render this server-side or use a template engine
301
- return `
302
- <div class="payment-card" data-payment-id="${payment.id}">
303
- <!-- Simplified payment card HTML -->
304
- <div class="p-4 bg-white dark:bg-gray-800 rounded-lg border">
305
- <div class="flex justify-between items-start mb-2">
306
- <span class="font-medium">#${payment.internal_payment_id || payment.id.substring(0, 8)}</span>
307
- <span class="px-2 py-1 text-xs rounded-full payment-status-${payment.status}">
308
- ${payment.status}
309
- </span>
310
- </div>
311
- <div class="text-xl font-bold mb-2">$${parseFloat(payment.amount_usd).toFixed(2)}</div>
312
- <div class="text-sm text-gray-500">${payment.provider} • ${new Date(payment.created_at).toLocaleDateString()}</div>
313
- </div>
314
- </div>
315
- `;
232
+ function refreshPayments() {
233
+ location.reload();
316
234
  }
317
235
 
318
- // Auto-refresh dashboard every 30 seconds
319
- setInterval(() => {
320
- if (document.visibilityState === 'visible') {
321
- loadPayments();
236
+ function exportPayments() {
237
+ if (window.notificationManager) {
238
+ window.notificationManager.info('Export functionality coming soon');
322
239
  }
323
- }, 30000);
240
+ }
324
241
 
325
- // Initialize dashboard
326
- document.addEventListener('DOMContentLoaded', () => {
327
- // Update URL parameters on filter change
328
- window.addEventListener('popstate', () => {
329
- // Handle browser back/forward
330
- const urlParams = new URLSearchParams(window.location.search);
331
- currentFilters.status = urlParams.get('status') || '';
332
- currentFilters.provider = urlParams.get('provider') || '';
333
- currentFilters.search = urlParams.get('search') || '';
334
- currentFilters.date = urlParams.get('date') || '';
335
-
336
- // Update form values
337
- document.getElementById('status-filter').value = currentFilters.status;
338
- document.getElementById('provider-filter').value = currentFilters.provider;
339
- document.getElementById('search-input').value = currentFilters.search;
340
- document.getElementById('date-filter').value = currentFilters.date;
341
-
342
- loadPayments();
343
- });
344
- });
242
+ // Debug panel toggle
243
+ function toggleDebug() {
244
+ const content = document.getElementById('debug-content');
245
+ const icon = document.getElementById('debug-icon');
246
+
247
+ if (content.classList.contains('hidden')) {
248
+ content.classList.remove('hidden');
249
+ icon.textContent = 'expand_less';
250
+ icon.classList.add('rotate-180');
251
+ } else {
252
+ content.classList.add('hidden');
253
+ icon.textContent = 'expand_more';
254
+ icon.classList.remove('rotate-180');
255
+ }
256
+ }
345
257
  </script>
346
- {% endblock %}
258
+ {% endblock %}
@@ -0,0 +1,35 @@
1
+ {% extends 'payments/base.html' %}
2
+
3
+ {% block title %}Payment Dashboard - Simple Test{% endblock %}
4
+
5
+ {% block header_title %}Payment Dashboard - Test{% endblock %}
6
+ {% block header_subtitle %}Simple test version{% endblock %}
7
+
8
+ {% block content %}
9
+ <div style="background: yellow; padding: 20px; margin: 20px; border: 2px solid red;">
10
+ <h1 style="color: black;">🔥 DASHBOARD TEST 🔥</h1>
11
+ <p style="color: black;"><strong>Payments count:</strong> {{ payments|length }}</p>
12
+ <p style="color: black;"><strong>Payment stats:</strong> {{ payment_stats }}</p>
13
+ <p style="color: black;"><strong>Provider stats:</strong> {{ provider_stats|length }}</p>
14
+
15
+ {% if payments %}
16
+ <h2 style="color: black;">Payments List:</h2>
17
+ <ul style="color: black;">
18
+ {% for payment in payments %}
19
+ <li>{{ payment.user.email }} - ${{ payment.amount_usd }} ({{ payment.status }})</li>
20
+ {% endfor %}
21
+ </ul>
22
+ {% else %}
23
+ <p style="color: red; font-size: 20px;">❌ NO PAYMENTS FOUND</p>
24
+ {% endif %}
25
+
26
+ {% if payment_stats %}
27
+ <h2 style="color: black;">Stats:</h2>
28
+ <p style="color: black;">Total: {{ payment_stats.total_payments_count }}</p>
29
+ <p style="color: black;">Pending: {{ payment_stats.pending_payments_count }}</p>
30
+ <p style="color: black;">Volume: ${{ payment_stats.total_volume }}</p>
31
+ {% else %}
32
+ <p style="color: red; font-size: 20px;">❌ NO STATS FOUND</p>
33
+ {% endif %}
34
+ </div>
35
+ {% endblock %}