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.
- django_cfg/__init__.py +1 -1
- django_cfg/apps/payments/admin/__init__.py +3 -2
- django_cfg/apps/payments/admin/balance_admin.py +18 -18
- django_cfg/apps/payments/admin/currencies_admin.py +319 -131
- django_cfg/apps/payments/admin/payments_admin.py +15 -4
- django_cfg/apps/payments/config/module.py +2 -2
- django_cfg/apps/payments/config/utils.py +2 -2
- django_cfg/apps/payments/decorators.py +2 -2
- django_cfg/apps/payments/management/commands/README.md +95 -127
- django_cfg/apps/payments/management/commands/currency_stats.py +5 -24
- django_cfg/apps/payments/management/commands/manage_currencies.py +229 -0
- django_cfg/apps/payments/management/commands/manage_providers.py +235 -0
- django_cfg/apps/payments/managers/__init__.py +3 -2
- django_cfg/apps/payments/managers/balance_manager.py +2 -2
- django_cfg/apps/payments/managers/currency_manager.py +272 -49
- django_cfg/apps/payments/managers/payment_manager.py +161 -13
- django_cfg/apps/payments/middleware/api_access.py +2 -2
- django_cfg/apps/payments/middleware/rate_limiting.py +8 -18
- django_cfg/apps/payments/middleware/usage_tracking.py +20 -17
- django_cfg/apps/payments/migrations/0002_network_providercurrency_and_more.py +241 -0
- django_cfg/apps/payments/migrations/0003_add_usd_rate_cache.py +30 -0
- django_cfg/apps/payments/models/__init__.py +3 -2
- django_cfg/apps/payments/models/currencies.py +187 -71
- django_cfg/apps/payments/models/payments.py +3 -2
- django_cfg/apps/payments/serializers/__init__.py +3 -2
- django_cfg/apps/payments/serializers/currencies.py +20 -12
- django_cfg/apps/payments/services/cache/simple_cache.py +2 -2
- django_cfg/apps/payments/services/core/balance_service.py +2 -2
- django_cfg/apps/payments/services/core/fallback_service.py +2 -2
- django_cfg/apps/payments/services/core/payment_service.py +3 -6
- django_cfg/apps/payments/services/core/subscription_service.py +4 -7
- django_cfg/apps/payments/services/internal_types.py +171 -7
- django_cfg/apps/payments/services/monitoring/api_schemas.py +58 -204
- django_cfg/apps/payments/services/monitoring/provider_health.py +2 -2
- django_cfg/apps/payments/services/providers/base.py +144 -43
- django_cfg/apps/payments/services/providers/cryptapi/__init__.py +4 -0
- django_cfg/apps/payments/services/providers/cryptapi/config.py +8 -0
- django_cfg/apps/payments/services/providers/cryptapi/models.py +192 -0
- django_cfg/apps/payments/services/providers/cryptapi/provider.py +439 -0
- django_cfg/apps/payments/services/providers/cryptomus/__init__.py +4 -0
- django_cfg/apps/payments/services/providers/cryptomus/models.py +176 -0
- django_cfg/apps/payments/services/providers/cryptomus/provider.py +429 -0
- django_cfg/apps/payments/services/providers/cryptomus/provider_v2.py +564 -0
- django_cfg/apps/payments/services/providers/models/__init__.py +34 -0
- django_cfg/apps/payments/services/providers/models/currencies.py +190 -0
- django_cfg/apps/payments/services/providers/nowpayments/__init__.py +4 -0
- django_cfg/apps/payments/services/providers/nowpayments/models.py +196 -0
- django_cfg/apps/payments/services/providers/nowpayments/provider.py +380 -0
- django_cfg/apps/payments/services/providers/registry.py +294 -11
- django_cfg/apps/payments/services/providers/stripe/__init__.py +4 -0
- django_cfg/apps/payments/services/providers/stripe/models.py +184 -0
- django_cfg/apps/payments/services/providers/stripe/provider.py +109 -0
- django_cfg/apps/payments/services/security/error_handler.py +6 -8
- django_cfg/apps/payments/services/security/payment_notifications.py +2 -2
- django_cfg/apps/payments/services/security/webhook_validator.py +3 -4
- django_cfg/apps/payments/signals/api_key_signals.py +2 -2
- django_cfg/apps/payments/signals/payment_signals.py +11 -5
- django_cfg/apps/payments/signals/subscription_signals.py +2 -2
- django_cfg/apps/payments/tasks/webhook_processing.py +2 -2
- django_cfg/apps/payments/templates/admin/payments/currency/change_list.html +50 -0
- django_cfg/apps/payments/templates/payments/base.html +4 -4
- django_cfg/apps/payments/templates/payments/components/payment_card.html +6 -6
- django_cfg/apps/payments/templates/payments/components/payment_qr_code.html +4 -4
- django_cfg/apps/payments/templates/payments/components/progress_bar.html +14 -7
- django_cfg/apps/payments/templates/payments/components/provider_stats.html +2 -2
- django_cfg/apps/payments/templates/payments/components/status_badge.html +8 -1
- django_cfg/apps/payments/templates/payments/components/status_overview.html +34 -30
- django_cfg/apps/payments/templates/payments/dashboard.html +202 -290
- django_cfg/apps/payments/templates/payments/dashboard_simple_test.html +35 -0
- django_cfg/apps/payments/templates/payments/payment_create.html +579 -0
- django_cfg/apps/payments/templates/payments/payment_detail.html +373 -0
- django_cfg/apps/payments/templates/payments/payment_list.html +354 -0
- django_cfg/apps/payments/templates/payments/stats.html +261 -0
- django_cfg/apps/payments/templates/payments/test.html +213 -0
- django_cfg/apps/payments/urls.py +3 -1
- django_cfg/apps/payments/{urls_templates.py → urls_admin.py} +6 -0
- django_cfg/apps/payments/utils/__init__.py +1 -3
- django_cfg/apps/payments/utils/billing_utils.py +2 -2
- django_cfg/apps/payments/utils/config_utils.py +2 -8
- django_cfg/apps/payments/utils/validation_utils.py +2 -2
- django_cfg/apps/payments/views/__init__.py +3 -2
- django_cfg/apps/payments/views/currency_views.py +31 -20
- django_cfg/apps/payments/views/payment_views.py +2 -2
- django_cfg/apps/payments/views/templates/ajax.py +141 -2
- django_cfg/apps/payments/views/templates/base.py +21 -13
- django_cfg/apps/payments/views/templates/payment_detail.py +1 -1
- django_cfg/apps/payments/views/templates/payment_management.py +34 -40
- django_cfg/apps/payments/views/templates/stats.py +8 -4
- django_cfg/apps/payments/views/webhook_views.py +2 -2
- django_cfg/apps/payments/viewsets.py +3 -2
- django_cfg/apps/tasks/urls.py +0 -2
- django_cfg/apps/tasks/urls_admin.py +14 -0
- django_cfg/apps/urls.py +4 -4
- django_cfg/core/config.py +35 -0
- django_cfg/models/payments.py +2 -8
- django_cfg/modules/django_currency/__init__.py +16 -11
- django_cfg/modules/django_currency/clients/__init__.py +4 -4
- django_cfg/modules/django_currency/clients/coinpaprika_client.py +289 -0
- django_cfg/modules/django_currency/clients/yahoo_client.py +157 -0
- django_cfg/modules/django_currency/core/__init__.py +1 -7
- django_cfg/modules/django_currency/core/converter.py +18 -23
- django_cfg/modules/django_currency/core/models.py +122 -11
- django_cfg/modules/django_currency/database/__init__.py +4 -4
- django_cfg/modules/django_currency/database/database_loader.py +190 -309
- django_cfg/modules/django_unfold/dashboard.py +7 -2
- django_cfg/template_archive/django_sample.zip +0 -0
- django_cfg/templates/admin/components/action_grid.html +9 -9
- django_cfg/templates/admin/components/metric_card.html +5 -5
- django_cfg/templates/admin/components/status_badge.html +2 -2
- django_cfg/templates/admin/layouts/dashboard_with_tabs.html +152 -24
- django_cfg/templates/admin/snippets/components/quick_actions.html +3 -3
- django_cfg/templates/admin/snippets/components/system_health.html +1 -1
- django_cfg/templates/admin/snippets/tabs/overview_tab.html +49 -52
- {django_cfg-1.2.29.dist-info → django_cfg-1.2.31.dist-info}/METADATA +2 -4
- {django_cfg-1.2.29.dist-info → django_cfg-1.2.31.dist-info}/RECORD +118 -96
- django_cfg/apps/payments/management/commands/populate_currencies.py +0 -246
- django_cfg/apps/payments/management/commands/update_currencies.py +0 -336
- django_cfg/apps/payments/services/providers/cryptapi.py +0 -273
- django_cfg/apps/payments/services/providers/cryptomus.py +0 -311
- django_cfg/apps/payments/services/providers/nowpayments.py +0 -293
- django_cfg/apps/payments/services/validators/__init__.py +0 -8
- django_cfg/modules/django_currency/clients/coingecko_client.py +0 -257
- django_cfg/modules/django_currency/clients/yfinance_client.py +0 -246
- {django_cfg-1.2.29.dist-info → django_cfg-1.2.31.dist-info}/WHEEL +0 -0
- {django_cfg-1.2.29.dist-info → django_cfg-1.2.31.dist-info}/entry_points.txt +0 -0
- {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="
|
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="
|
34
|
-
<!--
|
35
|
-
<div class="
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
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="
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
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
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
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
|
-
<!--
|
101
|
+
<!-- Sidebar Stats -->
|
109
102
|
<div class="space-y-6">
|
110
|
-
<!-- Provider
|
111
|
-
<div class="bg-white dark:bg-gray-800
|
112
|
-
<
|
113
|
-
|
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
|
-
<!--
|
117
|
-
<div class="bg-white dark:bg-gray-800
|
118
|
-
<
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
123
|
-
|
124
|
-
|
125
|
-
|
126
|
-
|
127
|
-
|
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-
|
134
|
-
<
|
135
|
-
|
136
|
-
|
137
|
-
|
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="
|
143
|
-
|
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
|
-
<!--
|
155
|
-
<div class="bg-white dark:bg-gray-800
|
156
|
-
<
|
157
|
-
|
158
|
-
|
159
|
-
|
160
|
-
|
161
|
-
|
162
|
-
|
163
|
-
|
164
|
-
|
165
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
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
|
-
//
|
193
|
-
|
194
|
-
|
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
|
-
|
210
|
-
|
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
|
224
|
-
|
225
|
-
document.
|
226
|
-
|
227
|
-
|
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
|
-
|
230
|
-
|
231
|
-
loadPayments();
|
213
|
+
// Auto-refresh timestamp every minute
|
214
|
+
setInterval(updateTimestamps, 60000);
|
232
215
|
}
|
233
216
|
|
234
|
-
function
|
235
|
-
const
|
236
|
-
|
237
|
-
|
238
|
-
|
239
|
-
|
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
|
-
|
294
|
-
|
295
|
-
|
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
|
299
|
-
|
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
|
-
|
319
|
-
|
320
|
-
|
321
|
-
loadPayments();
|
236
|
+
function exportPayments() {
|
237
|
+
if (window.notificationManager) {
|
238
|
+
window.notificationManager.info('Export functionality coming soon');
|
322
239
|
}
|
323
|
-
}
|
240
|
+
}
|
324
241
|
|
325
|
-
//
|
326
|
-
|
327
|
-
|
328
|
-
|
329
|
-
|
330
|
-
|
331
|
-
|
332
|
-
|
333
|
-
|
334
|
-
|
335
|
-
|
336
|
-
|
337
|
-
|
338
|
-
|
339
|
-
|
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 %}
|